ItemsService.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623
  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. // Default list type = children
  103. if (!string.IsNullOrEmpty(request.Ids))
  104. {
  105. request.Recursive = true;
  106. var query = GetItemsQuery(request, user);
  107. var result = await ((Folder)item).GetItems(query).ConfigureAwait(false);
  108. if (string.IsNullOrWhiteSpace(request.SortBy))
  109. {
  110. var ids = query.ItemIds.ToList();
  111. // Try to preserve order
  112. result.Items = result.Items.OrderBy(i => ids.IndexOf(i.Id.ToString("N"))).ToArray();
  113. }
  114. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  115. }
  116. if (request.Recursive)
  117. {
  118. var result = await ((Folder)item).GetItems(GetItemsQuery(request, user)).ConfigureAwait(false);
  119. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  120. }
  121. if (user == null)
  122. {
  123. var result = await ((Folder)item).GetItems(GetItemsQuery(request, null)).ConfigureAwait(false);
  124. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  125. }
  126. var userRoot = item as UserRootFolder;
  127. if (userRoot == null)
  128. {
  129. var result = await ((Folder)item).GetItems(GetItemsQuery(request, user)).ConfigureAwait(false);
  130. return new Tuple<QueryResult<BaseItem>, bool>(result, true);
  131. }
  132. IEnumerable<BaseItem> items = ((Folder)item).GetChildren(user, true);
  133. var itemsArray = items.ToArray();
  134. return new Tuple<QueryResult<BaseItem>, bool>(new QueryResult<BaseItem>
  135. {
  136. Items = itemsArray,
  137. TotalRecordCount = itemsArray.Length
  138. }, false);
  139. }
  140. private InternalItemsQuery GetItemsQuery(GetItems request, User user)
  141. {
  142. var query = new InternalItemsQuery
  143. {
  144. User = user,
  145. IsPlayed = request.IsPlayed,
  146. MediaTypes = request.GetMediaTypes(),
  147. IncludeItemTypes = request.GetIncludeItemTypes(),
  148. ExcludeItemTypes = request.GetExcludeItemTypes(),
  149. Recursive = request.Recursive,
  150. SortBy = request.GetOrderBy(),
  151. SortOrder = request.SortOrder ?? SortOrder.Ascending,
  152. Filter = i => ApplyAdditionalFilters(request, i, user, _libraryManager),
  153. Limit = request.Limit,
  154. StartIndex = request.StartIndex,
  155. IsMissing = request.IsMissing,
  156. IsVirtualUnaired = request.IsVirtualUnaired,
  157. IsUnaired = request.IsUnaired,
  158. CollapseBoxSetItems = request.CollapseBoxSetItems,
  159. NameLessThan = request.NameLessThan,
  160. NameStartsWith = request.NameStartsWith,
  161. NameStartsWithOrGreater = request.NameStartsWithOrGreater,
  162. HasImdbId = request.HasImdbId,
  163. IsYearMismatched = request.IsYearMismatched,
  164. IsPlaceHolder = request.IsPlaceHolder,
  165. IsLocked = request.IsLocked,
  166. IsInBoxSet = request.IsInBoxSet,
  167. IsHD = request.IsHD,
  168. Is3D = request.Is3D,
  169. HasTvdbId = request.HasTvdbId,
  170. HasTmdbId = request.HasTmdbId,
  171. HasOverview = request.HasOverview,
  172. HasOfficialRating = request.HasOfficialRating,
  173. HasParentalRating = request.HasParentalRating,
  174. HasSpecialFeature = request.HasSpecialFeature,
  175. HasSubtitles = request.HasSubtitles,
  176. HasThemeSong = request.HasThemeSong,
  177. HasThemeVideo = request.HasThemeVideo,
  178. HasTrailer = request.HasTrailer,
  179. Tags = request.GetTags(),
  180. OfficialRatings = request.GetOfficialRatings(),
  181. Genres = request.GetGenres(),
  182. Studios = request.GetStudios(),
  183. StudioIds = request.GetStudioIds(),
  184. Person = request.Person,
  185. PersonIds = request.GetPersonIds(),
  186. PersonTypes = request.GetPersonTypes(),
  187. Years = request.GetYears(),
  188. ImageTypes = request.GetImageTypes().ToArray(),
  189. VideoTypes = request.GetVideoTypes().ToArray(),
  190. AdjacentTo = request.AdjacentTo,
  191. ItemIds = request.GetItemIds(),
  192. MinPlayers = request.MinPlayers,
  193. MaxPlayers = request.MaxPlayers,
  194. MinCommunityRating = request.MinCommunityRating,
  195. MinCriticRating = request.MinCriticRating
  196. };
  197. if (!string.IsNullOrWhiteSpace(request.Ids))
  198. {
  199. query.CollapseBoxSetItems = false;
  200. }
  201. foreach (var filter in request.GetFilters())
  202. {
  203. switch (filter)
  204. {
  205. case ItemFilter.Dislikes:
  206. query.IsLiked = false;
  207. break;
  208. case ItemFilter.IsFavorite:
  209. query.IsFavorite = true;
  210. break;
  211. case ItemFilter.IsFavoriteOrLikes:
  212. query.IsFavoriteOrLiked = true;
  213. break;
  214. case ItemFilter.IsFolder:
  215. query.IsFolder = true;
  216. break;
  217. case ItemFilter.IsNotFolder:
  218. query.IsFolder = false;
  219. break;
  220. case ItemFilter.IsPlayed:
  221. query.IsPlayed = true;
  222. break;
  223. case ItemFilter.IsRecentlyAdded:
  224. break;
  225. case ItemFilter.IsResumable:
  226. query.IsResumable = true;
  227. break;
  228. case ItemFilter.IsUnplayed:
  229. query.IsPlayed = false;
  230. break;
  231. case ItemFilter.Likes:
  232. query.IsLiked = true;
  233. break;
  234. }
  235. }
  236. return query;
  237. }
  238. /// <summary>
  239. /// Applies filtering
  240. /// </summary>
  241. /// <param name="items">The items.</param>
  242. /// <param name="filter">The filter.</param>
  243. /// <param name="user">The user.</param>
  244. /// <param name="repository">The repository.</param>
  245. /// <returns>IEnumerable{BaseItem}.</returns>
  246. internal static IEnumerable<BaseItem> ApplyFilter(IEnumerable<BaseItem> items, ItemFilter filter, User user, IUserDataManager repository)
  247. {
  248. // Avoid implicitly captured closure
  249. var currentUser = user;
  250. switch (filter)
  251. {
  252. case ItemFilter.IsFavoriteOrLikes:
  253. return items.Where(item =>
  254. {
  255. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  256. if (userdata == null)
  257. {
  258. return false;
  259. }
  260. var likes = userdata.Likes ?? false;
  261. var favorite = userdata.IsFavorite;
  262. return likes || favorite;
  263. });
  264. case ItemFilter.Likes:
  265. return items.Where(item =>
  266. {
  267. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  268. return userdata != null && userdata.Likes.HasValue && userdata.Likes.Value;
  269. });
  270. case ItemFilter.Dislikes:
  271. return items.Where(item =>
  272. {
  273. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  274. return userdata != null && userdata.Likes.HasValue && !userdata.Likes.Value;
  275. });
  276. case ItemFilter.IsFavorite:
  277. return items.Where(item =>
  278. {
  279. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  280. return userdata != null && userdata.IsFavorite;
  281. });
  282. case ItemFilter.IsResumable:
  283. return items.Where(item =>
  284. {
  285. var userdata = repository.GetUserData(user.Id, item.GetUserDataKey());
  286. return userdata != null && userdata.PlaybackPositionTicks > 0;
  287. });
  288. case ItemFilter.IsPlayed:
  289. return items.Where(item => item.IsPlayed(currentUser));
  290. case ItemFilter.IsUnplayed:
  291. return items.Where(item => item.IsUnplayed(currentUser));
  292. case ItemFilter.IsFolder:
  293. return items.Where(item => item.IsFolder);
  294. case ItemFilter.IsNotFolder:
  295. return items.Where(item => !item.IsFolder);
  296. case ItemFilter.IsRecentlyAdded:
  297. return items.Where(item => (DateTime.UtcNow - item.DateCreated).TotalDays <= 10);
  298. }
  299. return items;
  300. }
  301. private bool ApplyAdditionalFilters(GetItems request, BaseItem i, User user, ILibraryManager libraryManager)
  302. {
  303. // Artists
  304. if (!string.IsNullOrEmpty(request.ArtistIds))
  305. {
  306. var artistIds = request.ArtistIds.Split(new[] { '|', ',' });
  307. var audio = i as IHasArtist;
  308. if (!(audio != null && artistIds.Any(id =>
  309. {
  310. var artistItem = libraryManager.GetItemById(id);
  311. return artistItem != null && audio.HasAnyArtist(artistItem.Name);
  312. })))
  313. {
  314. return false;
  315. }
  316. }
  317. // Artists
  318. if (!string.IsNullOrEmpty(request.Artists))
  319. {
  320. var artists = request.Artists.Split('|');
  321. var audio = i as IHasArtist;
  322. if (!(audio != null && artists.Any(audio.HasAnyArtist)))
  323. {
  324. return false;
  325. }
  326. }
  327. // Albums
  328. if (!string.IsNullOrEmpty(request.Albums))
  329. {
  330. var albums = request.Albums.Split('|');
  331. var audio = i as Audio;
  332. if (audio != null)
  333. {
  334. if (!albums.Any(a => string.Equals(a, audio.Album, StringComparison.OrdinalIgnoreCase)))
  335. {
  336. return false;
  337. }
  338. }
  339. var album = i as MusicAlbum;
  340. if (album != null)
  341. {
  342. if (!albums.Any(a => string.Equals(a, album.Name, StringComparison.OrdinalIgnoreCase)))
  343. {
  344. return false;
  345. }
  346. }
  347. var musicVideo = i as MusicVideo;
  348. if (musicVideo != null)
  349. {
  350. if (!albums.Any(a => string.Equals(a, musicVideo.Album, StringComparison.OrdinalIgnoreCase)))
  351. {
  352. return false;
  353. }
  354. }
  355. return false;
  356. }
  357. // Min index number
  358. if (request.MinIndexNumber.HasValue)
  359. {
  360. if (!(i.IndexNumber.HasValue && i.IndexNumber.Value >= request.MinIndexNumber.Value))
  361. {
  362. return false;
  363. }
  364. }
  365. // Min official rating
  366. if (!string.IsNullOrEmpty(request.MinOfficialRating))
  367. {
  368. var level = _localization.GetRatingLevel(request.MinOfficialRating);
  369. if (level.HasValue)
  370. {
  371. var rating = i.CustomRating;
  372. if (string.IsNullOrEmpty(rating))
  373. {
  374. rating = i.OfficialRating;
  375. }
  376. if (!string.IsNullOrEmpty(rating))
  377. {
  378. var itemLevel = _localization.GetRatingLevel(rating);
  379. if (!(!itemLevel.HasValue || itemLevel.Value >= level.Value))
  380. {
  381. return false;
  382. }
  383. }
  384. }
  385. }
  386. // Max official rating
  387. if (!string.IsNullOrEmpty(request.MaxOfficialRating))
  388. {
  389. var level = _localization.GetRatingLevel(request.MaxOfficialRating);
  390. if (level.HasValue)
  391. {
  392. var rating = i.CustomRating;
  393. if (string.IsNullOrEmpty(rating))
  394. {
  395. rating = i.OfficialRating;
  396. }
  397. if (!string.IsNullOrEmpty(rating))
  398. {
  399. var itemLevel = _localization.GetRatingLevel(rating);
  400. if (!(!itemLevel.HasValue || itemLevel.Value <= level.Value))
  401. {
  402. return false;
  403. }
  404. }
  405. }
  406. }
  407. // LocationTypes
  408. if (!string.IsNullOrEmpty(request.LocationTypes))
  409. {
  410. var vals = request.LocationTypes.Split(',');
  411. if (!vals.Contains(i.LocationType.ToString(), StringComparer.OrdinalIgnoreCase))
  412. {
  413. return false;
  414. }
  415. }
  416. // ExcludeLocationTypes
  417. if (!string.IsNullOrEmpty(request.ExcludeLocationTypes))
  418. {
  419. var vals = request.ExcludeLocationTypes.Split(',');
  420. if (vals.Contains(i.LocationType.ToString(), StringComparer.OrdinalIgnoreCase))
  421. {
  422. return false;
  423. }
  424. }
  425. if (!string.IsNullOrEmpty(request.AlbumArtistStartsWithOrGreater))
  426. {
  427. var ok = new[] { i }.OfType<IHasAlbumArtist>()
  428. .Any(p => string.Compare(request.AlbumArtistStartsWithOrGreater, p.AlbumArtists.FirstOrDefault(), StringComparison.CurrentCultureIgnoreCase) < 1);
  429. if (!ok)
  430. {
  431. return false;
  432. }
  433. }
  434. // Filter by Series Status
  435. if (!string.IsNullOrEmpty(request.SeriesStatus))
  436. {
  437. var vals = request.SeriesStatus.Split(',');
  438. var ok = new[] { i }.OfType<Series>().Any(p => p.Status.HasValue && vals.Contains(p.Status.Value.ToString(), StringComparer.OrdinalIgnoreCase));
  439. if (!ok)
  440. {
  441. return false;
  442. }
  443. }
  444. // Filter by Series AirDays
  445. if (!string.IsNullOrEmpty(request.AirDays))
  446. {
  447. var days = request.AirDays.Split(',').Select(d => (DayOfWeek)Enum.Parse(typeof(DayOfWeek), d, true));
  448. var ok = new[] { i }.OfType<Series>().Any(p => p.AirDays != null && days.Any(d => p.AirDays.Contains(d)));
  449. if (!ok)
  450. {
  451. return false;
  452. }
  453. }
  454. if (request.ParentIndexNumber.HasValue)
  455. {
  456. var filterValue = request.ParentIndexNumber.Value;
  457. var episode = i as Episode;
  458. if (episode != null)
  459. {
  460. if (episode.ParentIndexNumber.HasValue && episode.ParentIndexNumber.Value != filterValue)
  461. {
  462. return false;
  463. }
  464. }
  465. var song = i as Audio;
  466. if (song != null)
  467. {
  468. if (song.ParentIndexNumber.HasValue && song.ParentIndexNumber.Value != filterValue)
  469. {
  470. return false;
  471. }
  472. }
  473. }
  474. if (request.AiredDuringSeason.HasValue)
  475. {
  476. var episode = i as Episode;
  477. if (episode == null)
  478. {
  479. return false;
  480. }
  481. if (!Series.FilterEpisodesBySeason(new[] { episode }, request.AiredDuringSeason.Value, true).Any())
  482. {
  483. return false;
  484. }
  485. }
  486. if (!string.IsNullOrEmpty(request.MinPremiereDate))
  487. {
  488. var date = DateTime.Parse(request.MinPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  489. if (!(i.PremiereDate.HasValue && i.PremiereDate.Value >= date))
  490. {
  491. return false;
  492. }
  493. }
  494. if (!string.IsNullOrEmpty(request.MaxPremiereDate))
  495. {
  496. var date = DateTime.Parse(request.MaxPremiereDate, null, DateTimeStyles.RoundtripKind).ToUniversalTime();
  497. if (!(i.PremiereDate.HasValue && i.PremiereDate.Value <= date))
  498. {
  499. return false;
  500. }
  501. }
  502. return true;
  503. }
  504. }
  505. /// <summary>
  506. /// Class DateCreatedComparer
  507. /// </summary>
  508. public class DateCreatedComparer : IComparer<BaseItem>
  509. {
  510. /// <summary>
  511. /// Compares the specified x.
  512. /// </summary>
  513. /// <param name="x">The x.</param>
  514. /// <param name="y">The y.</param>
  515. /// <returns>System.Int32.</returns>
  516. public int Compare(BaseItem x, BaseItem y)
  517. {
  518. return x.DateCreated.CompareTo(y.DateCreated);
  519. }
  520. }
  521. }