| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327 | using System;using System.Collections.Generic;using System.Globalization;using System.Linq;using Jellyfin.Api.Extensions;using Jellyfin.Api.Helpers;using Jellyfin.Api.ModelBinders;using Jellyfin.Data.Enums;using Jellyfin.Database.Implementations.Entities;using Jellyfin.Database.Implementations.Enums;using Jellyfin.Extensions;using MediaBrowser.Common.Extensions;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Dto;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Library;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Querying;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Mvc;namespace Jellyfin.Api.Controllers;/// <summary>/// Movies controller./// </summary>[Authorize]public class MoviesController : BaseJellyfinApiController{    private readonly IUserManager _userManager;    private readonly ILibraryManager _libraryManager;    private readonly IDtoService _dtoService;    private readonly IServerConfigurationManager _serverConfigurationManager;    /// <summary>    /// Initializes a new instance of the <see cref="MoviesController"/> class.    /// </summary>    /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>    /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>    /// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param>    /// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>    public MoviesController(        IUserManager userManager,        ILibraryManager libraryManager,        IDtoService dtoService,        IServerConfigurationManager serverConfigurationManager)    {        _userManager = userManager;        _libraryManager = libraryManager;        _dtoService = dtoService;        _serverConfigurationManager = serverConfigurationManager;    }    /// <summary>    /// Gets movie recommendations.    /// </summary>    /// <param name="userId">Optional. Filter by user id, and attach user data.</param>    /// <param name="parentId">Specify this to localize the search to a specific item or folder. Omit to use the root.</param>    /// <param name="fields">Optional. The fields to return.</param>    /// <param name="categoryLimit">The max number of categories to return.</param>    /// <param name="itemLimit">The max number of items to return per category.</param>    /// <response code="200">Movie recommendations returned.</response>    /// <returns>The list of movie recommendations.</returns>    [HttpGet("Recommendations")]    public ActionResult<IEnumerable<RecommendationDto>> GetMovieRecommendations(        [FromQuery] Guid? userId,        [FromQuery] Guid? parentId,        [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] ItemFields[] fields,        [FromQuery] int categoryLimit = 5,        [FromQuery] int itemLimit = 8)    {        userId = RequestHelpers.GetUserId(User, userId);        var user = userId.IsNullOrEmpty()            ? null            : _userManager.GetUserById(userId.Value);        var dtoOptions = new DtoOptions { Fields = fields }            .AddClientFields(User);        var categories = new List<RecommendationDto>();        var parentIdGuid = parentId ?? Guid.Empty;        var query = new InternalItemsQuery(user)        {            IncludeItemTypes = new[]            {                BaseItemKind.Movie,                // nameof(Trailer),                // nameof(LiveTvProgram)            },            // IsMovie = true            OrderBy = new[] { (ItemSortBy.DatePlayed, SortOrder.Descending), (ItemSortBy.Random, SortOrder.Descending) },            Limit = 7,            ParentId = parentIdGuid,            Recursive = true,            IsPlayed = true,            DtoOptions = dtoOptions        };        var recentlyPlayedMovies = _libraryManager.GetItemList(query);        var itemTypes = new List<BaseItemKind> { BaseItemKind.Movie };        if (_serverConfigurationManager.Configuration.EnableExternalContentInSuggestions)        {            itemTypes.Add(BaseItemKind.Trailer);            itemTypes.Add(BaseItemKind.LiveTvProgram);        }        var likedMovies = _libraryManager.GetItemList(new InternalItemsQuery(user)        {            IncludeItemTypes = itemTypes.ToArray(),            IsMovie = true,            OrderBy = new[] { (ItemSortBy.Random, SortOrder.Descending) },            Limit = 10,            IsFavoriteOrLiked = true,            ExcludeItemIds = recentlyPlayedMovies.Select(i => i.Id).ToArray(),            EnableGroupByMetadataKey = true,            ParentId = parentIdGuid,            Recursive = true,            DtoOptions = dtoOptions        });        var mostRecentMovies = recentlyPlayedMovies.Take(Math.Min(recentlyPlayedMovies.Count, 6)).ToList();        // Get recently played directors        var recentDirectors = GetDirectors(mostRecentMovies)            .ToList();        // Get recently played actors        var recentActors = GetActors(mostRecentMovies)            .ToList();        var similarToRecentlyPlayed = GetSimilarTo(user, recentlyPlayedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToRecentlyPlayed).GetEnumerator();        var similarToLiked = GetSimilarTo(user, likedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToLikedItem).GetEnumerator();        var hasDirectorFromRecentlyPlayed = GetWithDirector(user, recentDirectors, itemLimit, dtoOptions, RecommendationType.HasDirectorFromRecentlyPlayed).GetEnumerator();        var hasActorFromRecentlyPlayed = GetWithActor(user, recentActors, itemLimit, dtoOptions, RecommendationType.HasActorFromRecentlyPlayed).GetEnumerator();        var categoryTypes = new List<IEnumerator<RecommendationDto>>            {                // Give this extra weight                similarToRecentlyPlayed,                similarToRecentlyPlayed,                // Give this extra weight                similarToLiked,                similarToLiked,                hasDirectorFromRecentlyPlayed,                hasActorFromRecentlyPlayed            };        while (categories.Count < categoryLimit)        {            var allEmpty = true;            foreach (var category in categoryTypes)            {                if (category.MoveNext())                {                    categories.Add(category.Current);                    allEmpty = false;                    if (categories.Count >= categoryLimit)                    {                        break;                    }                }            }            if (allEmpty)            {                break;            }        }        return Ok(categories.OrderBy(i => i.RecommendationType).AsEnumerable());    }    private IEnumerable<RecommendationDto> GetWithDirector(        User? user,        IEnumerable<string> names,        int itemLimit,        DtoOptions dtoOptions,        RecommendationType type)    {        var itemTypes = new List<BaseItemKind> { BaseItemKind.Movie };        if (_serverConfigurationManager.Configuration.EnableExternalContentInSuggestions)        {            itemTypes.Add(BaseItemKind.Trailer);            itemTypes.Add(BaseItemKind.LiveTvProgram);        }        foreach (var name in names)        {            var items = _libraryManager.GetItemList(                new InternalItemsQuery(user)                {                    Person = name,                    // Account for duplicates by IMDb id, since the database doesn't support this yet                    Limit = itemLimit + 2,                    PersonTypes = new[] { PersonType.Director },                    IncludeItemTypes = itemTypes.ToArray(),                    IsMovie = true,                    EnableGroupByMetadataKey = true,                    DtoOptions = dtoOptions                }).DistinctBy(i => i.GetProviderId(MediaBrowser.Model.Entities.MetadataProvider.Imdb) ?? Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture))                .Take(itemLimit)                .ToList();            if (items.Count > 0)            {                var returnItems = _dtoService.GetBaseItemDtos(items, dtoOptions, user);                yield return new RecommendationDto                {                    BaselineItemName = name,                    CategoryId = name.GetMD5(),                    RecommendationType = type,                    Items = returnItems                };            }        }    }    private IEnumerable<RecommendationDto> GetWithActor(User? user, IEnumerable<string> names, int itemLimit, DtoOptions dtoOptions, RecommendationType type)    {        var itemTypes = new List<BaseItemKind> { BaseItemKind.Movie };        if (_serverConfigurationManager.Configuration.EnableExternalContentInSuggestions)        {            itemTypes.Add(BaseItemKind.Trailer);            itemTypes.Add(BaseItemKind.LiveTvProgram);        }        foreach (var name in names)        {            var items = _libraryManager.GetItemList(new InternalItemsQuery(user)            {                Person = name,                // Account for duplicates by IMDb id, since the database doesn't support this yet                Limit = itemLimit + 2,                IncludeItemTypes = itemTypes.ToArray(),                IsMovie = true,                EnableGroupByMetadataKey = true,                DtoOptions = dtoOptions            }).DistinctBy(i => i.GetProviderId(MediaBrowser.Model.Entities.MetadataProvider.Imdb) ?? Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture))                .Take(itemLimit)                .ToList();            if (items.Count > 0)            {                var returnItems = _dtoService.GetBaseItemDtos(items, dtoOptions, user);                yield return new RecommendationDto                {                    BaselineItemName = name,                    CategoryId = name.GetMD5(),                    RecommendationType = type,                    Items = returnItems                };            }        }    }    private IEnumerable<RecommendationDto> GetSimilarTo(User? user, IEnumerable<BaseItem> baselineItems, int itemLimit, DtoOptions dtoOptions, RecommendationType type)    {        var itemTypes = new List<BaseItemKind> { BaseItemKind.Movie };        if (_serverConfigurationManager.Configuration.EnableExternalContentInSuggestions)        {            itemTypes.Add(BaseItemKind.Trailer);            itemTypes.Add(BaseItemKind.LiveTvProgram);        }        foreach (var item in baselineItems)        {            var similar = _libraryManager.GetItemList(new InternalItemsQuery(user)            {                Limit = itemLimit,                IncludeItemTypes = itemTypes.ToArray(),                IsMovie = true,                EnableGroupByMetadataKey = true,                DtoOptions = dtoOptions            });            if (similar.Count > 0)            {                var returnItems = _dtoService.GetBaseItemDtos(similar, dtoOptions, user);                yield return new RecommendationDto                {                    BaselineItemName = item.Name,                    CategoryId = item.Id,                    RecommendationType = type,                    Items = returnItems                };            }        }    }    private IEnumerable<string> GetActors(IEnumerable<BaseItem> items)    {        var people = _libraryManager.GetPeople(new InternalPeopleQuery(Array.Empty<string>(), new[] { PersonType.Director })        {            MaxListOrder = 3        });        var itemIds = items.Select(i => i.Id).ToList();        return people            .Where(i => itemIds.Contains(i.ItemId))            .Select(i => i.Name)            .DistinctNames();    }    private IEnumerable<string> GetDirectors(IEnumerable<BaseItem> items)    {        var people = _libraryManager.GetPeople(new InternalPeopleQuery(            new[] { PersonType.Director },            Array.Empty<string>()));        var itemIds = items.Select(i => i.Id).ToList();        return people            .Where(i => itemIds.Contains(i.ItemId))            .Select(i => i.Name)            .DistinctNames();    }}
 |