| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833 | using System;using System.ComponentModel.DataAnnotations;using System.Linq;using Jellyfin.Api.Constants;using Jellyfin.Api.Extensions;using Jellyfin.Api.Helpers;using Jellyfin.Api.ModelBinders;using Jellyfin.Data.Enums;using MediaBrowser.Controller.Dto;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Library;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Globalization;using MediaBrowser.Model.Querying;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;using Microsoft.Extensions.Logging;namespace Jellyfin.Api.Controllers{    /// <summary>    /// The items controller.    /// </summary>    [Route("")]    [Authorize(Policy = Policies.DefaultAuthorization)]    public class ItemsController : BaseJellyfinApiController    {        private readonly IUserManager _userManager;        private readonly ILibraryManager _libraryManager;        private readonly ILocalizationManager _localization;        private readonly IDtoService _dtoService;        private readonly ILogger<ItemsController> _logger;        /// <summary>        /// Initializes a new instance of the <see cref="ItemsController"/> 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="localization">Instance of the <see cref="ILocalizationManager"/> interface.</param>        /// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param>        /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param>        public ItemsController(            IUserManager userManager,            ILibraryManager libraryManager,            ILocalizationManager localization,            IDtoService dtoService,            ILogger<ItemsController> logger)        {            _userManager = userManager;            _libraryManager = libraryManager;            _localization = localization;            _dtoService = dtoService;            _logger = logger;        }        /// <summary>        /// Gets items based on a query.        /// </summary>        /// <param name="userId">The user id supplied as query parameter.</param>        /// <param name="maxOfficialRating">Optional filter by maximum official rating (PG, PG-13, TV-MA, etc).</param>        /// <param name="hasThemeSong">Optional filter by items with theme songs.</param>        /// <param name="hasThemeVideo">Optional filter by items with theme videos.</param>        /// <param name="hasSubtitles">Optional filter by items with subtitles.</param>        /// <param name="hasSpecialFeature">Optional filter by items with special features.</param>        /// <param name="hasTrailer">Optional filter by items with trailers.</param>        /// <param name="adjacentTo">Optional. Return items that are siblings of a supplied item.</param>        /// <param name="parentIndexNumber">Optional filter by parent index number.</param>        /// <param name="hasParentalRating">Optional filter by items that have or do not have a parental rating.</param>        /// <param name="isHd">Optional filter by items that are HD or not.</param>        /// <param name="is4K">Optional filter by items that are 4K or not.</param>        /// <param name="locationTypes">Optional. If specified, results will be filtered based on LocationType. This allows multiple, comma delimited.</param>        /// <param name="excludeLocationTypes">Optional. If specified, results will be filtered based on the LocationType. This allows multiple, comma delimited.</param>        /// <param name="isMissing">Optional filter by items that are missing episodes or not.</param>        /// <param name="isUnaired">Optional filter by items that are unaired episodes or not.</param>        /// <param name="minCommunityRating">Optional filter by minimum community rating.</param>        /// <param name="minCriticRating">Optional filter by minimum critic rating.</param>        /// <param name="minPremiereDate">Optional. The minimum premiere date. Format = ISO.</param>        /// <param name="minDateLastSaved">Optional. The minimum last saved date. Format = ISO.</param>        /// <param name="minDateLastSavedForUser">Optional. The minimum last saved date for the current user. Format = ISO.</param>        /// <param name="maxPremiereDate">Optional. The maximum premiere date. Format = ISO.</param>        /// <param name="hasOverview">Optional filter by items that have an overview or not.</param>        /// <param name="hasImdbId">Optional filter by items that have an imdb id or not.</param>        /// <param name="hasTmdbId">Optional filter by items that have a tmdb id or not.</param>        /// <param name="hasTvdbId">Optional filter by items that have a tvdb id or not.</param>        /// <param name="excludeItemIds">Optional. If specified, results will be filtered by excluding item ids. This allows multiple, comma delimited.</param>        /// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>        /// <param name="limit">Optional. The maximum number of records to return.</param>        /// <param name="recursive">When searching within folders, this determines whether or not the search will be recursive. true/false.</param>        /// <param name="searchTerm">Optional. Filter based on a search term.</param>        /// <param name="sortOrder">Sort Order - Ascending,Descending.</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. Specify additional fields of information to return in the output. This allows multiple, comma delimited. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines.</param>        /// <param name="excludeItemTypes">Optional. If specified, results will be filtered based on item type. This allows multiple, comma delimited.</param>        /// <param name="includeItemTypes">Optional. If specified, results will be filtered based on the item type. This allows multiple, comma delimited.</param>        /// <param name="filters">Optional. Specify additional filters to apply. This allows multiple, comma delimited. Options: IsFolder, IsNotFolder, IsUnplayed, IsPlayed, IsFavorite, IsResumable, Likes, Dislikes.</param>        /// <param name="isFavorite">Optional filter by items that are marked as favorite, or not.</param>        /// <param name="mediaTypes">Optional filter by MediaType. Allows multiple, comma delimited.</param>        /// <param name="imageTypes">Optional. If specified, results will be filtered based on those containing image types. This allows multiple, comma delimited.</param>        /// <param name="sortBy">Optional. Specify one or more sort orders, comma delimited. Options: Album, AlbumArtist, Artist, Budget, CommunityRating, CriticRating, DateCreated, DatePlayed, PlayCount, PremiereDate, ProductionYear, SortName, Random, Revenue, Runtime.</param>        /// <param name="isPlayed">Optional filter by items that are played, or not.</param>        /// <param name="genres">Optional. If specified, results will be filtered based on genre. This allows multiple, pipe delimited.</param>        /// <param name="officialRatings">Optional. If specified, results will be filtered based on OfficialRating. This allows multiple, pipe delimited.</param>        /// <param name="tags">Optional. If specified, results will be filtered based on tag. This allows multiple, pipe delimited.</param>        /// <param name="years">Optional. If specified, results will be filtered based on production year. This allows multiple, comma delimited.</param>        /// <param name="enableUserData">Optional, include user data.</param>        /// <param name="imageTypeLimit">Optional, the max number of images to return, per image type.</param>        /// <param name="enableImageTypes">Optional. The image types to include in the output.</param>        /// <param name="person">Optional. If specified, results will be filtered to include only those containing the specified person.</param>        /// <param name="personIds">Optional. If specified, results will be filtered to include only those containing the specified person id.</param>        /// <param name="personTypes">Optional. If specified, along with Person, results will be filtered to include only those containing the specified person and PersonType. Allows multiple, comma-delimited.</param>        /// <param name="studios">Optional. If specified, results will be filtered based on studio. This allows multiple, pipe delimited.</param>        /// <param name="artists">Optional. If specified, results will be filtered based on artists. This allows multiple, pipe delimited.</param>        /// <param name="excludeArtistIds">Optional. If specified, results will be filtered based on artist id. This allows multiple, pipe delimited.</param>        /// <param name="artistIds">Optional. If specified, results will be filtered to include only those containing the specified artist id.</param>        /// <param name="albumArtistIds">Optional. If specified, results will be filtered to include only those containing the specified album artist id.</param>        /// <param name="contributingArtistIds">Optional. If specified, results will be filtered to include only those containing the specified contributing artist id.</param>        /// <param name="albums">Optional. If specified, results will be filtered based on album. This allows multiple, pipe delimited.</param>        /// <param name="albumIds">Optional. If specified, results will be filtered based on album id. This allows multiple, pipe delimited.</param>        /// <param name="ids">Optional. If specific items are needed, specify a list of item id's to retrieve. This allows multiple, comma delimited.</param>        /// <param name="videoTypes">Optional filter by VideoType (videofile, dvd, bluray, iso). Allows multiple, comma delimited.</param>        /// <param name="minOfficialRating">Optional filter by minimum official rating (PG, PG-13, TV-MA, etc).</param>        /// <param name="isLocked">Optional filter by items that are locked.</param>        /// <param name="isPlaceHolder">Optional filter by items that are placeholders.</param>        /// <param name="hasOfficialRating">Optional filter by items that have official ratings.</param>        /// <param name="collapseBoxSetItems">Whether or not to hide items behind their boxsets.</param>        /// <param name="minWidth">Optional. Filter by the minimum width of the item.</param>        /// <param name="minHeight">Optional. Filter by the minimum height of the item.</param>        /// <param name="maxWidth">Optional. Filter by the maximum width of the item.</param>        /// <param name="maxHeight">Optional. Filter by the maximum height of the item.</param>        /// <param name="is3D">Optional filter by items that are 3D, or not.</param>        /// <param name="seriesStatus">Optional filter by Series Status. Allows multiple, comma delimited.</param>        /// <param name="nameStartsWithOrGreater">Optional filter by items whose name is sorted equally or greater than a given input string.</param>        /// <param name="nameStartsWith">Optional filter by items whose name is sorted equally than a given input string.</param>        /// <param name="nameLessThan">Optional filter by items whose name is equally or lesser than a given input string.</param>        /// <param name="studioIds">Optional. If specified, results will be filtered based on studio id. This allows multiple, pipe delimited.</param>        /// <param name="genreIds">Optional. If specified, results will be filtered based on genre id. This allows multiple, pipe delimited.</param>        /// <param name="enableTotalRecordCount">Optional. Enable the total record count.</param>        /// <param name="enableImages">Optional, include image information in output.</param>        /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the items.</returns>        [HttpGet("Items")]        [ProducesResponseType(StatusCodes.Status200OK)]        public ActionResult<QueryResult<BaseItemDto>> GetItems(            [FromQuery] Guid? userId,            [FromQuery] string? maxOfficialRating,            [FromQuery] bool? hasThemeSong,            [FromQuery] bool? hasThemeVideo,            [FromQuery] bool? hasSubtitles,            [FromQuery] bool? hasSpecialFeature,            [FromQuery] bool? hasTrailer,            [FromQuery] string? adjacentTo,            [FromQuery] int? parentIndexNumber,            [FromQuery] bool? hasParentalRating,            [FromQuery] bool? isHd,            [FromQuery] bool? is4K,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] LocationType[] locationTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] LocationType[] excludeLocationTypes,            [FromQuery] bool? isMissing,            [FromQuery] bool? isUnaired,            [FromQuery] double? minCommunityRating,            [FromQuery] double? minCriticRating,            [FromQuery] DateTime? minPremiereDate,            [FromQuery] DateTime? minDateLastSaved,            [FromQuery] DateTime? minDateLastSavedForUser,            [FromQuery] DateTime? maxPremiereDate,            [FromQuery] bool? hasOverview,            [FromQuery] bool? hasImdbId,            [FromQuery] bool? hasTmdbId,            [FromQuery] bool? hasTvdbId,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] excludeItemIds,            [FromQuery] int? startIndex,            [FromQuery] int? limit,            [FromQuery] bool? recursive,            [FromQuery] string? searchTerm,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] SortOrder[] sortOrder,            [FromQuery] Guid? parentId,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields[] fields,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] excludeItemTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFilter[] filters,            [FromQuery] bool? isFavorite,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] mediaTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType[] imageTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] sortBy,            [FromQuery] bool? isPlayed,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] genres,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] officialRatings,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] tags,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] int[] years,            [FromQuery] bool? enableUserData,            [FromQuery] int? imageTypeLimit,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType[] enableImageTypes,            [FromQuery] string? person,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] personIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] personTypes,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] studios,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] artists,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] excludeArtistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] artistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] albumArtistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] contributingArtistIds,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] albums,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] albumIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] ids,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] VideoType[] videoTypes,            [FromQuery] string? minOfficialRating,            [FromQuery] bool? isLocked,            [FromQuery] bool? isPlaceHolder,            [FromQuery] bool? hasOfficialRating,            [FromQuery] bool? collapseBoxSetItems,            [FromQuery] int? minWidth,            [FromQuery] int? minHeight,            [FromQuery] int? maxWidth,            [FromQuery] int? maxHeight,            [FromQuery] bool? is3D,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] SeriesStatus[] seriesStatus,            [FromQuery] string? nameStartsWithOrGreater,            [FromQuery] string? nameStartsWith,            [FromQuery] string? nameLessThan,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] studioIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] genreIds,            [FromQuery] bool enableTotalRecordCount = true,            [FromQuery] bool? enableImages = true)        {            var user = userId.HasValue && !userId.Equals(Guid.Empty)                ? _userManager.GetUserById(userId.Value)                : null;            var dtoOptions = new DtoOptions { Fields = fields }                .AddClientFields(Request)                .AddAdditionalDtoOptions(enableImages, enableUserData, imageTypeLimit, enableImageTypes);            if (includeItemTypes.Length == 1                && (includeItemTypes[0] == BaseItemKind.Playlist                    || includeItemTypes[0] == BaseItemKind.BoxSet))            {                parentId = null;            }            var item = _libraryManager.GetParentItem(parentId, userId);            QueryResult<BaseItem> result;            if (!(item is Folder folder))            {                folder = _libraryManager.GetUserRootFolder();            }            string? collectionType = null;            if (folder is IHasCollectionType hasCollectionType)            {                collectionType = hasCollectionType.CollectionType;            }            if (string.Equals(collectionType, CollectionType.Playlists, StringComparison.OrdinalIgnoreCase))            {                recursive = true;                includeItemTypes = new[] { BaseItemKind.Playlist };            }            var enabledChannels = user!.GetPreferenceValues<Guid>(PreferenceKind.EnabledChannels);            bool isInEnabledFolder = Array.IndexOf(user.GetPreferenceValues<Guid>(PreferenceKind.EnabledFolders), item.Id) != -1                                     // Assume all folders inside an EnabledChannel are enabled                                     || Array.IndexOf(enabledChannels, item.Id) != -1                                     // Assume all items inside an EnabledChannel are enabled                                     || Array.IndexOf(enabledChannels, item.ChannelId) != -1;            var collectionFolders = _libraryManager.GetCollectionFolders(item);            foreach (var collectionFolder in collectionFolders)            {                if (user.GetPreferenceValues<Guid>(PreferenceKind.EnabledFolders).Contains(collectionFolder.Id))                {                    isInEnabledFolder = true;                }            }            if (item is not UserRootFolder                && !isInEnabledFolder                && !user.HasPermission(PermissionKind.EnableAllFolders)                && !user.HasPermission(PermissionKind.EnableAllChannels)                && !string.Equals(collectionType, CollectionType.Folders, StringComparison.OrdinalIgnoreCase))            {                _logger.LogWarning("{UserName} is not permitted to access Library {ItemName}.", user.Username, item.Name);                return Unauthorized($"{user.Username} is not permitted to access Library {item.Name}.");            }            if ((recursive.HasValue && recursive.Value) || ids.Length != 0 || !(item is UserRootFolder))            {                var query = new InternalItemsQuery(user!)                {                    IsPlayed = isPlayed,                    MediaTypes = mediaTypes,                    IncludeItemTypes = RequestHelpers.GetItemTypeStrings(includeItemTypes),                    ExcludeItemTypes = RequestHelpers.GetItemTypeStrings(excludeItemTypes),                    Recursive = recursive ?? false,                    OrderBy = RequestHelpers.GetOrderBy(sortBy, sortOrder),                    IsFavorite = isFavorite,                    Limit = limit,                    StartIndex = startIndex,                    IsMissing = isMissing,                    IsUnaired = isUnaired,                    CollapseBoxSetItems = collapseBoxSetItems,                    NameLessThan = nameLessThan,                    NameStartsWith = nameStartsWith,                    NameStartsWithOrGreater = nameStartsWithOrGreater,                    HasImdbId = hasImdbId,                    IsPlaceHolder = isPlaceHolder,                    IsLocked = isLocked,                    MinWidth = minWidth,                    MinHeight = minHeight,                    MaxWidth = maxWidth,                    MaxHeight = maxHeight,                    Is3D = is3D,                    HasTvdbId = hasTvdbId,                    HasTmdbId = hasTmdbId,                    HasOverview = hasOverview,                    HasOfficialRating = hasOfficialRating,                    HasParentalRating = hasParentalRating,                    HasSpecialFeature = hasSpecialFeature,                    HasSubtitles = hasSubtitles,                    HasThemeSong = hasThemeSong,                    HasThemeVideo = hasThemeVideo,                    HasTrailer = hasTrailer,                    IsHD = isHd,                    Is4K = is4K,                    Tags = tags,                    OfficialRatings = officialRatings,                    Genres = genres,                    ArtistIds = artistIds,                    AlbumArtistIds = albumArtistIds,                    ContributingArtistIds = contributingArtistIds,                    GenreIds = genreIds,                    StudioIds = studioIds,                    Person = person,                    PersonIds = personIds,                    PersonTypes = personTypes,                    Years = years,                    ImageTypes = imageTypes,                    VideoTypes = videoTypes,                    AdjacentTo = adjacentTo,                    ItemIds = ids,                    MinCommunityRating = minCommunityRating,                    MinCriticRating = minCriticRating,                    ParentId = parentId ?? Guid.Empty,                    ParentIndexNumber = parentIndexNumber,                    EnableTotalRecordCount = enableTotalRecordCount,                    ExcludeItemIds = excludeItemIds,                    DtoOptions = dtoOptions,                    SearchTerm = searchTerm,                    MinDateLastSaved = minDateLastSaved?.ToUniversalTime(),                    MinDateLastSavedForUser = minDateLastSavedForUser?.ToUniversalTime(),                    MinPremiereDate = minPremiereDate?.ToUniversalTime(),                    MaxPremiereDate = maxPremiereDate?.ToUniversalTime(),                };                if (ids.Length != 0 || !string.IsNullOrWhiteSpace(searchTerm))                {                    query.CollapseBoxSetItems = false;                }                foreach (var filter in filters)                {                    switch (filter)                    {                        case ItemFilter.Dislikes:                            query.IsLiked = false;                            break;                        case ItemFilter.IsFavorite:                            query.IsFavorite = true;                            break;                        case ItemFilter.IsFavoriteOrLikes:                            query.IsFavoriteOrLiked = true;                            break;                        case ItemFilter.IsFolder:                            query.IsFolder = true;                            break;                        case ItemFilter.IsNotFolder:                            query.IsFolder = false;                            break;                        case ItemFilter.IsPlayed:                            query.IsPlayed = true;                            break;                        case ItemFilter.IsResumable:                            query.IsResumable = true;                            break;                        case ItemFilter.IsUnplayed:                            query.IsPlayed = false;                            break;                        case ItemFilter.Likes:                            query.IsLiked = true;                            break;                    }                }                // Filter by Series Status                if (seriesStatus.Length != 0)                {                    query.SeriesStatuses = seriesStatus;                }                // ExcludeLocationTypes                if (excludeLocationTypes.Any(t => t == LocationType.Virtual))                {                    query.IsVirtualItem = false;                }                if (locationTypes.Length > 0 && locationTypes.Length < 4)                {                    query.IsVirtualItem = locationTypes.Contains(LocationType.Virtual);                }                // Min official rating                if (!string.IsNullOrWhiteSpace(minOfficialRating))                {                    query.MinParentalRating = _localization.GetRatingLevel(minOfficialRating);                }                // Max official rating                if (!string.IsNullOrWhiteSpace(maxOfficialRating))                {                    query.MaxParentalRating = _localization.GetRatingLevel(maxOfficialRating);                }                // Artists                if (artists.Length != 0)                {                    query.ArtistIds = artists.Select(i =>                    {                        try                        {                            return _libraryManager.GetArtist(i, new DtoOptions(false));                        }                        catch                        {                            return null;                        }                    }).Where(i => i != null).Select(i => i!.Id).ToArray();                }                // ExcludeArtistIds                if (excludeArtistIds.Length != 0)                {                    query.ExcludeArtistIds = excludeArtistIds;                }                if (albumIds.Length != 0)                {                    query.AlbumIds = albumIds;                }                // Albums                if (albums.Length != 0)                {                    query.AlbumIds = albums.SelectMany(i =>                    {                        return _libraryManager.GetItemIds(new InternalItemsQuery { IncludeItemTypes = new[] { nameof(MusicAlbum) }, Name = i, Limit = 1 });                    }).ToArray();                }                // Studios                if (studios.Length != 0)                {                    query.StudioIds = studios.Select(i =>                    {                        try                        {                            return _libraryManager.GetStudio(i);                        }                        catch                        {                            return null;                        }                    }).Where(i => i != null).Select(i => i!.Id).ToArray();                }                // Apply default sorting if none requested                if (query.OrderBy.Count == 0)                {                    // Albums by artist                    if (query.ArtistIds.Length > 0 && query.IncludeItemTypes.Length == 1 && string.Equals(query.IncludeItemTypes[0], "MusicAlbum", StringComparison.OrdinalIgnoreCase))                    {                        query.OrderBy = new[] { new ValueTuple<string, SortOrder>(ItemSortBy.ProductionYear, SortOrder.Descending), new ValueTuple<string, SortOrder>(ItemSortBy.SortName, SortOrder.Ascending) };                    }                }                result = folder.GetItems(query);            }            else            {                var itemsArray = folder.GetChildren(user, true);                result = new QueryResult<BaseItem> { Items = itemsArray, TotalRecordCount = itemsArray.Count, StartIndex = 0 };            }            return new QueryResult<BaseItemDto> { StartIndex = startIndex.GetValueOrDefault(), TotalRecordCount = result.TotalRecordCount, Items = _dtoService.GetBaseItemDtos(result.Items, dtoOptions, user) };        }        /// <summary>        /// Gets items based on a query.        /// </summary>        /// <param name="userId">The user id supplied as query parameter.</param>        /// <param name="maxOfficialRating">Optional filter by maximum official rating (PG, PG-13, TV-MA, etc).</param>        /// <param name="hasThemeSong">Optional filter by items with theme songs.</param>        /// <param name="hasThemeVideo">Optional filter by items with theme videos.</param>        /// <param name="hasSubtitles">Optional filter by items with subtitles.</param>        /// <param name="hasSpecialFeature">Optional filter by items with special features.</param>        /// <param name="hasTrailer">Optional filter by items with trailers.</param>        /// <param name="adjacentTo">Optional. Return items that are siblings of a supplied item.</param>        /// <param name="parentIndexNumber">Optional filter by parent index number.</param>        /// <param name="hasParentalRating">Optional filter by items that have or do not have a parental rating.</param>        /// <param name="isHd">Optional filter by items that are HD or not.</param>        /// <param name="is4K">Optional filter by items that are 4K or not.</param>        /// <param name="locationTypes">Optional. If specified, results will be filtered based on LocationType. This allows multiple, comma delimeted.</param>        /// <param name="excludeLocationTypes">Optional. If specified, results will be filtered based on the LocationType. This allows multiple, comma delimeted.</param>        /// <param name="isMissing">Optional filter by items that are missing episodes or not.</param>        /// <param name="isUnaired">Optional filter by items that are unaired episodes or not.</param>        /// <param name="minCommunityRating">Optional filter by minimum community rating.</param>        /// <param name="minCriticRating">Optional filter by minimum critic rating.</param>        /// <param name="minPremiereDate">Optional. The minimum premiere date. Format = ISO.</param>        /// <param name="minDateLastSaved">Optional. The minimum last saved date. Format = ISO.</param>        /// <param name="minDateLastSavedForUser">Optional. The minimum last saved date for the current user. Format = ISO.</param>        /// <param name="maxPremiereDate">Optional. The maximum premiere date. Format = ISO.</param>        /// <param name="hasOverview">Optional filter by items that have an overview or not.</param>        /// <param name="hasImdbId">Optional filter by items that have an imdb id or not.</param>        /// <param name="hasTmdbId">Optional filter by items that have a tmdb id or not.</param>        /// <param name="hasTvdbId">Optional filter by items that have a tvdb id or not.</param>        /// <param name="excludeItemIds">Optional. If specified, results will be filtered by exxcluding item ids. This allows multiple, comma delimeted.</param>        /// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>        /// <param name="limit">Optional. The maximum number of records to return.</param>        /// <param name="recursive">When searching within folders, this determines whether or not the search will be recursive. true/false.</param>        /// <param name="searchTerm">Optional. Filter based on a search term.</param>        /// <param name="sortOrder">Sort Order - Ascending,Descending.</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. Specify additional fields of information to return in the output. This allows multiple, comma delimeted. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines.</param>        /// <param name="excludeItemTypes">Optional. If specified, results will be filtered based on item type. This allows multiple, comma delimeted.</param>        /// <param name="includeItemTypes">Optional. If specified, results will be filtered based on the item type. This allows multiple, comma delimeted.</param>        /// <param name="filters">Optional. Specify additional filters to apply. This allows multiple, comma delimeted. Options: IsFolder, IsNotFolder, IsUnplayed, IsPlayed, IsFavorite, IsResumable, Likes, Dislikes.</param>        /// <param name="isFavorite">Optional filter by items that are marked as favorite, or not.</param>        /// <param name="mediaTypes">Optional filter by MediaType. Allows multiple, comma delimited.</param>        /// <param name="imageTypes">Optional. If specified, results will be filtered based on those containing image types. This allows multiple, comma delimited.</param>        /// <param name="sortBy">Optional. Specify one or more sort orders, comma delimeted. Options: Album, AlbumArtist, Artist, Budget, CommunityRating, CriticRating, DateCreated, DatePlayed, PlayCount, PremiereDate, ProductionYear, SortName, Random, Revenue, Runtime.</param>        /// <param name="isPlayed">Optional filter by items that are played, or not.</param>        /// <param name="genres">Optional. If specified, results will be filtered based on genre. This allows multiple, pipe delimeted.</param>        /// <param name="officialRatings">Optional. If specified, results will be filtered based on OfficialRating. This allows multiple, pipe delimeted.</param>        /// <param name="tags">Optional. If specified, results will be filtered based on tag. This allows multiple, pipe delimeted.</param>        /// <param name="years">Optional. If specified, results will be filtered based on production year. This allows multiple, comma delimeted.</param>        /// <param name="enableUserData">Optional, include user data.</param>        /// <param name="imageTypeLimit">Optional, the max number of images to return, per image type.</param>        /// <param name="enableImageTypes">Optional. The image types to include in the output.</param>        /// <param name="person">Optional. If specified, results will be filtered to include only those containing the specified person.</param>        /// <param name="personIds">Optional. If specified, results will be filtered to include only those containing the specified person id.</param>        /// <param name="personTypes">Optional. If specified, along with Person, results will be filtered to include only those containing the specified person and PersonType. Allows multiple, comma-delimited.</param>        /// <param name="studios">Optional. If specified, results will be filtered based on studio. This allows multiple, pipe delimeted.</param>        /// <param name="artists">Optional. If specified, results will be filtered based on artists. This allows multiple, pipe delimeted.</param>        /// <param name="excludeArtistIds">Optional. If specified, results will be filtered based on artist id. This allows multiple, pipe delimeted.</param>        /// <param name="artistIds">Optional. If specified, results will be filtered to include only those containing the specified artist id.</param>        /// <param name="albumArtistIds">Optional. If specified, results will be filtered to include only those containing the specified album artist id.</param>        /// <param name="contributingArtistIds">Optional. If specified, results will be filtered to include only those containing the specified contributing artist id.</param>        /// <param name="albums">Optional. If specified, results will be filtered based on album. This allows multiple, pipe delimeted.</param>        /// <param name="albumIds">Optional. If specified, results will be filtered based on album id. This allows multiple, pipe delimeted.</param>        /// <param name="ids">Optional. If specific items are needed, specify a list of item id's to retrieve. This allows multiple, comma delimited.</param>        /// <param name="videoTypes">Optional filter by VideoType (videofile, dvd, bluray, iso). Allows multiple, comma delimeted.</param>        /// <param name="minOfficialRating">Optional filter by minimum official rating (PG, PG-13, TV-MA, etc).</param>        /// <param name="isLocked">Optional filter by items that are locked.</param>        /// <param name="isPlaceHolder">Optional filter by items that are placeholders.</param>        /// <param name="hasOfficialRating">Optional filter by items that have official ratings.</param>        /// <param name="collapseBoxSetItems">Whether or not to hide items behind their boxsets.</param>        /// <param name="minWidth">Optional. Filter by the minimum width of the item.</param>        /// <param name="minHeight">Optional. Filter by the minimum height of the item.</param>        /// <param name="maxWidth">Optional. Filter by the maximum width of the item.</param>        /// <param name="maxHeight">Optional. Filter by the maximum height of the item.</param>        /// <param name="is3D">Optional filter by items that are 3D, or not.</param>        /// <param name="seriesStatus">Optional filter by Series Status. Allows multiple, comma delimeted.</param>        /// <param name="nameStartsWithOrGreater">Optional filter by items whose name is sorted equally or greater than a given input string.</param>        /// <param name="nameStartsWith">Optional filter by items whose name is sorted equally than a given input string.</param>        /// <param name="nameLessThan">Optional filter by items whose name is equally or lesser than a given input string.</param>        /// <param name="studioIds">Optional. If specified, results will be filtered based on studio id. This allows multiple, pipe delimeted.</param>        /// <param name="genreIds">Optional. If specified, results will be filtered based on genre id. This allows multiple, pipe delimeted.</param>        /// <param name="enableTotalRecordCount">Optional. Enable the total record count.</param>        /// <param name="enableImages">Optional, include image information in output.</param>        /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the items.</returns>        [HttpGet("Users/{userId}/Items")]        [ProducesResponseType(StatusCodes.Status200OK)]        public ActionResult<QueryResult<BaseItemDto>> GetItemsByUserId(            [FromRoute] Guid userId,            [FromQuery] string? maxOfficialRating,            [FromQuery] bool? hasThemeSong,            [FromQuery] bool? hasThemeVideo,            [FromQuery] bool? hasSubtitles,            [FromQuery] bool? hasSpecialFeature,            [FromQuery] bool? hasTrailer,            [FromQuery] string? adjacentTo,            [FromQuery] int? parentIndexNumber,            [FromQuery] bool? hasParentalRating,            [FromQuery] bool? isHd,            [FromQuery] bool? is4K,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] LocationType[] locationTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] LocationType[] excludeLocationTypes,            [FromQuery] bool? isMissing,            [FromQuery] bool? isUnaired,            [FromQuery] double? minCommunityRating,            [FromQuery] double? minCriticRating,            [FromQuery] DateTime? minPremiereDate,            [FromQuery] DateTime? minDateLastSaved,            [FromQuery] DateTime? minDateLastSavedForUser,            [FromQuery] DateTime? maxPremiereDate,            [FromQuery] bool? hasOverview,            [FromQuery] bool? hasImdbId,            [FromQuery] bool? hasTmdbId,            [FromQuery] bool? hasTvdbId,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] excludeItemIds,            [FromQuery] int? startIndex,            [FromQuery] int? limit,            [FromQuery] bool? recursive,            [FromQuery] string? searchTerm,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] SortOrder[] sortOrder,            [FromQuery] Guid? parentId,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields[] fields,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] excludeItemTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFilter[] filters,            [FromQuery] bool? isFavorite,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] mediaTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType[] imageTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] sortBy,            [FromQuery] bool? isPlayed,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] genres,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] officialRatings,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] tags,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] int[] years,            [FromQuery] bool? enableUserData,            [FromQuery] int? imageTypeLimit,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType[] enableImageTypes,            [FromQuery] string? person,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] personIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] personTypes,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] studios,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] artists,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] excludeArtistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] artistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] albumArtistIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] contributingArtistIds,            [FromQuery, ModelBinder(typeof(PipeDelimitedArrayModelBinder))] string[] albums,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] albumIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] ids,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] VideoType[] videoTypes,            [FromQuery] string? minOfficialRating,            [FromQuery] bool? isLocked,            [FromQuery] bool? isPlaceHolder,            [FromQuery] bool? hasOfficialRating,            [FromQuery] bool? collapseBoxSetItems,            [FromQuery] int? minWidth,            [FromQuery] int? minHeight,            [FromQuery] int? maxWidth,            [FromQuery] int? maxHeight,            [FromQuery] bool? is3D,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] SeriesStatus[] seriesStatus,            [FromQuery] string? nameStartsWithOrGreater,            [FromQuery] string? nameStartsWith,            [FromQuery] string? nameLessThan,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] studioIds,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] Guid[] genreIds,            [FromQuery] bool enableTotalRecordCount = true,            [FromQuery] bool? enableImages = true)        {            return GetItems(                userId,                maxOfficialRating,                hasThemeSong,                hasThemeVideo,                hasSubtitles,                hasSpecialFeature,                hasTrailer,                adjacentTo,                parentIndexNumber,                hasParentalRating,                isHd,                is4K,                locationTypes,                excludeLocationTypes,                isMissing,                isUnaired,                minCommunityRating,                minCriticRating,                minPremiereDate,                minDateLastSaved,                minDateLastSavedForUser,                maxPremiereDate,                hasOverview,                hasImdbId,                hasTmdbId,                hasTvdbId,                excludeItemIds,                startIndex,                limit,                recursive,                searchTerm,                sortOrder,                parentId,                fields,                excludeItemTypes,                includeItemTypes,                filters,                isFavorite,                mediaTypes,                imageTypes,                sortBy,                isPlayed,                genres,                officialRatings,                tags,                years,                enableUserData,                imageTypeLimit,                enableImageTypes,                person,                personIds,                personTypes,                studios,                artists,                excludeArtistIds,                artistIds,                albumArtistIds,                contributingArtistIds,                albums,                albumIds,                ids,                videoTypes,                minOfficialRating,                isLocked,                isPlaceHolder,                hasOfficialRating,                collapseBoxSetItems,                minWidth,                minHeight,                maxWidth,                maxHeight,                is3D,                seriesStatus,                nameStartsWithOrGreater,                nameStartsWith,                nameLessThan,                studioIds,                genreIds,                enableTotalRecordCount,                enableImages);        }        /// <summary>        /// Gets items based on a query.        /// </summary>        /// <param name="userId">The user id.</param>        /// <param name="startIndex">The start index.</param>        /// <param name="limit">The item limit.</param>        /// <param name="searchTerm">The search term.</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. Specify additional fields of information to return in the output. This allows multiple, comma delimited. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines.</param>        /// <param name="mediaTypes">Optional. Filter by MediaType. Allows multiple, comma delimited.</param>        /// <param name="enableUserData">Optional. Include user data.</param>        /// <param name="imageTypeLimit">Optional. The max number of images to return, per image type.</param>        /// <param name="enableImageTypes">Optional. The image types to include in the output.</param>        /// <param name="excludeItemTypes">Optional. If specified, results will be filtered based on item type. This allows multiple, comma delimited.</param>        /// <param name="includeItemTypes">Optional. If specified, results will be filtered based on the item type. This allows multiple, comma delimited.</param>        /// <param name="enableTotalRecordCount">Optional. Enable the total record count.</param>        /// <param name="enableImages">Optional. Include image information in output.</param>        /// <response code="200">Items returned.</response>        /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the items that are resumable.</returns>        [HttpGet("Users/{userId}/Items/Resume")]        [ProducesResponseType(StatusCodes.Status200OK)]        public ActionResult<QueryResult<BaseItemDto>> GetResumeItems(            [FromRoute, Required] Guid userId,            [FromQuery] int? startIndex,            [FromQuery] int? limit,            [FromQuery] string? searchTerm,            [FromQuery] Guid? parentId,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields[] fields,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] mediaTypes,            [FromQuery] bool? enableUserData,            [FromQuery] int? imageTypeLimit,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType[] enableImageTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] excludeItemTypes,            [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,            [FromQuery] bool enableTotalRecordCount = true,            [FromQuery] bool? enableImages = true)        {            var user = _userManager.GetUserById(userId);            var parentIdGuid = parentId ?? Guid.Empty;            var dtoOptions = new DtoOptions { Fields = fields }                .AddClientFields(Request)                .AddAdditionalDtoOptions(enableImages, enableUserData, imageTypeLimit, enableImageTypes);            var ancestorIds = Array.Empty<Guid>();            var excludeFolderIds = user.GetPreferenceValues<Guid>(PreferenceKind.LatestItemExcludes);            if (parentIdGuid.Equals(Guid.Empty) && excludeFolderIds.Length > 0)            {                ancestorIds = _libraryManager.GetUserRootFolder().GetChildren(user, true)                    .Where(i => i is Folder)                    .Where(i => !excludeFolderIds.Contains(i.Id))                    .Select(i => i.Id)                    .ToArray();            }            var itemsResult = _libraryManager.GetItemsResult(new InternalItemsQuery(user)            {                OrderBy = new[] { (ItemSortBy.DatePlayed, SortOrder.Descending) },                IsResumable = true,                StartIndex = startIndex,                Limit = limit,                ParentId = parentIdGuid,                Recursive = true,                DtoOptions = dtoOptions,                MediaTypes = mediaTypes,                IsVirtualItem = false,                CollapseBoxSetItems = false,                EnableTotalRecordCount = enableTotalRecordCount,                AncestorIds = ancestorIds,                IncludeItemTypes = RequestHelpers.GetItemTypeStrings(includeItemTypes),                ExcludeItemTypes = RequestHelpers.GetItemTypeStrings(excludeItemTypes),                SearchTerm = searchTerm            });            var returnItems = _dtoService.GetBaseItemDtos(itemsResult.Items, dtoOptions, user);            return new QueryResult<BaseItemDto>            {                StartIndex = startIndex.GetValueOrDefault(),                TotalRecordCount = itemsResult.TotalRecordCount,                Items = returnItems            };        }    }}
 |