ItemsService.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using Jellyfin.Data.Enums;
  6. using MediaBrowser.Controller.Configuration;
  7. using MediaBrowser.Controller.Dto;
  8. using MediaBrowser.Controller.Entities;
  9. using MediaBrowser.Controller.Entities.Audio;
  10. using MediaBrowser.Controller.Library;
  11. using MediaBrowser.Controller.Net;
  12. using MediaBrowser.Model.Dto;
  13. using MediaBrowser.Model.Entities;
  14. using MediaBrowser.Model.Globalization;
  15. using MediaBrowser.Model.Querying;
  16. using MediaBrowser.Model.Services;
  17. using Microsoft.Extensions.Logging;
  18. namespace MediaBrowser.Api.UserLibrary
  19. {
  20. /// <summary>
  21. /// Class GetItems
  22. /// </summary>
  23. [Route("/Items", "GET", Summary = "Gets items based on a query.")]
  24. [Route("/Users/{UserId}/Items", "GET", Summary = "Gets items based on a query.")]
  25. public class GetItems : BaseItemsRequest, IReturn<QueryResult<BaseItemDto>>
  26. {
  27. }
  28. [Route("/Users/{UserId}/Items/Resume", "GET", Summary = "Gets items based on a query.")]
  29. public class GetResumeItems : BaseItemsRequest, IReturn<QueryResult<BaseItemDto>>
  30. {
  31. }
  32. /// <summary>
  33. /// Class ItemsService
  34. /// </summary>
  35. [Authenticated]
  36. public class ItemsService : BaseApiService
  37. {
  38. /// <summary>
  39. /// The _user manager
  40. /// </summary>
  41. private readonly IUserManager _userManager;
  42. /// <summary>
  43. /// The _library manager
  44. /// </summary>
  45. private readonly ILibraryManager _libraryManager;
  46. private readonly ILocalizationManager _localization;
  47. private readonly IDtoService _dtoService;
  48. private readonly IAuthorizationContext _authContext;
  49. /// <summary>
  50. /// Initializes a new instance of the <see cref="ItemsService" /> class.
  51. /// </summary>
  52. /// <param name="userManager">The user manager.</param>
  53. /// <param name="libraryManager">The library manager.</param>
  54. /// <param name="localization">The localization.</param>
  55. /// <param name="dtoService">The dto service.</param>
  56. public ItemsService(
  57. ILogger<ItemsService> logger,
  58. IServerConfigurationManager serverConfigurationManager,
  59. IHttpResultFactory httpResultFactory,
  60. IUserManager userManager,
  61. ILibraryManager libraryManager,
  62. ILocalizationManager localization,
  63. IDtoService dtoService,
  64. IAuthorizationContext authContext)
  65. : base(logger, serverConfigurationManager, httpResultFactory)
  66. {
  67. _userManager = userManager;
  68. _libraryManager = libraryManager;
  69. _localization = localization;
  70. _dtoService = dtoService;
  71. _authContext = authContext;
  72. }
  73. public object Get(GetResumeItems request)
  74. {
  75. var user = _userManager.GetUserById(request.UserId);
  76. var parentIdGuid = string.IsNullOrWhiteSpace(request.ParentId) ? Guid.Empty : new Guid(request.ParentId);
  77. var options = GetDtoOptions(_authContext, request);
  78. var ancestorIds = Array.Empty<Guid>();
  79. var excludeFolderIds = user.GetPreference(PreferenceKind.LatestItemExcludes);
  80. if (parentIdGuid.Equals(Guid.Empty) && excludeFolderIds.Length > 0)
  81. {
  82. ancestorIds = _libraryManager.GetUserRootFolder().GetChildren(user, true)
  83. .Where(i => i is Folder)
  84. .Where(i => !excludeFolderIds.Contains(i.Id.ToString("N", CultureInfo.InvariantCulture)))
  85. .Select(i => i.Id)
  86. .ToArray();
  87. }
  88. var itemsResult = _libraryManager.GetItemsResult(new InternalItemsQuery(user)
  89. {
  90. OrderBy = new[] { (ItemSortBy.DatePlayed, SortOrder.Descending) },
  91. IsResumable = true,
  92. StartIndex = request.StartIndex,
  93. Limit = request.Limit,
  94. ParentId = parentIdGuid,
  95. Recursive = true,
  96. DtoOptions = options,
  97. MediaTypes = request.GetMediaTypes(),
  98. IsVirtualItem = false,
  99. CollapseBoxSetItems = false,
  100. EnableTotalRecordCount = request.EnableTotalRecordCount,
  101. AncestorIds = ancestorIds,
  102. IncludeItemTypes = request.GetIncludeItemTypes(),
  103. ExcludeItemTypes = request.GetExcludeItemTypes(),
  104. SearchTerm = request.SearchTerm
  105. });
  106. var returnItems = _dtoService.GetBaseItemDtos(itemsResult.Items, options, user);
  107. var result = new QueryResult<BaseItemDto>
  108. {
  109. StartIndex = request.StartIndex.GetValueOrDefault(),
  110. TotalRecordCount = itemsResult.TotalRecordCount,
  111. Items = returnItems
  112. };
  113. return ToOptimizedResult(result);
  114. }
  115. /// <summary>
  116. /// Gets the specified request.
  117. /// </summary>
  118. /// <param name="request">The request.</param>
  119. /// <returns>System.Object.</returns>
  120. public object Get(GetItems request)
  121. {
  122. if (request == null)
  123. {
  124. throw new ArgumentNullException(nameof(request));
  125. }
  126. var result = GetItems(request);
  127. return ToOptimizedResult(result);
  128. }
  129. /// <summary>
  130. /// Gets the items.
  131. /// </summary>
  132. /// <param name="request">The request.</param>
  133. private QueryResult<BaseItemDto> GetItems(GetItems request)
  134. {
  135. var user = request.UserId == Guid.Empty ? null : _userManager.GetUserById(request.UserId);
  136. var dtoOptions = GetDtoOptions(_authContext, request);
  137. var result = GetQueryResult(request, dtoOptions, user);
  138. if (result == null)
  139. {
  140. throw new InvalidOperationException("GetItemsToSerialize returned null");
  141. }
  142. if (result.Items == null)
  143. {
  144. throw new InvalidOperationException("GetItemsToSerialize result.Items returned null");
  145. }
  146. var dtoList = _dtoService.GetBaseItemDtos(result.Items, dtoOptions, user);
  147. return new QueryResult<BaseItemDto>
  148. {
  149. StartIndex = request.StartIndex.GetValueOrDefault(),
  150. TotalRecordCount = result.TotalRecordCount,
  151. Items = dtoList
  152. };
  153. }
  154. /// <summary>
  155. /// Gets the items to serialize.
  156. /// </summary>
  157. private QueryResult<BaseItem> GetQueryResult(GetItems request, DtoOptions dtoOptions, Jellyfin.Data.Entities.User user)
  158. {
  159. if (string.Equals(request.IncludeItemTypes, "Playlist", StringComparison.OrdinalIgnoreCase)
  160. || string.Equals(request.IncludeItemTypes, "BoxSet", StringComparison.OrdinalIgnoreCase))
  161. {
  162. request.ParentId = null;
  163. }
  164. BaseItem item = null;
  165. if (!string.IsNullOrEmpty(request.ParentId))
  166. {
  167. item = _libraryManager.GetItemById(request.ParentId);
  168. }
  169. if (item == null)
  170. {
  171. item = _libraryManager.GetUserRootFolder();
  172. }
  173. if (!(item is Folder folder))
  174. {
  175. folder = _libraryManager.GetUserRootFolder();
  176. }
  177. if (folder is IHasCollectionType hasCollectionType
  178. && string.Equals(hasCollectionType.CollectionType, CollectionType.Playlists, StringComparison.OrdinalIgnoreCase))
  179. {
  180. request.Recursive = true;
  181. request.IncludeItemTypes = "Playlist";
  182. }
  183. bool isInEnabledFolder = user.GetPreference(PreferenceKind.EnabledFolders).Any(i => new Guid(i) == item.Id)
  184. // Assume all folders inside an EnabledChannel are enabled
  185. || user.GetPreference(PreferenceKind.EnabledChannels).Any(i => new Guid(i) == item.Id);
  186. var collectionFolders = _libraryManager.GetCollectionFolders(item);
  187. foreach (var collectionFolder in collectionFolders)
  188. {
  189. if (user.GetPreference(PreferenceKind.EnabledFolders).Contains(
  190. collectionFolder.Id.ToString("N", CultureInfo.InvariantCulture),
  191. StringComparer.OrdinalIgnoreCase))
  192. {
  193. isInEnabledFolder = true;
  194. }
  195. }
  196. if (!(item is UserRootFolder)
  197. && !isInEnabledFolder
  198. && !user.HasPermission(PermissionKind.EnableAllFolders)
  199. && !user.HasPermission(PermissionKind.EnableAllChannels))
  200. {
  201. Logger.LogWarning("{UserName} is not permitted to access Library {ItemName}.", user.Username, item.Name);
  202. return new QueryResult<BaseItem>
  203. {
  204. Items = Array.Empty<BaseItem>(),
  205. TotalRecordCount = 0,
  206. StartIndex = 0
  207. };
  208. }
  209. if (request.Recursive || !string.IsNullOrEmpty(request.Ids) || !(item is UserRootFolder))
  210. {
  211. return folder.GetItems(GetItemsQuery(request, dtoOptions, user));
  212. }
  213. var itemsArray = folder.GetChildren(user, true);
  214. return new QueryResult<BaseItem>
  215. {
  216. Items = itemsArray,
  217. TotalRecordCount = itemsArray.Count,
  218. StartIndex = 0
  219. };
  220. }
  221. private InternalItemsQuery GetItemsQuery(GetItems request, DtoOptions dtoOptions, Jellyfin.Data.Entities.User user)
  222. {
  223. var query = new InternalItemsQuery(user)
  224. {
  225. IsPlayed = request.IsPlayed,
  226. MediaTypes = request.GetMediaTypes(),
  227. IncludeItemTypes = request.GetIncludeItemTypes(),
  228. ExcludeItemTypes = request.GetExcludeItemTypes(),
  229. Recursive = request.Recursive,
  230. OrderBy = request.GetOrderBy(),
  231. IsFavorite = request.IsFavorite,
  232. Limit = request.Limit,
  233. StartIndex = request.StartIndex,
  234. IsMissing = request.IsMissing,
  235. IsUnaired = request.IsUnaired,
  236. CollapseBoxSetItems = request.CollapseBoxSetItems,
  237. NameLessThan = request.NameLessThan,
  238. NameStartsWith = request.NameStartsWith,
  239. NameStartsWithOrGreater = request.NameStartsWithOrGreater,
  240. HasImdbId = request.HasImdbId,
  241. IsPlaceHolder = request.IsPlaceHolder,
  242. IsLocked = request.IsLocked,
  243. MinWidth = request.MinWidth,
  244. MinHeight = request.MinHeight,
  245. MaxWidth = request.MaxWidth,
  246. MaxHeight = request.MaxHeight,
  247. Is3D = request.Is3D,
  248. HasTvdbId = request.HasTvdbId,
  249. HasTmdbId = request.HasTmdbId,
  250. HasOverview = request.HasOverview,
  251. HasOfficialRating = request.HasOfficialRating,
  252. HasParentalRating = request.HasParentalRating,
  253. HasSpecialFeature = request.HasSpecialFeature,
  254. HasSubtitles = request.HasSubtitles,
  255. HasThemeSong = request.HasThemeSong,
  256. HasThemeVideo = request.HasThemeVideo,
  257. HasTrailer = request.HasTrailer,
  258. IsHD = request.IsHD,
  259. Is4K = request.Is4K,
  260. Tags = request.GetTags(),
  261. OfficialRatings = request.GetOfficialRatings(),
  262. Genres = request.GetGenres(),
  263. ArtistIds = GetGuids(request.ArtistIds),
  264. AlbumArtistIds = GetGuids(request.AlbumArtistIds),
  265. ContributingArtistIds = GetGuids(request.ContributingArtistIds),
  266. GenreIds = GetGuids(request.GenreIds),
  267. StudioIds = GetGuids(request.StudioIds),
  268. Person = request.Person,
  269. PersonIds = GetGuids(request.PersonIds),
  270. PersonTypes = request.GetPersonTypes(),
  271. Years = request.GetYears(),
  272. ImageTypes = request.GetImageTypes(),
  273. VideoTypes = request.GetVideoTypes(),
  274. AdjacentTo = request.AdjacentTo,
  275. ItemIds = GetGuids(request.Ids),
  276. MinCommunityRating = request.MinCommunityRating,
  277. MinCriticRating = request.MinCriticRating,
  278. ParentId = string.IsNullOrWhiteSpace(request.ParentId) ? Guid.Empty : new Guid(request.ParentId),
  279. ParentIndexNumber = request.ParentIndexNumber,
  280. EnableTotalRecordCount = request.EnableTotalRecordCount,
  281. ExcludeItemIds = GetGuids(request.ExcludeItemIds),
  282. DtoOptions = dtoOptions,
  283. SearchTerm = request.SearchTerm
  284. };
  285. if (!string.IsNullOrWhiteSpace(request.Ids) || !string.IsNullOrWhiteSpace(request.SearchTerm))
  286. {
  287. query.CollapseBoxSetItems = false;
  288. }
  289. foreach (var filter in request.GetFilters())
  290. {
  291. switch (filter)
  292. {
  293. case ItemFilter.Dislikes:
  294. query.IsLiked = false;
  295. break;
  296. case ItemFilter.IsFavorite:
  297. query.IsFavorite = true;
  298. break;
  299. case ItemFilter.IsFavoriteOrLikes:
  300. query.IsFavoriteOrLiked = true;
  301. break;
  302. case ItemFilter.IsFolder:
  303. query.IsFolder = true;
  304. break;
  305. case ItemFilter.IsNotFolder:
  306. query.IsFolder = false;
  307. break;
  308. case ItemFilter.IsPlayed:
  309. query.IsPlayed = true;
  310. break;
  311. case ItemFilter.IsResumable:
  312. query.IsResumable = true;
  313. break;
  314. case ItemFilter.IsUnplayed:
  315. query.IsPlayed = false;
  316. break;
  317. case ItemFilter.Likes:
  318. query.IsLiked = true;
  319. break;
  320. }
  321. }
  322. if (!string.IsNullOrEmpty(request.MinDateLastSaved))
  323. {
  324. query.MinDateLastSaved = DateTime.Parse(request.MinDateLastSaved, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  325. }
  326. if (!string.IsNullOrEmpty(request.MinDateLastSavedForUser))
  327. {
  328. query.MinDateLastSavedForUser = DateTime.Parse(request.MinDateLastSavedForUser, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  329. }
  330. if (!string.IsNullOrEmpty(request.MinPremiereDate))
  331. {
  332. query.MinPremiereDate = DateTime.Parse(request.MinPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  333. }
  334. if (!string.IsNullOrEmpty(request.MaxPremiereDate))
  335. {
  336. query.MaxPremiereDate = DateTime.Parse(request.MaxPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  337. }
  338. // Filter by Series Status
  339. if (!string.IsNullOrEmpty(request.SeriesStatus))
  340. {
  341. query.SeriesStatuses = request.SeriesStatus.Split(',').Select(d => (SeriesStatus)Enum.Parse(typeof(SeriesStatus), d, true)).ToArray();
  342. }
  343. // ExcludeLocationTypes
  344. if (!string.IsNullOrEmpty(request.ExcludeLocationTypes))
  345. {
  346. var excludeLocationTypes = request.ExcludeLocationTypes.Split(',').Select(d => (LocationType)Enum.Parse(typeof(LocationType), d, true)).ToArray();
  347. if (excludeLocationTypes.Contains(LocationType.Virtual))
  348. {
  349. query.IsVirtualItem = false;
  350. }
  351. }
  352. if (!string.IsNullOrEmpty(request.LocationTypes))
  353. {
  354. var requestedLocationTypes =
  355. request.LocationTypes.Split(',');
  356. if (requestedLocationTypes.Length > 0 && requestedLocationTypes.Length < 4)
  357. {
  358. query.IsVirtualItem = requestedLocationTypes.Contains(LocationType.Virtual.ToString());
  359. }
  360. }
  361. // Min official rating
  362. if (!string.IsNullOrWhiteSpace(request.MinOfficialRating))
  363. {
  364. query.MinParentalRating = _localization.GetRatingLevel(request.MinOfficialRating);
  365. }
  366. // Max official rating
  367. if (!string.IsNullOrWhiteSpace(request.MaxOfficialRating))
  368. {
  369. query.MaxParentalRating = _localization.GetRatingLevel(request.MaxOfficialRating);
  370. }
  371. // Artists
  372. if (!string.IsNullOrEmpty(request.Artists))
  373. {
  374. query.ArtistIds = request.Artists.Split('|').Select(i =>
  375. {
  376. try
  377. {
  378. return _libraryManager.GetArtist(i, new DtoOptions(false));
  379. }
  380. catch
  381. {
  382. return null;
  383. }
  384. }).Where(i => i != null).Select(i => i.Id).ToArray();
  385. }
  386. // ExcludeArtistIds
  387. if (!string.IsNullOrWhiteSpace(request.ExcludeArtistIds))
  388. {
  389. query.ExcludeArtistIds = GetGuids(request.ExcludeArtistIds);
  390. }
  391. if (!string.IsNullOrWhiteSpace(request.AlbumIds))
  392. {
  393. query.AlbumIds = GetGuids(request.AlbumIds);
  394. }
  395. // Albums
  396. if (!string.IsNullOrEmpty(request.Albums))
  397. {
  398. query.AlbumIds = request.Albums.Split('|').SelectMany(i =>
  399. {
  400. return _libraryManager.GetItemIds(new InternalItemsQuery
  401. {
  402. IncludeItemTypes = new[] { typeof(MusicAlbum).Name },
  403. Name = i,
  404. Limit = 1
  405. });
  406. }).ToArray();
  407. }
  408. // Studios
  409. if (!string.IsNullOrEmpty(request.Studios))
  410. {
  411. query.StudioIds = request.Studios.Split('|').Select(i =>
  412. {
  413. try
  414. {
  415. return _libraryManager.GetStudio(i);
  416. }
  417. catch
  418. {
  419. return null;
  420. }
  421. }).Where(i => i != null).Select(i => i.Id).ToArray();
  422. }
  423. // Apply default sorting if none requested
  424. if (query.OrderBy.Count == 0)
  425. {
  426. // Albums by artist
  427. if (query.ArtistIds.Length > 0 && query.IncludeItemTypes.Length == 1 && string.Equals(query.IncludeItemTypes[0], "MusicAlbum", StringComparison.OrdinalIgnoreCase))
  428. {
  429. query.OrderBy = new[]
  430. {
  431. new ValueTuple<string, SortOrder>(ItemSortBy.ProductionYear, SortOrder.Descending),
  432. new ValueTuple<string, SortOrder>(ItemSortBy.SortName, SortOrder.Ascending)
  433. };
  434. }
  435. }
  436. return query;
  437. }
  438. }
  439. /// <summary>
  440. /// Class DateCreatedComparer
  441. /// </summary>
  442. public class DateCreatedComparer : IComparer<BaseItem>
  443. {
  444. /// <summary>
  445. /// Compares the specified x.
  446. /// </summary>
  447. /// <param name="x">The x.</param>
  448. /// <param name="y">The y.</param>
  449. /// <returns>System.Int32.</returns>
  450. public int Compare(BaseItem x, BaseItem y)
  451. {
  452. return x.DateCreated.CompareTo(y.DateCreated);
  453. }
  454. }
  455. }