PersonsService.cs 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. using MediaBrowser.Controller.Dto;
  2. using MediaBrowser.Controller.Entities;
  3. using MediaBrowser.Controller.Entities.Audio;
  4. using MediaBrowser.Controller.Entities.Movies;
  5. using MediaBrowser.Controller.Entities.TV;
  6. using MediaBrowser.Controller.Library;
  7. using MediaBrowser.Controller.Persistence;
  8. using MediaBrowser.Model.Dto;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Querying;
  11. using ServiceStack.ServiceHost;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Linq;
  15. using System.Threading.Tasks;
  16. namespace MediaBrowser.Api.UserLibrary
  17. {
  18. /// <summary>
  19. /// Class GetPersons
  20. /// </summary>
  21. [Route("/Persons", "GET")]
  22. [Api(Description = "Gets all persons from a given item, folder, or the entire library")]
  23. public class GetPersons : GetItemsByName
  24. {
  25. /// <summary>
  26. /// Gets or sets the person types.
  27. /// </summary>
  28. /// <value>The person types.</value>
  29. public string PersonTypes { get; set; }
  30. }
  31. /// <summary>
  32. /// Class GetPersonItemCounts
  33. /// </summary>
  34. [Route("/Persons/{Name}/Counts", "GET")]
  35. [Api(Description = "Gets item counts of library items that a person appears in")]
  36. public class GetPersonItemCounts : IReturn<ItemByNameCounts>
  37. {
  38. /// <summary>
  39. /// Gets or sets the user id.
  40. /// </summary>
  41. /// <value>The user id.</value>
  42. [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  43. public Guid UserId { get; set; }
  44. /// <summary>
  45. /// Gets or sets the name.
  46. /// </summary>
  47. /// <value>The name.</value>
  48. [ApiMember(Name = "Name", Description = "The person name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  49. public string Name { get; set; }
  50. }
  51. /// <summary>
  52. /// Class GetPerson
  53. /// </summary>
  54. [Route("/Persons/{Name}", "GET")]
  55. [Api(Description = "Gets a person, by name")]
  56. public class GetPerson : IReturn<BaseItemDto>
  57. {
  58. /// <summary>
  59. /// Gets or sets the name.
  60. /// </summary>
  61. /// <value>The name.</value>
  62. [ApiMember(Name = "Name", Description = "The person name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  63. public string Name { get; set; }
  64. /// <summary>
  65. /// Gets or sets the user id.
  66. /// </summary>
  67. /// <value>The user id.</value>
  68. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  69. public Guid? UserId { get; set; }
  70. }
  71. /// <summary>
  72. /// Class PersonsService
  73. /// </summary>
  74. public class PersonsService : BaseItemsByNameService<Person>
  75. {
  76. /// <summary>
  77. /// Initializes a new instance of the <see cref="PersonsService" /> class.
  78. /// </summary>
  79. /// <param name="userManager">The user manager.</param>
  80. /// <param name="libraryManager">The library manager.</param>
  81. /// <param name="userDataRepository">The user data repository.</param>
  82. /// <param name="itemRepo">The item repo.</param>
  83. public PersonsService(IUserManager userManager, ILibraryManager libraryManager, IUserDataRepository userDataRepository, IItemRepository itemRepo)
  84. : base(userManager, libraryManager, userDataRepository, itemRepo)
  85. {
  86. }
  87. /// <summary>
  88. /// Gets the specified request.
  89. /// </summary>
  90. /// <param name="request">The request.</param>
  91. /// <returns>System.Object.</returns>
  92. public object Get(GetPerson request)
  93. {
  94. var result = GetItem(request).Result;
  95. return ToOptimizedResult(result);
  96. }
  97. /// <summary>
  98. /// Gets the item.
  99. /// </summary>
  100. /// <param name="request">The request.</param>
  101. /// <returns>Task{BaseItemDto}.</returns>
  102. private async Task<BaseItemDto> GetItem(GetPerson request)
  103. {
  104. var item = await GetPerson(request.Name, LibraryManager).ConfigureAwait(false);
  105. // Get everything
  106. var fields = Enum.GetNames(typeof(ItemFields)).Select(i => (ItemFields)Enum.Parse(typeof(ItemFields), i, true));
  107. var builder = new DtoBuilder(Logger, LibraryManager, UserDataRepository, ItemRepository);
  108. if (request.UserId.HasValue)
  109. {
  110. var user = UserManager.GetUserById(request.UserId.Value);
  111. return await builder.GetBaseItemDto(item, fields.ToList(), user).ConfigureAwait(false);
  112. }
  113. return await builder.GetBaseItemDto(item, fields.ToList()).ConfigureAwait(false);
  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(GetPersons request)
  121. {
  122. var result = GetResult(request).Result;
  123. return ToOptimizedResult(result);
  124. }
  125. /// <summary>
  126. /// Gets the specified request.
  127. /// </summary>
  128. /// <param name="request">The request.</param>
  129. /// <returns>System.Object.</returns>
  130. public object Get(GetPersonItemCounts request)
  131. {
  132. var name = DeSlugPersonName(request.Name, LibraryManager);
  133. var items = GetItems(request.UserId).Where(i => i.People != null && i.People.Any(p => string.Equals(p.Name, name, StringComparison.OrdinalIgnoreCase))).ToList();
  134. var counts = new ItemByNameCounts
  135. {
  136. TotalCount = items.Count,
  137. TrailerCount = items.OfType<Trailer>().Count(),
  138. MovieCount = items.OfType<Movie>().Count(),
  139. SeriesCount = items.OfType<Series>().Count(),
  140. GameCount = items.OfType<Game>().Count(),
  141. SongCount = items.OfType<Audio>().Count(),
  142. AlbumCount = items.OfType<MusicAlbum>().Count(),
  143. EpisodeCount = items.OfType<Episode>().Count(),
  144. MusicVideoCount = items.OfType<MusicVideo>().Count(),
  145. AdultVideoCount = items.OfType<AdultVideo>().Count()
  146. };
  147. return ToOptimizedResult(counts);
  148. }
  149. /// <summary>
  150. /// Gets all items.
  151. /// </summary>
  152. /// <param name="request">The request.</param>
  153. /// <param name="items">The items.</param>
  154. /// <returns>IEnumerable{Tuple{System.StringFunc{System.Int32}}}.</returns>
  155. protected override IEnumerable<IbnStub<Person>> GetAllItems(GetItemsByName request, IEnumerable<BaseItem> items)
  156. {
  157. var inputPersonTypes = ((GetPersons)request).PersonTypes;
  158. var personTypes = string.IsNullOrEmpty(inputPersonTypes) ? new string[] { } : inputPersonTypes.Split(',');
  159. var itemsList = items.ToList();
  160. // Either get all people, or all people filtered by a specific person type
  161. var allPeople = GetAllPeople(itemsList, personTypes);
  162. return allPeople
  163. .Select(i => i.Name)
  164. .Distinct(StringComparer.OrdinalIgnoreCase)
  165. .Select(name => new IbnStub<Person>(name, () =>
  166. {
  167. if (personTypes.Length == 0)
  168. {
  169. return itemsList.Where(i => i.People.Any(p => p.Name.Equals(name, StringComparison.OrdinalIgnoreCase)));
  170. }
  171. return itemsList.Where(i => i.People.Any(p => p.Name.Equals(name, StringComparison.OrdinalIgnoreCase) && (personTypes.Contains(p.Type ?? string.Empty, StringComparer.OrdinalIgnoreCase) || personTypes.Contains(p.Role ?? string.Empty, StringComparer.OrdinalIgnoreCase))));
  172. }, GetEntity)
  173. );
  174. }
  175. /// <summary>
  176. /// Gets all people.
  177. /// </summary>
  178. /// <param name="itemsList">The items list.</param>
  179. /// <param name="personTypes">The person types.</param>
  180. /// <returns>IEnumerable{PersonInfo}.</returns>
  181. private IEnumerable<PersonInfo> GetAllPeople(IEnumerable<BaseItem> itemsList, string[] personTypes)
  182. {
  183. var people = itemsList.SelectMany(i => i.People.OrderBy(p => p.Type));
  184. return personTypes.Length == 0 ?
  185. people :
  186. people.Where(p => personTypes.Contains(p.Type ?? string.Empty, StringComparer.OrdinalIgnoreCase) || personTypes.Contains(p.Role ?? string.Empty, StringComparer.OrdinalIgnoreCase));
  187. }
  188. /// <summary>
  189. /// Gets the entity.
  190. /// </summary>
  191. /// <param name="name">The name.</param>
  192. /// <returns>Task{Genre}.</returns>
  193. protected Task<Person> GetEntity(string name)
  194. {
  195. return LibraryManager.GetPerson(name);
  196. }
  197. }
  198. }