ItemsService.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620
  1. using MediaBrowser.Controller.Collections;
  2. using MediaBrowser.Controller.Dto;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Audio;
  5. using MediaBrowser.Controller.Entities.TV;
  6. using MediaBrowser.Controller.Library;
  7. using MediaBrowser.Controller.Localization;
  8. using MediaBrowser.Controller.Net;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Querying;
  11. using ServiceStack;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Globalization;
  15. using System.Linq;
  16. using System.Threading.Tasks;
  17. namespace MediaBrowser.Api.UserLibrary
  18. {
  19. /// <summary>
  20. /// Class GetItems
  21. /// </summary>
  22. [Route("/Items", "GET", Summary = "Gets items based on a query.")]
  23. [Route("/Users/{UserId}/Items", "GET", Summary = "Gets items based on a query.")]
  24. public class GetItems : BaseItemsRequest, IReturn<ItemsResult>
  25. {
  26. }
  27. /// <summary>
  28. /// Class ItemsService
  29. /// </summary>
  30. [Authenticated]
  31. public class ItemsService : BaseApiService
  32. {
  33. /// <summary>
  34. /// The _user manager
  35. /// </summary>
  36. private readonly IUserManager _userManager;
  37. private readonly IUserDataManager _userDataRepository;
  38. /// <summary>
  39. /// The _library manager
  40. /// </summary>
  41. private readonly ILibraryManager _libraryManager;
  42. private readonly ILocalizationManager _localization;
  43. private readonly IDtoService _dtoService;
  44. private readonly ICollectionManager _collectionManager;
  45. /// <summary>
  46. /// Initializes a new instance of the <see cref="ItemsService" /> class.
  47. /// </summary>
  48. /// <param name="userManager">The user manager.</param>
  49. /// <param name="libraryManager">The library manager.</param>
  50. /// <param name="userDataRepository">The user data repository.</param>
  51. /// <param name="localization">The localization.</param>
  52. /// <param name="dtoService">The dto service.</param>
  53. /// <param name="collectionManager">The collection manager.</param>
  54. public ItemsService(IUserManager userManager, ILibraryManager libraryManager, IUserDataManager userDataRepository, ILocalizationManager localization, IDtoService dtoService, ICollectionManager collectionManager)
  55. {
  56. _userManager = userManager;
  57. _libraryManager = libraryManager;
  58. _userDataRepository = userDataRepository;
  59. _localization = localization;
  60. _dtoService = dtoService;
  61. _collectionManager = collectionManager;
  62. }
  63. /// <summary>
  64. /// Gets the specified request.
  65. /// </summary>
  66. /// <param name="request">The request.</param>
  67. /// <returns>System.Object.</returns>
  68. public async Task<object> Get(GetItems request)
  69. {
  70. var result = await GetItems(request).ConfigureAwait(false);
  71. return ToOptimizedSerializedResultUsingCache(result);
  72. }
  73. /// <summary>
  74. /// Gets the items.
  75. /// </summary>
  76. /// <param name="request">The request.</param>
  77. /// <returns>Task{ItemsResult}.</returns>
  78. private async Task<ItemsResult> GetItems(GetItems request)
  79. {
  80. var parentItem = string.IsNullOrEmpty(request.ParentId) ? null : _libraryManager.GetItemById(request.ParentId);
  81. var user = !string.IsNullOrWhiteSpace(request.UserId) ? _userManager.GetUserById(request.UserId) : null;
  82. var result = await GetItemsToSerialize(request, user, parentItem).ConfigureAwait(false);
  83. var dtoOptions = GetDtoOptions(request);
  84. return new ItemsResult
  85. {
  86. TotalRecordCount = result.Item1.TotalRecordCount,
  87. Items = _dtoService.GetBaseItemDtos(result.Item1.Items, dtoOptions, user).ToArray()
  88. };
  89. }
  90. /// <summary>
  91. /// Gets the items to serialize.
  92. /// </summary>
  93. /// <param name="request">The request.</param>
  94. /// <param name="user">The user.</param>
  95. /// <param name="parentItem">The parent item.</param>
  96. /// <returns>IEnumerable{BaseItem}.</returns>
  97. private async Task<Tuple<QueryResult<BaseItem>, bool>> GetItemsToSerialize(GetItems request, User user, BaseItem parentItem)
  98. {
  99. var item = string.IsNullOrEmpty(request.ParentId) ?
  100. user == null ? _libraryManager.RootFolder : user.RootFolder :
  101. parentItem;
  102. if (string.Equals(request.IncludeItemTypes, "Playlist", StringComparison.OrdinalIgnoreCase))
  103. {
  104. item = user == null ? _libraryManager.RootFolder : user.RootFolder;
  105. }
  106. // Default list type = children
  107. if (!string.IsNullOrEmpty(request.Ids))
  108. {
  109. request.Recursive = true;
  110. var query = GetItemsQuery(request, user);
  111. var result = await ((Folder)item).GetItems(query).ConfigureAwait(false);
  112. if (string.IsNullOrWhiteSpace(request.SortBy))
  113. {
  114. var ids = query.ItemIds.ToList();
  115. // Try to preserve order
  116. result.Items = result.Items.OrderBy(i => ids.IndexOf(i.Id.ToString("N"))).ToArray();
  117. }
  118. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  119. }
  120. if (request.Recursive)
  121. {
  122. var result = await ((Folder)item).GetItems(GetItemsQuery(request, user)).ConfigureAwait(false);
  123. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  124. }
  125. if (user == null)
  126. {
  127. var result = await ((Folder)item).GetItems(GetItemsQuery(request, null)).ConfigureAwait(false);
  128. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  129. }
  130. var userRoot = item as UserRootFolder;
  131. if (userRoot == null)
  132. {
  133. var result = await ((Folder)item).GetItems(GetItemsQuery(request, user)).ConfigureAwait(false);
  134. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  135. }
  136. IEnumerable<BaseItem> items = ((Folder)item).GetChildren(user, true);
  137. var itemsArray = items.ToArray();
  138. return new Tuple<QueryResult<BaseItem>, bool>(new QueryResult<BaseItem>
  139. {
  140. Items = itemsArray,
  141. TotalRecordCount = itemsArray.Length
  142. }, false);
  143. }
  144. private InternalItemsQuery GetItemsQuery(GetItems request, User user)
  145. {
  146. var query = new InternalItemsQuery
  147. {
  148. User = user,
  149. IsPlayed = request.IsPlayed,
  150. MediaTypes = request.GetMediaTypes(),
  151. IncludeItemTypes = request.GetIncludeItemTypes(),
  152. ExcludeItemTypes = request.GetExcludeItemTypes(),
  153. Recursive = request.Recursive,
  154. SortBy = request.GetOrderBy(),
  155. SortOrder = request.SortOrder ?? SortOrder.Ascending,
  156. Filter = i => ApplyAdditionalFilters(request, i, user, _libraryManager),
  157. Limit = request.Limit,
  158. StartIndex = request.StartIndex,
  159. IsMissing = request.IsMissing,
  160. IsVirtualUnaired = request.IsVirtualUnaired,
  161. IsUnaired = request.IsUnaired,
  162. CollapseBoxSetItems = request.CollapseBoxSetItems,
  163. NameLessThan = request.NameLessThan,
  164. NameStartsWith = request.NameStartsWith,
  165. NameStartsWithOrGreater = request.NameStartsWithOrGreater,
  166. HasImdbId = request.HasImdbId,
  167. IsYearMismatched = request.IsYearMismatched,
  168. IsPlaceHolder = request.IsPlaceHolder,
  169. IsLocked = request.IsLocked,
  170. IsInBoxSet = request.IsInBoxSet,
  171. IsHD = request.IsHD,
  172. Is3D = request.Is3D,
  173. HasTvdbId = request.HasTvdbId,
  174. HasTmdbId = request.HasTmdbId,
  175. HasOverview = request.HasOverview,
  176. HasOfficialRating = request.HasOfficialRating,
  177. HasParentalRating = request.HasParentalRating,
  178. HasSpecialFeature = request.HasSpecialFeature,
  179. HasSubtitles = request.HasSubtitles,
  180. HasThemeSong = request.HasThemeSong,
  181. HasThemeVideo = request.HasThemeVideo,
  182. HasTrailer = request.HasTrailer,
  183. Tags = request.GetTags(),
  184. OfficialRatings = request.GetOfficialRatings(),
  185. Genres = request.GetGenres(),
  186. GenreIds = request.GetGenreIds(),
  187. Studios = request.GetStudios(),
  188. StudioIds = request.GetStudioIds(),
  189. Person = request.Person,
  190. PersonIds = request.GetPersonIds(),
  191. PersonTypes = request.GetPersonTypes(),
  192. Years = request.GetYears(),
  193. ImageTypes = request.GetImageTypes().ToArray(),
  194. VideoTypes = request.GetVideoTypes().ToArray(),
  195. AdjacentTo = request.AdjacentTo,
  196. ItemIds = request.GetItemIds(),
  197. MinPlayers = request.MinPlayers,
  198. MaxPlayers = request.MaxPlayers,
  199. MinCommunityRating = request.MinCommunityRating,
  200. MinCriticRating = request.MinCriticRating
  201. };
  202. if (!string.IsNullOrWhiteSpace(request.Ids))
  203. {
  204. query.CollapseBoxSetItems = false;
  205. }
  206. foreach (var filter in request.GetFilters())
  207. {
  208. switch (filter)
  209. {
  210. case ItemFilter.Dislikes:
  211. query.IsLiked = false;
  212. break;
  213. case ItemFilter.IsFavorite:
  214. query.IsFavorite = true;
  215. break;
  216. case ItemFilter.IsFavoriteOrLikes:
  217. query.IsFavoriteOrLiked = true;
  218. break;
  219. case ItemFilter.IsFolder:
  220. query.IsFolder = true;
  221. break;
  222. case ItemFilter.IsNotFolder:
  223. query.IsFolder = false;
  224. break;
  225. case ItemFilter.IsPlayed:
  226. query.IsPlayed = true;
  227. break;
  228. case ItemFilter.IsRecentlyAdded:
  229. break;
  230. case ItemFilter.IsResumable:
  231. query.IsResumable = true;
  232. break;
  233. case ItemFilter.IsUnplayed:
  234. query.IsPlayed = false;
  235. break;
  236. case ItemFilter.Likes:
  237. query.IsLiked = true;
  238. break;
  239. }
  240. }
  241. return query;
  242. }
  243. /// <summary>
  244. /// Applies filtering
  245. /// </summary>
  246. /// <param name="items">The items.</param>
  247. /// <param name="filter">The filter.</param>
  248. /// <param name="user">The user.</param>
  249. /// <param name="repository">The repository.</param>
  250. /// <returns>IEnumerable{BaseItem}.</returns>
  251. internal static IEnumerable<BaseItem> ApplyFilter(IEnumerable<BaseItem> items, ItemFilter filter, User user, IUserDataManager repository)
  252. {
  253. // Avoid implicitly captured closure
  254. var currentUser = user;
  255. switch (filter)
  256. {
  257. case ItemFilter.IsFavoriteOrLikes:
  258. return items.Where(item =>
  259. {
  260. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  261. if (userdata == null)
  262. {
  263. return false;
  264. }
  265. var likes = userdata.Likes ?? false;
  266. var favorite = userdata.IsFavorite;
  267. return likes || favorite;
  268. });
  269. case ItemFilter.Likes:
  270. return items.Where(item =>
  271. {
  272. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  273. return userdata != null && userdata.Likes.HasValue && userdata.Likes.Value;
  274. });
  275. case ItemFilter.Dislikes:
  276. return items.Where(item =>
  277. {
  278. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  279. return userdata != null && userdata.Likes.HasValue && !userdata.Likes.Value;
  280. });
  281. case ItemFilter.IsFavorite:
  282. return items.Where(item =>
  283. {
  284. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  285. return userdata != null && userdata.IsFavorite;
  286. });
  287. case ItemFilter.IsResumable:
  288. return items.Where(item =>
  289. {
  290. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  291. return userdata != null && userdata.PlaybackPositionTicks > 0;
  292. });
  293. case ItemFilter.IsPlayed:
  294. return items.Where(item => item.IsPlayed(currentUser));
  295. case ItemFilter.IsUnplayed:
  296. return items.Where(item => item.IsUnplayed(currentUser));
  297. case ItemFilter.IsFolder:
  298. return items.Where(item => item.IsFolder);
  299. case ItemFilter.IsNotFolder:
  300. return items.Where(item => !item.IsFolder);
  301. case ItemFilter.IsRecentlyAdded:
  302. return items.Where(item => (DateTime.UtcNow - item.DateCreated).TotalDays <= 10);
  303. }
  304. return items;
  305. }
  306. private bool ApplyAdditionalFilters(GetItems request, BaseItem i, User user, ILibraryManager libraryManager)
  307. {
  308. // Artists
  309. if (!string.IsNullOrEmpty(request.ArtistIds))
  310. {
  311. var artistIds = request.ArtistIds.Split(new[] { '|', ',' });
  312. var audio = i as IHasArtist;
  313. if (!(audio != null && artistIds.Any(id =>
  314. {
  315. var artistItem = libraryManager.GetItemById(id);
  316. return artistItem != null && audio.HasAnyArtist(artistItem.Name);
  317. })))
  318. {
  319. return false;
  320. }
  321. }
  322. // Artists
  323. if (!string.IsNullOrEmpty(request.Artists))
  324. {
  325. var artists = request.Artists.Split('|');
  326. var audio = i as IHasArtist;
  327. if (!(audio != null && artists.Any(audio.HasAnyArtist)))
  328. {
  329. return false;
  330. }
  331. }
  332. // Albums
  333. if (!string.IsNullOrEmpty(request.Albums))
  334. {
  335. var albums = request.Albums.Split('|');
  336. var audio = i as Audio;
  337. if (audio != null)
  338. {
  339. if (!albums.Any(a => string.Equals(a, audio.Album, StringComparison.OrdinalIgnoreCase)))
  340. {
  341. return false;
  342. }
  343. }
  344. var album = i as MusicAlbum;
  345. if (album != null)
  346. {
  347. if (!albums.Any(a => string.Equals(a, album.Name, StringComparison.OrdinalIgnoreCase)))
  348. {
  349. return false;
  350. }
  351. }
  352. var musicVideo = i as MusicVideo;
  353. if (musicVideo != null)
  354. {
  355. if (!albums.Any(a => string.Equals(a, musicVideo.Album, StringComparison.OrdinalIgnoreCase)))
  356. {
  357. return false;
  358. }
  359. }
  360. return false;
  361. }
  362. // Min official rating
  363. if (!string.IsNullOrEmpty(request.MinOfficialRating))
  364. {
  365. var level = _localization.GetRatingLevel(request.MinOfficialRating);
  366. if (level.HasValue)
  367. {
  368. var rating = i.CustomRating;
  369. if (string.IsNullOrEmpty(rating))
  370. {
  371. rating = i.OfficialRating;
  372. }
  373. if (!string.IsNullOrEmpty(rating))
  374. {
  375. var itemLevel = _localization.GetRatingLevel(rating);
  376. if (!(!itemLevel.HasValue || itemLevel.Value >= level.Value))
  377. {
  378. return false;
  379. }
  380. }
  381. }
  382. }
  383. // Max official rating
  384. if (!string.IsNullOrEmpty(request.MaxOfficialRating))
  385. {
  386. var level = _localization.GetRatingLevel(request.MaxOfficialRating);
  387. if (level.HasValue)
  388. {
  389. var rating = i.CustomRating;
  390. if (string.IsNullOrEmpty(rating))
  391. {
  392. rating = i.OfficialRating;
  393. }
  394. if (!string.IsNullOrEmpty(rating))
  395. {
  396. var itemLevel = _localization.GetRatingLevel(rating);
  397. if (!(!itemLevel.HasValue || itemLevel.Value <= level.Value))
  398. {
  399. return false;
  400. }
  401. }
  402. }
  403. }
  404. // LocationTypes
  405. if (!string.IsNullOrEmpty(request.LocationTypes))
  406. {
  407. var vals = request.LocationTypes.Split(',');
  408. if (!vals.Contains(i.LocationType.ToString(), StringComparer.OrdinalIgnoreCase))
  409. {
  410. return false;
  411. }
  412. }
  413. // ExcludeLocationTypes
  414. if (!string.IsNullOrEmpty(request.ExcludeLocationTypes))
  415. {
  416. var vals = request.ExcludeLocationTypes.Split(',');
  417. if (vals.Contains(i.LocationType.ToString(), StringComparer.OrdinalIgnoreCase))
  418. {
  419. return false;
  420. }
  421. }
  422. if (!string.IsNullOrEmpty(request.AlbumArtistStartsWithOrGreater))
  423. {
  424. var ok = new[] { i }.OfType<IHasAlbumArtist>()
  425. .Any(p => string.Compare(request.AlbumArtistStartsWithOrGreater, p.AlbumArtists.FirstOrDefault(), StringComparison.CurrentCultureIgnoreCase) < 1);
  426. if (!ok)
  427. {
  428. return false;
  429. }
  430. }
  431. // Filter by Series Status
  432. if (!string.IsNullOrEmpty(request.SeriesStatus))
  433. {
  434. var vals = request.SeriesStatus.Split(',');
  435. var ok = new[] { i }.OfType<Series>().Any(p => p.Status.HasValue && vals.Contains(p.Status.Value.ToString(), StringComparer.OrdinalIgnoreCase));
  436. if (!ok)
  437. {
  438. return false;
  439. }
  440. }
  441. // Filter by Series AirDays
  442. if (!string.IsNullOrEmpty(request.AirDays))
  443. {
  444. var days = request.AirDays.Split(',').Select(d => (DayOfWeek)Enum.Parse(typeof(DayOfWeek), d, true));
  445. var ok = new[] { i }.OfType<Series>().Any(p => p.AirDays != null && days.Any(d => p.AirDays.Contains(d)));
  446. if (!ok)
  447. {
  448. return false;
  449. }
  450. }
  451. if (request.ParentIndexNumber.HasValue)
  452. {
  453. var filterValue = request.ParentIndexNumber.Value;
  454. var episode = i as Episode;
  455. if (episode != null)
  456. {
  457. if (episode.ParentIndexNumber.HasValue && episode.ParentIndexNumber.Value != filterValue)
  458. {
  459. return false;
  460. }
  461. }
  462. var song = i as Audio;
  463. if (song != null)
  464. {
  465. if (song.ParentIndexNumber.HasValue && song.ParentIndexNumber.Value != filterValue)
  466. {
  467. return false;
  468. }
  469. }
  470. }
  471. if (request.AiredDuringSeason.HasValue)
  472. {
  473. var episode = i as Episode;
  474. if (episode == null)
  475. {
  476. return false;
  477. }
  478. if (!Series.FilterEpisodesBySeason(new[] { episode }, request.AiredDuringSeason.Value, true).Any())
  479. {
  480. return false;
  481. }
  482. }
  483. if (!string.IsNullOrEmpty(request.MinPremiereDate))
  484. {
  485. var date = DateTime.Parse(request.MinPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  486. if (!(i.PremiereDate.HasValue && i.PremiereDate.Value >= date))
  487. {
  488. return false;
  489. }
  490. }
  491. if (!string.IsNullOrEmpty(request.MaxPremiereDate))
  492. {
  493. var date = DateTime.Parse(request.MaxPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  494. if (!(i.PremiereDate.HasValue && i.PremiereDate.Value <= date))
  495. {
  496. return false;
  497. }
  498. }
  499. return true;
  500. }
  501. }
  502. /// <summary>
  503. /// Class DateCreatedComparer
  504. /// </summary>
  505. public class DateCreatedComparer : IComparer<BaseItem>
  506. {
  507. /// <summary>
  508. /// Compares the specified x.
  509. /// </summary>
  510. /// <param name="x">The x.</param>
  511. /// <param name="y">The y.</param>
  512. /// <returns>System.Int32.</returns>
  513. public int Compare(BaseItem x, BaseItem y)
  514. {
  515. return x.DateCreated.CompareTo(y.DateCreated);
  516. }
  517. }
  518. }