LibraryService.cs 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Net;
  6. using System.Text.RegularExpressions;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using MediaBrowser.Api.Movies;
  10. using MediaBrowser.Common.Extensions;
  11. using MediaBrowser.Common.Progress;
  12. using MediaBrowser.Controller.Configuration;
  13. using MediaBrowser.Controller.Dto;
  14. using MediaBrowser.Controller.Entities;
  15. using MediaBrowser.Controller.Entities.Audio;
  16. using MediaBrowser.Controller.Entities.Movies;
  17. using MediaBrowser.Controller.Entities.TV;
  18. using MediaBrowser.Controller.Library;
  19. using MediaBrowser.Controller.Net;
  20. using MediaBrowser.Controller.Providers;
  21. using MediaBrowser.Model.Activity;
  22. using MediaBrowser.Model.Configuration;
  23. using MediaBrowser.Model.Dto;
  24. using MediaBrowser.Model.Entities;
  25. using MediaBrowser.Model.Globalization;
  26. using MediaBrowser.Model.Querying;
  27. using MediaBrowser.Model.Services;
  28. using Microsoft.Extensions.Logging;
  29. using Microsoft.Net.Http.Headers;
  30. namespace MediaBrowser.Api.Library
  31. {
  32. [Route("/Items/{Id}/File", "GET", Summary = "Gets the original file of an item")]
  33. [Authenticated]
  34. public class GetFile
  35. {
  36. /// <summary>
  37. /// Gets or sets the id.
  38. /// </summary>
  39. /// <value>The id.</value>
  40. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  41. public string Id { get; set; }
  42. }
  43. /// <summary>
  44. /// Class GetCriticReviews
  45. /// </summary>
  46. [Route("/Items/{Id}/CriticReviews", "GET", Summary = "Gets critic reviews for an item")]
  47. [Authenticated]
  48. public class GetCriticReviews : IReturn<QueryResult<BaseItemDto>>
  49. {
  50. /// <summary>
  51. /// Gets or sets the id.
  52. /// </summary>
  53. /// <value>The id.</value>
  54. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  55. public string Id { get; set; }
  56. /// <summary>
  57. /// Skips over a given number of items within the results. Use for paging.
  58. /// </summary>
  59. /// <value>The start index.</value>
  60. [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")]
  61. public int? StartIndex { 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. }
  69. /// <summary>
  70. /// Class GetThemeSongs
  71. /// </summary>
  72. [Route("/Items/{Id}/ThemeSongs", "GET", Summary = "Gets theme songs for an item")]
  73. [Authenticated]
  74. public class GetThemeSongs : IReturn<ThemeMediaResult>
  75. {
  76. /// <summary>
  77. /// Gets or sets the user id.
  78. /// </summary>
  79. /// <value>The user id.</value>
  80. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  81. public Guid UserId { get; set; }
  82. /// <summary>
  83. /// Gets or sets the id.
  84. /// </summary>
  85. /// <value>The id.</value>
  86. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  87. public string Id { get; set; }
  88. [ApiMember(Name = "InheritFromParent", Description = "Determines whether or not parent items should be searched for theme media.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  89. public bool InheritFromParent { get; set; }
  90. }
  91. /// <summary>
  92. /// Class GetThemeVideos
  93. /// </summary>
  94. [Route("/Items/{Id}/ThemeVideos", "GET", Summary = "Gets theme videos for an item")]
  95. [Authenticated]
  96. public class GetThemeVideos : IReturn<ThemeMediaResult>
  97. {
  98. /// <summary>
  99. /// Gets or sets the user id.
  100. /// </summary>
  101. /// <value>The user id.</value>
  102. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  103. public Guid UserId { get; set; }
  104. /// <summary>
  105. /// Gets or sets the id.
  106. /// </summary>
  107. /// <value>The id.</value>
  108. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  109. public string Id { get; set; }
  110. [ApiMember(Name = "InheritFromParent", Description = "Determines whether or not parent items should be searched for theme media.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  111. public bool InheritFromParent { get; set; }
  112. }
  113. /// <summary>
  114. /// Class GetThemeVideos
  115. /// </summary>
  116. [Route("/Items/{Id}/ThemeMedia", "GET", Summary = "Gets theme videos and songs for an item")]
  117. [Authenticated]
  118. public class GetThemeMedia : IReturn<AllThemeMediaResult>
  119. {
  120. /// <summary>
  121. /// Gets or sets the user id.
  122. /// </summary>
  123. /// <value>The user id.</value>
  124. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  125. public Guid UserId { get; set; }
  126. /// <summary>
  127. /// Gets or sets the id.
  128. /// </summary>
  129. /// <value>The id.</value>
  130. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  131. public string Id { get; set; }
  132. [ApiMember(Name = "InheritFromParent", Description = "Determines whether or not parent items should be searched for theme media.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  133. public bool InheritFromParent { get; set; }
  134. }
  135. [Route("/Library/Refresh", "POST", Summary = "Starts a library scan")]
  136. [Authenticated(Roles = "Admin")]
  137. public class RefreshLibrary : IReturnVoid
  138. {
  139. }
  140. [Route("/Items/{Id}", "DELETE", Summary = "Deletes an item from the library and file system")]
  141. [Authenticated]
  142. public class DeleteItem : IReturnVoid
  143. {
  144. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  145. public string Id { get; set; }
  146. }
  147. [Route("/Items", "DELETE", Summary = "Deletes an item from the library and file system")]
  148. [Authenticated]
  149. public class DeleteItems : IReturnVoid
  150. {
  151. [ApiMember(Name = "Ids", Description = "Ids", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "DELETE")]
  152. public string Ids { get; set; }
  153. }
  154. [Route("/Items/Counts", "GET")]
  155. [Authenticated]
  156. public class GetItemCounts : IReturn<ItemCounts>
  157. {
  158. [ApiMember(Name = "UserId", Description = "Optional. Get counts from a specific user's library.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  159. public Guid UserId { get; set; }
  160. [ApiMember(Name = "IsFavorite", Description = "Optional. Get counts of favorite items", IsRequired = false, DataType = "bool", ParameterType = "query", Verb = "GET")]
  161. public bool? IsFavorite { get; set; }
  162. }
  163. [Route("/Items/{Id}/Ancestors", "GET", Summary = "Gets all parents of an item")]
  164. [Authenticated]
  165. public class GetAncestors : IReturn<BaseItemDto[]>
  166. {
  167. /// <summary>
  168. /// Gets or sets the user id.
  169. /// </summary>
  170. /// <value>The user id.</value>
  171. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  172. public Guid UserId { get; set; }
  173. /// <summary>
  174. /// Gets or sets the id.
  175. /// </summary>
  176. /// <value>The id.</value>
  177. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  178. public string Id { get; set; }
  179. }
  180. /// <summary>
  181. /// Class GetPhyscialPaths
  182. /// </summary>
  183. [Route("/Library/PhysicalPaths", "GET", Summary = "Gets a list of physical paths from virtual folders")]
  184. [Authenticated(Roles = "Admin")]
  185. public class GetPhyscialPaths : IReturn<List<string>>
  186. {
  187. }
  188. [Route("/Library/MediaFolders", "GET", Summary = "Gets all user media folders.")]
  189. [Authenticated]
  190. public class GetMediaFolders : IReturn<QueryResult<BaseItemDto>>
  191. {
  192. [ApiMember(Name = "IsHidden", Description = "Optional. Filter by folders that are marked hidden, or not.", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  193. public bool? IsHidden { get; set; }
  194. }
  195. [Route("/Library/Series/Added", "POST", Summary = "Reports that new episodes of a series have been added by an external source")]
  196. [Route("/Library/Series/Updated", "POST", Summary = "Reports that new episodes of a series have been added by an external source")]
  197. [Authenticated]
  198. public class PostUpdatedSeries : IReturnVoid
  199. {
  200. [ApiMember(Name = "TvdbId", Description = "Tvdb Id", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "POST")]
  201. public string TvdbId { get; set; }
  202. }
  203. [Route("/Library/Movies/Added", "POST", Summary = "Reports that new movies have been added by an external source")]
  204. [Route("/Library/Movies/Updated", "POST", Summary = "Reports that new movies have been added by an external source")]
  205. [Authenticated]
  206. public class PostUpdatedMovies : IReturnVoid
  207. {
  208. [ApiMember(Name = "TmdbId", Description = "Tmdb Id", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "POST")]
  209. public string TmdbId { get; set; }
  210. [ApiMember(Name = "ImdbId", Description = "Imdb Id", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "POST")]
  211. public string ImdbId { get; set; }
  212. }
  213. public class MediaUpdateInfo
  214. {
  215. public string Path { get; set; }
  216. // Created, Modified, Deleted
  217. public string UpdateType { get; set; }
  218. }
  219. [Route("/Library/Media/Updated", "POST", Summary = "Reports that new movies have been added by an external source")]
  220. [Authenticated]
  221. public class PostUpdatedMedia : IReturnVoid
  222. {
  223. [ApiMember(Name = "Updates", Description = "A list of updated media paths", IsRequired = false, DataType = "string", ParameterType = "body", Verb = "POST")]
  224. public List<MediaUpdateInfo> Updates { get; set; }
  225. }
  226. [Route("/Items/{Id}/Download", "GET", Summary = "Downloads item media")]
  227. [Authenticated(Roles = "download")]
  228. public class GetDownload
  229. {
  230. /// <summary>
  231. /// Gets or sets the id.
  232. /// </summary>
  233. /// <value>The id.</value>
  234. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  235. public string Id { get; set; }
  236. }
  237. [Route("/Artists/{Id}/Similar", "GET", Summary = "Finds albums similar to a given album.")]
  238. [Route("/Items/{Id}/Similar", "GET", Summary = "Gets similar items")]
  239. [Route("/Albums/{Id}/Similar", "GET", Summary = "Finds albums similar to a given album.")]
  240. [Route("/Shows/{Id}/Similar", "GET", Summary = "Finds tv shows similar to a given one.")]
  241. [Route("/Movies/{Id}/Similar", "GET", Summary = "Finds movies and trailers similar to a given movie.")]
  242. [Route("/Trailers/{Id}/Similar", "GET", Summary = "Finds movies and trailers similar to a given trailer.")]
  243. [Authenticated]
  244. public class GetSimilarItems : BaseGetSimilarItemsFromItem
  245. {
  246. }
  247. [Route("/Libraries/AvailableOptions", "GET")]
  248. [Authenticated(AllowBeforeStartupWizard = true)]
  249. public class GetLibraryOptionsInfo : IReturn<LibraryOptionsResult>
  250. {
  251. public string LibraryContentType { get; set; }
  252. public bool IsNewLibrary { get; set; }
  253. }
  254. public class LibraryOptionInfo
  255. {
  256. public string Name { get; set; }
  257. public bool DefaultEnabled { get; set; }
  258. }
  259. public class LibraryOptionsResult
  260. {
  261. public LibraryOptionInfo[] MetadataSavers { get; set; }
  262. public LibraryOptionInfo[] MetadataReaders { get; set; }
  263. public LibraryOptionInfo[] SubtitleFetchers { get; set; }
  264. public LibraryTypeOptions[] TypeOptions { get; set; }
  265. }
  266. public class LibraryTypeOptions
  267. {
  268. public string Type { get; set; }
  269. public LibraryOptionInfo[] MetadataFetchers { get; set; }
  270. public LibraryOptionInfo[] ImageFetchers { get; set; }
  271. public ImageType[] SupportedImageTypes { get; set; }
  272. public ImageOption[] DefaultImageOptions { get; set; }
  273. }
  274. /// <summary>
  275. /// Class LibraryService
  276. /// </summary>
  277. public class LibraryService : BaseApiService
  278. {
  279. private readonly IProviderManager _providerManager;
  280. private readonly ILibraryManager _libraryManager;
  281. private readonly IUserManager _userManager;
  282. private readonly IDtoService _dtoService;
  283. private readonly IAuthorizationContext _authContext;
  284. private readonly IActivityManager _activityManager;
  285. private readonly ILocalizationManager _localization;
  286. private readonly ILibraryMonitor _libraryMonitor;
  287. /// <summary>
  288. /// Initializes a new instance of the <see cref="LibraryService" /> class.
  289. /// </summary>
  290. public LibraryService(
  291. ILogger<LibraryService> logger,
  292. IServerConfigurationManager serverConfigurationManager,
  293. IHttpResultFactory httpResultFactory,
  294. IProviderManager providerManager,
  295. ILibraryManager libraryManager,
  296. IUserManager userManager,
  297. IDtoService dtoService,
  298. IAuthorizationContext authContext,
  299. IActivityManager activityManager,
  300. ILocalizationManager localization,
  301. ILibraryMonitor libraryMonitor)
  302. : base(logger, serverConfigurationManager, httpResultFactory)
  303. {
  304. _providerManager = providerManager;
  305. _libraryManager = libraryManager;
  306. _userManager = userManager;
  307. _dtoService = dtoService;
  308. _authContext = authContext;
  309. _activityManager = activityManager;
  310. _localization = localization;
  311. _libraryMonitor = libraryMonitor;
  312. }
  313. private string[] GetRepresentativeItemTypes(string contentType)
  314. {
  315. return contentType switch
  316. {
  317. CollectionType.BoxSets => new[] {"BoxSet"},
  318. CollectionType.Playlists => new[] {"Playlist"},
  319. CollectionType.Movies => new[] {"Movie"},
  320. CollectionType.TvShows => new[] {"Series", "Season", "Episode"},
  321. CollectionType.Books => new[] {"Book"},
  322. CollectionType.Music => new[] {"MusicAlbum", "MusicArtist", "Audio", "MusicVideo"},
  323. CollectionType.HomeVideos => new[] {"Video", "Photo"},
  324. CollectionType.Photos => new[] {"Video", "Photo"},
  325. CollectionType.MusicVideos => new[] {"MusicVideo"},
  326. _ => new[] {"Series", "Season", "Episode", "Movie"}
  327. };
  328. }
  329. private bool IsSaverEnabledByDefault(string name, string[] itemTypes, bool isNewLibrary)
  330. {
  331. if (isNewLibrary)
  332. {
  333. return false;
  334. }
  335. var metadataOptions = ServerConfigurationManager.Configuration.MetadataOptions
  336. .Where(i => itemTypes.Contains(i.ItemType ?? string.Empty, StringComparer.OrdinalIgnoreCase))
  337. .ToArray();
  338. if (metadataOptions.Length == 0)
  339. {
  340. return true;
  341. }
  342. return metadataOptions.Any(i => !i.DisabledMetadataSavers.Contains(name, StringComparer.OrdinalIgnoreCase));
  343. }
  344. private bool IsMetadataFetcherEnabledByDefault(string name, string type, bool isNewLibrary)
  345. {
  346. if (isNewLibrary)
  347. {
  348. if (string.Equals(name, "TheMovieDb", StringComparison.OrdinalIgnoreCase))
  349. {
  350. return !(string.Equals(type, "Season", StringComparison.OrdinalIgnoreCase)
  351. || string.Equals(type, "Episode", StringComparison.OrdinalIgnoreCase)
  352. || string.Equals(type, "MusicVideo", StringComparison.OrdinalIgnoreCase));
  353. }
  354. return string.Equals(name, "TheTVDB", StringComparison.OrdinalIgnoreCase)
  355. || string.Equals(name, "TheAudioDB", StringComparison.OrdinalIgnoreCase)
  356. || string.Equals(name, "MusicBrainz", StringComparison.OrdinalIgnoreCase);
  357. }
  358. var metadataOptions = ServerConfigurationManager.Configuration.MetadataOptions
  359. .Where(i => string.Equals(i.ItemType, type, StringComparison.OrdinalIgnoreCase))
  360. .ToArray();
  361. return metadataOptions.Length == 0
  362. || metadataOptions.Any(i => !i.DisabledMetadataFetchers.Contains(name, StringComparer.OrdinalIgnoreCase));
  363. }
  364. private bool IsImageFetcherEnabledByDefault(string name, string type, bool isNewLibrary)
  365. {
  366. if (isNewLibrary)
  367. {
  368. if (string.Equals(name, "TheMovieDb", StringComparison.OrdinalIgnoreCase))
  369. {
  370. return !string.Equals(type, "Series", StringComparison.OrdinalIgnoreCase)
  371. && !string.Equals(type, "Season", StringComparison.OrdinalIgnoreCase)
  372. && !string.Equals(type, "Episode", StringComparison.OrdinalIgnoreCase)
  373. && !string.Equals(type, "MusicVideo", StringComparison.OrdinalIgnoreCase);
  374. }
  375. return string.Equals(name, "TheTVDB", StringComparison.OrdinalIgnoreCase)
  376. || string.Equals(name, "Screen Grabber", StringComparison.OrdinalIgnoreCase)
  377. || string.Equals(name, "TheAudioDB", StringComparison.OrdinalIgnoreCase)
  378. || string.Equals(name, "Emby Designs", StringComparison.OrdinalIgnoreCase)
  379. || string.Equals(name, "Image Extractor", StringComparison.OrdinalIgnoreCase);
  380. }
  381. var metadataOptions = ServerConfigurationManager.Configuration.MetadataOptions
  382. .Where(i => string.Equals(i.ItemType, type, StringComparison.OrdinalIgnoreCase))
  383. .ToArray();
  384. if (metadataOptions.Length == 0)
  385. {
  386. return true;
  387. }
  388. return metadataOptions.Any(i => !i.DisabledImageFetchers.Contains(name, StringComparer.OrdinalIgnoreCase));
  389. }
  390. public object Get(GetLibraryOptionsInfo request)
  391. {
  392. var result = new LibraryOptionsResult();
  393. var types = GetRepresentativeItemTypes(request.LibraryContentType);
  394. var isNewLibrary = request.IsNewLibrary;
  395. var typesList = types.ToList();
  396. var plugins = _providerManager.GetAllMetadataPlugins()
  397. .Where(i => types.Contains(i.ItemType, StringComparer.OrdinalIgnoreCase))
  398. .OrderBy(i => typesList.IndexOf(i.ItemType))
  399. .ToList();
  400. result.MetadataSavers = plugins
  401. .SelectMany(i => i.Plugins.Where(p => p.Type == MetadataPluginType.MetadataSaver))
  402. .Select(i => new LibraryOptionInfo
  403. {
  404. Name = i.Name,
  405. DefaultEnabled = IsSaverEnabledByDefault(i.Name, types, isNewLibrary)
  406. })
  407. .GroupBy(i => i.Name, StringComparer.OrdinalIgnoreCase)
  408. .Select(x => x.First())
  409. .ToArray();
  410. result.MetadataReaders = plugins
  411. .SelectMany(i => i.Plugins.Where(p => p.Type == MetadataPluginType.LocalMetadataProvider))
  412. .Select(i => new LibraryOptionInfo
  413. {
  414. Name = i.Name,
  415. DefaultEnabled = true
  416. })
  417. .GroupBy(i => i.Name, StringComparer.OrdinalIgnoreCase)
  418. .Select(x => x.First())
  419. .ToArray();
  420. result.SubtitleFetchers = plugins
  421. .SelectMany(i => i.Plugins.Where(p => p.Type == MetadataPluginType.SubtitleFetcher))
  422. .Select(i => new LibraryOptionInfo
  423. {
  424. Name = i.Name,
  425. DefaultEnabled = true
  426. })
  427. .GroupBy(i => i.Name, StringComparer.OrdinalIgnoreCase)
  428. .Select(x => x.First())
  429. .ToArray();
  430. var typeOptions = new List<LibraryTypeOptions>();
  431. foreach (var type in types)
  432. {
  433. TypeOptions.DefaultImageOptions.TryGetValue(type, out var defaultImageOptions);
  434. typeOptions.Add(new LibraryTypeOptions
  435. {
  436. Type = type,
  437. MetadataFetchers = plugins
  438. .Where(i => string.Equals(i.ItemType, type, StringComparison.OrdinalIgnoreCase))
  439. .SelectMany(i => i.Plugins.Where(p => p.Type == MetadataPluginType.MetadataFetcher))
  440. .Select(i => new LibraryOptionInfo
  441. {
  442. Name = i.Name,
  443. DefaultEnabled = IsMetadataFetcherEnabledByDefault(i.Name, type, isNewLibrary)
  444. })
  445. .GroupBy(i => i.Name, StringComparer.OrdinalIgnoreCase)
  446. .Select(x => x.First())
  447. .ToArray(),
  448. ImageFetchers = plugins
  449. .Where(i => string.Equals(i.ItemType, type, StringComparison.OrdinalIgnoreCase))
  450. .SelectMany(i => i.Plugins.Where(p => p.Type == MetadataPluginType.ImageFetcher))
  451. .Select(i => new LibraryOptionInfo
  452. {
  453. Name = i.Name,
  454. DefaultEnabled = IsImageFetcherEnabledByDefault(i.Name, type, isNewLibrary)
  455. })
  456. .GroupBy(i => i.Name, StringComparer.OrdinalIgnoreCase)
  457. .Select(x => x.First())
  458. .ToArray(),
  459. SupportedImageTypes = plugins
  460. .Where(i => string.Equals(i.ItemType, type, StringComparison.OrdinalIgnoreCase))
  461. .SelectMany(i => i.SupportedImageTypes ?? Array.Empty<ImageType>())
  462. .Distinct()
  463. .ToArray(),
  464. DefaultImageOptions = defaultImageOptions ?? Array.Empty<ImageOption>()
  465. });
  466. }
  467. result.TypeOptions = typeOptions.ToArray();
  468. return result;
  469. }
  470. public object Get(GetSimilarItems request)
  471. {
  472. var item = string.IsNullOrEmpty(request.Id) ?
  473. (!request.UserId.Equals(Guid.Empty) ? _libraryManager.GetUserRootFolder() :
  474. _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id);
  475. var program = item as IHasProgramAttributes;
  476. if (item is Movie || (program != null && program.IsMovie) || item is Trailer)
  477. {
  478. return new MoviesService(
  479. Logger,
  480. ServerConfigurationManager,
  481. ResultFactory,
  482. _userManager,
  483. _libraryManager,
  484. _dtoService,
  485. _authContext)
  486. {
  487. Request = Request,
  488. }.GetSimilarItemsResult(request);
  489. }
  490. if (program != null && program.IsSeries)
  491. {
  492. return GetSimilarItemsResult(request, new[] { typeof(Series).Name });
  493. }
  494. if (item is Episode || (item is IItemByName && !(item is MusicArtist)))
  495. {
  496. return new QueryResult<BaseItemDto>();
  497. }
  498. return GetSimilarItemsResult(request, new[] { item.GetType().Name });
  499. }
  500. private QueryResult<BaseItemDto> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request, string[] includeItemTypes)
  501. {
  502. var user = !request.UserId.Equals(Guid.Empty) ? _userManager.GetUserById(request.UserId) : null;
  503. var item = string.IsNullOrEmpty(request.Id) ?
  504. (!request.UserId.Equals(Guid.Empty) ? _libraryManager.GetUserRootFolder() :
  505. _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id);
  506. var dtoOptions = GetDtoOptions(_authContext, request);
  507. var query = new InternalItemsQuery(user)
  508. {
  509. Limit = request.Limit,
  510. IncludeItemTypes = includeItemTypes,
  511. SimilarTo = item,
  512. DtoOptions = dtoOptions,
  513. EnableTotalRecordCount = false
  514. };
  515. // ExcludeArtistIds
  516. if (!string.IsNullOrEmpty(request.ExcludeArtistIds))
  517. {
  518. query.ExcludeArtistIds = GetGuids(request.ExcludeArtistIds);
  519. }
  520. List<BaseItem> itemsResult;
  521. if (item is MusicArtist)
  522. {
  523. query.IncludeItemTypes = Array.Empty<string>();
  524. itemsResult = _libraryManager.GetArtists(query).Items.Select(i => i.Item1).ToList();
  525. }
  526. else
  527. {
  528. itemsResult = _libraryManager.GetItemList(query);
  529. }
  530. var returnList = _dtoService.GetBaseItemDtos(itemsResult, dtoOptions, user);
  531. var result = new QueryResult<BaseItemDto>
  532. {
  533. Items = returnList,
  534. TotalRecordCount = itemsResult.Count
  535. };
  536. return result;
  537. }
  538. public object Get(GetMediaFolders request)
  539. {
  540. var items = _libraryManager.GetUserRootFolder().Children.Concat(_libraryManager.RootFolder.VirtualChildren).OrderBy(i => i.SortName).ToList();
  541. if (request.IsHidden.HasValue)
  542. {
  543. var val = request.IsHidden.Value;
  544. items = items.Where(i => i.IsHidden == val).ToList();
  545. }
  546. var dtoOptions = GetDtoOptions(_authContext, request);
  547. var result = new QueryResult<BaseItemDto>
  548. {
  549. TotalRecordCount = items.Count,
  550. Items = items.Select(i => _dtoService.GetBaseItemDto(i, dtoOptions)).ToArray()
  551. };
  552. return result;
  553. }
  554. public void Post(PostUpdatedSeries request)
  555. {
  556. var series = _libraryManager.GetItemList(new InternalItemsQuery
  557. {
  558. IncludeItemTypes = new[] { typeof(Series).Name },
  559. DtoOptions = new DtoOptions(false)
  560. {
  561. EnableImages = false
  562. }
  563. }).Where(i => string.Equals(request.TvdbId, i.GetProviderId(MetadataProviders.Tvdb), StringComparison.OrdinalIgnoreCase)).ToArray();
  564. foreach (var item in series)
  565. {
  566. _libraryMonitor.ReportFileSystemChanged(item.Path);
  567. }
  568. }
  569. public void Post(PostUpdatedMedia request)
  570. {
  571. if (request.Updates != null)
  572. {
  573. foreach (var item in request.Updates)
  574. {
  575. _libraryMonitor.ReportFileSystemChanged(item.Path);
  576. }
  577. }
  578. }
  579. public void Post(PostUpdatedMovies request)
  580. {
  581. var movies = _libraryManager.GetItemList(new InternalItemsQuery
  582. {
  583. IncludeItemTypes = new[] { typeof(Movie).Name },
  584. DtoOptions = new DtoOptions(false)
  585. {
  586. EnableImages = false
  587. }
  588. });
  589. if (!string.IsNullOrWhiteSpace(request.ImdbId))
  590. {
  591. movies = movies.Where(i => string.Equals(request.ImdbId, i.GetProviderId(MetadataProviders.Imdb), StringComparison.OrdinalIgnoreCase)).ToList();
  592. }
  593. else if (!string.IsNullOrWhiteSpace(request.TmdbId))
  594. {
  595. movies = movies.Where(i => string.Equals(request.TmdbId, i.GetProviderId(MetadataProviders.Tmdb), StringComparison.OrdinalIgnoreCase)).ToList();
  596. }
  597. else
  598. {
  599. movies = new List<BaseItem>();
  600. }
  601. foreach (var item in movies)
  602. {
  603. _libraryMonitor.ReportFileSystemChanged(item.Path);
  604. }
  605. }
  606. public Task<object> Get(GetDownload request)
  607. {
  608. var item = _libraryManager.GetItemById(request.Id);
  609. var auth = _authContext.GetAuthorizationInfo(Request);
  610. var user = auth.User;
  611. if (user != null)
  612. {
  613. if (!item.CanDownload(user))
  614. {
  615. throw new ArgumentException("Item does not support downloading");
  616. }
  617. }
  618. else
  619. {
  620. if (!item.CanDownload())
  621. {
  622. throw new ArgumentException("Item does not support downloading");
  623. }
  624. }
  625. var headers = new Dictionary<string, string>();
  626. if (user != null)
  627. {
  628. LogDownload(item, user, auth);
  629. }
  630. var path = item.Path;
  631. // Quotes are valid in linux. They'll possibly cause issues here
  632. var filename = (Path.GetFileName(path) ?? string.Empty).Replace("\"", string.Empty);
  633. if (!string.IsNullOrWhiteSpace(filename))
  634. {
  635. // Kestrel doesn't support non-ASCII characters in headers
  636. if (Regex.IsMatch(filename, @"[^\p{IsBasicLatin}]"))
  637. {
  638. // Manually encoding non-ASCII characters, following https://tools.ietf.org/html/rfc5987#section-3.2.2
  639. headers[HeaderNames.ContentDisposition] = "attachment; filename*=UTF-8''" + WebUtility.UrlEncode(filename);
  640. }
  641. else
  642. {
  643. headers[HeaderNames.ContentDisposition] = "attachment; filename=\"" + filename + "\"";
  644. }
  645. }
  646. return ResultFactory.GetStaticFileResult(Request, new StaticFileResultOptions
  647. {
  648. Path = path,
  649. ResponseHeaders = headers
  650. });
  651. }
  652. private void LogDownload(BaseItem item, User user, AuthorizationInfo auth)
  653. {
  654. try
  655. {
  656. _activityManager.Create(new Jellyfin.Data.Entities.ActivityLog(
  657. string.Format(_localization.GetLocalizedString("UserDownloadingItemWithValues"), user.Name, item.Name),
  658. "UserDownloadingContent",
  659. auth.UserId,
  660. DateTime.UtcNow,
  661. LogLevel.Trace)
  662. {
  663. ShortOverview = string.Format(_localization.GetLocalizedString("AppDeviceValues"), auth.Client, auth.Device),
  664. });
  665. }
  666. catch
  667. {
  668. // Logged at lower levels
  669. }
  670. }
  671. public Task<object> Get(GetFile request)
  672. {
  673. var item = _libraryManager.GetItemById(request.Id);
  674. return ResultFactory.GetStaticFileResult(Request, item.Path);
  675. }
  676. /// <summary>
  677. /// Gets the specified request.
  678. /// </summary>
  679. /// <param name="request">The request.</param>
  680. /// <returns>System.Object.</returns>
  681. public object Get(GetPhyscialPaths request)
  682. {
  683. var result = _libraryManager.RootFolder.Children
  684. .SelectMany(c => c.PhysicalLocations)
  685. .ToList();
  686. return ToOptimizedResult(result);
  687. }
  688. /// <summary>
  689. /// Gets the specified request.
  690. /// </summary>
  691. /// <param name="request">The request.</param>
  692. /// <returns>System.Object.</returns>
  693. public object Get(GetAncestors request)
  694. {
  695. var result = GetAncestors(request);
  696. return ToOptimizedResult(result);
  697. }
  698. /// <summary>
  699. /// Gets the ancestors.
  700. /// </summary>
  701. /// <param name="request">The request.</param>
  702. /// <returns>Task{BaseItemDto[]}.</returns>
  703. public List<BaseItemDto> GetAncestors(GetAncestors request)
  704. {
  705. var item = _libraryManager.GetItemById(request.Id);
  706. var baseItemDtos = new List<BaseItemDto>();
  707. var user = !request.UserId.Equals(Guid.Empty) ? _userManager.GetUserById(request.UserId) : null;
  708. var dtoOptions = GetDtoOptions(_authContext, request);
  709. BaseItem parent = item.GetParent();
  710. while (parent != null)
  711. {
  712. if (user != null)
  713. {
  714. parent = TranslateParentItem(parent, user);
  715. }
  716. baseItemDtos.Add(_dtoService.GetBaseItemDto(parent, dtoOptions, user));
  717. parent = parent.GetParent();
  718. }
  719. return baseItemDtos;
  720. }
  721. private BaseItem TranslateParentItem(BaseItem item, User user)
  722. {
  723. return item.GetParent() is AggregateFolder
  724. ? _libraryManager.GetUserRootFolder().GetChildren(user, true)
  725. .FirstOrDefault(i => i.PhysicalLocations.Contains(item.Path))
  726. : item;
  727. }
  728. /// <summary>
  729. /// Gets the specified request.
  730. /// </summary>
  731. /// <param name="request">The request.</param>
  732. /// <returns>System.Object.</returns>
  733. public object Get(GetCriticReviews request)
  734. {
  735. return new QueryResult<BaseItemDto>();
  736. }
  737. /// <summary>
  738. /// Gets the specified request.
  739. /// </summary>
  740. /// <param name="request">The request.</param>
  741. /// <returns>System.Object.</returns>
  742. public object Get(GetItemCounts request)
  743. {
  744. var user = request.UserId.Equals(Guid.Empty) ? null : _userManager.GetUserById(request.UserId);
  745. var counts = new ItemCounts
  746. {
  747. AlbumCount = GetCount(typeof(MusicAlbum), user, request),
  748. EpisodeCount = GetCount(typeof(Episode), user, request),
  749. MovieCount = GetCount(typeof(Movie), user, request),
  750. SeriesCount = GetCount(typeof(Series), user, request),
  751. SongCount = GetCount(typeof(Audio), user, request),
  752. MusicVideoCount = GetCount(typeof(MusicVideo), user, request),
  753. BoxSetCount = GetCount(typeof(BoxSet), user, request),
  754. BookCount = GetCount(typeof(Book), user, request)
  755. };
  756. return ToOptimizedResult(counts);
  757. }
  758. private int GetCount(Type type, User user, GetItemCounts request)
  759. {
  760. var query = new InternalItemsQuery(user)
  761. {
  762. IncludeItemTypes = new[] { type.Name },
  763. Limit = 0,
  764. Recursive = true,
  765. IsVirtualItem = false,
  766. IsFavorite = request.IsFavorite,
  767. DtoOptions = new DtoOptions(false)
  768. {
  769. EnableImages = false
  770. }
  771. };
  772. return _libraryManager.GetItemsResult(query).TotalRecordCount;
  773. }
  774. /// <summary>
  775. /// Posts the specified request.
  776. /// </summary>
  777. /// <param name="request">The request.</param>
  778. public async Task Post(RefreshLibrary request)
  779. {
  780. try
  781. {
  782. await _libraryManager.ValidateMediaLibrary(new SimpleProgress<double>(), CancellationToken.None).ConfigureAwait(false);
  783. }
  784. catch (Exception ex)
  785. {
  786. Logger.LogError(ex, "Error refreshing library");
  787. }
  788. }
  789. /// <summary>
  790. /// Deletes the specified request.
  791. /// </summary>
  792. /// <param name="request">The request.</param>
  793. public void Delete(DeleteItems request)
  794. {
  795. var ids = string.IsNullOrWhiteSpace(request.Ids)
  796. ? Array.Empty<string>()
  797. : request.Ids.Split(',');
  798. foreach (var i in ids)
  799. {
  800. var item = _libraryManager.GetItemById(i);
  801. var auth = _authContext.GetAuthorizationInfo(Request);
  802. var user = auth.User;
  803. if (!item.CanDelete(user))
  804. {
  805. if (ids.Length > 1)
  806. {
  807. throw new SecurityException("Unauthorized access");
  808. }
  809. continue;
  810. }
  811. _libraryManager.DeleteItem(item, new DeleteOptions
  812. {
  813. DeleteFileLocation = true
  814. }, true);
  815. }
  816. }
  817. /// <summary>
  818. /// Deletes the specified request.
  819. /// </summary>
  820. /// <param name="request">The request.</param>
  821. public void Delete(DeleteItem request)
  822. {
  823. Delete(new DeleteItems
  824. {
  825. Ids = request.Id
  826. });
  827. }
  828. public object Get(GetThemeMedia request)
  829. {
  830. var themeSongs = GetThemeSongs(new GetThemeSongs
  831. {
  832. InheritFromParent = request.InheritFromParent,
  833. Id = request.Id,
  834. UserId = request.UserId
  835. });
  836. var themeVideos = GetThemeVideos(new GetThemeVideos
  837. {
  838. InheritFromParent = request.InheritFromParent,
  839. Id = request.Id,
  840. UserId = request.UserId
  841. });
  842. return ToOptimizedResult(new AllThemeMediaResult
  843. {
  844. ThemeSongsResult = themeSongs,
  845. ThemeVideosResult = themeVideos,
  846. SoundtrackSongsResult = new ThemeMediaResult()
  847. });
  848. }
  849. /// <summary>
  850. /// Gets the specified request.
  851. /// </summary>
  852. /// <param name="request">The request.</param>
  853. /// <returns>System.Object.</returns>
  854. public object Get(GetThemeSongs request)
  855. {
  856. var result = GetThemeSongs(request);
  857. return ToOptimizedResult(result);
  858. }
  859. private ThemeMediaResult GetThemeSongs(GetThemeSongs request)
  860. {
  861. var user = !request.UserId.Equals(Guid.Empty) ? _userManager.GetUserById(request.UserId) : null;
  862. var item = string.IsNullOrEmpty(request.Id)
  863. ? (!request.UserId.Equals(Guid.Empty)
  864. ? _libraryManager.GetUserRootFolder()
  865. : _libraryManager.RootFolder)
  866. : _libraryManager.GetItemById(request.Id);
  867. if (item == null)
  868. {
  869. throw new ResourceNotFoundException("Item not found.");
  870. }
  871. IEnumerable<BaseItem> themeItems;
  872. while (true)
  873. {
  874. themeItems = item.GetThemeSongs();
  875. if (themeItems.Any() || !request.InheritFromParent)
  876. {
  877. break;
  878. }
  879. var parent = item.GetParent();
  880. if (parent == null)
  881. {
  882. break;
  883. }
  884. item = parent;
  885. }
  886. var dtoOptions = GetDtoOptions(_authContext, request);
  887. var items = themeItems
  888. .Select(i => _dtoService.GetBaseItemDto(i, dtoOptions, user, item))
  889. .ToArray();
  890. return new ThemeMediaResult
  891. {
  892. Items = items,
  893. TotalRecordCount = items.Length,
  894. OwnerId = item.Id
  895. };
  896. }
  897. /// <summary>
  898. /// Gets the specified request.
  899. /// </summary>
  900. /// <param name="request">The request.</param>
  901. /// <returns>System.Object.</returns>
  902. public object Get(GetThemeVideos request)
  903. {
  904. return ToOptimizedResult(GetThemeVideos(request));
  905. }
  906. public ThemeMediaResult GetThemeVideos(GetThemeVideos request)
  907. {
  908. var user = !request.UserId.Equals(Guid.Empty) ? _userManager.GetUserById(request.UserId) : null;
  909. var item = string.IsNullOrEmpty(request.Id)
  910. ? (!request.UserId.Equals(Guid.Empty)
  911. ? _libraryManager.GetUserRootFolder()
  912. : _libraryManager.RootFolder)
  913. : _libraryManager.GetItemById(request.Id);
  914. if (item == null)
  915. {
  916. throw new ResourceNotFoundException("Item not found.");
  917. }
  918. IEnumerable<BaseItem> themeItems;
  919. while (true)
  920. {
  921. themeItems = item.GetThemeVideos();
  922. if (themeItems.Any() || !request.InheritFromParent)
  923. {
  924. break;
  925. }
  926. var parent = item.GetParent();
  927. if (parent == null)
  928. {
  929. break;
  930. }
  931. item = parent;
  932. }
  933. var dtoOptions = GetDtoOptions(_authContext, request);
  934. var items = themeItems
  935. .Select(i => _dtoService.GetBaseItemDto(i, dtoOptions, user, item))
  936. .ToArray();
  937. return new ThemeMediaResult
  938. {
  939. Items = items,
  940. TotalRecordCount = items.Length,
  941. OwnerId = item.Id
  942. };
  943. }
  944. }
  945. }