LiveTvService.cs 57 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using MediaBrowser.Api.UserLibrary;
  10. using MediaBrowser.Common.Configuration;
  11. using MediaBrowser.Common.Net;
  12. using MediaBrowser.Controller.Configuration;
  13. using MediaBrowser.Controller.Dto;
  14. using MediaBrowser.Controller.Entities;
  15. using MediaBrowser.Controller.Entities.TV;
  16. using MediaBrowser.Controller.Library;
  17. using MediaBrowser.Controller.LiveTv;
  18. using MediaBrowser.Controller.Net;
  19. using MediaBrowser.Model.Cryptography;
  20. using MediaBrowser.Model.Dto;
  21. using MediaBrowser.Model.Entities;
  22. using MediaBrowser.Model.IO;
  23. using MediaBrowser.Model.LiveTv;
  24. using MediaBrowser.Model.Querying;
  25. using MediaBrowser.Model.Services;
  26. using MediaBrowser.Model.System;
  27. using Microsoft.Net.Http.Headers;
  28. namespace MediaBrowser.Api.LiveTv
  29. {
  30. /// <summary>
  31. /// This is insecure right now to avoid windows phone refactoring
  32. /// </summary>
  33. [Route("/LiveTv/Info", "GET", Summary = "Gets available live tv services.")]
  34. [Authenticated]
  35. public class GetLiveTvInfo : IReturn<LiveTvInfo>
  36. {
  37. }
  38. [Route("/LiveTv/Channels", "GET", Summary = "Gets available live tv channels.")]
  39. [Authenticated]
  40. public class GetChannels : IReturn<QueryResult<BaseItemDto>>, IHasDtoOptions
  41. {
  42. [ApiMember(Name = "Type", Description = "Optional filter by channel type.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  43. public ChannelType? Type { get; set; }
  44. [ApiMember(Name = "UserId", Description = "Optional filter by user and attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  45. public Guid UserId { get; set; }
  46. /// <summary>
  47. /// Skips over a given number of items within the results. Use for paging.
  48. /// </summary>
  49. /// <value>The start index.</value>
  50. [ApiMember(Name = "StartIndex", Description = "Optional. The record index to start at. All items with a lower index will be dropped from the results.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  51. public int? StartIndex { get; set; }
  52. [ApiMember(Name = "IsMovie", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  53. public bool? IsMovie { get; set; }
  54. [ApiMember(Name = "IsSeries", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  55. public bool? IsSeries { get; set; }
  56. [ApiMember(Name = "IsNews", Description = "Optional filter for news.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  57. public bool? IsNews { get; set; }
  58. [ApiMember(Name = "IsKids", Description = "Optional filter for kids.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  59. public bool? IsKids { get; set; }
  60. [ApiMember(Name = "IsSports", Description = "Optional filter for sports.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  61. public bool? IsSports { get; set; }
  62. /// <summary>
  63. /// The maximum number of items to return
  64. /// </summary>
  65. /// <value>The limit.</value>
  66. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  67. public int? Limit { get; set; }
  68. [ApiMember(Name = "IsFavorite", Description = "Filter by channels that are favorites, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  69. public bool? IsFavorite { get; set; }
  70. [ApiMember(Name = "IsLiked", Description = "Filter by channels that are liked, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  71. public bool? IsLiked { get; set; }
  72. [ApiMember(Name = "IsDisliked", Description = "Filter by channels that are disliked, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  73. public bool? IsDisliked { get; set; }
  74. [ApiMember(Name = "EnableFavoriteSorting", Description = "Incorporate favorite and like status into channel sorting.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  75. public bool EnableFavoriteSorting { get; set; }
  76. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  77. public bool? EnableImages { get; set; }
  78. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  79. public int? ImageTypeLimit { get; set; }
  80. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  81. public string EnableImageTypes { get; set; }
  82. /// <summary>
  83. /// Fields to return within the items, in addition to basic information
  84. /// </summary>
  85. /// <value>The fields.</value>
  86. [ApiMember(Name = "Fields", Description = "Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  87. public string Fields { get; set; }
  88. [ApiMember(Name = "AddCurrentProgram", Description = "Optional. Adds current program info to each channel", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  89. public bool AddCurrentProgram { get; set; }
  90. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  91. public bool? EnableUserData { get; set; }
  92. public string SortBy { get; set; }
  93. public SortOrder? SortOrder { get; set; }
  94. /// <summary>
  95. /// Gets the order by.
  96. /// </summary>
  97. /// <returns>IEnumerable{ItemSortBy}.</returns>
  98. public string[] GetOrderBy()
  99. {
  100. var val = SortBy;
  101. if (string.IsNullOrEmpty(val))
  102. {
  103. return Array.Empty<string>();
  104. }
  105. return val.Split(',');
  106. }
  107. public GetChannels()
  108. {
  109. AddCurrentProgram = true;
  110. }
  111. }
  112. [Route("/LiveTv/Channels/{Id}", "GET", Summary = "Gets a live tv channel")]
  113. [Authenticated]
  114. public class GetChannel : IReturn<BaseItemDto>
  115. {
  116. /// <summary>
  117. /// Gets or sets the id.
  118. /// </summary>
  119. /// <value>The id.</value>
  120. [ApiMember(Name = "Id", Description = "Channel Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  121. public string Id { get; set; }
  122. [ApiMember(Name = "UserId", Description = "Optional attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  123. public Guid UserId { get; set; }
  124. }
  125. [Route("/LiveTv/Recordings", "GET", Summary = "Gets live tv recordings")]
  126. [Authenticated]
  127. public class GetRecordings : IReturn<QueryResult<BaseItemDto>>, IHasDtoOptions
  128. {
  129. [ApiMember(Name = "ChannelId", Description = "Optional filter by channel id.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  130. public string ChannelId { get; set; }
  131. [ApiMember(Name = "UserId", Description = "Optional filter by user and attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  132. public Guid UserId { get; set; }
  133. [ApiMember(Name = "StartIndex", Description = "Optional. The record index to start at. All items with a lower index will be dropped from the results.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  134. public int? StartIndex { get; set; }
  135. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  136. public int? Limit { get; set; }
  137. [ApiMember(Name = "Status", Description = "Optional filter by recording status.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  138. public RecordingStatus? Status { get; set; }
  139. [ApiMember(Name = "Status", Description = "Optional filter by recordings that are in progress, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  140. public bool? IsInProgress { get; set; }
  141. [ApiMember(Name = "SeriesTimerId", Description = "Optional filter by recordings belonging to a series timer", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  142. public string SeriesTimerId { get; set; }
  143. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  144. public bool? EnableImages { get; set; }
  145. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  146. public int? ImageTypeLimit { get; set; }
  147. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  148. public string EnableImageTypes { get; set; }
  149. /// <summary>
  150. /// Fields to return within the items, in addition to basic information
  151. /// </summary>
  152. /// <value>The fields.</value>
  153. [ApiMember(Name = "Fields", Description = "Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  154. public string Fields { get; set; }
  155. public bool EnableTotalRecordCount { get; set; }
  156. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  157. public bool? EnableUserData { get; set; }
  158. public bool? IsMovie { get; set; }
  159. public bool? IsSeries { get; set; }
  160. public bool? IsKids { get; set; }
  161. public bool? IsSports { get; set; }
  162. public bool? IsNews { get; set; }
  163. public bool? IsLibraryItem { get; set; }
  164. public GetRecordings()
  165. {
  166. EnableTotalRecordCount = true;
  167. }
  168. }
  169. [Route("/LiveTv/Recordings/Series", "GET", Summary = "Gets live tv recordings")]
  170. [Authenticated]
  171. public class GetRecordingSeries : IReturn<QueryResult<BaseItemDto>>, IHasDtoOptions
  172. {
  173. [ApiMember(Name = "ChannelId", Description = "Optional filter by channel id.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  174. public string ChannelId { get; set; }
  175. [ApiMember(Name = "UserId", Description = "Optional filter by user and attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  176. public string UserId { get; set; }
  177. [ApiMember(Name = "GroupId", Description = "Optional filter by recording group.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  178. public string GroupId { get; set; }
  179. [ApiMember(Name = "StartIndex", Description = "Optional. The record index to start at. All items with a lower index will be dropped from the results.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  180. public int? StartIndex { get; set; }
  181. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  182. public int? Limit { get; set; }
  183. [ApiMember(Name = "Status", Description = "Optional filter by recording status.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  184. public RecordingStatus? Status { get; set; }
  185. [ApiMember(Name = "Status", Description = "Optional filter by recordings that are in progress, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  186. public bool? IsInProgress { get; set; }
  187. [ApiMember(Name = "SeriesTimerId", Description = "Optional filter by recordings belonging to a series timer", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  188. public string SeriesTimerId { get; set; }
  189. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  190. public bool? EnableImages { get; set; }
  191. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  192. public int? ImageTypeLimit { get; set; }
  193. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  194. public string EnableImageTypes { get; set; }
  195. /// <summary>
  196. /// Fields to return within the items, in addition to basic information
  197. /// </summary>
  198. /// <value>The fields.</value>
  199. [ApiMember(Name = "Fields", Description = "Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  200. public string Fields { get; set; }
  201. public bool EnableTotalRecordCount { get; set; }
  202. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  203. public bool? EnableUserData { get; set; }
  204. public GetRecordingSeries()
  205. {
  206. EnableTotalRecordCount = true;
  207. }
  208. }
  209. [Route("/LiveTv/Recordings/Groups", "GET", Summary = "Gets live tv recording groups")]
  210. [Authenticated]
  211. public class GetRecordingGroups : IReturn<QueryResult<BaseItemDto>>
  212. {
  213. [ApiMember(Name = "UserId", Description = "Optional filter by user and attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  214. public string UserId { get; set; }
  215. }
  216. [Route("/LiveTv/Recordings/Folders", "GET", Summary = "Gets recording folders")]
  217. [Authenticated]
  218. public class GetRecordingFolders : IReturn<BaseItemDto[]>
  219. {
  220. [ApiMember(Name = "UserId", Description = "Optional filter by user and attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  221. public Guid UserId { get; set; }
  222. }
  223. [Route("/LiveTv/Recordings/{Id}", "GET", Summary = "Gets a live tv recording")]
  224. [Authenticated]
  225. public class GetRecording : IReturn<BaseItemDto>
  226. {
  227. [ApiMember(Name = "Id", Description = "Recording Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  228. public string Id { get; set; }
  229. [ApiMember(Name = "UserId", Description = "Optional attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  230. public Guid UserId { get; set; }
  231. }
  232. [Route("/LiveTv/Tuners/{Id}/Reset", "POST", Summary = "Resets a tv tuner")]
  233. [Authenticated]
  234. public class ResetTuner : IReturnVoid
  235. {
  236. [ApiMember(Name = "Id", Description = "Tuner Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  237. public string Id { get; set; }
  238. }
  239. [Route("/LiveTv/Timers/{Id}", "GET", Summary = "Gets a live tv timer")]
  240. [Authenticated]
  241. public class GetTimer : IReturn<TimerInfoDto>
  242. {
  243. [ApiMember(Name = "Id", Description = "Timer Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  244. public string Id { get; set; }
  245. }
  246. [Route("/LiveTv/Timers/Defaults", "GET", Summary = "Gets default values for a new timer")]
  247. [Authenticated]
  248. public class GetDefaultTimer : IReturn<SeriesTimerInfoDto>
  249. {
  250. [ApiMember(Name = "ProgramId", Description = "Optional, to attach default values based on a program.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  251. public string ProgramId { get; set; }
  252. }
  253. [Route("/LiveTv/Timers", "GET", Summary = "Gets live tv timers")]
  254. [Authenticated]
  255. public class GetTimers : IReturn<QueryResult<TimerInfoDto>>
  256. {
  257. [ApiMember(Name = "ChannelId", Description = "Optional filter by channel id.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  258. public string ChannelId { get; set; }
  259. [ApiMember(Name = "SeriesTimerId", Description = "Optional filter by timers belonging to a series timer", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  260. public string SeriesTimerId { get; set; }
  261. public bool? IsActive { get; set; }
  262. public bool? IsScheduled { get; set; }
  263. }
  264. [Route("/LiveTv/Programs", "GET,POST", Summary = "Gets available live tv epgs..")]
  265. [Authenticated]
  266. public class GetPrograms : IReturn<QueryResult<BaseItemDto>>, IHasDtoOptions
  267. {
  268. [ApiMember(Name = "ChannelIds", Description = "The channels to return guide information for.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  269. public string ChannelIds { get; set; }
  270. [ApiMember(Name = "UserId", Description = "Optional filter by user id.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  271. public Guid UserId { get; set; }
  272. [ApiMember(Name = "MinStartDate", Description = "Optional. The minimum premiere date. Format = ISO", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  273. public string MinStartDate { get; set; }
  274. [ApiMember(Name = "HasAired", Description = "Optional. Filter by programs that have completed airing, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  275. public bool? HasAired { get; set; }
  276. public bool? IsAiring { get; set; }
  277. [ApiMember(Name = "MaxStartDate", Description = "Optional. The maximum premiere date. Format = ISO", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  278. public string MaxStartDate { get; set; }
  279. [ApiMember(Name = "MinEndDate", Description = "Optional. The minimum premiere date. Format = ISO", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  280. public string MinEndDate { get; set; }
  281. [ApiMember(Name = "MaxEndDate", Description = "Optional. The maximum premiere date. Format = ISO", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  282. public string MaxEndDate { get; set; }
  283. [ApiMember(Name = "IsMovie", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  284. public bool? IsMovie { get; set; }
  285. [ApiMember(Name = "IsSeries", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  286. public bool? IsSeries { get; set; }
  287. [ApiMember(Name = "IsNews", Description = "Optional filter for news.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  288. public bool? IsNews { get; set; }
  289. [ApiMember(Name = "IsKids", Description = "Optional filter for kids.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  290. public bool? IsKids { get; set; }
  291. [ApiMember(Name = "IsSports", Description = "Optional filter for sports.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  292. public bool? IsSports { get; set; }
  293. [ApiMember(Name = "StartIndex", Description = "Optional. The record index to start at. All items with a lower index will be dropped from the results.", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  294. public int? StartIndex { get; set; }
  295. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  296. public int? Limit { get; set; }
  297. [ApiMember(Name = "SortBy", Description = "Optional. Specify one or more sort orders, comma delimeted. Options: Name, StartDate", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  298. public string SortBy { get; set; }
  299. [ApiMember(Name = "SortOrder", Description = "Sort Order - Ascending,Descending", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  300. public string SortOrder { get; set; }
  301. [ApiMember(Name = "Genres", Description = "The genres to return guide information for.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  302. public string Genres { get; set; }
  303. [ApiMember(Name = "GenreIds", Description = "The genres to return guide information for.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  304. public string GenreIds { get; set; }
  305. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  306. public bool? EnableImages { get; set; }
  307. public bool EnableTotalRecordCount { get; set; }
  308. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  309. public int? ImageTypeLimit { get; set; }
  310. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  311. public string EnableImageTypes { get; set; }
  312. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  313. public bool? EnableUserData { get; set; }
  314. public string SeriesTimerId { get; set; }
  315. public Guid LibrarySeriesId { get; set; }
  316. /// <summary>
  317. /// Fields to return within the items, in addition to basic information
  318. /// </summary>
  319. /// <value>The fields.</value>
  320. [ApiMember(Name = "Fields", Description = "Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  321. public string Fields { get; set; }
  322. public GetPrograms()
  323. {
  324. EnableTotalRecordCount = true;
  325. }
  326. }
  327. [Route("/LiveTv/Programs/Recommended", "GET", Summary = "Gets available live tv epgs..")]
  328. [Authenticated]
  329. public class GetRecommendedPrograms : IReturn<QueryResult<BaseItemDto>>, IHasDtoOptions
  330. {
  331. public bool EnableTotalRecordCount { get; set; }
  332. public GetRecommendedPrograms()
  333. {
  334. EnableTotalRecordCount = true;
  335. }
  336. [ApiMember(Name = "UserId", Description = "Optional filter by user id.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  337. public Guid UserId { get; set; }
  338. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  339. public int? Limit { get; set; }
  340. [ApiMember(Name = "IsAiring", Description = "Optional. Filter by programs that are currently airing, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  341. public bool? IsAiring { get; set; }
  342. [ApiMember(Name = "HasAired", Description = "Optional. Filter by programs that have completed airing, or not.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  343. public bool? HasAired { get; set; }
  344. [ApiMember(Name = "IsSeries", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  345. public bool? IsSeries { get; set; }
  346. [ApiMember(Name = "IsMovie", Description = "Optional filter for movies.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  347. public bool? IsMovie { get; set; }
  348. [ApiMember(Name = "IsNews", Description = "Optional filter for news.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  349. public bool? IsNews { get; set; }
  350. [ApiMember(Name = "IsKids", Description = "Optional filter for kids.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  351. public bool? IsKids { get; set; }
  352. [ApiMember(Name = "IsSports", Description = "Optional filter for sports.", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET,POST")]
  353. public bool? IsSports { get; set; }
  354. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  355. public bool? EnableImages { get; set; }
  356. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  357. public int? ImageTypeLimit { get; set; }
  358. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  359. public string EnableImageTypes { get; set; }
  360. [ApiMember(Name = "GenreIds", Description = "The genres to return guide information for.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  361. public string GenreIds { get; set; }
  362. /// <summary>
  363. /// Fields to return within the items, in addition to basic information
  364. /// </summary>
  365. /// <value>The fields.</value>
  366. [ApiMember(Name = "Fields", Description = "Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET", AllowMultiple = true)]
  367. public string Fields { get; set; }
  368. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  369. public bool? EnableUserData { get; set; }
  370. }
  371. [Route("/LiveTv/Programs/{Id}", "GET", Summary = "Gets a live tv program")]
  372. [Authenticated]
  373. public class GetProgram : IReturn<BaseItemDto>
  374. {
  375. [ApiMember(Name = "Id", Description = "Program Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  376. public string Id { get; set; }
  377. [ApiMember(Name = "UserId", Description = "Optional attach user data.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  378. public Guid UserId { get; set; }
  379. }
  380. [Route("/LiveTv/Recordings/{Id}", "DELETE", Summary = "Deletes a live tv recording")]
  381. [Authenticated]
  382. public class DeleteRecording : IReturnVoid
  383. {
  384. [ApiMember(Name = "Id", Description = "Recording Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  385. public Guid Id { get; set; }
  386. }
  387. [Route("/LiveTv/Timers/{Id}", "DELETE", Summary = "Cancels a live tv timer")]
  388. [Authenticated]
  389. public class CancelTimer : IReturnVoid
  390. {
  391. [ApiMember(Name = "Id", Description = "Timer Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  392. public string Id { get; set; }
  393. }
  394. [Route("/LiveTv/Timers/{Id}", "POST", Summary = "Updates a live tv timer")]
  395. [Authenticated]
  396. public class UpdateTimer : TimerInfoDto, IReturnVoid
  397. {
  398. }
  399. [Route("/LiveTv/Timers", "POST", Summary = "Creates a live tv timer")]
  400. [Authenticated]
  401. public class CreateTimer : TimerInfoDto, IReturnVoid
  402. {
  403. }
  404. [Route("/LiveTv/SeriesTimers/{Id}", "GET", Summary = "Gets a live tv series timer")]
  405. [Authenticated]
  406. public class GetSeriesTimer : IReturn<TimerInfoDto>
  407. {
  408. [ApiMember(Name = "Id", Description = "Timer Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  409. public string Id { get; set; }
  410. }
  411. [Route("/LiveTv/SeriesTimers", "GET", Summary = "Gets live tv series timers")]
  412. [Authenticated]
  413. public class GetSeriesTimers : IReturn<QueryResult<SeriesTimerInfoDto>>
  414. {
  415. [ApiMember(Name = "SortBy", Description = "Optional. Sort by SortName or Priority", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  416. public string SortBy { get; set; }
  417. [ApiMember(Name = "SortOrder", Description = "Optional. Sort in Ascending or Descending order", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET,POST")]
  418. public SortOrder SortOrder { get; set; }
  419. }
  420. [Route("/LiveTv/SeriesTimers/{Id}", "DELETE", Summary = "Cancels a live tv series timer")]
  421. [Authenticated]
  422. public class CancelSeriesTimer : IReturnVoid
  423. {
  424. [ApiMember(Name = "Id", Description = "Timer Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  425. public string Id { get; set; }
  426. }
  427. [Route("/LiveTv/SeriesTimers/{Id}", "POST", Summary = "Updates a live tv series timer")]
  428. [Authenticated]
  429. public class UpdateSeriesTimer : SeriesTimerInfoDto, IReturnVoid
  430. {
  431. }
  432. [Route("/LiveTv/SeriesTimers", "POST", Summary = "Creates a live tv series timer")]
  433. [Authenticated]
  434. public class CreateSeriesTimer : SeriesTimerInfoDto, IReturnVoid
  435. {
  436. }
  437. [Route("/LiveTv/Recordings/Groups/{Id}", "GET", Summary = "Gets a recording group")]
  438. [Authenticated]
  439. public class GetRecordingGroup : IReturn<BaseItemDto>
  440. {
  441. [ApiMember(Name = "Id", Description = "Recording group Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  442. public string Id { get; set; }
  443. }
  444. [Route("/LiveTv/GuideInfo", "GET", Summary = "Gets guide info")]
  445. [Authenticated]
  446. public class GetGuideInfo : IReturn<GuideInfo>
  447. {
  448. }
  449. [Route("/LiveTv/TunerHosts", "POST", Summary = "Adds a tuner host")]
  450. [Authenticated]
  451. public class AddTunerHost : TunerHostInfo, IReturn<TunerHostInfo>
  452. {
  453. }
  454. [Route("/LiveTv/TunerHosts", "DELETE", Summary = "Deletes a tuner host")]
  455. [Authenticated]
  456. public class DeleteTunerHost : IReturnVoid
  457. {
  458. [ApiMember(Name = "Id", Description = "Tuner host id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "DELETE")]
  459. public string Id { get; set; }
  460. }
  461. [Route("/LiveTv/ListingProviders/Default", "GET")]
  462. [Authenticated]
  463. public class GetDefaultListingProvider : ListingsProviderInfo, IReturn<ListingsProviderInfo>
  464. {
  465. }
  466. [Route("/LiveTv/ListingProviders", "POST", Summary = "Adds a listing provider")]
  467. [Authenticated]
  468. public class AddListingProvider : ListingsProviderInfo, IReturn<ListingsProviderInfo>
  469. {
  470. public bool ValidateLogin { get; set; }
  471. public bool ValidateListings { get; set; }
  472. public string Pw { get; set; }
  473. }
  474. [Route("/LiveTv/ListingProviders", "DELETE", Summary = "Deletes a listing provider")]
  475. [Authenticated]
  476. public class DeleteListingProvider : IReturnVoid
  477. {
  478. [ApiMember(Name = "Id", Description = "Provider id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "DELETE")]
  479. public string Id { get; set; }
  480. }
  481. [Route("/LiveTv/ListingProviders/Lineups", "GET", Summary = "Gets available lineups")]
  482. [Authenticated]
  483. public class GetLineups : IReturn<List<NameIdPair>>
  484. {
  485. [ApiMember(Name = "Id", Description = "Provider id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  486. public string Id { get; set; }
  487. [ApiMember(Name = "Type", Description = "Provider Type", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  488. public string Type { get; set; }
  489. [ApiMember(Name = "Location", Description = "Location", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  490. public string Location { get; set; }
  491. [ApiMember(Name = "Country", Description = "Country", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  492. public string Country { get; set; }
  493. }
  494. [Route("/LiveTv/ListingProviders/SchedulesDirect/Countries", "GET", Summary = "Gets available lineups")]
  495. [Authenticated]
  496. public class GetSchedulesDirectCountries
  497. {
  498. }
  499. [Route("/LiveTv/ChannelMappingOptions")]
  500. [Authenticated]
  501. public class GetChannelMappingOptions
  502. {
  503. [ApiMember(Name = "Id", Description = "Provider id", IsRequired = true, DataType = "string", ParameterType = "query")]
  504. public string ProviderId { get; set; }
  505. }
  506. [Route("/LiveTv/ChannelMappings")]
  507. [Authenticated]
  508. public class SetChannelMapping
  509. {
  510. [ApiMember(Name = "Id", Description = "Provider id", IsRequired = true, DataType = "string", ParameterType = "query")]
  511. public string ProviderId { get; set; }
  512. public string TunerChannelId { get; set; }
  513. public string ProviderChannelId { get; set; }
  514. }
  515. public class ChannelMappingOptions
  516. {
  517. public List<TunerChannelMapping> TunerChannels { get; set; }
  518. public List<NameIdPair> ProviderChannels { get; set; }
  519. public NameValuePair[] Mappings { get; set; }
  520. public string ProviderName { get; set; }
  521. }
  522. [Route("/LiveTv/LiveStreamFiles/{Id}/stream.{Container}", "GET", Summary = "Gets a live tv channel")]
  523. public class GetLiveStreamFile
  524. {
  525. public string Id { get; set; }
  526. public string Container { get; set; }
  527. }
  528. [Route("/LiveTv/LiveRecordings/{Id}/stream", "GET", Summary = "Gets a live tv channel")]
  529. public class GetLiveRecordingFile
  530. {
  531. public string Id { get; set; }
  532. }
  533. [Route("/LiveTv/TunerHosts/Types", "GET")]
  534. [Authenticated]
  535. public class GetTunerHostTypes : IReturn<List<NameIdPair>>
  536. {
  537. }
  538. [Route("/LiveTv/Tuners/Discvover", "GET")]
  539. [Authenticated]
  540. public class DiscoverTuners : IReturn<List<TunerHostInfo>>
  541. {
  542. public bool NewDevicesOnly { get; set; }
  543. }
  544. public class LiveTvService : BaseApiService
  545. {
  546. private readonly ILiveTvManager _liveTvManager;
  547. private readonly IUserManager _userManager;
  548. private readonly IServerConfigurationManager _config;
  549. private readonly IHttpClient _httpClient;
  550. private readonly ILibraryManager _libraryManager;
  551. private readonly IDtoService _dtoService;
  552. private readonly IFileSystem _fileSystem;
  553. private readonly IAuthorizationContext _authContext;
  554. private readonly ISessionContext _sessionContext;
  555. private ICryptoProvider _cryptographyProvider;
  556. private IStreamHelper _streamHelper;
  557. private IMediaSourceManager _mediaSourceManager;
  558. public LiveTvService(ICryptoProvider crypto, IMediaSourceManager mediaSourceManager, IStreamHelper streamHelper, ILiveTvManager liveTvManager, IUserManager userManager, IServerConfigurationManager config, IHttpClient httpClient, ILibraryManager libraryManager, IDtoService dtoService, IFileSystem fileSystem, IAuthorizationContext authContext, ISessionContext sessionContext)
  559. {
  560. _liveTvManager = liveTvManager;
  561. _userManager = userManager;
  562. _config = config;
  563. _httpClient = httpClient;
  564. _libraryManager = libraryManager;
  565. _dtoService = dtoService;
  566. _fileSystem = fileSystem;
  567. _authContext = authContext;
  568. _sessionContext = sessionContext;
  569. _cryptographyProvider = crypto;
  570. _streamHelper = streamHelper;
  571. _mediaSourceManager = mediaSourceManager;
  572. }
  573. public object Get(GetTunerHostTypes request)
  574. {
  575. var list = _liveTvManager.GetTunerHostTypes();
  576. return ToOptimizedResult(list);
  577. }
  578. public object Get(GetRecordingFolders request)
  579. {
  580. var user = request.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(request.UserId);
  581. var folders = _liveTvManager.GetRecordingFolders(user);
  582. var returnArray = _dtoService.GetBaseItemDtos(folders.ToArray(), new DtoOptions(), user);
  583. var result = new QueryResult<BaseItemDto>
  584. {
  585. Items = returnArray,
  586. TotalRecordCount = returnArray.Length
  587. };
  588. return ToOptimizedResult(result);
  589. }
  590. public object Get(GetLiveRecordingFile request)
  591. {
  592. var path = _liveTvManager.GetEmbyTvActiveRecordingPath(request.Id);
  593. if (string.IsNullOrWhiteSpace(path))
  594. {
  595. throw new FileNotFoundException();
  596. }
  597. var outputHeaders = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  598. {
  599. [HeaderNames.ContentType] = Model.Net.MimeTypes.GetMimeType(path)
  600. };
  601. return new ProgressiveFileCopier(_fileSystem, _streamHelper, path, outputHeaders, Logger)
  602. {
  603. AllowEndOfFile = false
  604. };
  605. }
  606. public async Task<object> Get(DiscoverTuners request)
  607. {
  608. var result = await _liveTvManager.DiscoverTuners(request.NewDevicesOnly, CancellationToken.None).ConfigureAwait(false);
  609. return ToOptimizedResult(result);
  610. }
  611. public async Task<object> Get(GetLiveStreamFile request)
  612. {
  613. var liveStreamInfo = await _mediaSourceManager.GetDirectStreamProviderByUniqueId(request.Id, CancellationToken.None).ConfigureAwait(false);
  614. var directStreamProvider = liveStreamInfo;
  615. var outputHeaders = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  616. {
  617. [HeaderNames.ContentType] = Model.Net.MimeTypes.GetMimeType("file." + request.Container)
  618. };
  619. return new ProgressiveFileCopier(directStreamProvider, _streamHelper, outputHeaders, Logger)
  620. {
  621. AllowEndOfFile = false
  622. };
  623. }
  624. public object Get(GetDefaultListingProvider request)
  625. {
  626. return ToOptimizedResult(new ListingsProviderInfo());
  627. }
  628. public async Task<object> Post(SetChannelMapping request)
  629. {
  630. return await _liveTvManager.SetChannelMapping(request.ProviderId, request.TunerChannelId, request.ProviderChannelId).ConfigureAwait(false);
  631. }
  632. public async Task<object> Get(GetChannelMappingOptions request)
  633. {
  634. var config = GetConfiguration();
  635. var listingsProviderInfo = config.ListingProviders.First(i => string.Equals(request.ProviderId, i.Id, StringComparison.OrdinalIgnoreCase));
  636. var listingsProviderName = _liveTvManager.ListingProviders.First(i => string.Equals(i.Type, listingsProviderInfo.Type, StringComparison.OrdinalIgnoreCase)).Name;
  637. var tunerChannels = await _liveTvManager.GetChannelsForListingsProvider(request.ProviderId, CancellationToken.None)
  638. .ConfigureAwait(false);
  639. var providerChannels = await _liveTvManager.GetChannelsFromListingsProviderData(request.ProviderId, CancellationToken.None)
  640. .ConfigureAwait(false);
  641. var mappings = listingsProviderInfo.ChannelMappings;
  642. var result = new ChannelMappingOptions
  643. {
  644. TunerChannels = tunerChannels.Select(i => _liveTvManager.GetTunerChannelMapping(i, mappings, providerChannels)).ToList(),
  645. ProviderChannels = providerChannels.Select(i => new NameIdPair
  646. {
  647. Name = i.Name,
  648. Id = i.Id
  649. }).ToList(),
  650. Mappings = mappings,
  651. ProviderName = listingsProviderName
  652. };
  653. return ToOptimizedResult(result);
  654. }
  655. public async Task<object> Get(GetSchedulesDirectCountries request)
  656. {
  657. // https://json.schedulesdirect.org/20141201/available/countries
  658. var response = await _httpClient.Get(new HttpRequestOptions
  659. {
  660. Url = "https://json.schedulesdirect.org/20141201/available/countries",
  661. BufferContent = false
  662. }).ConfigureAwait(false);
  663. return ResultFactory.GetResult(Request, response, "application/json");
  664. }
  665. private void AssertUserCanManageLiveTv()
  666. {
  667. var user = _sessionContext.GetUser(Request);
  668. if (user == null)
  669. {
  670. throw new SecurityException("Anonymous live tv management is not allowed.");
  671. }
  672. if (!user.Policy.EnableLiveTvManagement)
  673. {
  674. throw new SecurityException("The current user does not have permission to manage live tv.");
  675. }
  676. }
  677. public async Task<object> Post(AddListingProvider request)
  678. {
  679. if (request.Pw != null)
  680. {
  681. request.Password = GetHashedString(request.Pw);
  682. }
  683. request.Pw = null;
  684. var result = await _liveTvManager.SaveListingProvider(request, request.ValidateLogin, request.ValidateListings).ConfigureAwait(false);
  685. return ToOptimizedResult(result);
  686. }
  687. /// <summary>
  688. /// Gets the hashed string.
  689. /// </summary>
  690. private string GetHashedString(string str)
  691. {
  692. // legacy
  693. return BitConverter.ToString(
  694. _cryptographyProvider.ComputeSHA1(Encoding.UTF8.GetBytes(str)))
  695. .Replace("-", string.Empty).ToLowerInvariant();
  696. }
  697. public void Delete(DeleteListingProvider request)
  698. {
  699. _liveTvManager.DeleteListingsProvider(request.Id);
  700. }
  701. public async Task<object> Post(AddTunerHost request)
  702. {
  703. var result = await _liveTvManager.SaveTunerHost(request).ConfigureAwait(false);
  704. return ToOptimizedResult(result);
  705. }
  706. public void Delete(DeleteTunerHost request)
  707. {
  708. var config = GetConfiguration();
  709. config.TunerHosts = config.TunerHosts.Where(i => !string.Equals(request.Id, i.Id, StringComparison.OrdinalIgnoreCase)).ToArray();
  710. _config.SaveConfiguration("livetv", config);
  711. }
  712. private LiveTvOptions GetConfiguration()
  713. {
  714. return _config.GetConfiguration<LiveTvOptions>("livetv");
  715. }
  716. private void UpdateConfiguration(LiveTvOptions options)
  717. {
  718. _config.SaveConfiguration("livetv", options);
  719. }
  720. public async Task<object> Get(GetLineups request)
  721. {
  722. var info = await _liveTvManager.GetLineups(request.Type, request.Id, request.Country, request.Location).ConfigureAwait(false);
  723. return ToOptimizedResult(info);
  724. }
  725. public object Get(GetLiveTvInfo request)
  726. {
  727. var info = _liveTvManager.GetLiveTvInfo(CancellationToken.None);
  728. return ToOptimizedResult(info);
  729. }
  730. public object Get(GetChannels request)
  731. {
  732. var options = GetDtoOptions(_authContext, request);
  733. var channelResult = _liveTvManager.GetInternalChannels(new LiveTvChannelQuery
  734. {
  735. ChannelType = request.Type,
  736. UserId = request.UserId,
  737. StartIndex = request.StartIndex,
  738. Limit = request.Limit,
  739. IsFavorite = request.IsFavorite,
  740. IsLiked = request.IsLiked,
  741. IsDisliked = request.IsDisliked,
  742. EnableFavoriteSorting = request.EnableFavoriteSorting,
  743. IsMovie = request.IsMovie,
  744. IsSeries = request.IsSeries,
  745. IsNews = request.IsNews,
  746. IsKids = request.IsKids,
  747. IsSports = request.IsSports,
  748. SortBy = request.GetOrderBy(),
  749. SortOrder = request.SortOrder ?? SortOrder.Ascending,
  750. AddCurrentProgram = request.AddCurrentProgram
  751. }, options, CancellationToken.None);
  752. var user = request.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(request.UserId);
  753. RemoveFields(options);
  754. options.AddCurrentProgram = request.AddCurrentProgram;
  755. var returnArray = _dtoService.GetBaseItemDtos(channelResult.Items, options, user);
  756. var result = new QueryResult<BaseItemDto>
  757. {
  758. Items = returnArray,
  759. TotalRecordCount = channelResult.TotalRecordCount
  760. };
  761. return ToOptimizedResult(result);
  762. }
  763. private void RemoveFields(DtoOptions options)
  764. {
  765. var fields = options.Fields.ToList();
  766. fields.Remove(ItemFields.CanDelete);
  767. fields.Remove(ItemFields.CanDownload);
  768. fields.Remove(ItemFields.DisplayPreferencesId);
  769. fields.Remove(ItemFields.Etag);
  770. options.Fields = fields.ToArray();
  771. }
  772. public object Get(GetChannel request)
  773. {
  774. var user = _userManager.GetUserById(request.UserId);
  775. var item = string.IsNullOrEmpty(request.Id) ? _libraryManager.GetUserRootFolder() : _libraryManager.GetItemById(request.Id);
  776. var dtoOptions = GetDtoOptions(_authContext, request);
  777. var result = _dtoService.GetBaseItemDto(item, dtoOptions, user);
  778. return ToOptimizedResult(result);
  779. }
  780. public async Task<object> Get(GetPrograms request)
  781. {
  782. var user = request.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(request.UserId);
  783. var query = new InternalItemsQuery(user)
  784. {
  785. ChannelIds = ApiEntryPoint.Split(request.ChannelIds, ',', true).Select(i => new Guid(i)).ToArray(),
  786. HasAired = request.HasAired,
  787. IsAiring = request.IsAiring,
  788. EnableTotalRecordCount = request.EnableTotalRecordCount
  789. };
  790. if (!string.IsNullOrEmpty(request.MinStartDate))
  791. {
  792. query.MinStartDate = DateTime.Parse(request.MinStartDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  793. }
  794. if (!string.IsNullOrEmpty(request.MinEndDate))
  795. {
  796. query.MinEndDate = DateTime.Parse(request.MinEndDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  797. }
  798. if (!string.IsNullOrEmpty(request.MaxStartDate))
  799. {
  800. query.MaxStartDate = DateTime.Parse(request.MaxStartDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  801. }
  802. if (!string.IsNullOrEmpty(request.MaxEndDate))
  803. {
  804. query.MaxEndDate = DateTime.Parse(request.MaxEndDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  805. }
  806. query.StartIndex = request.StartIndex;
  807. query.Limit = request.Limit;
  808. query.OrderBy = BaseItemsRequest.GetOrderBy(request.SortBy, request.SortOrder);
  809. query.IsNews = request.IsNews;
  810. query.IsMovie = request.IsMovie;
  811. query.IsSeries = request.IsSeries;
  812. query.IsKids = request.IsKids;
  813. query.IsSports = request.IsSports;
  814. query.SeriesTimerId = request.SeriesTimerId;
  815. query.Genres = (request.Genres ?? string.Empty).Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
  816. query.GenreIds = GetGuids(request.GenreIds);
  817. if (!request.LibrarySeriesId.Equals(Guid.Empty))
  818. {
  819. query.IsSeries = true;
  820. var series = _libraryManager.GetItemById(request.LibrarySeriesId) as Series;
  821. if (series != null)
  822. {
  823. query.Name = series.Name;
  824. }
  825. }
  826. var result = await _liveTvManager.GetPrograms(query, GetDtoOptions(_authContext, request), CancellationToken.None).ConfigureAwait(false);
  827. return ToOptimizedResult(result);
  828. }
  829. public object Get(GetRecommendedPrograms request)
  830. {
  831. var user = _userManager.GetUserById(request.UserId);
  832. var query = new InternalItemsQuery(user)
  833. {
  834. IsAiring = request.IsAiring,
  835. Limit = request.Limit,
  836. HasAired = request.HasAired,
  837. IsSeries = request.IsSeries,
  838. IsMovie = request.IsMovie,
  839. IsKids = request.IsKids,
  840. IsNews = request.IsNews,
  841. IsSports = request.IsSports,
  842. EnableTotalRecordCount = request.EnableTotalRecordCount
  843. };
  844. query.GenreIds = GetGuids(request.GenreIds);
  845. var result = _liveTvManager.GetRecommendedPrograms(query, GetDtoOptions(_authContext, request), CancellationToken.None);
  846. return ToOptimizedResult(result);
  847. }
  848. public object Post(GetPrograms request)
  849. {
  850. return Get(request);
  851. }
  852. public object Get(GetRecordings request)
  853. {
  854. var options = GetDtoOptions(_authContext, request);
  855. var result = _liveTvManager.GetRecordings(new RecordingQuery
  856. {
  857. ChannelId = request.ChannelId,
  858. UserId = request.UserId,
  859. StartIndex = request.StartIndex,
  860. Limit = request.Limit,
  861. Status = request.Status,
  862. SeriesTimerId = request.SeriesTimerId,
  863. IsInProgress = request.IsInProgress,
  864. EnableTotalRecordCount = request.EnableTotalRecordCount,
  865. IsMovie = request.IsMovie,
  866. IsNews = request.IsNews,
  867. IsSeries = request.IsSeries,
  868. IsKids = request.IsKids,
  869. IsSports = request.IsSports,
  870. IsLibraryItem = request.IsLibraryItem,
  871. Fields = request.GetItemFields(),
  872. ImageTypeLimit = request.ImageTypeLimit,
  873. EnableImages = request.EnableImages
  874. }, options);
  875. return ToOptimizedResult(result);
  876. }
  877. public object Get(GetRecordingSeries request)
  878. {
  879. return ToOptimizedResult(new QueryResult<BaseItemDto>());
  880. }
  881. public object Get(GetRecording request)
  882. {
  883. var user = _userManager.GetUserById(request.UserId);
  884. var item = string.IsNullOrEmpty(request.Id) ? _libraryManager.GetUserRootFolder() : _libraryManager.GetItemById(request.Id);
  885. var dtoOptions = GetDtoOptions(_authContext, request);
  886. var result = _dtoService.GetBaseItemDto(item, dtoOptions, user);
  887. return ToOptimizedResult(result);
  888. }
  889. public async Task<object> Get(GetTimer request)
  890. {
  891. var result = await _liveTvManager.GetTimer(request.Id, CancellationToken.None).ConfigureAwait(false);
  892. return ToOptimizedResult(result);
  893. }
  894. public async Task<object> Get(GetTimers request)
  895. {
  896. var result = await _liveTvManager.GetTimers(new TimerQuery
  897. {
  898. ChannelId = request.ChannelId,
  899. SeriesTimerId = request.SeriesTimerId,
  900. IsActive = request.IsActive,
  901. IsScheduled = request.IsScheduled
  902. }, CancellationToken.None).ConfigureAwait(false);
  903. return ToOptimizedResult(result);
  904. }
  905. public void Delete(DeleteRecording request)
  906. {
  907. AssertUserCanManageLiveTv();
  908. _libraryManager.DeleteItem(_libraryManager.GetItemById(request.Id), new DeleteOptions
  909. {
  910. DeleteFileLocation = false
  911. });
  912. }
  913. public Task Delete(CancelTimer request)
  914. {
  915. AssertUserCanManageLiveTv();
  916. return _liveTvManager.CancelTimer(request.Id);
  917. }
  918. public Task Post(UpdateTimer request)
  919. {
  920. AssertUserCanManageLiveTv();
  921. return _liveTvManager.UpdateTimer(request, CancellationToken.None);
  922. }
  923. public async Task<object> Get(GetSeriesTimers request)
  924. {
  925. var result = await _liveTvManager.GetSeriesTimers(new SeriesTimerQuery
  926. {
  927. SortOrder = request.SortOrder,
  928. SortBy = request.SortBy
  929. }, CancellationToken.None).ConfigureAwait(false);
  930. return ToOptimizedResult(result);
  931. }
  932. public async Task<object> Get(GetSeriesTimer request)
  933. {
  934. var result = await _liveTvManager.GetSeriesTimer(request.Id, CancellationToken.None).ConfigureAwait(false);
  935. return ToOptimizedResult(result);
  936. }
  937. public Task Delete(CancelSeriesTimer request)
  938. {
  939. AssertUserCanManageLiveTv();
  940. return _liveTvManager.CancelSeriesTimer(request.Id);
  941. }
  942. public Task Post(UpdateSeriesTimer request)
  943. {
  944. AssertUserCanManageLiveTv();
  945. return _liveTvManager.UpdateSeriesTimer(request, CancellationToken.None);
  946. }
  947. public async Task<object> Get(GetDefaultTimer request)
  948. {
  949. if (string.IsNullOrEmpty(request.ProgramId))
  950. {
  951. var result = await _liveTvManager.GetNewTimerDefaults(CancellationToken.None).ConfigureAwait(false);
  952. return ToOptimizedResult(result);
  953. }
  954. else
  955. {
  956. var result = await _liveTvManager.GetNewTimerDefaults(request.ProgramId, CancellationToken.None).ConfigureAwait(false);
  957. return ToOptimizedResult(result);
  958. }
  959. }
  960. public async Task<object> Get(GetProgram request)
  961. {
  962. var user = request.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(request.UserId);
  963. var result = await _liveTvManager.GetProgram(request.Id, CancellationToken.None, user).ConfigureAwait(false);
  964. return ToOptimizedResult(result);
  965. }
  966. public Task Post(CreateSeriesTimer request)
  967. {
  968. AssertUserCanManageLiveTv();
  969. return _liveTvManager.CreateSeriesTimer(request, CancellationToken.None);
  970. }
  971. public Task Post(CreateTimer request)
  972. {
  973. AssertUserCanManageLiveTv();
  974. return _liveTvManager.CreateTimer(request, CancellationToken.None);
  975. }
  976. public object Get(GetRecordingGroups request)
  977. {
  978. return ToOptimizedResult(new QueryResult<BaseItemDto>());
  979. }
  980. public object Get(GetRecordingGroup request)
  981. {
  982. throw new FileNotFoundException();
  983. }
  984. public object Get(GetGuideInfo request)
  985. {
  986. return ToOptimizedResult(_liveTvManager.GetGuideInfo());
  987. }
  988. public Task Post(ResetTuner request)
  989. {
  990. AssertUserCanManageLiveTv();
  991. return _liveTvManager.ResetTuner(request.Id, CancellationToken.None);
  992. }
  993. }
  994. }