BaseItemsByNameService.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. using MediaBrowser.Controller.Dto;
  2. using MediaBrowser.Controller.Entities;
  3. using MediaBrowser.Controller.Library;
  4. using MediaBrowser.Controller.Persistence;
  5. using MediaBrowser.Model.Dto;
  6. using MediaBrowser.Model.Querying;
  7. using ServiceStack.ServiceHost;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. using System.Threading.Tasks;
  12. namespace MediaBrowser.Api.UserLibrary
  13. {
  14. /// <summary>
  15. /// Class BaseItemsByNameService
  16. /// </summary>
  17. /// <typeparam name="TItemType">The type of the T item type.</typeparam>
  18. public abstract class BaseItemsByNameService<TItemType> : BaseApiService
  19. where TItemType : BaseItem, IItemByName
  20. {
  21. /// <summary>
  22. /// The _user manager
  23. /// </summary>
  24. protected readonly IUserManager UserManager;
  25. /// <summary>
  26. /// The library manager
  27. /// </summary>
  28. protected readonly ILibraryManager LibraryManager;
  29. protected readonly IUserDataRepository UserDataRepository;
  30. protected readonly IItemRepository ItemRepository;
  31. protected IDtoService DtoService { get; private set; }
  32. /// <summary>
  33. /// Initializes a new instance of the <see cref="BaseItemsByNameService{TItemType}" /> class.
  34. /// </summary>
  35. /// <param name="userManager">The user manager.</param>
  36. /// <param name="libraryManager">The library manager.</param>
  37. /// <param name="userDataRepository">The user data repository.</param>
  38. /// <param name="itemRepository">The item repository.</param>
  39. /// <param name="dtoService">The dto service.</param>
  40. protected BaseItemsByNameService(IUserManager userManager, ILibraryManager libraryManager, IUserDataRepository userDataRepository, IItemRepository itemRepository, IDtoService dtoService)
  41. {
  42. UserManager = userManager;
  43. LibraryManager = libraryManager;
  44. UserDataRepository = userDataRepository;
  45. ItemRepository = itemRepository;
  46. DtoService = dtoService;
  47. }
  48. /// <summary>
  49. /// Gets the specified request.
  50. /// </summary>
  51. /// <param name="request">The request.</param>
  52. /// <returns>Task{ItemsResult}.</returns>
  53. protected async Task<ItemsResult> GetResult(GetItemsByName request)
  54. {
  55. User user = null;
  56. BaseItem item;
  57. if (request.UserId.HasValue)
  58. {
  59. user = UserManager.GetUserById(request.UserId.Value);
  60. item = string.IsNullOrEmpty(request.ParentId) ? user.RootFolder : DtoService.GetItemByDtoId(request.ParentId, user.Id);
  61. }
  62. else
  63. {
  64. item = string.IsNullOrEmpty(request.ParentId) ? LibraryManager.RootFolder : DtoService.GetItemByDtoId(request.ParentId);
  65. }
  66. IEnumerable<BaseItem> items;
  67. if (item.IsFolder)
  68. {
  69. var folder = (Folder)item;
  70. if (request.UserId.HasValue)
  71. {
  72. items = request.Recursive ? folder.GetRecursiveChildren(user) : folder.GetChildren(user, true);
  73. }
  74. else
  75. {
  76. items = request.Recursive ? folder.RecursiveChildren : folder.Children;
  77. }
  78. }
  79. else
  80. {
  81. items = new[] { item };
  82. }
  83. items = FilterItems(request, items);
  84. var ibnItemTasks = GetAllItems(request, items);
  85. var extractedItems = await Task.WhenAll(ibnItemTasks).ConfigureAwait(false);
  86. var filteredItems = FilterItems(request, extractedItems, user);
  87. filteredItems = ItemsService.ApplySortOrder(request, filteredItems, user, LibraryManager).Cast<TItemType>();
  88. var ibnItemsArray = filteredItems.ToArray();
  89. IEnumerable<TItemType> ibnItems = ibnItemsArray;
  90. var result = new ItemsResult
  91. {
  92. TotalRecordCount = ibnItemsArray.Length
  93. };
  94. if (request.StartIndex.HasValue || request.Limit.HasValue)
  95. {
  96. if (request.StartIndex.HasValue)
  97. {
  98. ibnItems = ibnItems.Skip(request.StartIndex.Value);
  99. }
  100. if (request.Limit.HasValue)
  101. {
  102. ibnItems = ibnItems.Take(request.Limit.Value);
  103. }
  104. }
  105. var fields = request.GetItemFields().ToList();
  106. var tasks = ibnItems.Select(i => GetDto(i, user, fields));
  107. var resultItems = await Task.WhenAll(tasks).ConfigureAwait(false);
  108. result.Items = resultItems.Where(i => i != null).ToArray();
  109. return result;
  110. }
  111. /// <summary>
  112. /// Filters the items.
  113. /// </summary>
  114. /// <param name="request">The request.</param>
  115. /// <param name="items">The items.</param>
  116. /// <param name="user">The user.</param>
  117. /// <returns>IEnumerable{`0}.</returns>
  118. private IEnumerable<TItemType> FilterItems(GetItemsByName request, IEnumerable<TItemType> items, User user)
  119. {
  120. if (!string.IsNullOrEmpty(request.NameStartsWithOrGreater))
  121. {
  122. items = items.Where(i => string.Compare(request.NameStartsWithOrGreater, i.Name, StringComparison.CurrentCultureIgnoreCase) < 1);
  123. }
  124. var imageTypes = request.GetImageTypes().ToArray();
  125. if (imageTypes.Length > 0)
  126. {
  127. items = items.Where(item => imageTypes.Any(imageType => ItemsService.HasImage(item, imageType)));
  128. }
  129. var filters = request.GetFilters().ToList();
  130. if (filters.Count == 0)
  131. {
  132. return items;
  133. }
  134. items = items.AsParallel();
  135. if (filters.Contains(ItemFilter.Dislikes))
  136. {
  137. items = items.Where(i =>
  138. {
  139. var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
  140. return userdata != null && userdata.Likes.HasValue && !userdata.Likes.Value;
  141. });
  142. }
  143. if (filters.Contains(ItemFilter.Likes))
  144. {
  145. items = items.Where(i =>
  146. {
  147. var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
  148. return userdata != null && userdata.Likes.HasValue && userdata.Likes.Value;
  149. });
  150. }
  151. if (filters.Contains(ItemFilter.IsFavoriteOrLikes))
  152. {
  153. items = items.Where(i =>
  154. {
  155. var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
  156. var likes = userdata.Likes ?? false;
  157. var favorite = userdata.IsFavorite;
  158. return likes || favorite;
  159. });
  160. }
  161. if (filters.Contains(ItemFilter.IsFavorite))
  162. {
  163. items = items.Where(i =>
  164. {
  165. var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
  166. return userdata != null && userdata.Likes.HasValue && userdata.IsFavorite;
  167. });
  168. }
  169. return items.AsEnumerable();
  170. }
  171. /// <summary>
  172. /// Filters the items.
  173. /// </summary>
  174. /// <param name="request">The request.</param>
  175. /// <param name="items">The items.</param>
  176. /// <returns>IEnumerable{BaseItem}.</returns>
  177. protected virtual IEnumerable<BaseItem> FilterItems(GetItemsByName request, IEnumerable<BaseItem> items)
  178. {
  179. // Exclude item types
  180. if (!string.IsNullOrEmpty(request.ExcludeItemTypes))
  181. {
  182. var vals = request.ExcludeItemTypes.Split(',');
  183. items = items.Where(f => !vals.Contains(f.GetType().Name, StringComparer.OrdinalIgnoreCase));
  184. }
  185. // Include item types
  186. if (!string.IsNullOrEmpty(request.IncludeItemTypes))
  187. {
  188. var vals = request.IncludeItemTypes.Split(',');
  189. items = items.Where(f => vals.Contains(f.GetType().Name, StringComparer.OrdinalIgnoreCase));
  190. }
  191. // Include MediaTypes
  192. if (!string.IsNullOrEmpty(request.MediaTypes))
  193. {
  194. var vals = request.MediaTypes.Split(',');
  195. items = items.Where(f => vals.Contains(f.MediaType ?? string.Empty, StringComparer.OrdinalIgnoreCase));
  196. }
  197. return items;
  198. }
  199. /// <summary>
  200. /// Gets all items.
  201. /// </summary>
  202. /// <param name="request">The request.</param>
  203. /// <param name="items">The items.</param>
  204. /// <returns>IEnumerable{Task{`0}}.</returns>
  205. protected abstract IEnumerable<Task<TItemType>> GetAllItems(GetItemsByName request, IEnumerable<BaseItem> items);
  206. /// <summary>
  207. /// Gets the dto.
  208. /// </summary>
  209. /// <param name="item">The item.</param>
  210. /// <param name="user">The user.</param>
  211. /// <param name="fields">The fields.</param>
  212. /// <returns>Task{DtoBaseItem}.</returns>
  213. private async Task<BaseItemDto> GetDto(TItemType item, User user, List<ItemFields> fields)
  214. {
  215. var dto = user == null ? await DtoService.GetBaseItemDto(item, fields).ConfigureAwait(false) :
  216. await DtoService.GetBaseItemDto(item, fields, user).ConfigureAwait(false);
  217. return dto;
  218. }
  219. /// <summary>
  220. /// Gets the items.
  221. /// </summary>
  222. /// <param name="userId">The user id.</param>
  223. /// <returns>IEnumerable{BaseItem}.</returns>
  224. protected IEnumerable<BaseItem> GetItems(Guid? userId)
  225. {
  226. if (userId.HasValue)
  227. {
  228. var user = UserManager.GetUserById(userId.Value);
  229. return UserManager.GetUserById(userId.Value).RootFolder.GetRecursiveChildren(user);
  230. }
  231. return LibraryManager.RootFolder.RecursiveChildren;
  232. }
  233. }
  234. /// <summary>
  235. /// Class GetItemsByName
  236. /// </summary>
  237. public class GetItemsByName : BaseItemsRequest, IReturn<ItemsResult>
  238. {
  239. /// <summary>
  240. /// Gets or sets the user id.
  241. /// </summary>
  242. /// <value>The user id.</value>
  243. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  244. public Guid? UserId { get; set; }
  245. [ApiMember(Name = "NameStartsWithOrGreater", Description = "Optional filter by items whose name is sorted equally or greater than a given input string.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  246. public string NameStartsWithOrGreater { get; set; }
  247. public GetItemsByName()
  248. {
  249. Recursive = true;
  250. }
  251. }
  252. }