BaseItemsByNameService.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. using MediaBrowser.Controller.Entities;
  2. using MediaBrowser.Controller.Library;
  3. using MediaBrowser.Model.Dto;
  4. using MediaBrowser.Model.Entities;
  5. using MediaBrowser.Server.Implementations.HttpServer;
  6. using ServiceStack.ServiceHost;
  7. using System;
  8. using System.Collections.Generic;
  9. using System.IO;
  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> : BaseRestService
  19. where TItemType : BaseItem
  20. {
  21. /// <summary>
  22. /// The _user manager
  23. /// </summary>
  24. protected readonly IUserManager UserManager;
  25. protected readonly ILibraryManager LibraryManager;
  26. /// <summary>
  27. /// Initializes a new instance of the <see cref="BaseItemsByNameService{TItemType}" /> class.
  28. /// </summary>
  29. /// <param name="userManager">The user manager.</param>
  30. protected BaseItemsByNameService(IUserManager userManager, ILibraryManager libraryManager)
  31. {
  32. UserManager = userManager;
  33. LibraryManager = libraryManager;
  34. }
  35. /// <summary>
  36. /// Gets the specified request.
  37. /// </summary>
  38. /// <param name="request">The request.</param>
  39. /// <returns>Task{ItemsResult}.</returns>
  40. protected async Task<ItemsResult> GetResult(GetItemsByName request)
  41. {
  42. var user = UserManager.GetUserById(request.UserId);
  43. var item = string.IsNullOrEmpty(request.Id) ? user.RootFolder : DtoBuilder.GetItemByClientId(request.Id, UserManager, LibraryManager, user.Id);
  44. IEnumerable<BaseItem> items;
  45. if (item.IsFolder)
  46. {
  47. var folder = (Folder)item;
  48. items = request.Recursive ? folder.GetRecursiveChildren(user) : folder.GetChildren(user);
  49. }
  50. else
  51. {
  52. items = new[] { item };
  53. }
  54. var ibnItemsArray = GetAllItems(request, items, user).ToArray();
  55. IEnumerable<Tuple<string, Func<int>>> ibnItems = ibnItemsArray;
  56. var result = new ItemsResult
  57. {
  58. TotalRecordCount = ibnItemsArray.Length
  59. };
  60. if (request.StartIndex.HasValue || request.Limit.HasValue)
  61. {
  62. if (request.StartIndex.HasValue)
  63. {
  64. ibnItems = ibnItems.Skip(request.StartIndex.Value);
  65. }
  66. if (request.Limit.HasValue)
  67. {
  68. ibnItems = ibnItems.Take(request.Limit.Value);
  69. }
  70. }
  71. var fields = GetItemFields(request).ToList();
  72. var tasks = ibnItems.Select(i => GetDto(i, user, fields));
  73. var resultItems = await Task.WhenAll(tasks).ConfigureAwait(false);
  74. result.Items = resultItems.Where(i => i != null).ToArray();
  75. return result;
  76. }
  77. /// <summary>
  78. /// Gets the item fields.
  79. /// </summary>
  80. /// <param name="request">The request.</param>
  81. /// <returns>IEnumerable{ItemFields}.</returns>
  82. private IEnumerable<ItemFields> GetItemFields(GetItemsByName request)
  83. {
  84. var val = request.Fields;
  85. if (string.IsNullOrEmpty(val))
  86. {
  87. return new ItemFields[] { };
  88. }
  89. return val.Split(',').Select(v => (ItemFields)Enum.Parse(typeof(ItemFields), v, true));
  90. }
  91. /// <summary>
  92. /// Gets all items.
  93. /// </summary>
  94. /// <param name="request">The request.</param>
  95. /// <param name="items">The items.</param>
  96. /// <param name="user">The user.</param>
  97. /// <returns>IEnumerable{Tuple{System.StringFunc{System.Int32}}}.</returns>
  98. protected abstract IEnumerable<Tuple<string, Func<int>>> GetAllItems(GetItemsByName request, IEnumerable<BaseItem> items, User user);
  99. /// <summary>
  100. /// Gets the entity.
  101. /// </summary>
  102. /// <param name="name">The name.</param>
  103. /// <returns>Task{BaseItem}.</returns>
  104. protected abstract Task<TItemType> GetEntity(string name);
  105. /// <summary>
  106. /// Gets the dto.
  107. /// </summary>
  108. /// <param name="stub">The stub.</param>
  109. /// <param name="user">The user.</param>
  110. /// <param name="fields">The fields.</param>
  111. /// <returns>Task{DtoBaseItem}.</returns>
  112. private async Task<BaseItemDto> GetDto(Tuple<string, Func<int>> stub, User user, List<ItemFields> fields)
  113. {
  114. BaseItem item;
  115. try
  116. {
  117. item = await GetEntity(stub.Item1).ConfigureAwait(false);
  118. }
  119. catch (IOException ex)
  120. {
  121. Logger.ErrorException("Error getting IBN item {0}", ex, stub.Item1);
  122. return null;
  123. }
  124. var dto = await new DtoBuilder(Logger).GetBaseItemDto(item, user, fields, LibraryManager).ConfigureAwait(false);
  125. dto.ChildCount = stub.Item2();
  126. return dto;
  127. }
  128. }
  129. /// <summary>
  130. /// Class GetItemsByName
  131. /// </summary>
  132. public class GetItemsByName : IReturn<ItemsResult>
  133. {
  134. /// <summary>
  135. /// Gets or sets the user id.
  136. /// </summary>
  137. /// <value>The user id.</value>
  138. [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  139. public Guid UserId { get; set; }
  140. /// <summary>
  141. /// Gets or sets the start index.
  142. /// </summary>
  143. /// <value>The start index.</value>
  144. [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")]
  145. public int? StartIndex { get; set; }
  146. /// <summary>
  147. /// Gets or sets the size of the page.
  148. /// </summary>
  149. /// <value>The size of the page.</value>
  150. [ApiMember(Name = "Limit", Description = "Optional. The maximum number of records to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  151. public int? Limit { get; set; }
  152. /// <summary>
  153. /// Gets or sets a value indicating whether this <see cref="GetItemsByName" /> is recursive.
  154. /// </summary>
  155. /// <value><c>true</c> if recursive; otherwise, <c>false</c>.</value>
  156. [ApiMember(Name = "Recursive", Description = "When searching within folders, this determines whether or not the search will be recursive. true/false", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  157. public bool Recursive { get; set; }
  158. /// <summary>
  159. /// Gets or sets the sort order.
  160. /// </summary>
  161. /// <value>The sort order.</value>
  162. [ApiMember(Name = "SortOrder", Description = "Sort Order - Ascending,Descending", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  163. public SortOrder? SortOrder { get; set; }
  164. /// <summary>
  165. /// If specified the search will be localized within a specific item or folder
  166. /// </summary>
  167. /// <value>The item id.</value>
  168. [ApiMember(Name = "Id", Description = "If specified the search will be localized within a specific item or folder", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  169. public string Id { get; set; }
  170. /// <summary>
  171. /// Fields to return within the items, in addition to basic information
  172. /// </summary>
  173. /// <value>The fields.</value>
  174. public string Fields { get; set; }
  175. }
  176. }