MoviesService.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using Jellyfin.Data.Entities;
  6. using MediaBrowser.Common.Extensions;
  7. using MediaBrowser.Controller.Configuration;
  8. using MediaBrowser.Controller.Dto;
  9. using MediaBrowser.Controller.Entities;
  10. using MediaBrowser.Controller.Library;
  11. using MediaBrowser.Controller.LiveTv;
  12. using MediaBrowser.Controller.Net;
  13. using MediaBrowser.Model.Dto;
  14. using MediaBrowser.Model.Entities;
  15. using MediaBrowser.Model.Querying;
  16. using MediaBrowser.Model.Services;
  17. using Microsoft.Extensions.Logging;
  18. using MetadataProvider = MediaBrowser.Model.Entities.MetadataProvider;
  19. using Movie = MediaBrowser.Controller.Entities.Movies.Movie;
  20. namespace MediaBrowser.Api.Movies
  21. {
  22. [Route("/Movies/Recommendations", "GET", Summary = "Gets movie recommendations")]
  23. public class GetMovieRecommendations : IReturn<RecommendationDto[]>, IHasDtoOptions
  24. {
  25. [ApiMember(Name = "CategoryLimit", Description = "The max number of categories to return", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  26. public int CategoryLimit { get; set; }
  27. [ApiMember(Name = "ItemLimit", Description = "The max number of items to return per category", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  28. public int ItemLimit { get; set; }
  29. /// <summary>
  30. /// Gets or sets the user id.
  31. /// </summary>
  32. /// <value>The user id.</value>
  33. [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  34. public Guid UserId { get; set; }
  35. /// <summary>
  36. /// Specify this to localize the search to a specific item or folder. Omit to use the root.
  37. /// </summary>
  38. /// <value>The parent id.</value>
  39. [ApiMember(Name = "ParentId", Description = "Specify this to localize the search to a specific item or folder. Omit to use the root", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  40. public string ParentId { get; set; }
  41. [ApiMember(Name = "EnableImages", Description = "Optional, include image information in output", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  42. public bool? EnableImages { get; set; }
  43. [ApiMember(Name = "EnableUserData", Description = "Optional, include user data", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "GET")]
  44. public bool? EnableUserData { get; set; }
  45. [ApiMember(Name = "ImageTypeLimit", Description = "Optional, the max number of images to return, per image type", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "GET")]
  46. public int? ImageTypeLimit { get; set; }
  47. [ApiMember(Name = "EnableImageTypes", Description = "Optional. The image types to include in the output.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  48. public string EnableImageTypes { get; set; }
  49. public GetMovieRecommendations()
  50. {
  51. CategoryLimit = 5;
  52. ItemLimit = 8;
  53. }
  54. public string Fields { get; set; }
  55. }
  56. /// <summary>
  57. /// Class MoviesService
  58. /// </summary>
  59. [Authenticated]
  60. public class MoviesService : BaseApiService
  61. {
  62. /// <summary>
  63. /// The _user manager
  64. /// </summary>
  65. private readonly IUserManager _userManager;
  66. private readonly ILibraryManager _libraryManager;
  67. private readonly IDtoService _dtoService;
  68. private readonly IAuthorizationContext _authContext;
  69. /// <summary>
  70. /// Initializes a new instance of the <see cref="MoviesService" /> class.
  71. /// </summary>
  72. public MoviesService(
  73. ILogger<MoviesService> logger,
  74. IServerConfigurationManager serverConfigurationManager,
  75. IHttpResultFactory httpResultFactory,
  76. IUserManager userManager,
  77. ILibraryManager libraryManager,
  78. IDtoService dtoService,
  79. IAuthorizationContext authContext)
  80. : base(logger, serverConfigurationManager, httpResultFactory)
  81. {
  82. _userManager = userManager;
  83. _libraryManager = libraryManager;
  84. _dtoService = dtoService;
  85. _authContext = authContext;
  86. }
  87. public object Get(GetMovieRecommendations request)
  88. {
  89. var user = _userManager.GetUserById(request.UserId);
  90. var dtoOptions = GetDtoOptions(_authContext, request);
  91. var result = GetRecommendationCategories(user, request.ParentId, request.CategoryLimit, request.ItemLimit, dtoOptions);
  92. return ToOptimizedResult(result);
  93. }
  94. public QueryResult<BaseItemDto> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request)
  95. {
  96. var user = !request.UserId.Equals(Guid.Empty) ? _userManager.GetUserById(request.UserId) : null;
  97. var item = string.IsNullOrEmpty(request.Id) ?
  98. (!request.UserId.Equals(Guid.Empty) ? _libraryManager.GetUserRootFolder() :
  99. _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id);
  100. var itemTypes = new List<string> { typeof(Movie).Name };
  101. if (ServerConfigurationManager.Configuration.EnableExternalContentInSuggestions)
  102. {
  103. itemTypes.Add(typeof(Trailer).Name);
  104. itemTypes.Add(typeof(LiveTvProgram).Name);
  105. }
  106. var dtoOptions = GetDtoOptions(_authContext, request);
  107. var itemsResult = _libraryManager.GetItemList(new InternalItemsQuery(user)
  108. {
  109. Limit = request.Limit,
  110. IncludeItemTypes = itemTypes.ToArray(),
  111. IsMovie = true,
  112. SimilarTo = item,
  113. EnableGroupByMetadataKey = true,
  114. DtoOptions = dtoOptions
  115. });
  116. var returnList = _dtoService.GetBaseItemDtos(itemsResult, dtoOptions, user);
  117. var result = new QueryResult<BaseItemDto>
  118. {
  119. Items = returnList,
  120. TotalRecordCount = itemsResult.Count
  121. };
  122. return result;
  123. }
  124. private IEnumerable<RecommendationDto> GetRecommendationCategories(User user, string parentId, int categoryLimit, int itemLimit, DtoOptions dtoOptions)
  125. {
  126. var categories = new List<RecommendationDto>();
  127. var parentIdGuid = string.IsNullOrWhiteSpace(parentId) ? Guid.Empty : new Guid(parentId);
  128. var query = new InternalItemsQuery(user)
  129. {
  130. IncludeItemTypes = new[]
  131. {
  132. typeof(Movie).Name,
  133. //typeof(Trailer).Name,
  134. //typeof(LiveTvProgram).Name
  135. },
  136. // IsMovie = true
  137. OrderBy = new[] { ItemSortBy.DatePlayed, ItemSortBy.Random }.Select(i => new ValueTuple<string, SortOrder>(i, SortOrder.Descending)).ToArray(),
  138. Limit = 7,
  139. ParentId = parentIdGuid,
  140. Recursive = true,
  141. IsPlayed = true,
  142. DtoOptions = dtoOptions
  143. };
  144. var recentlyPlayedMovies = _libraryManager.GetItemList(query);
  145. var itemTypes = new List<string> { typeof(Movie).Name };
  146. if (ServerConfigurationManager.Configuration.EnableExternalContentInSuggestions)
  147. {
  148. itemTypes.Add(typeof(Trailer).Name);
  149. itemTypes.Add(typeof(LiveTvProgram).Name);
  150. }
  151. var likedMovies = _libraryManager.GetItemList(new InternalItemsQuery(user)
  152. {
  153. IncludeItemTypes = itemTypes.ToArray(),
  154. IsMovie = true,
  155. OrderBy = new[] { ItemSortBy.Random }.Select(i => new ValueTuple<string, SortOrder>(i, SortOrder.Descending)).ToArray(),
  156. Limit = 10,
  157. IsFavoriteOrLiked = true,
  158. ExcludeItemIds = recentlyPlayedMovies.Select(i => i.Id).ToArray(),
  159. EnableGroupByMetadataKey = true,
  160. ParentId = parentIdGuid,
  161. Recursive = true,
  162. DtoOptions = dtoOptions
  163. });
  164. var mostRecentMovies = recentlyPlayedMovies.Take(6).ToList();
  165. // Get recently played directors
  166. var recentDirectors = GetDirectors(mostRecentMovies)
  167. .ToList();
  168. // Get recently played actors
  169. var recentActors = GetActors(mostRecentMovies)
  170. .ToList();
  171. var similarToRecentlyPlayed = GetSimilarTo(user, recentlyPlayedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToRecentlyPlayed).GetEnumerator();
  172. var similarToLiked = GetSimilarTo(user, likedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToLikedItem).GetEnumerator();
  173. var hasDirectorFromRecentlyPlayed = GetWithDirector(user, recentDirectors, itemLimit, dtoOptions, RecommendationType.HasDirectorFromRecentlyPlayed).GetEnumerator();
  174. var hasActorFromRecentlyPlayed = GetWithActor(user, recentActors, itemLimit, dtoOptions, RecommendationType.HasActorFromRecentlyPlayed).GetEnumerator();
  175. var categoryTypes = new List<IEnumerator<RecommendationDto>>
  176. {
  177. // Give this extra weight
  178. similarToRecentlyPlayed,
  179. similarToRecentlyPlayed,
  180. // Give this extra weight
  181. similarToLiked,
  182. similarToLiked,
  183. hasDirectorFromRecentlyPlayed,
  184. hasActorFromRecentlyPlayed
  185. };
  186. while (categories.Count < categoryLimit)
  187. {
  188. var allEmpty = true;
  189. foreach (var category in categoryTypes)
  190. {
  191. if (category.MoveNext())
  192. {
  193. categories.Add(category.Current);
  194. allEmpty = false;
  195. if (categories.Count >= categoryLimit)
  196. {
  197. break;
  198. }
  199. }
  200. }
  201. if (allEmpty)
  202. {
  203. break;
  204. }
  205. }
  206. return categories.OrderBy(i => i.RecommendationType);
  207. }
  208. private IEnumerable<RecommendationDto> GetWithDirector(
  209. User user,
  210. IEnumerable<string> names,
  211. int itemLimit,
  212. DtoOptions dtoOptions,
  213. RecommendationType type)
  214. {
  215. var itemTypes = new List<string> { typeof(Movie).Name };
  216. if (ServerConfigurationManager.Configuration.EnableExternalContentInSuggestions)
  217. {
  218. itemTypes.Add(typeof(Trailer).Name);
  219. itemTypes.Add(typeof(LiveTvProgram).Name);
  220. }
  221. foreach (var name in names)
  222. {
  223. var items = _libraryManager.GetItemList(new InternalItemsQuery(user)
  224. {
  225. Person = name,
  226. // Account for duplicates by imdb id, since the database doesn't support this yet
  227. Limit = itemLimit + 2,
  228. PersonTypes = new[] { PersonType.Director },
  229. IncludeItemTypes = itemTypes.ToArray(),
  230. IsMovie = true,
  231. EnableGroupByMetadataKey = true,
  232. DtoOptions = dtoOptions
  233. }).GroupBy(i => i.GetProviderId(MetadataProvider.Imdb) ?? Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture))
  234. .Select(x => x.First())
  235. .Take(itemLimit)
  236. .ToList();
  237. if (items.Count > 0)
  238. {
  239. var returnItems = _dtoService.GetBaseItemDtos(items, dtoOptions, user);
  240. yield return new RecommendationDto
  241. {
  242. BaselineItemName = name,
  243. CategoryId = name.GetMD5(),
  244. RecommendationType = type,
  245. Items = returnItems
  246. };
  247. }
  248. }
  249. }
  250. private IEnumerable<RecommendationDto> GetWithActor(User user, IEnumerable<string> names, int itemLimit, DtoOptions dtoOptions, RecommendationType type)
  251. {
  252. var itemTypes = new List<string> { typeof(Movie).Name };
  253. if (ServerConfigurationManager.Configuration.EnableExternalContentInSuggestions)
  254. {
  255. itemTypes.Add(typeof(Trailer).Name);
  256. itemTypes.Add(typeof(LiveTvProgram).Name);
  257. }
  258. foreach (var name in names)
  259. {
  260. var items = _libraryManager.GetItemList(new InternalItemsQuery(user)
  261. {
  262. Person = name,
  263. // Account for duplicates by imdb id, since the database doesn't support this yet
  264. Limit = itemLimit + 2,
  265. IncludeItemTypes = itemTypes.ToArray(),
  266. IsMovie = true,
  267. EnableGroupByMetadataKey = true,
  268. DtoOptions = dtoOptions
  269. }).GroupBy(i => i.GetProviderId(MetadataProvider.Imdb) ?? Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture))
  270. .Select(x => x.First())
  271. .Take(itemLimit)
  272. .ToList();
  273. if (items.Count > 0)
  274. {
  275. var returnItems = _dtoService.GetBaseItemDtos(items, dtoOptions, user);
  276. yield return new RecommendationDto
  277. {
  278. BaselineItemName = name,
  279. CategoryId = name.GetMD5(),
  280. RecommendationType = type,
  281. Items = returnItems
  282. };
  283. }
  284. }
  285. }
  286. private IEnumerable<RecommendationDto> GetSimilarTo(User user, List<BaseItem> baselineItems, int itemLimit, DtoOptions dtoOptions, RecommendationType type)
  287. {
  288. var itemTypes = new List<string> { typeof(Movie).Name };
  289. if (ServerConfigurationManager.Configuration.EnableExternalContentInSuggestions)
  290. {
  291. itemTypes.Add(typeof(Trailer).Name);
  292. itemTypes.Add(typeof(LiveTvProgram).Name);
  293. }
  294. foreach (var item in baselineItems)
  295. {
  296. var similar = _libraryManager.GetItemList(new InternalItemsQuery(user)
  297. {
  298. Limit = itemLimit,
  299. IncludeItemTypes = itemTypes.ToArray(),
  300. IsMovie = true,
  301. SimilarTo = item,
  302. EnableGroupByMetadataKey = true,
  303. DtoOptions = dtoOptions
  304. });
  305. if (similar.Count > 0)
  306. {
  307. var returnItems = _dtoService.GetBaseItemDtos(similar, dtoOptions, user);
  308. yield return new RecommendationDto
  309. {
  310. BaselineItemName = item.Name,
  311. CategoryId = item.Id,
  312. RecommendationType = type,
  313. Items = returnItems
  314. };
  315. }
  316. }
  317. }
  318. private IEnumerable<string> GetActors(List<BaseItem> items)
  319. {
  320. var people = _libraryManager.GetPeople(new InternalPeopleQuery
  321. {
  322. ExcludePersonTypes = new[]
  323. {
  324. PersonType.Director
  325. },
  326. MaxListOrder = 3
  327. });
  328. var itemIds = items.Select(i => i.Id).ToList();
  329. return people
  330. .Where(i => itemIds.Contains(i.ItemId))
  331. .Select(i => i.Name)
  332. .DistinctNames();
  333. }
  334. private IEnumerable<string> GetDirectors(List<BaseItem> items)
  335. {
  336. var people = _libraryManager.GetPeople(new InternalPeopleQuery
  337. {
  338. PersonTypes = new[]
  339. {
  340. PersonType.Director
  341. }
  342. });
  343. var itemIds = items.Select(i => i.Id).ToList();
  344. return people
  345. .Where(i => itemIds.Contains(i.ItemId))
  346. .Select(i => i.Name)
  347. .DistinctNames();
  348. }
  349. }
  350. }