LiveTvManager.cs 92 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523
  1. using MediaBrowser.Common.Configuration;
  2. using MediaBrowser.Common.Extensions;
  3. using MediaBrowser.Common.Progress;
  4. using MediaBrowser.Controller.Configuration;
  5. using MediaBrowser.Controller.Drawing;
  6. using MediaBrowser.Controller.Dto;
  7. using MediaBrowser.Controller.Entities;
  8. using MediaBrowser.Controller.Library;
  9. using MediaBrowser.Controller.LiveTv;
  10. using MediaBrowser.Controller.Persistence;
  11. using MediaBrowser.Controller.Providers;
  12. using MediaBrowser.Controller.Sorting;
  13. using MediaBrowser.Model.Dto;
  14. using MediaBrowser.Model.Entities;
  15. using MediaBrowser.Model.LiveTv;
  16. using Microsoft.Extensions.Logging;
  17. using MediaBrowser.Model.Querying;
  18. using MediaBrowser.Model.Serialization;
  19. using System;
  20. using System.Collections.Generic;
  21. using System.Linq;
  22. using System.Threading;
  23. using System.Threading.Tasks;
  24. using MediaBrowser.Model.IO;
  25. using MediaBrowser.Common.Security;
  26. using MediaBrowser.Controller.Entities.Movies;
  27. using MediaBrowser.Controller.Entities.TV;
  28. using MediaBrowser.Model.Events;
  29. using MediaBrowser.Model.Extensions;
  30. using MediaBrowser.Model.Globalization;
  31. using MediaBrowser.Model.Tasks;
  32. using Emby.Server.Implementations.LiveTv.Listings;
  33. using MediaBrowser.Controller.Channels;
  34. using Emby.Server.Implementations.Library;
  35. using MediaBrowser.Controller;
  36. using MediaBrowser.Common.Net;
  37. namespace Emby.Server.Implementations.LiveTv
  38. {
  39. /// <summary>
  40. /// Class LiveTvManager
  41. /// </summary>
  42. public class LiveTvManager : ILiveTvManager, IDisposable
  43. {
  44. private readonly IServerConfigurationManager _config;
  45. private readonly ILogger _logger;
  46. private readonly IItemRepository _itemRepo;
  47. private readonly IUserManager _userManager;
  48. private readonly IUserDataManager _userDataManager;
  49. private readonly ILibraryManager _libraryManager;
  50. private readonly ITaskManager _taskManager;
  51. private readonly IJsonSerializer _jsonSerializer;
  52. private readonly IProviderManager _providerManager;
  53. private readonly ISecurityManager _security;
  54. private readonly Func<IChannelManager> _channelManager;
  55. private readonly IDtoService _dtoService;
  56. private readonly ILocalizationManager _localization;
  57. private readonly LiveTvDtoService _tvDtoService;
  58. private ILiveTvService[] _services = new ILiveTvService[] { };
  59. private ITunerHost[] _tunerHosts = Array.Empty<ITunerHost>();
  60. private IListingsProvider[] _listingProviders = Array.Empty<IListingsProvider>();
  61. private readonly IFileSystem _fileSystem;
  62. public event EventHandler<GenericEventArgs<TimerEventInfo>> SeriesTimerCancelled;
  63. public event EventHandler<GenericEventArgs<TimerEventInfo>> TimerCancelled;
  64. public event EventHandler<GenericEventArgs<TimerEventInfo>> TimerCreated;
  65. public event EventHandler<GenericEventArgs<TimerEventInfo>> SeriesTimerCreated;
  66. public string GetEmbyTvActiveRecordingPath(string id)
  67. {
  68. return EmbyTV.EmbyTV.Current.GetActiveRecordingPath(id);
  69. }
  70. private IServerApplicationHost _appHost;
  71. private IHttpClient _httpClient;
  72. public LiveTvManager(IServerApplicationHost appHost, IHttpClient httpClient, IServerConfigurationManager config, ILogger logger, IItemRepository itemRepo, IImageProcessor imageProcessor, IUserDataManager userDataManager, IDtoService dtoService, IUserManager userManager, ILibraryManager libraryManager, ITaskManager taskManager, ILocalizationManager localization, IJsonSerializer jsonSerializer, IProviderManager providerManager, IFileSystem fileSystem, ISecurityManager security, Func<IChannelManager> channelManager)
  73. {
  74. _appHost = appHost;
  75. _config = config;
  76. _logger = logger;
  77. _itemRepo = itemRepo;
  78. _userManager = userManager;
  79. _libraryManager = libraryManager;
  80. _taskManager = taskManager;
  81. _localization = localization;
  82. _jsonSerializer = jsonSerializer;
  83. _providerManager = providerManager;
  84. _fileSystem = fileSystem;
  85. _security = security;
  86. _dtoService = dtoService;
  87. _userDataManager = userDataManager;
  88. _channelManager = channelManager;
  89. _httpClient = httpClient;
  90. _tvDtoService = new LiveTvDtoService(dtoService, imageProcessor, logger, appHost, _libraryManager);
  91. }
  92. /// <summary>
  93. /// Gets the services.
  94. /// </summary>
  95. /// <value>The services.</value>
  96. public IReadOnlyList<ILiveTvService> Services
  97. {
  98. get { return _services; }
  99. }
  100. private LiveTvOptions GetConfiguration()
  101. {
  102. return _config.GetConfiguration<LiveTvOptions>("livetv");
  103. }
  104. /// <summary>
  105. /// Adds the parts.
  106. /// </summary>
  107. /// <param name="services">The services.</param>
  108. /// <param name="tunerHosts">The tuner hosts.</param>
  109. /// <param name="listingProviders">The listing providers.</param>
  110. public void AddParts(IEnumerable<ILiveTvService> services, IEnumerable<ITunerHost> tunerHosts, IEnumerable<IListingsProvider> listingProviders)
  111. {
  112. _services = services.ToArray();
  113. _tunerHosts = tunerHosts.Where(i => i.IsSupported).ToArray();
  114. _listingProviders = listingProviders.ToArray();
  115. foreach (var service in _services)
  116. {
  117. service.DataSourceChanged += service_DataSourceChanged;
  118. if (service is EmbyTV.EmbyTV embyTv)
  119. {
  120. embyTv.TimerCreated += EmbyTv_TimerCreated;
  121. embyTv.TimerCancelled += EmbyTv_TimerCancelled;
  122. }
  123. }
  124. }
  125. private void EmbyTv_TimerCancelled(object sender, GenericEventArgs<string> e)
  126. {
  127. var timerId = e.Argument;
  128. TimerCancelled?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  129. {
  130. Argument = new TimerEventInfo
  131. {
  132. Id = timerId
  133. }
  134. });
  135. }
  136. private void EmbyTv_TimerCreated(object sender, GenericEventArgs<TimerInfo> e)
  137. {
  138. var timer = e.Argument;
  139. var service = sender as ILiveTvService;
  140. TimerCreated?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  141. {
  142. Argument = new TimerEventInfo
  143. {
  144. ProgramId = _tvDtoService.GetInternalProgramId(timer.ProgramId),
  145. Id = timer.Id
  146. }
  147. });
  148. }
  149. public ITunerHost[] TunerHosts
  150. {
  151. get { return _tunerHosts; }
  152. }
  153. public IListingsProvider[] ListingProviders
  154. {
  155. get { return _listingProviders; }
  156. }
  157. public List<NameIdPair> GetTunerHostTypes()
  158. {
  159. return _tunerHosts.OrderBy(i => i.Name).Select(i => new NameIdPair
  160. {
  161. Name = i.Name,
  162. Id = i.Type
  163. }).ToList();
  164. }
  165. public Task<List<TunerHostInfo>> DiscoverTuners(bool newDevicesOnly, CancellationToken cancellationToken)
  166. {
  167. return EmbyTV.EmbyTV.Current.DiscoverTuners(newDevicesOnly, cancellationToken);
  168. }
  169. void service_DataSourceChanged(object sender, EventArgs e)
  170. {
  171. if (!_isDisposed)
  172. {
  173. _taskManager.CancelIfRunningAndQueue<RefreshChannelsScheduledTask>();
  174. }
  175. }
  176. public QueryResult<BaseItem> GetInternalChannels(LiveTvChannelQuery query, DtoOptions dtoOptions, CancellationToken cancellationToken)
  177. {
  178. var user = query.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(query.UserId);
  179. var topFolder = GetInternalLiveTvFolder(cancellationToken);
  180. var internalQuery = new InternalItemsQuery(user)
  181. {
  182. IsMovie = query.IsMovie,
  183. IsNews = query.IsNews,
  184. IsKids = query.IsKids,
  185. IsSports = query.IsSports,
  186. IsSeries = query.IsSeries,
  187. IncludeItemTypes = new[] { typeof(LiveTvChannel).Name },
  188. TopParentIds = new[] { topFolder.Id },
  189. IsFavorite = query.IsFavorite,
  190. IsLiked = query.IsLiked,
  191. StartIndex = query.StartIndex,
  192. Limit = query.Limit,
  193. DtoOptions = dtoOptions
  194. };
  195. var orderBy = internalQuery.OrderBy.ToList();
  196. orderBy.AddRange(query.SortBy.Select(i => new ValueTuple<string, SortOrder>(i, query.SortOrder ?? SortOrder.Ascending)));
  197. if (query.EnableFavoriteSorting)
  198. {
  199. orderBy.Insert(0, new ValueTuple<string, SortOrder>(ItemSortBy.IsFavoriteOrLiked, SortOrder.Descending));
  200. }
  201. if (!internalQuery.OrderBy.Any(i => string.Equals(i.Item1, ItemSortBy.SortName, StringComparison.OrdinalIgnoreCase)))
  202. {
  203. orderBy.Add(new ValueTuple<string, SortOrder>(ItemSortBy.SortName, SortOrder.Ascending));
  204. }
  205. internalQuery.OrderBy = orderBy.ToArray();
  206. return _libraryManager.GetItemsResult(internalQuery);
  207. }
  208. public async Task<Tuple<MediaSourceInfo, ILiveStream>> GetChannelStream(string id, string mediaSourceId, List<ILiveStream> currentLiveStreams, CancellationToken cancellationToken)
  209. {
  210. if (string.Equals(id, mediaSourceId, StringComparison.OrdinalIgnoreCase))
  211. {
  212. mediaSourceId = null;
  213. }
  214. var channel = (LiveTvChannel)_libraryManager.GetItemById(id);
  215. bool isVideo = channel.ChannelType == ChannelType.TV;
  216. ILiveTvService service = GetService(channel);
  217. _logger.LogInformation("Opening channel stream from {0}, external channel Id: {1}", service.Name, channel.ExternalId);
  218. MediaSourceInfo info;
  219. ILiveStream liveStream;
  220. if (service is ISupportsDirectStreamProvider supportsManagedStream)
  221. {
  222. liveStream = await supportsManagedStream.GetChannelStreamWithDirectStreamProvider(channel.ExternalId, mediaSourceId, currentLiveStreams, cancellationToken).ConfigureAwait(false);
  223. info = liveStream.MediaSource;
  224. }
  225. else
  226. {
  227. info = await service.GetChannelStream(channel.ExternalId, mediaSourceId, cancellationToken).ConfigureAwait(false);
  228. var openedId = info.Id;
  229. Func<Task> closeFn = () => service.CloseLiveStream(openedId, CancellationToken.None);
  230. liveStream = new ExclusiveLiveStream(info, closeFn);
  231. var startTime = DateTime.UtcNow;
  232. await liveStream.Open(cancellationToken).ConfigureAwait(false);
  233. var endTime = DateTime.UtcNow;
  234. _logger.LogInformation("Live stream opened after {0}ms", (endTime - startTime).TotalMilliseconds);
  235. }
  236. info.RequiresClosing = true;
  237. var idPrefix = service.GetType().FullName.GetMD5().ToString("N") + "_";
  238. info.LiveStreamId = idPrefix + info.Id;
  239. Normalize(info, service, isVideo);
  240. return new Tuple<MediaSourceInfo, ILiveStream>(info, liveStream);
  241. }
  242. public async Task<IEnumerable<MediaSourceInfo>> GetChannelMediaSources(BaseItem item, CancellationToken cancellationToken)
  243. {
  244. var baseItem = (LiveTvChannel)item;
  245. var service = GetService(baseItem);
  246. var sources = await service.GetChannelStreamMediaSources(baseItem.ExternalId, cancellationToken).ConfigureAwait(false);
  247. if (sources.Count == 0)
  248. {
  249. throw new NotImplementedException();
  250. }
  251. foreach (var source in sources)
  252. {
  253. Normalize(source, service, baseItem.ChannelType == ChannelType.TV);
  254. }
  255. return sources;
  256. }
  257. private ILiveTvService GetService(LiveTvChannel item)
  258. {
  259. var name = item.ServiceName;
  260. return GetService(name);
  261. }
  262. private ILiveTvService GetService(LiveTvProgram item)
  263. {
  264. var channel = _libraryManager.GetItemById(item.ChannelId) as LiveTvChannel;
  265. return GetService(channel);
  266. }
  267. private ILiveTvService GetService(string name)
  268. {
  269. return _services.FirstOrDefault(i => string.Equals(i.Name, name, StringComparison.OrdinalIgnoreCase));
  270. }
  271. private void Normalize(MediaSourceInfo mediaSource, ILiveTvService service, bool isVideo)
  272. {
  273. // Not all of the plugins are setting this
  274. mediaSource.IsInfiniteStream = true;
  275. if (mediaSource.MediaStreams.Count == 0)
  276. {
  277. if (isVideo)
  278. {
  279. mediaSource.MediaStreams.AddRange(new List<MediaStream>
  280. {
  281. new MediaStream
  282. {
  283. Type = MediaStreamType.Video,
  284. // Set the index to -1 because we don't know the exact index of the video stream within the container
  285. Index = -1,
  286. // Set to true if unknown to enable deinterlacing
  287. IsInterlaced = true
  288. },
  289. new MediaStream
  290. {
  291. Type = MediaStreamType.Audio,
  292. // Set the index to -1 because we don't know the exact index of the audio stream within the container
  293. Index = -1
  294. }
  295. });
  296. }
  297. else
  298. {
  299. mediaSource.MediaStreams.AddRange(new List<MediaStream>
  300. {
  301. new MediaStream
  302. {
  303. Type = MediaStreamType.Audio,
  304. // Set the index to -1 because we don't know the exact index of the audio stream within the container
  305. Index = -1
  306. }
  307. });
  308. }
  309. }
  310. // Clean some bad data coming from providers
  311. foreach (var stream in mediaSource.MediaStreams)
  312. {
  313. if (stream.BitRate.HasValue && stream.BitRate <= 0)
  314. {
  315. stream.BitRate = null;
  316. }
  317. if (stream.Channels.HasValue && stream.Channels <= 0)
  318. {
  319. stream.Channels = null;
  320. }
  321. if (stream.AverageFrameRate.HasValue && stream.AverageFrameRate <= 0)
  322. {
  323. stream.AverageFrameRate = null;
  324. }
  325. if (stream.RealFrameRate.HasValue && stream.RealFrameRate <= 0)
  326. {
  327. stream.RealFrameRate = null;
  328. }
  329. if (stream.Width.HasValue && stream.Width <= 0)
  330. {
  331. stream.Width = null;
  332. }
  333. if (stream.Height.HasValue && stream.Height <= 0)
  334. {
  335. stream.Height = null;
  336. }
  337. if (stream.SampleRate.HasValue && stream.SampleRate <= 0)
  338. {
  339. stream.SampleRate = null;
  340. }
  341. if (stream.Level.HasValue && stream.Level <= 0)
  342. {
  343. stream.Level = null;
  344. }
  345. }
  346. var indexes = mediaSource.MediaStreams.Select(i => i.Index).Distinct().ToList();
  347. // If there are duplicate stream indexes, set them all to unknown
  348. if (indexes.Count != mediaSource.MediaStreams.Count)
  349. {
  350. foreach (var stream in mediaSource.MediaStreams)
  351. {
  352. stream.Index = -1;
  353. }
  354. }
  355. // Set the total bitrate if not already supplied
  356. mediaSource.InferTotalBitrate();
  357. if (!(service is EmbyTV.EmbyTV))
  358. {
  359. // We can't trust that we'll be able to direct stream it through emby server, no matter what the provider says
  360. //mediaSource.SupportsDirectPlay = false;
  361. //mediaSource.SupportsDirectStream = false;
  362. mediaSource.SupportsTranscoding = true;
  363. foreach (var stream in mediaSource.MediaStreams)
  364. {
  365. if (stream.Type == MediaStreamType.Video && string.IsNullOrWhiteSpace(stream.NalLengthSize))
  366. {
  367. stream.NalLengthSize = "0";
  368. }
  369. if (stream.Type == MediaStreamType.Video)
  370. {
  371. stream.IsInterlaced = true;
  372. }
  373. }
  374. }
  375. }
  376. private const string ExternalServiceTag = "ExternalServiceId";
  377. private LiveTvChannel GetChannel(ChannelInfo channelInfo, string serviceName, BaseItem parentFolder, CancellationToken cancellationToken)
  378. {
  379. var parentFolderId = parentFolder.Id;
  380. var isNew = false;
  381. var forceUpdate = false;
  382. var id = _tvDtoService.GetInternalChannelId(serviceName, channelInfo.Id);
  383. var item = _libraryManager.GetItemById(id) as LiveTvChannel;
  384. if (item == null)
  385. {
  386. item = new LiveTvChannel
  387. {
  388. Name = channelInfo.Name,
  389. Id = id,
  390. DateCreated = DateTime.UtcNow
  391. };
  392. isNew = true;
  393. }
  394. if (channelInfo.Tags != null)
  395. {
  396. if (!channelInfo.Tags.SequenceEqual(item.Tags, StringComparer.OrdinalIgnoreCase))
  397. {
  398. isNew = true;
  399. }
  400. item.Tags = channelInfo.Tags;
  401. }
  402. if (!item.ParentId.Equals(parentFolderId))
  403. {
  404. isNew = true;
  405. }
  406. item.ParentId = parentFolderId;
  407. item.ChannelType = channelInfo.ChannelType;
  408. item.ServiceName = serviceName;
  409. if (!string.Equals(item.GetProviderId(ExternalServiceTag), serviceName, StringComparison.OrdinalIgnoreCase))
  410. {
  411. forceUpdate = true;
  412. }
  413. item.SetProviderId(ExternalServiceTag, serviceName);
  414. if (!string.Equals(channelInfo.Id, item.ExternalId, StringComparison.Ordinal))
  415. {
  416. forceUpdate = true;
  417. }
  418. item.ExternalId = channelInfo.Id;
  419. if (!string.Equals(channelInfo.Number, item.Number, StringComparison.Ordinal))
  420. {
  421. forceUpdate = true;
  422. }
  423. item.Number = channelInfo.Number;
  424. if (!string.Equals(channelInfo.Name, item.Name, StringComparison.Ordinal))
  425. {
  426. forceUpdate = true;
  427. }
  428. item.Name = channelInfo.Name;
  429. if (!item.HasImage(ImageType.Primary))
  430. {
  431. if (!string.IsNullOrWhiteSpace(channelInfo.ImagePath))
  432. {
  433. item.SetImagePath(ImageType.Primary, channelInfo.ImagePath);
  434. forceUpdate = true;
  435. }
  436. else if (!string.IsNullOrWhiteSpace(channelInfo.ImageUrl))
  437. {
  438. item.SetImagePath(ImageType.Primary, channelInfo.ImageUrl);
  439. forceUpdate = true;
  440. }
  441. }
  442. if (isNew)
  443. {
  444. _libraryManager.CreateItem(item, parentFolder);
  445. }
  446. else if (forceUpdate)
  447. {
  448. _libraryManager.UpdateItem(item, parentFolder, ItemUpdateType.MetadataImport, cancellationToken);
  449. }
  450. return item;
  451. }
  452. private const string EtagKey = "ProgramEtag";
  453. private Tuple<LiveTvProgram, bool, bool> GetProgram(ProgramInfo info, Dictionary<Guid, LiveTvProgram> allExistingPrograms, LiveTvChannel channel, ChannelType channelType, string serviceName, CancellationToken cancellationToken)
  454. {
  455. var id = _tvDtoService.GetInternalProgramId(info.Id);
  456. var isNew = false;
  457. var forceUpdate = false;
  458. LiveTvProgram item;
  459. if (!allExistingPrograms.TryGetValue(id, out item))
  460. {
  461. isNew = true;
  462. item = new LiveTvProgram
  463. {
  464. Name = info.Name,
  465. Id = id,
  466. DateCreated = DateTime.UtcNow,
  467. DateModified = DateTime.UtcNow
  468. };
  469. if (!string.IsNullOrEmpty(info.Etag))
  470. {
  471. item.SetProviderId(EtagKey, info.Etag);
  472. }
  473. }
  474. if (!string.Equals(info.ShowId, item.ShowId, StringComparison.OrdinalIgnoreCase))
  475. {
  476. item.ShowId = info.ShowId;
  477. forceUpdate = true;
  478. }
  479. var seriesId = info.SeriesId;
  480. if (!item.ParentId.Equals(channel.Id))
  481. {
  482. forceUpdate = true;
  483. }
  484. item.ParentId = channel.Id;
  485. //item.ChannelType = channelType;
  486. item.Audio = info.Audio;
  487. item.ChannelId = channel.Id;
  488. item.CommunityRating = item.CommunityRating ?? info.CommunityRating;
  489. if ((item.CommunityRating ?? 0).Equals(0))
  490. {
  491. item.CommunityRating = null;
  492. }
  493. item.EpisodeTitle = info.EpisodeTitle;
  494. item.ExternalId = info.Id;
  495. if (!string.IsNullOrWhiteSpace(seriesId) && !string.Equals(item.ExternalSeriesId, seriesId, StringComparison.Ordinal))
  496. {
  497. forceUpdate = true;
  498. }
  499. item.ExternalSeriesId = seriesId;
  500. var isSeries = info.IsSeries || !string.IsNullOrEmpty(info.EpisodeTitle);
  501. if (isSeries || !string.IsNullOrEmpty(info.EpisodeTitle))
  502. {
  503. item.SeriesName = info.Name;
  504. }
  505. var tags = new List<string>();
  506. if (info.IsLive)
  507. {
  508. tags.Add("Live");
  509. }
  510. if (info.IsPremiere)
  511. {
  512. tags.Add("Premiere");
  513. }
  514. if (info.IsNews)
  515. {
  516. tags.Add("News");
  517. }
  518. if (info.IsSports)
  519. {
  520. tags.Add("Sports");
  521. }
  522. if (info.IsKids)
  523. {
  524. tags.Add("Kids");
  525. }
  526. if (info.IsRepeat)
  527. {
  528. tags.Add("Repeat");
  529. }
  530. if (info.IsMovie)
  531. {
  532. tags.Add("Movie");
  533. }
  534. if (isSeries)
  535. {
  536. tags.Add("Series");
  537. }
  538. item.Tags = tags.ToArray();
  539. item.Genres = info.Genres.ToArray();
  540. if (info.IsHD ?? false)
  541. {
  542. item.Width = 1280;
  543. item.Height = 720;
  544. }
  545. item.IsMovie = info.IsMovie;
  546. item.IsRepeat = info.IsRepeat;
  547. if (item.IsSeries != isSeries)
  548. {
  549. forceUpdate = true;
  550. }
  551. item.IsSeries = isSeries;
  552. item.Name = info.Name;
  553. item.OfficialRating = item.OfficialRating ?? info.OfficialRating;
  554. item.Overview = item.Overview ?? info.Overview;
  555. item.RunTimeTicks = (info.EndDate - info.StartDate).Ticks;
  556. item.ProviderIds = info.ProviderIds;
  557. foreach (var providerId in info.SeriesProviderIds)
  558. {
  559. info.ProviderIds["Series" + providerId.Key] = providerId.Value;
  560. }
  561. if (item.StartDate != info.StartDate)
  562. {
  563. forceUpdate = true;
  564. }
  565. item.StartDate = info.StartDate;
  566. if (item.EndDate != info.EndDate)
  567. {
  568. forceUpdate = true;
  569. }
  570. item.EndDate = info.EndDate;
  571. item.ProductionYear = info.ProductionYear;
  572. if (!isSeries || info.IsRepeat)
  573. {
  574. item.PremiereDate = info.OriginalAirDate;
  575. }
  576. item.IndexNumber = info.EpisodeNumber;
  577. item.ParentIndexNumber = info.SeasonNumber;
  578. if (!item.HasImage(ImageType.Primary))
  579. {
  580. if (!string.IsNullOrWhiteSpace(info.ImagePath))
  581. {
  582. item.SetImage(new ItemImageInfo
  583. {
  584. Path = info.ImagePath,
  585. Type = ImageType.Primary
  586. }, 0);
  587. }
  588. else if (!string.IsNullOrWhiteSpace(info.ImageUrl))
  589. {
  590. item.SetImage(new ItemImageInfo
  591. {
  592. Path = info.ImageUrl,
  593. Type = ImageType.Primary
  594. }, 0);
  595. }
  596. }
  597. if (!item.HasImage(ImageType.Thumb))
  598. {
  599. if (!string.IsNullOrWhiteSpace(info.ThumbImageUrl))
  600. {
  601. item.SetImage(new ItemImageInfo
  602. {
  603. Path = info.ThumbImageUrl,
  604. Type = ImageType.Thumb
  605. }, 0);
  606. }
  607. }
  608. if (!item.HasImage(ImageType.Logo))
  609. {
  610. if (!string.IsNullOrWhiteSpace(info.LogoImageUrl))
  611. {
  612. item.SetImage(new ItemImageInfo
  613. {
  614. Path = info.LogoImageUrl,
  615. Type = ImageType.Logo
  616. }, 0);
  617. }
  618. }
  619. if (!item.HasImage(ImageType.Backdrop))
  620. {
  621. if (!string.IsNullOrWhiteSpace(info.BackdropImageUrl))
  622. {
  623. item.SetImage(new ItemImageInfo
  624. {
  625. Path = info.BackdropImageUrl,
  626. Type = ImageType.Backdrop
  627. }, 0);
  628. }
  629. }
  630. var isUpdated = false;
  631. if (isNew)
  632. {
  633. }
  634. else if (forceUpdate || string.IsNullOrWhiteSpace(info.Etag))
  635. {
  636. isUpdated = true;
  637. }
  638. else
  639. {
  640. var etag = info.Etag;
  641. if (!string.Equals(etag, item.GetProviderId(EtagKey), StringComparison.OrdinalIgnoreCase))
  642. {
  643. item.SetProviderId(EtagKey, etag);
  644. isUpdated = true;
  645. }
  646. }
  647. if (isNew || isUpdated)
  648. {
  649. item.OnMetadataChanged();
  650. }
  651. return new Tuple<LiveTvProgram, bool, bool>(item, isNew, isUpdated);
  652. }
  653. public async Task<BaseItemDto> GetProgram(string id, CancellationToken cancellationToken, User user = null)
  654. {
  655. var program = _libraryManager.GetItemById(id);
  656. var dto = _dtoService.GetBaseItemDto(program, new DtoOptions(), user);
  657. var list = new List<Tuple<BaseItemDto, string, string>>() {
  658. new Tuple<BaseItemDto, string, string>(dto, program.ExternalId, program.ExternalSeriesId)
  659. };
  660. await AddRecordingInfo(list, cancellationToken).ConfigureAwait(false);
  661. return dto;
  662. }
  663. public async Task<QueryResult<BaseItemDto>> GetPrograms(InternalItemsQuery query, DtoOptions options, CancellationToken cancellationToken)
  664. {
  665. var user = query.User;
  666. var topFolder = GetInternalLiveTvFolder(cancellationToken);
  667. if (query.OrderBy.Length == 0)
  668. {
  669. if (query.IsAiring ?? false)
  670. {
  671. // Unless something else was specified, order by start date to take advantage of a specialized index
  672. query.OrderBy = new ValueTuple<string, SortOrder>[]
  673. {
  674. new ValueTuple<string, SortOrder>(ItemSortBy.StartDate, SortOrder.Ascending)
  675. };
  676. }
  677. else
  678. {
  679. // Unless something else was specified, order by start date to take advantage of a specialized index
  680. query.OrderBy = new ValueTuple<string, SortOrder>[]
  681. {
  682. new ValueTuple<string, SortOrder>(ItemSortBy.StartDate, SortOrder.Ascending)
  683. };
  684. }
  685. }
  686. RemoveFields(options);
  687. var internalQuery = new InternalItemsQuery(user)
  688. {
  689. IncludeItemTypes = new[] { typeof(LiveTvProgram).Name },
  690. MinEndDate = query.MinEndDate,
  691. MinStartDate = query.MinStartDate,
  692. MaxEndDate = query.MaxEndDate,
  693. MaxStartDate = query.MaxStartDate,
  694. ChannelIds = query.ChannelIds,
  695. IsMovie = query.IsMovie,
  696. IsSeries = query.IsSeries,
  697. IsSports = query.IsSports,
  698. IsKids = query.IsKids,
  699. IsNews = query.IsNews,
  700. Genres = query.Genres,
  701. GenreIds = query.GenreIds,
  702. StartIndex = query.StartIndex,
  703. Limit = query.Limit,
  704. OrderBy = query.OrderBy,
  705. EnableTotalRecordCount = query.EnableTotalRecordCount,
  706. TopParentIds = new[] { topFolder.Id },
  707. Name = query.Name,
  708. DtoOptions = options,
  709. HasAired = query.HasAired,
  710. IsAiring = query.IsAiring
  711. };
  712. if (!string.IsNullOrWhiteSpace(query.SeriesTimerId))
  713. {
  714. var seriesTimers = await GetSeriesTimersInternal(new SeriesTimerQuery { }, cancellationToken).ConfigureAwait(false);
  715. var seriesTimer = seriesTimers.Items.FirstOrDefault(i => string.Equals(_tvDtoService.GetInternalSeriesTimerId(i.Id).ToString("N"), query.SeriesTimerId, StringComparison.OrdinalIgnoreCase));
  716. if (seriesTimer != null)
  717. {
  718. internalQuery.ExternalSeriesId = seriesTimer.SeriesId;
  719. if (string.IsNullOrWhiteSpace(seriesTimer.SeriesId))
  720. {
  721. // Better to return nothing than every program in the database
  722. return new QueryResult<BaseItemDto>();
  723. }
  724. }
  725. else
  726. {
  727. // Better to return nothing than every program in the database
  728. return new QueryResult<BaseItemDto>();
  729. }
  730. }
  731. var queryResult = _libraryManager.QueryItems(internalQuery);
  732. var returnArray = _dtoService.GetBaseItemDtos(queryResult.Items, options, user);
  733. var result = new QueryResult<BaseItemDto>
  734. {
  735. Items = returnArray,
  736. TotalRecordCount = queryResult.TotalRecordCount
  737. };
  738. return result;
  739. }
  740. public QueryResult<BaseItem> GetRecommendedProgramsInternal(InternalItemsQuery query, DtoOptions options, CancellationToken cancellationToken)
  741. {
  742. var user = query.User;
  743. var topFolder = GetInternalLiveTvFolder(cancellationToken);
  744. var internalQuery = new InternalItemsQuery(user)
  745. {
  746. IncludeItemTypes = new[] { typeof(LiveTvProgram).Name },
  747. IsAiring = query.IsAiring,
  748. HasAired = query.HasAired,
  749. IsNews = query.IsNews,
  750. IsMovie = query.IsMovie,
  751. IsSeries = query.IsSeries,
  752. IsSports = query.IsSports,
  753. IsKids = query.IsKids,
  754. EnableTotalRecordCount = query.EnableTotalRecordCount,
  755. OrderBy = new[] { new ValueTuple<string, SortOrder>(ItemSortBy.StartDate, SortOrder.Ascending) },
  756. TopParentIds = new[] { topFolder.Id },
  757. DtoOptions = options,
  758. GenreIds = query.GenreIds
  759. };
  760. if (query.Limit.HasValue)
  761. {
  762. internalQuery.Limit = Math.Max(query.Limit.Value * 4, 200);
  763. }
  764. var programList = _libraryManager.QueryItems(internalQuery).Items;
  765. var totalCount = programList.Length;
  766. IOrderedEnumerable<LiveTvProgram> orderedPrograms = programList.Cast<LiveTvProgram>().OrderBy(i => i.StartDate.Date);
  767. if (query.IsAiring ?? false)
  768. {
  769. orderedPrograms = orderedPrograms
  770. .ThenByDescending(i => GetRecommendationScore(i, user, true));
  771. }
  772. IEnumerable<BaseItem> programs = orderedPrograms;
  773. if (query.Limit.HasValue)
  774. {
  775. programs = programs.Take(query.Limit.Value);
  776. }
  777. return new QueryResult<BaseItem>
  778. {
  779. Items = programs.ToArray(),
  780. TotalRecordCount = totalCount
  781. };
  782. }
  783. public QueryResult<BaseItemDto> GetRecommendedPrograms(InternalItemsQuery query, DtoOptions options, CancellationToken cancellationToken)
  784. {
  785. if (!(query.IsAiring ?? false))
  786. {
  787. return GetPrograms(query, options, cancellationToken).Result;
  788. }
  789. RemoveFields(options);
  790. var internalResult = GetRecommendedProgramsInternal(query, options, cancellationToken);
  791. return new QueryResult<BaseItemDto>
  792. {
  793. Items = _dtoService.GetBaseItemDtos(internalResult.Items, options, query.User),
  794. TotalRecordCount = internalResult.TotalRecordCount
  795. };
  796. }
  797. private int GetRecommendationScore(LiveTvProgram program, User user, bool factorChannelWatchCount)
  798. {
  799. var score = 0;
  800. if (program.IsLive)
  801. {
  802. score++;
  803. }
  804. if (program.IsSeries && !program.IsRepeat)
  805. {
  806. score++;
  807. }
  808. var channel = _libraryManager.GetItemById(program.ChannelId);
  809. if (channel == null)
  810. {
  811. return score;
  812. }
  813. var channelUserdata = _userDataManager.GetUserData(user, channel);
  814. if (channelUserdata.Likes.HasValue)
  815. {
  816. score += channelUserdata.Likes.Value ? 2 : -2;
  817. }
  818. if (channelUserdata.IsFavorite)
  819. {
  820. score += 3;
  821. }
  822. if (factorChannelWatchCount)
  823. {
  824. score += channelUserdata.PlayCount;
  825. }
  826. return score;
  827. }
  828. private async Task AddRecordingInfo(IEnumerable<Tuple<BaseItemDto, string, string>> programs, CancellationToken cancellationToken)
  829. {
  830. var timers = new Dictionary<string, List<TimerInfo>>();
  831. var seriesTimers = new Dictionary<string, List<SeriesTimerInfo>>();
  832. TimerInfo[] timerList = null;
  833. SeriesTimerInfo[] seriesTimerList = null;
  834. foreach (var programTuple in programs)
  835. {
  836. var program = programTuple.Item1;
  837. var externalProgramId = programTuple.Item2;
  838. string externalSeriesId = programTuple.Item3;
  839. if (timerList == null)
  840. {
  841. timerList = (await GetTimersInternal(new TimerQuery(), cancellationToken).ConfigureAwait(false)).Items;
  842. }
  843. var timer = timerList.FirstOrDefault(i => string.Equals(i.ProgramId, externalProgramId, StringComparison.OrdinalIgnoreCase));
  844. var foundSeriesTimer = false;
  845. if (timer != null)
  846. {
  847. if (timer.Status != RecordingStatus.Cancelled && timer.Status != RecordingStatus.Error)
  848. {
  849. program.TimerId = _tvDtoService.GetInternalTimerId(timer.Id);
  850. program.Status = timer.Status.ToString();
  851. }
  852. if (!string.IsNullOrEmpty(timer.SeriesTimerId))
  853. {
  854. program.SeriesTimerId = _tvDtoService.GetInternalSeriesTimerId(timer.SeriesTimerId)
  855. .ToString("N");
  856. foundSeriesTimer = true;
  857. }
  858. }
  859. if (foundSeriesTimer || string.IsNullOrWhiteSpace(externalSeriesId))
  860. {
  861. continue;
  862. }
  863. if (seriesTimerList == null)
  864. {
  865. seriesTimerList = (await GetSeriesTimersInternal(new SeriesTimerQuery(), cancellationToken).ConfigureAwait(false)).Items;
  866. }
  867. var seriesTimer = seriesTimerList.FirstOrDefault(i => string.Equals(i.SeriesId, externalSeriesId, StringComparison.OrdinalIgnoreCase));
  868. if (seriesTimer != null)
  869. {
  870. program.SeriesTimerId = _tvDtoService.GetInternalSeriesTimerId(seriesTimer.Id)
  871. .ToString("N");
  872. }
  873. }
  874. }
  875. internal Task RefreshChannels(IProgress<double> progress, CancellationToken cancellationToken)
  876. {
  877. return RefreshChannelsInternal(progress, cancellationToken);
  878. }
  879. private async Task RefreshChannelsInternal(IProgress<double> progress, CancellationToken cancellationToken)
  880. {
  881. await EmbyTV.EmbyTV.Current.CreateRecordingFolders().ConfigureAwait(false);
  882. await EmbyTV.EmbyTV.Current.ScanForTunerDeviceChanges(cancellationToken).ConfigureAwait(false);
  883. var numComplete = 0;
  884. double progressPerService = _services.Length == 0
  885. ? 0
  886. : 1.0 / _services.Length;
  887. var newChannelIdList = new List<Guid>();
  888. var newProgramIdList = new List<Guid>();
  889. var cleanDatabase = true;
  890. foreach (var service in _services)
  891. {
  892. cancellationToken.ThrowIfCancellationRequested();
  893. _logger.LogDebug("Refreshing guide from {name}", service.Name);
  894. try
  895. {
  896. var innerProgress = new ActionableProgress<double>();
  897. innerProgress.RegisterAction(p => progress.Report(p * progressPerService));
  898. var idList = await RefreshChannelsInternal(service, innerProgress, cancellationToken).ConfigureAwait(false);
  899. newChannelIdList.AddRange(idList.Item1);
  900. newProgramIdList.AddRange(idList.Item2);
  901. }
  902. catch (OperationCanceledException)
  903. {
  904. throw;
  905. }
  906. catch (Exception ex)
  907. {
  908. cleanDatabase = false;
  909. _logger.LogError(ex, "Error refreshing channels for service");
  910. }
  911. numComplete++;
  912. double percent = numComplete;
  913. percent /= _services.Length;
  914. progress.Report(100 * percent);
  915. }
  916. if (cleanDatabase)
  917. {
  918. CleanDatabaseInternal(newChannelIdList.ToArray(), new[] { typeof(LiveTvChannel).Name }, progress, cancellationToken);
  919. CleanDatabaseInternal(newProgramIdList.ToArray(), new[] { typeof(LiveTvProgram).Name }, progress, cancellationToken);
  920. }
  921. var coreService = _services.OfType<EmbyTV.EmbyTV>().FirstOrDefault();
  922. if (coreService != null)
  923. {
  924. await coreService.RefreshSeriesTimers(cancellationToken, new SimpleProgress<double>()).ConfigureAwait(false);
  925. await coreService.RefreshTimers(cancellationToken, new SimpleProgress<double>()).ConfigureAwait(false);
  926. }
  927. // Load these now which will prefetch metadata
  928. var dtoOptions = new DtoOptions();
  929. var fields = dtoOptions.Fields.ToList();
  930. fields.Remove(ItemFields.BasicSyncInfo);
  931. dtoOptions.Fields = fields.ToArray();
  932. progress.Report(100);
  933. }
  934. private async Task<Tuple<List<Guid>, List<Guid>>> RefreshChannelsInternal(ILiveTvService service, IProgress<double> progress, CancellationToken cancellationToken)
  935. {
  936. progress.Report(10);
  937. var allChannelsList = (await service.GetChannelsAsync(cancellationToken).ConfigureAwait(false))
  938. .Select(i => new Tuple<string, ChannelInfo>(service.Name, i))
  939. .ToList();
  940. var list = new List<LiveTvChannel>();
  941. var numComplete = 0;
  942. var parentFolder = GetInternalLiveTvFolder(cancellationToken);
  943. foreach (var channelInfo in allChannelsList)
  944. {
  945. cancellationToken.ThrowIfCancellationRequested();
  946. try
  947. {
  948. var item = GetChannel(channelInfo.Item2, channelInfo.Item1, parentFolder, cancellationToken);
  949. list.Add(item);
  950. }
  951. catch (OperationCanceledException)
  952. {
  953. throw;
  954. }
  955. catch (Exception ex)
  956. {
  957. _logger.LogError(ex, "Error getting channel information for {name}", channelInfo.Item2.Name);
  958. }
  959. numComplete++;
  960. double percent = numComplete;
  961. percent /= allChannelsList.Count;
  962. progress.Report(5 * percent + 10);
  963. }
  964. progress.Report(15);
  965. numComplete = 0;
  966. var programs = new List<Guid>();
  967. var channels = new List<Guid>();
  968. var guideDays = GetGuideDays();
  969. _logger.LogInformation("Refreshing guide with {0} days of guide data", guideDays);
  970. cancellationToken.ThrowIfCancellationRequested();
  971. foreach (var currentChannel in list)
  972. {
  973. channels.Add(currentChannel.Id);
  974. cancellationToken.ThrowIfCancellationRequested();
  975. try
  976. {
  977. var start = DateTime.UtcNow.AddHours(-1);
  978. var end = start.AddDays(guideDays);
  979. var isMovie = false;
  980. var isSports = false;
  981. var isNews = false;
  982. var isKids = false;
  983. var iSSeries = false;
  984. var channelPrograms = (await service.GetProgramsAsync(currentChannel.ExternalId, start, end, cancellationToken).ConfigureAwait(false)).ToList();
  985. var existingPrograms = _libraryManager.GetItemList(new InternalItemsQuery
  986. {
  987. IncludeItemTypes = new string[] { typeof(LiveTvProgram).Name },
  988. ChannelIds = new Guid[] { currentChannel.Id },
  989. DtoOptions = new DtoOptions(true)
  990. }).Cast<LiveTvProgram>().ToDictionary(i => i.Id);
  991. var newPrograms = new List<LiveTvProgram>();
  992. var updatedPrograms = new List<BaseItem>();
  993. foreach (var program in channelPrograms)
  994. {
  995. var programTuple = GetProgram(program, existingPrograms, currentChannel, currentChannel.ChannelType, service.Name, cancellationToken);
  996. var programItem = programTuple.Item1;
  997. if (programTuple.Item2)
  998. {
  999. newPrograms.Add(programItem);
  1000. }
  1001. else if (programTuple.Item3)
  1002. {
  1003. updatedPrograms.Add(programItem);
  1004. }
  1005. programs.Add(programItem.Id);
  1006. isMovie |= program.IsMovie;
  1007. iSSeries |= program.IsSeries;
  1008. isSports |= program.IsSports;
  1009. isNews |= program.IsNews;
  1010. isKids |= program.IsKids;
  1011. }
  1012. _logger.LogDebug("Channel {0} has {1} new programs and {2} updated programs", currentChannel.Name, newPrograms.Count, updatedPrograms.Count);
  1013. if (newPrograms.Count > 0)
  1014. {
  1015. _libraryManager.CreateItems(newPrograms, null, cancellationToken);
  1016. }
  1017. if (updatedPrograms.Count > 0)
  1018. {
  1019. _libraryManager.UpdateItems(updatedPrograms, currentChannel, ItemUpdateType.MetadataImport, cancellationToken);
  1020. }
  1021. currentChannel.IsMovie = isMovie;
  1022. currentChannel.IsNews = isNews;
  1023. currentChannel.IsSports = isSports;
  1024. currentChannel.IsSeries = iSSeries;
  1025. if (isKids)
  1026. {
  1027. currentChannel.AddTag("Kids");
  1028. }
  1029. //currentChannel.UpdateToRepository(ItemUpdateType.MetadataImport, cancellationToken);
  1030. await currentChannel.RefreshMetadata(new MetadataRefreshOptions(new DirectoryService(_logger, _fileSystem))
  1031. {
  1032. ForceSave = true
  1033. }, cancellationToken).ConfigureAwait(false);
  1034. }
  1035. catch (OperationCanceledException)
  1036. {
  1037. throw;
  1038. }
  1039. catch (Exception ex)
  1040. {
  1041. _logger.LogError(ex, "Error getting programs for channel {name}", currentChannel.Name);
  1042. }
  1043. numComplete++;
  1044. double percent = numComplete / (double) allChannelsList.Count;
  1045. progress.Report(85 * percent + 15);
  1046. }
  1047. progress.Report(100);
  1048. return new Tuple<List<Guid>, List<Guid>>(channels, programs);
  1049. }
  1050. private void CleanDatabaseInternal(Guid[] currentIdList, string[] validTypes, IProgress<double> progress, CancellationToken cancellationToken)
  1051. {
  1052. var list = _itemRepo.GetItemIdsList(new InternalItemsQuery
  1053. {
  1054. IncludeItemTypes = validTypes,
  1055. DtoOptions = new DtoOptions(false)
  1056. });
  1057. var numComplete = 0;
  1058. foreach (var itemId in list)
  1059. {
  1060. cancellationToken.ThrowIfCancellationRequested();
  1061. if (itemId.Equals(Guid.Empty))
  1062. {
  1063. // Somehow some invalid data got into the db. It probably predates the boundary checking
  1064. continue;
  1065. }
  1066. if (!currentIdList.Contains(itemId))
  1067. {
  1068. var item = _libraryManager.GetItemById(itemId);
  1069. if (item != null)
  1070. {
  1071. _libraryManager.DeleteItem(item, new DeleteOptions
  1072. {
  1073. DeleteFileLocation = false,
  1074. DeleteFromExternalProvider = false
  1075. }, false);
  1076. }
  1077. }
  1078. numComplete++;
  1079. double percent = numComplete / (double) list.Count;
  1080. progress.Report(100 * percent);
  1081. }
  1082. }
  1083. private const int MaxGuideDays = 14;
  1084. private double GetGuideDays()
  1085. {
  1086. var config = GetConfiguration();
  1087. if (config.GuideDays.HasValue)
  1088. {
  1089. return Math.Max(1, Math.Min(config.GuideDays.Value, MaxGuideDays));
  1090. }
  1091. return 7;
  1092. }
  1093. private QueryResult<BaseItem> GetEmbyRecordings(RecordingQuery query, DtoOptions dtoOptions, User user)
  1094. {
  1095. if (user == null)
  1096. {
  1097. return new QueryResult<BaseItem>();
  1098. }
  1099. var folderIds = GetRecordingFolders(user, true)
  1100. .Select(i => i.Id)
  1101. .ToList();
  1102. var excludeItemTypes = new List<string>();
  1103. if (folderIds.Count == 0)
  1104. {
  1105. return new QueryResult<BaseItem>();
  1106. }
  1107. var includeItemTypes = new List<string>();
  1108. var genres = new List<string>();
  1109. if (query.IsMovie.HasValue)
  1110. {
  1111. if (query.IsMovie.Value)
  1112. {
  1113. includeItemTypes.Add(typeof(Movie).Name);
  1114. }
  1115. else
  1116. {
  1117. excludeItemTypes.Add(typeof(Movie).Name);
  1118. }
  1119. }
  1120. if (query.IsSeries.HasValue)
  1121. {
  1122. if (query.IsSeries.Value)
  1123. {
  1124. includeItemTypes.Add(typeof(Episode).Name);
  1125. }
  1126. else
  1127. {
  1128. excludeItemTypes.Add(typeof(Episode).Name);
  1129. }
  1130. }
  1131. if (query.IsSports ?? false)
  1132. {
  1133. genres.Add("Sports");
  1134. }
  1135. if (query.IsKids ?? false)
  1136. {
  1137. genres.Add("Kids");
  1138. genres.Add("Children");
  1139. genres.Add("Family");
  1140. }
  1141. var limit = query.Limit;
  1142. if (query.IsInProgress ?? false)
  1143. {
  1144. // limit = (query.Limit ?? 10) * 2;
  1145. limit = null;
  1146. //var allActivePaths = EmbyTV.EmbyTV.Current.GetAllActiveRecordings().Select(i => i.Path).ToArray();
  1147. //var items = allActivePaths.Select(i => _libraryManager.FindByPath(i, false)).Where(i => i != null).ToArray();
  1148. //return new QueryResult<BaseItem>
  1149. //{
  1150. // Items = items,
  1151. // TotalRecordCount = items.Length
  1152. //};
  1153. dtoOptions.Fields = dtoOptions.Fields.Concat(new[] { ItemFields.Tags }).Distinct().ToArray();
  1154. }
  1155. var result = _libraryManager.GetItemsResult(new InternalItemsQuery(user)
  1156. {
  1157. MediaTypes = new[] { MediaType.Video },
  1158. Recursive = true,
  1159. AncestorIds = folderIds.ToArray(),
  1160. IsFolder = false,
  1161. IsVirtualItem = false,
  1162. Limit = limit,
  1163. StartIndex = query.StartIndex,
  1164. OrderBy = new[] { new ValueTuple<string, SortOrder>(ItemSortBy.DateCreated, SortOrder.Descending) },
  1165. EnableTotalRecordCount = query.EnableTotalRecordCount,
  1166. IncludeItemTypes = includeItemTypes.ToArray(),
  1167. ExcludeItemTypes = excludeItemTypes.ToArray(),
  1168. Genres = genres.ToArray(),
  1169. DtoOptions = dtoOptions
  1170. });
  1171. if (query.IsInProgress ?? false)
  1172. {
  1173. result.Items = result
  1174. .Items
  1175. .OfType<Video>()
  1176. .Where(i => !i.IsCompleteMedia)
  1177. .ToArray();
  1178. result.TotalRecordCount = result.Items.Length;
  1179. }
  1180. return result;
  1181. }
  1182. public Task AddInfoToProgramDto(List<Tuple<BaseItem, BaseItemDto>> tuples, ItemFields[] fields, User user = null)
  1183. {
  1184. var programTuples = new List<Tuple<BaseItemDto, string, string>>();
  1185. var hasChannelImage = fields.Contains(ItemFields.ChannelImage);
  1186. var hasChannelInfo = fields.Contains(ItemFields.ChannelInfo);
  1187. foreach (var tuple in tuples)
  1188. {
  1189. var program = (LiveTvProgram)tuple.Item1;
  1190. var dto = tuple.Item2;
  1191. dto.StartDate = program.StartDate;
  1192. dto.EpisodeTitle = program.EpisodeTitle;
  1193. dto.IsRepeat |= program.IsRepeat;
  1194. dto.IsMovie |= program.IsMovie;
  1195. dto.IsSeries |= program.IsSeries;
  1196. dto.IsSports |= program.IsSports;
  1197. dto.IsLive |= program.IsLive;
  1198. dto.IsNews |= program.IsNews;
  1199. dto.IsKids |= program.IsKids;
  1200. dto.IsPremiere |= program.IsPremiere;
  1201. if (hasChannelInfo || hasChannelImage)
  1202. {
  1203. var channel = _libraryManager.GetItemById(program.ChannelId);
  1204. if (channel is LiveTvChannel liveChannel)
  1205. {
  1206. dto.ChannelName = liveChannel.Name;
  1207. dto.MediaType = liveChannel.MediaType;
  1208. dto.ChannelNumber = liveChannel.Number;
  1209. if (hasChannelImage && liveChannel.HasImage(ImageType.Primary))
  1210. {
  1211. dto.ChannelPrimaryImageTag = _tvDtoService.GetImageTag(liveChannel);
  1212. }
  1213. }
  1214. }
  1215. programTuples.Add(new Tuple<BaseItemDto, string, string>(dto, program.ExternalId, program.ExternalSeriesId));
  1216. }
  1217. return AddRecordingInfo(programTuples, CancellationToken.None);
  1218. }
  1219. public ActiveRecordingInfo GetActiveRecordingInfo(string path)
  1220. {
  1221. return EmbyTV.EmbyTV.Current.GetActiveRecordingInfo(path);
  1222. }
  1223. public void AddInfoToRecordingDto(BaseItem item, BaseItemDto dto, ActiveRecordingInfo activeRecordingInfo, User user = null)
  1224. {
  1225. var service = EmbyTV.EmbyTV.Current;
  1226. var info = activeRecordingInfo.Timer;
  1227. var channel = string.IsNullOrWhiteSpace(info.ChannelId) ? null : _libraryManager.GetItemById(_tvDtoService.GetInternalChannelId(service.Name, info.ChannelId));
  1228. dto.SeriesTimerId = string.IsNullOrEmpty(info.SeriesTimerId)
  1229. ? null
  1230. : _tvDtoService.GetInternalSeriesTimerId(info.SeriesTimerId).ToString("N");
  1231. dto.TimerId = string.IsNullOrEmpty(info.Id)
  1232. ? null
  1233. : _tvDtoService.GetInternalTimerId(info.Id);
  1234. var startDate = info.StartDate;
  1235. var endDate = info.EndDate;
  1236. dto.StartDate = startDate;
  1237. dto.EndDate = endDate;
  1238. dto.Status = info.Status.ToString();
  1239. dto.IsRepeat = info.IsRepeat;
  1240. dto.EpisodeTitle = info.EpisodeTitle;
  1241. dto.IsMovie = info.IsMovie;
  1242. dto.IsSeries = info.IsSeries;
  1243. dto.IsSports = info.IsSports;
  1244. dto.IsLive = info.IsLive;
  1245. dto.IsNews = info.IsNews;
  1246. dto.IsKids = info.IsKids;
  1247. dto.IsPremiere = info.IsPremiere;
  1248. if (info.Status == RecordingStatus.InProgress)
  1249. {
  1250. startDate = info.StartDate.AddSeconds(0 - info.PrePaddingSeconds);
  1251. endDate = info.EndDate.AddSeconds(info.PostPaddingSeconds);
  1252. var now = DateTime.UtcNow.Ticks;
  1253. var start = startDate.Ticks;
  1254. var end = endDate.Ticks;
  1255. var pct = now - start;
  1256. pct /= end;
  1257. pct *= 100;
  1258. dto.CompletionPercentage = pct;
  1259. }
  1260. if (channel != null)
  1261. {
  1262. dto.ChannelName = channel.Name;
  1263. if (channel.HasImage(ImageType.Primary))
  1264. {
  1265. dto.ChannelPrimaryImageTag = _tvDtoService.GetImageTag(channel);
  1266. }
  1267. }
  1268. }
  1269. public QueryResult<BaseItemDto> GetRecordings(RecordingQuery query, DtoOptions options)
  1270. {
  1271. var user = query.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(query.UserId);
  1272. RemoveFields(options);
  1273. var internalResult = GetEmbyRecordings(query, options, user);
  1274. var returnArray = _dtoService.GetBaseItemDtos(internalResult.Items, options, user);
  1275. return new QueryResult<BaseItemDto>
  1276. {
  1277. Items = returnArray,
  1278. TotalRecordCount = internalResult.TotalRecordCount
  1279. };
  1280. }
  1281. private async Task<QueryResult<TimerInfo>> GetTimersInternal(TimerQuery query, CancellationToken cancellationToken)
  1282. {
  1283. var tasks = _services.Select(async i =>
  1284. {
  1285. try
  1286. {
  1287. var recs = await i.GetTimersAsync(cancellationToken).ConfigureAwait(false);
  1288. return recs.Select(r => new Tuple<TimerInfo, ILiveTvService>(r, i));
  1289. }
  1290. catch (Exception ex)
  1291. {
  1292. _logger.LogError(ex, "Error getting recordings");
  1293. return new List<Tuple<TimerInfo, ILiveTvService>>();
  1294. }
  1295. });
  1296. var results = await Task.WhenAll(tasks).ConfigureAwait(false);
  1297. var timers = results.SelectMany(i => i.ToList());
  1298. if (query.IsActive.HasValue)
  1299. {
  1300. if (query.IsActive.Value)
  1301. {
  1302. timers = timers.Where(i => i.Item1.Status == RecordingStatus.InProgress);
  1303. }
  1304. else
  1305. {
  1306. timers = timers.Where(i => i.Item1.Status != RecordingStatus.InProgress);
  1307. }
  1308. }
  1309. if (query.IsScheduled.HasValue)
  1310. {
  1311. if (query.IsScheduled.Value)
  1312. {
  1313. timers = timers.Where(i => i.Item1.Status == RecordingStatus.New);
  1314. }
  1315. else
  1316. {
  1317. timers = timers.Where(i => i.Item1.Status != RecordingStatus.New);
  1318. }
  1319. }
  1320. if (!string.IsNullOrEmpty(query.ChannelId))
  1321. {
  1322. var guid = new Guid(query.ChannelId);
  1323. timers = timers.Where(i => guid == _tvDtoService.GetInternalChannelId(i.Item2.Name, i.Item1.ChannelId));
  1324. }
  1325. if (!string.IsNullOrEmpty(query.SeriesTimerId))
  1326. {
  1327. var guid = new Guid(query.SeriesTimerId);
  1328. timers = timers
  1329. .Where(i => _tvDtoService.GetInternalSeriesTimerId(i.Item1.SeriesTimerId) == guid);
  1330. }
  1331. if (!string.IsNullOrEmpty(query.Id))
  1332. {
  1333. var guid = new Guid(query.Id);
  1334. timers = timers
  1335. .Where(i => string.Equals(_tvDtoService.GetInternalTimerId(i.Item1.Id), query.Id, StringComparison.OrdinalIgnoreCase));
  1336. }
  1337. var returnArray = timers
  1338. .Select(i => i.Item1)
  1339. .OrderBy(i => i.StartDate)
  1340. .ToArray();
  1341. return new QueryResult<TimerInfo>
  1342. {
  1343. Items = returnArray,
  1344. TotalRecordCount = returnArray.Length
  1345. };
  1346. }
  1347. public async Task<QueryResult<TimerInfoDto>> GetTimers(TimerQuery query, CancellationToken cancellationToken)
  1348. {
  1349. var tasks = _services.Select(async i =>
  1350. {
  1351. try
  1352. {
  1353. var recs = await i.GetTimersAsync(cancellationToken).ConfigureAwait(false);
  1354. return recs.Select(r => new Tuple<TimerInfo, ILiveTvService>(r, i));
  1355. }
  1356. catch (Exception ex)
  1357. {
  1358. _logger.LogError(ex, "Error getting recordings");
  1359. return new List<Tuple<TimerInfo, ILiveTvService>>();
  1360. }
  1361. });
  1362. var results = await Task.WhenAll(tasks).ConfigureAwait(false);
  1363. var timers = results.SelectMany(i => i.ToList());
  1364. if (query.IsActive.HasValue)
  1365. {
  1366. if (query.IsActive.Value)
  1367. {
  1368. timers = timers.Where(i => i.Item1.Status == RecordingStatus.InProgress);
  1369. }
  1370. else
  1371. {
  1372. timers = timers.Where(i => i.Item1.Status != RecordingStatus.InProgress);
  1373. }
  1374. }
  1375. if (query.IsScheduled.HasValue)
  1376. {
  1377. if (query.IsScheduled.Value)
  1378. {
  1379. timers = timers.Where(i => i.Item1.Status == RecordingStatus.New);
  1380. }
  1381. else
  1382. {
  1383. timers = timers.Where(i => i.Item1.Status != RecordingStatus.New);
  1384. }
  1385. }
  1386. if (!string.IsNullOrEmpty(query.ChannelId))
  1387. {
  1388. var guid = new Guid(query.ChannelId);
  1389. timers = timers.Where(i => guid == _tvDtoService.GetInternalChannelId(i.Item2.Name, i.Item1.ChannelId));
  1390. }
  1391. if (!string.IsNullOrEmpty(query.SeriesTimerId))
  1392. {
  1393. var guid = new Guid(query.SeriesTimerId);
  1394. timers = timers
  1395. .Where(i => _tvDtoService.GetInternalSeriesTimerId(i.Item1.SeriesTimerId) == guid);
  1396. }
  1397. if (!string.IsNullOrEmpty(query.Id))
  1398. {
  1399. timers = timers
  1400. .Where(i => string.Equals(_tvDtoService.GetInternalTimerId(i.Item1.Id), query.Id, StringComparison.OrdinalIgnoreCase));
  1401. }
  1402. var returnList = new List<TimerInfoDto>();
  1403. foreach (var i in timers)
  1404. {
  1405. var program = string.IsNullOrEmpty(i.Item1.ProgramId) ?
  1406. null :
  1407. _libraryManager.GetItemById(_tvDtoService.GetInternalProgramId(i.Item1.ProgramId)) as LiveTvProgram;
  1408. var channel = string.IsNullOrEmpty(i.Item1.ChannelId) ? null : _libraryManager.GetItemById(_tvDtoService.GetInternalChannelId(i.Item2.Name, i.Item1.ChannelId));
  1409. returnList.Add(_tvDtoService.GetTimerInfoDto(i.Item1, i.Item2, program, channel));
  1410. }
  1411. var returnArray = returnList
  1412. .OrderBy(i => i.StartDate)
  1413. .ToArray();
  1414. return new QueryResult<TimerInfoDto>
  1415. {
  1416. Items = returnArray,
  1417. TotalRecordCount = returnArray.Length
  1418. };
  1419. }
  1420. public async Task CancelTimer(string id)
  1421. {
  1422. var timer = await GetTimer(id, CancellationToken.None).ConfigureAwait(false);
  1423. if (timer == null)
  1424. {
  1425. throw new ResourceNotFoundException(string.Format("Timer with Id {0} not found", id));
  1426. }
  1427. var service = GetService(timer.ServiceName);
  1428. await service.CancelTimerAsync(timer.ExternalId, CancellationToken.None).ConfigureAwait(false);
  1429. if (!(service is EmbyTV.EmbyTV))
  1430. {
  1431. TimerCancelled?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  1432. {
  1433. Argument = new TimerEventInfo
  1434. {
  1435. Id = id
  1436. }
  1437. });
  1438. }
  1439. }
  1440. public async Task CancelSeriesTimer(string id)
  1441. {
  1442. var timer = await GetSeriesTimer(id, CancellationToken.None).ConfigureAwait(false);
  1443. if (timer == null)
  1444. {
  1445. throw new ResourceNotFoundException(string.Format("SeriesTimer with Id {0} not found", id));
  1446. }
  1447. var service = GetService(timer.ServiceName);
  1448. await service.CancelSeriesTimerAsync(timer.ExternalId, CancellationToken.None).ConfigureAwait(false);
  1449. SeriesTimerCancelled?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  1450. {
  1451. Argument = new TimerEventInfo
  1452. {
  1453. Id = id
  1454. }
  1455. });
  1456. }
  1457. public async Task<TimerInfoDto> GetTimer(string id, CancellationToken cancellationToken)
  1458. {
  1459. var results = await GetTimers(new TimerQuery
  1460. {
  1461. Id = id
  1462. }, cancellationToken).ConfigureAwait(false);
  1463. return results.Items.FirstOrDefault(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase));
  1464. }
  1465. public async Task<SeriesTimerInfoDto> GetSeriesTimer(string id, CancellationToken cancellationToken)
  1466. {
  1467. var results = await GetSeriesTimers(new SeriesTimerQuery(), cancellationToken).ConfigureAwait(false);
  1468. return results.Items.FirstOrDefault(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase));
  1469. }
  1470. private async Task<QueryResult<SeriesTimerInfo>> GetSeriesTimersInternal(SeriesTimerQuery query, CancellationToken cancellationToken)
  1471. {
  1472. var tasks = _services.Select(async i =>
  1473. {
  1474. try
  1475. {
  1476. var recs = await i.GetSeriesTimersAsync(cancellationToken).ConfigureAwait(false);
  1477. return recs.Select(r =>
  1478. {
  1479. r.ServiceName = i.Name;
  1480. return new Tuple<SeriesTimerInfo, ILiveTvService>(r, i);
  1481. });
  1482. }
  1483. catch (Exception ex)
  1484. {
  1485. _logger.LogError(ex, "Error getting recordings");
  1486. return new List<Tuple<SeriesTimerInfo, ILiveTvService>>();
  1487. }
  1488. });
  1489. var results = await Task.WhenAll(tasks).ConfigureAwait(false);
  1490. var timers = results.SelectMany(i => i.ToList());
  1491. if (string.Equals(query.SortBy, "Priority", StringComparison.OrdinalIgnoreCase))
  1492. {
  1493. timers = query.SortOrder == SortOrder.Descending ?
  1494. timers.OrderBy(i => i.Item1.Priority).ThenByStringDescending(i => i.Item1.Name) :
  1495. timers.OrderByDescending(i => i.Item1.Priority).ThenByString(i => i.Item1.Name);
  1496. }
  1497. else
  1498. {
  1499. timers = query.SortOrder == SortOrder.Descending ?
  1500. timers.OrderByStringDescending(i => i.Item1.Name) :
  1501. timers.OrderByString(i => i.Item1.Name);
  1502. }
  1503. var returnArray = timers
  1504. .Select(i =>
  1505. {
  1506. return i.Item1;
  1507. })
  1508. .ToArray();
  1509. return new QueryResult<SeriesTimerInfo>
  1510. {
  1511. Items = returnArray,
  1512. TotalRecordCount = returnArray.Length
  1513. };
  1514. }
  1515. public async Task<QueryResult<SeriesTimerInfoDto>> GetSeriesTimers(SeriesTimerQuery query, CancellationToken cancellationToken)
  1516. {
  1517. var tasks = _services.Select(async i =>
  1518. {
  1519. try
  1520. {
  1521. var recs = await i.GetSeriesTimersAsync(cancellationToken).ConfigureAwait(false);
  1522. return recs.Select(r => new Tuple<SeriesTimerInfo, ILiveTvService>(r, i));
  1523. }
  1524. catch (Exception ex)
  1525. {
  1526. _logger.LogError(ex, "Error getting recordings");
  1527. return new List<Tuple<SeriesTimerInfo, ILiveTvService>>();
  1528. }
  1529. });
  1530. var results = await Task.WhenAll(tasks).ConfigureAwait(false);
  1531. var timers = results.SelectMany(i => i.ToList());
  1532. if (string.Equals(query.SortBy, "Priority", StringComparison.OrdinalIgnoreCase))
  1533. {
  1534. timers = query.SortOrder == SortOrder.Descending ?
  1535. timers.OrderBy(i => i.Item1.Priority).ThenByStringDescending(i => i.Item1.Name) :
  1536. timers.OrderByDescending(i => i.Item1.Priority).ThenByString(i => i.Item1.Name);
  1537. }
  1538. else
  1539. {
  1540. timers = query.SortOrder == SortOrder.Descending ?
  1541. timers.OrderByStringDescending(i => i.Item1.Name) :
  1542. timers.OrderByString(i => i.Item1.Name);
  1543. }
  1544. var returnArray = timers
  1545. .Select(i =>
  1546. {
  1547. string channelName = null;
  1548. if (!string.IsNullOrEmpty(i.Item1.ChannelId))
  1549. {
  1550. var internalChannelId = _tvDtoService.GetInternalChannelId(i.Item2.Name, i.Item1.ChannelId);
  1551. var channel = _libraryManager.GetItemById(internalChannelId);
  1552. channelName = channel == null ? null : channel.Name;
  1553. }
  1554. return _tvDtoService.GetSeriesTimerInfoDto(i.Item1, i.Item2, channelName);
  1555. })
  1556. .ToArray();
  1557. return new QueryResult<SeriesTimerInfoDto>
  1558. {
  1559. Items = returnArray,
  1560. TotalRecordCount = returnArray.Length
  1561. };
  1562. }
  1563. public BaseItem GetLiveTvChannel(TimerInfo timer, ILiveTvService service)
  1564. {
  1565. var internalChannelId = _tvDtoService.GetInternalChannelId(service.Name, timer.ChannelId);
  1566. return _libraryManager.GetItemById(internalChannelId);
  1567. }
  1568. public void AddChannelInfo(List<Tuple<BaseItemDto, LiveTvChannel>> tuples, DtoOptions options, User user)
  1569. {
  1570. var now = DateTime.UtcNow;
  1571. var channelIds = tuples.Select(i => i.Item2.Id).Distinct().ToArray();
  1572. var programs = options.AddCurrentProgram ? _libraryManager.GetItemList(new InternalItemsQuery(user)
  1573. {
  1574. IncludeItemTypes = new[] { typeof(LiveTvProgram).Name },
  1575. ChannelIds = channelIds,
  1576. MaxStartDate = now,
  1577. MinEndDate = now,
  1578. Limit = channelIds.Length,
  1579. OrderBy = new[] { new ValueTuple<string, SortOrder>(ItemSortBy.StartDate, SortOrder.Ascending) },
  1580. TopParentIds = new[] { GetInternalLiveTvFolder(CancellationToken.None).Id },
  1581. DtoOptions = options
  1582. }) : new List<BaseItem>();
  1583. RemoveFields(options);
  1584. var currentProgramsList = new List<BaseItem>();
  1585. var currentChannelsDict = new Dictionary<Guid, BaseItemDto>();
  1586. var addCurrentProgram = options.AddCurrentProgram;
  1587. foreach (var tuple in tuples)
  1588. {
  1589. var dto = tuple.Item1;
  1590. var channel = tuple.Item2;
  1591. dto.Number = channel.Number;
  1592. dto.ChannelNumber = channel.Number;
  1593. dto.ChannelType = channel.ChannelType;
  1594. currentChannelsDict[dto.Id] = dto;
  1595. if (addCurrentProgram)
  1596. {
  1597. var currentProgram = programs.FirstOrDefault(i => channel.Id.Equals(i.ChannelId));
  1598. if (currentProgram != null)
  1599. {
  1600. currentProgramsList.Add(currentProgram);
  1601. }
  1602. }
  1603. }
  1604. if (addCurrentProgram)
  1605. {
  1606. var currentProgramDtos = _dtoService.GetBaseItemDtos(currentProgramsList, options, user);
  1607. foreach (var programDto in currentProgramDtos)
  1608. {
  1609. BaseItemDto channelDto;
  1610. if (currentChannelsDict.TryGetValue(programDto.ChannelId, out channelDto))
  1611. {
  1612. channelDto.CurrentProgram = programDto;
  1613. }
  1614. }
  1615. }
  1616. }
  1617. private async Task<Tuple<SeriesTimerInfo, ILiveTvService>> GetNewTimerDefaultsInternal(CancellationToken cancellationToken, LiveTvProgram program = null)
  1618. {
  1619. ILiveTvService service = null;
  1620. ProgramInfo programInfo = null;
  1621. if(program != null)
  1622. {
  1623. service = GetService(program);
  1624. var channel = _libraryManager.GetItemById(program.ChannelId);
  1625. programInfo = new ProgramInfo
  1626. {
  1627. Audio = program.Audio,
  1628. ChannelId = channel.ExternalId,
  1629. CommunityRating = program.CommunityRating,
  1630. EndDate = program.EndDate ?? DateTime.MinValue,
  1631. EpisodeTitle = program.EpisodeTitle,
  1632. Genres = program.Genres.ToList(),
  1633. Id = program.ExternalId,
  1634. IsHD = program.IsHD,
  1635. IsKids = program.IsKids,
  1636. IsLive = program.IsLive,
  1637. IsMovie = program.IsMovie,
  1638. IsNews = program.IsNews,
  1639. IsPremiere = program.IsPremiere,
  1640. IsRepeat = program.IsRepeat,
  1641. IsSeries = program.IsSeries,
  1642. IsSports = program.IsSports,
  1643. OriginalAirDate = program.PremiereDate,
  1644. Overview = program.Overview,
  1645. StartDate = program.StartDate,
  1646. //ImagePath = program.ExternalImagePath,
  1647. Name = program.Name,
  1648. OfficialRating = program.OfficialRating
  1649. };
  1650. }
  1651. if (service == null)
  1652. {
  1653. service = _services.First();
  1654. }
  1655. var info = await service.GetNewTimerDefaultsAsync(cancellationToken, programInfo).ConfigureAwait(false);
  1656. info.RecordAnyTime = true;
  1657. info.Days = new List<DayOfWeek>
  1658. {
  1659. DayOfWeek.Sunday,
  1660. DayOfWeek.Monday,
  1661. DayOfWeek.Tuesday,
  1662. DayOfWeek.Wednesday,
  1663. DayOfWeek.Thursday,
  1664. DayOfWeek.Friday,
  1665. DayOfWeek.Saturday
  1666. };
  1667. info.Id = null;
  1668. return new Tuple<SeriesTimerInfo, ILiveTvService>(info, service);
  1669. }
  1670. public async Task<SeriesTimerInfoDto> GetNewTimerDefaults(CancellationToken cancellationToken)
  1671. {
  1672. var info = await GetNewTimerDefaultsInternal(cancellationToken).ConfigureAwait(false);
  1673. var obj = _tvDtoService.GetSeriesTimerInfoDto(info.Item1, info.Item2, null);
  1674. return obj;
  1675. }
  1676. public async Task<SeriesTimerInfoDto> GetNewTimerDefaults(string programId, CancellationToken cancellationToken)
  1677. {
  1678. var program = (LiveTvProgram)_libraryManager.GetItemById(programId);
  1679. var programDto = await GetProgram(programId, cancellationToken).ConfigureAwait(false);
  1680. var defaults = await GetNewTimerDefaultsInternal(cancellationToken, program).ConfigureAwait(false);
  1681. var info = _tvDtoService.GetSeriesTimerInfoDto(defaults.Item1, defaults.Item2, null);
  1682. info.Days = defaults.Item1.Days.ToArray();
  1683. info.DayPattern = _tvDtoService.GetDayPattern(info.Days);
  1684. info.Name = program.Name;
  1685. info.ChannelId = programDto.ChannelId;
  1686. info.ChannelName = programDto.ChannelName;
  1687. info.StartDate = program.StartDate;
  1688. info.Name = program.Name;
  1689. info.Overview = program.Overview;
  1690. info.ProgramId = programDto.Id.ToString("N");
  1691. info.ExternalProgramId = program.ExternalId;
  1692. if (program.EndDate.HasValue)
  1693. {
  1694. info.EndDate = program.EndDate.Value;
  1695. }
  1696. return info;
  1697. }
  1698. public async Task CreateTimer(TimerInfoDto timer, CancellationToken cancellationToken)
  1699. {
  1700. var service = GetService(timer.ServiceName);
  1701. var info = await _tvDtoService.GetTimerInfo(timer, true, this, cancellationToken).ConfigureAwait(false);
  1702. // Set priority from default values
  1703. var defaultValues = await service.GetNewTimerDefaultsAsync(cancellationToken).ConfigureAwait(false);
  1704. info.Priority = defaultValues.Priority;
  1705. string newTimerId = null;
  1706. if (service is ISupportsNewTimerIds supportsNewTimerIds)
  1707. {
  1708. newTimerId = await supportsNewTimerIds.CreateTimer(info, cancellationToken).ConfigureAwait(false);
  1709. newTimerId = _tvDtoService.GetInternalTimerId(newTimerId);
  1710. }
  1711. else
  1712. {
  1713. await service.CreateTimerAsync(info, cancellationToken).ConfigureAwait(false);
  1714. }
  1715. _logger.LogInformation("New recording scheduled");
  1716. if (!(service is EmbyTV.EmbyTV))
  1717. {
  1718. TimerCreated?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  1719. {
  1720. Argument = new TimerEventInfo
  1721. {
  1722. ProgramId = _tvDtoService.GetInternalProgramId(info.ProgramId),
  1723. Id = newTimerId
  1724. }
  1725. });
  1726. }
  1727. }
  1728. public async Task CreateSeriesTimer(SeriesTimerInfoDto timer, CancellationToken cancellationToken)
  1729. {
  1730. var registration = await GetRegistrationInfo("seriesrecordings").ConfigureAwait(false);
  1731. if (!registration.IsValid)
  1732. {
  1733. _logger.LogInformation("Creating series recordings requires an active Emby Premiere subscription.");
  1734. return;
  1735. }
  1736. var service = GetService(timer.ServiceName);
  1737. var info = await _tvDtoService.GetSeriesTimerInfo(timer, true, this, cancellationToken).ConfigureAwait(false);
  1738. // Set priority from default values
  1739. var defaultValues = await service.GetNewTimerDefaultsAsync(cancellationToken).ConfigureAwait(false);
  1740. info.Priority = defaultValues.Priority;
  1741. string newTimerId = null;
  1742. if (service is ISupportsNewTimerIds supportsNewTimerIds)
  1743. {
  1744. newTimerId = await supportsNewTimerIds.CreateSeriesTimer(info, cancellationToken).ConfigureAwait(false);
  1745. newTimerId = _tvDtoService.GetInternalSeriesTimerId(newTimerId).ToString("N");
  1746. }
  1747. else
  1748. {
  1749. await service.CreateSeriesTimerAsync(info, cancellationToken).ConfigureAwait(false);
  1750. }
  1751. SeriesTimerCreated?.Invoke(this, new GenericEventArgs<TimerEventInfo>
  1752. {
  1753. Argument = new TimerEventInfo
  1754. {
  1755. ProgramId = _tvDtoService.GetInternalProgramId(info.ProgramId),
  1756. Id = newTimerId
  1757. }
  1758. });
  1759. }
  1760. public async Task UpdateTimer(TimerInfoDto timer, CancellationToken cancellationToken)
  1761. {
  1762. var info = await _tvDtoService.GetTimerInfo(timer, false, this, cancellationToken).ConfigureAwait(false);
  1763. var service = GetService(timer.ServiceName);
  1764. await service.UpdateTimerAsync(info, cancellationToken).ConfigureAwait(false);
  1765. }
  1766. public async Task UpdateSeriesTimer(SeriesTimerInfoDto timer, CancellationToken cancellationToken)
  1767. {
  1768. var info = await _tvDtoService.GetSeriesTimerInfo(timer, false, this, cancellationToken).ConfigureAwait(false);
  1769. var service = GetService(timer.ServiceName);
  1770. await service.UpdateSeriesTimerAsync(info, cancellationToken).ConfigureAwait(false);
  1771. }
  1772. public GuideInfo GetGuideInfo()
  1773. {
  1774. var startDate = DateTime.UtcNow;
  1775. var endDate = startDate.AddDays(GetGuideDays());
  1776. return new GuideInfo
  1777. {
  1778. StartDate = startDate,
  1779. EndDate = endDate
  1780. };
  1781. }
  1782. /// <summary>
  1783. /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
  1784. /// </summary>
  1785. public void Dispose()
  1786. {
  1787. Dispose(true);
  1788. }
  1789. private bool _isDisposed = false;
  1790. /// <summary>
  1791. /// Releases unmanaged and - optionally - managed resources.
  1792. /// </summary>
  1793. /// <param name="dispose"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  1794. protected virtual void Dispose(bool dispose)
  1795. {
  1796. if (dispose)
  1797. {
  1798. _isDisposed = true;
  1799. }
  1800. }
  1801. private LiveTvServiceInfo[] GetServiceInfos()
  1802. {
  1803. return Services.Select(GetServiceInfo).ToArray();
  1804. }
  1805. private LiveTvServiceInfo GetServiceInfo(ILiveTvService service)
  1806. {
  1807. return new LiveTvServiceInfo
  1808. {
  1809. Name = service.Name
  1810. };
  1811. }
  1812. public LiveTvInfo GetLiveTvInfo(CancellationToken cancellationToken)
  1813. {
  1814. var services = GetServiceInfos();
  1815. var info = new LiveTvInfo
  1816. {
  1817. Services = services,
  1818. IsEnabled = services.Length > 0
  1819. };
  1820. info.EnabledUsers = _userManager.Users
  1821. .Where(IsLiveTvEnabled)
  1822. .Select(i => i.Id.ToString("N"))
  1823. .ToArray();
  1824. return info;
  1825. }
  1826. private bool IsLiveTvEnabled(User user)
  1827. {
  1828. return user.Policy.EnableLiveTvAccess && (Services.Count > 1 || GetConfiguration().TunerHosts.Length > 0);
  1829. }
  1830. public IEnumerable<User> GetEnabledUsers()
  1831. {
  1832. return _userManager.Users
  1833. .Where(IsLiveTvEnabled);
  1834. }
  1835. /// <summary>
  1836. /// Resets the tuner.
  1837. /// </summary>
  1838. /// <param name="id">The identifier.</param>
  1839. /// <param name="cancellationToken">The cancellation token.</param>
  1840. /// <returns>Task.</returns>
  1841. public Task ResetTuner(string id, CancellationToken cancellationToken)
  1842. {
  1843. var parts = id.Split(new[] { '_' }, 2);
  1844. var service = _services.FirstOrDefault(i => string.Equals(i.GetType().FullName.GetMD5().ToString("N"), parts[0], StringComparison.OrdinalIgnoreCase));
  1845. if (service == null)
  1846. {
  1847. throw new ArgumentException("Service not found.");
  1848. }
  1849. return service.ResetTuner(parts[1], cancellationToken);
  1850. }
  1851. private void RemoveFields(DtoOptions options)
  1852. {
  1853. var fields = options.Fields.ToList();
  1854. fields.Remove(ItemFields.CanDelete);
  1855. fields.Remove(ItemFields.CanDownload);
  1856. fields.Remove(ItemFields.DisplayPreferencesId);
  1857. fields.Remove(ItemFields.Etag);
  1858. options.Fields = fields.ToArray();
  1859. }
  1860. public Folder GetInternalLiveTvFolder(CancellationToken cancellationToken)
  1861. {
  1862. var name = _localization.GetLocalizedString("HeaderLiveTV");
  1863. return _libraryManager.GetNamedView(name, CollectionType.LiveTv, name);
  1864. }
  1865. public async Task<TunerHostInfo> SaveTunerHost(TunerHostInfo info, bool dataSourceChanged = true)
  1866. {
  1867. info = _jsonSerializer.DeserializeFromString<TunerHostInfo>(_jsonSerializer.SerializeToString(info));
  1868. var provider = _tunerHosts.FirstOrDefault(i => string.Equals(info.Type, i.Type, StringComparison.OrdinalIgnoreCase));
  1869. if (provider == null)
  1870. {
  1871. throw new ResourceNotFoundException();
  1872. }
  1873. if (provider is IConfigurableTunerHost configurable)
  1874. {
  1875. await configurable.Validate(info).ConfigureAwait(false);
  1876. }
  1877. var config = GetConfiguration();
  1878. var list = config.TunerHosts.ToList();
  1879. var index = list.FindIndex(i => string.Equals(i.Id, info.Id, StringComparison.OrdinalIgnoreCase));
  1880. if (index == -1 || string.IsNullOrWhiteSpace(info.Id))
  1881. {
  1882. info.Id = Guid.NewGuid().ToString("N");
  1883. list.Add(info);
  1884. config.TunerHosts = list.ToArray();
  1885. }
  1886. else
  1887. {
  1888. config.TunerHosts[index] = info;
  1889. }
  1890. _config.SaveConfiguration("livetv", config);
  1891. if (dataSourceChanged)
  1892. {
  1893. _taskManager.CancelIfRunningAndQueue<RefreshChannelsScheduledTask>();
  1894. }
  1895. return info;
  1896. }
  1897. public async Task<ListingsProviderInfo> SaveListingProvider(ListingsProviderInfo info, bool validateLogin, bool validateListings)
  1898. {
  1899. // Hack to make the object a pure ListingsProviderInfo instead of an AddListingProvider
  1900. // ServerConfiguration.SaveConfiguration crashes during xml serialization for AddListingProvider
  1901. info = _jsonSerializer.DeserializeFromString<ListingsProviderInfo>(_jsonSerializer.SerializeToString(info));
  1902. IListingsProvider provider = _listingProviders.FirstOrDefault(i => string.Equals(info.Type, i.Type, StringComparison.OrdinalIgnoreCase));
  1903. if (provider == null)
  1904. {
  1905. throw new ResourceNotFoundException(
  1906. string.Format("Couldn't find provider of type: '{0}'", info.Type)
  1907. );
  1908. }
  1909. await provider.Validate(info, validateLogin, validateListings).ConfigureAwait(false);
  1910. LiveTvOptions config = GetConfiguration();
  1911. List<ListingsProviderInfo> list = config.ListingProviders.ToList();
  1912. int index = list.FindIndex(i => string.Equals(i.Id, info.Id, StringComparison.OrdinalIgnoreCase));
  1913. if (index == -1 || string.IsNullOrWhiteSpace(info.Id))
  1914. {
  1915. info.Id = Guid.NewGuid().ToString("N");
  1916. list.Add(info);
  1917. config.ListingProviders = list.ToArray();
  1918. }
  1919. else
  1920. {
  1921. config.ListingProviders[index] = info;
  1922. }
  1923. _config.SaveConfiguration("livetv", config);
  1924. _taskManager.CancelIfRunningAndQueue<RefreshChannelsScheduledTask>();
  1925. return info;
  1926. }
  1927. public void DeleteListingsProvider(string id)
  1928. {
  1929. var config = GetConfiguration();
  1930. config.ListingProviders = config.ListingProviders.Where(i => !string.Equals(id, i.Id, StringComparison.OrdinalIgnoreCase)).ToArray();
  1931. _config.SaveConfiguration("livetv", config);
  1932. _taskManager.CancelIfRunningAndQueue<RefreshChannelsScheduledTask>();
  1933. }
  1934. public async Task<TunerChannelMapping> SetChannelMapping(string providerId, string tunerChannelId, string providerChannelId)
  1935. {
  1936. var config = GetConfiguration();
  1937. var listingsProviderInfo = config.ListingProviders.First(i => string.Equals(providerId, i.Id, StringComparison.OrdinalIgnoreCase));
  1938. listingsProviderInfo.ChannelMappings = listingsProviderInfo.ChannelMappings.Where(i => !string.Equals(i.Name, tunerChannelId, StringComparison.OrdinalIgnoreCase)).ToArray();
  1939. if (!string.Equals(tunerChannelId, providerChannelId, StringComparison.OrdinalIgnoreCase))
  1940. {
  1941. var list = listingsProviderInfo.ChannelMappings.ToList();
  1942. list.Add(new NameValuePair
  1943. {
  1944. Name = tunerChannelId,
  1945. Value = providerChannelId
  1946. });
  1947. listingsProviderInfo.ChannelMappings = list.ToArray();
  1948. }
  1949. _config.SaveConfiguration("livetv", config);
  1950. var tunerChannels = await GetChannelsForListingsProvider(providerId, CancellationToken.None)
  1951. .ConfigureAwait(false);
  1952. var providerChannels = await GetChannelsFromListingsProviderData(providerId, CancellationToken.None)
  1953. .ConfigureAwait(false);
  1954. var mappings = listingsProviderInfo.ChannelMappings;
  1955. var tunerChannelMappings =
  1956. tunerChannels.Select(i => GetTunerChannelMapping(i, mappings, providerChannels)).ToList();
  1957. _taskManager.CancelIfRunningAndQueue<RefreshChannelsScheduledTask>();
  1958. return tunerChannelMappings.First(i => string.Equals(i.Id, tunerChannelId, StringComparison.OrdinalIgnoreCase));
  1959. }
  1960. public TunerChannelMapping GetTunerChannelMapping(ChannelInfo tunerChannel, NameValuePair[] mappings, List<ChannelInfo> epgChannels)
  1961. {
  1962. var result = new TunerChannelMapping
  1963. {
  1964. Name = tunerChannel.Name,
  1965. Id = tunerChannel.Id
  1966. };
  1967. if (!string.IsNullOrWhiteSpace(tunerChannel.Number))
  1968. {
  1969. result.Name = tunerChannel.Number + " " + result.Name;
  1970. }
  1971. var providerChannel = EmbyTV.EmbyTV.Current.GetEpgChannelFromTunerChannel(mappings, tunerChannel, epgChannels);
  1972. if (providerChannel != null)
  1973. {
  1974. result.ProviderChannelName = providerChannel.Name;
  1975. result.ProviderChannelId = providerChannel.Id;
  1976. }
  1977. return result;
  1978. }
  1979. public Task<List<NameIdPair>> GetLineups(string providerType, string providerId, string country, string location)
  1980. {
  1981. var config = GetConfiguration();
  1982. if (string.IsNullOrWhiteSpace(providerId))
  1983. {
  1984. var provider = _listingProviders.FirstOrDefault(i => string.Equals(providerType, i.Type, StringComparison.OrdinalIgnoreCase));
  1985. if (provider == null)
  1986. {
  1987. throw new ResourceNotFoundException();
  1988. }
  1989. return provider.GetLineups(null, country, location);
  1990. }
  1991. else
  1992. {
  1993. var info = config.ListingProviders.FirstOrDefault(i => string.Equals(i.Id, providerId, StringComparison.OrdinalIgnoreCase));
  1994. var provider = _listingProviders.FirstOrDefault(i => string.Equals(info.Type, i.Type, StringComparison.OrdinalIgnoreCase));
  1995. if (provider == null)
  1996. {
  1997. throw new ResourceNotFoundException();
  1998. }
  1999. return provider.GetLineups(info, country, location);
  2000. }
  2001. }
  2002. public Task<MBRegistrationRecord> GetRegistrationInfo(string feature)
  2003. {
  2004. if (string.Equals(feature, "seriesrecordings", StringComparison.OrdinalIgnoreCase))
  2005. {
  2006. feature = "embytvseriesrecordings";
  2007. }
  2008. if (string.Equals(feature, "dvr-l", StringComparison.OrdinalIgnoreCase))
  2009. {
  2010. var config = GetConfiguration();
  2011. if (config.TunerHosts.Length > 0 &&
  2012. config.ListingProviders.Count(i => (i.EnableAllTuners || i.EnabledTuners.Length > 0) && string.Equals(i.Type, SchedulesDirect.TypeName, StringComparison.OrdinalIgnoreCase)) > 0)
  2013. {
  2014. return Task.FromResult(new MBRegistrationRecord
  2015. {
  2016. IsRegistered = true,
  2017. IsValid = true
  2018. });
  2019. }
  2020. }
  2021. return _security.GetRegistrationStatus(feature);
  2022. }
  2023. public Task<List<ChannelInfo>> GetChannelsForListingsProvider(string id, CancellationToken cancellationToken)
  2024. {
  2025. var info = GetConfiguration().ListingProviders.First(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase));
  2026. return EmbyTV.EmbyTV.Current.GetChannelsForListingsProvider(info, cancellationToken);
  2027. }
  2028. public Task<List<ChannelInfo>> GetChannelsFromListingsProviderData(string id, CancellationToken cancellationToken)
  2029. {
  2030. var info = GetConfiguration().ListingProviders.First(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase));
  2031. var provider = _listingProviders.First(i => string.Equals(i.Type, info.Type, StringComparison.OrdinalIgnoreCase));
  2032. return provider.GetChannels(info, cancellationToken);
  2033. }
  2034. public Guid GetInternalChannelId(string serviceName, string externalId)
  2035. {
  2036. return _tvDtoService.GetInternalChannelId(serviceName, externalId);
  2037. }
  2038. public Guid GetInternalProgramId(string externalId)
  2039. {
  2040. return _tvDtoService.GetInternalProgramId(externalId);
  2041. }
  2042. public List<BaseItem> GetRecordingFolders(User user)
  2043. {
  2044. return GetRecordingFolders(user, false);
  2045. }
  2046. private List<BaseItem> GetRecordingFolders(User user, bool refreshChannels)
  2047. {
  2048. var folders = EmbyTV.EmbyTV.Current.GetRecordingFolders()
  2049. .SelectMany(i => i.Locations)
  2050. .Distinct(StringComparer.OrdinalIgnoreCase)
  2051. .Select(i => _libraryManager.FindByPath(i, true))
  2052. .Where(i => i != null)
  2053. .Where(i => i.IsVisibleStandalone(user))
  2054. .SelectMany(i => _libraryManager.GetCollectionFolders(i))
  2055. .DistinctBy(i => i.Id)
  2056. .OrderBy(i => i.SortName)
  2057. .ToList();
  2058. folders.AddRange(_channelManager().GetChannelsInternal(new MediaBrowser.Model.Channels.ChannelQuery
  2059. {
  2060. UserId = user.Id,
  2061. IsRecordingsFolder = true,
  2062. RefreshLatestChannelItems = refreshChannels
  2063. }).Items);
  2064. return folders.Cast<BaseItem>().ToList();
  2065. }
  2066. }
  2067. }