jellyfin-server/Jellyfin.Api/Controllers/FilterController.cs

218 lines
8.7 KiB
C#
Raw Normal View History

2020-11-21 13:26:03 +00:00
using System;
2020-06-08 04:04:59 +00:00
using System.Linq;
2020-06-22 13:44:11 +00:00
using Jellyfin.Api.Constants;
2021-02-11 00:09:23 +00:00
using Jellyfin.Api.Helpers;
2020-11-17 03:37:43 +00:00
using Jellyfin.Api.ModelBinders;
2021-02-11 00:09:23 +00:00
using Jellyfin.Data.Enums;
2020-06-08 04:04:59 +00:00
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Querying;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
namespace Jellyfin.Api.Controllers
{
/// <summary>
/// Filters controller.
/// </summary>
2020-08-04 18:48:53 +00:00
[Route("")]
2020-06-22 13:44:11 +00:00
[Authorize(Policy = Policies.DefaultAuthorization)]
2020-06-08 04:04:59 +00:00
public class FilterController : BaseJellyfinApiController
{
private readonly ILibraryManager _libraryManager;
private readonly IUserManager _userManager;
/// <summary>
/// Initializes a new instance of the <see cref="FilterController"/> class.
/// </summary>
/// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
/// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
public FilterController(ILibraryManager libraryManager, IUserManager userManager)
{
_libraryManager = libraryManager;
_userManager = userManager;
}
/// <summary>
/// Gets legacy query filters.
/// </summary>
/// <param name="userId">Optional. User id.</param>
/// <param name="parentId">Optional. Parent id.</param>
/// <param name="includeItemTypes">Optional. If specified, results will be filtered based on item type. This allows multiple, comma delimited.</param>
/// <param name="mediaTypes">Optional. Filter by MediaType. Allows multiple, comma delimited.</param>
2020-06-08 13:37:16 +00:00
/// <response code="200">Legacy filters retrieved.</response>
2020-06-08 04:04:59 +00:00
/// <returns>Legacy query filters.</returns>
2020-08-04 18:48:53 +00:00
[HttpGet("Items/Filters")]
2020-06-08 04:04:59 +00:00
[ProducesResponseType(StatusCodes.Status200OK)]
public ActionResult<QueryFiltersLegacy> GetQueryFiltersLegacy(
[FromQuery] Guid? userId,
[FromQuery] Guid? parentId,
2021-02-11 00:09:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,
2020-11-17 03:37:43 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] string[] mediaTypes)
2020-06-08 04:04:59 +00:00
{
var user = userId.HasValue && !userId.Equals(Guid.Empty)
? _userManager.GetUserById(userId.Value)
: null;
2020-06-08 04:04:59 +00:00
2020-12-02 14:11:42 +00:00
BaseItem? item = null;
if (includeItemTypes.Length != 1
2021-02-11 00:09:23 +00:00
|| !(includeItemTypes[0] == BaseItemKind.BoxSet
|| includeItemTypes[0] == BaseItemKind.Playlist
|| includeItemTypes[0] == BaseItemKind.Trailer
|| includeItemTypes[0] == BaseItemKind.Program))
2020-06-08 04:04:59 +00:00
{
2020-12-02 14:11:42 +00:00
item = _libraryManager.GetParentItem(parentId, user?.Id);
2020-12-01 19:10:11 +00:00
}
2020-06-08 04:04:59 +00:00
var query = new InternalItemsQuery
{
User = user,
2020-11-17 03:37:43 +00:00
MediaTypes = mediaTypes,
2021-02-11 00:09:23 +00:00
IncludeItemTypes = RequestHelpers.GetItemTypeStrings(includeItemTypes),
2020-06-08 04:04:59 +00:00
Recursive = true,
EnableTotalRecordCount = false,
DtoOptions = new DtoOptions
{
Fields = new[] { ItemFields.Genres, ItemFields.Tags },
EnableImages = false,
EnableUserData = false
}
};
2020-12-02 15:04:08 +00:00
if (item is not Folder folder)
{
return new QueryFiltersLegacy();
}
2020-12-02 14:55:34 +00:00
var itemList = folder.GetItemList(query);
2020-06-08 04:04:59 +00:00
return new QueryFiltersLegacy
{
Years = itemList.Select(i => i.ProductionYear ?? -1)
.Where(i => i > 0)
.Distinct()
.OrderBy(i => i)
.ToArray(),
Genres = itemList.SelectMany(i => i.Genres)
.DistinctNames()
.OrderBy(i => i)
.ToArray(),
Tags = itemList
.SelectMany(i => i.Tags)
.Distinct(StringComparer.OrdinalIgnoreCase)
.OrderBy(i => i)
.ToArray(),
OfficialRatings = itemList
.Select(i => i.OfficialRating)
.Where(i => !string.IsNullOrWhiteSpace(i))
.Distinct(StringComparer.OrdinalIgnoreCase)
.OrderBy(i => i)
.ToArray()
};
}
/// <summary>
/// Gets query filters.
/// </summary>
/// <param name="userId">Optional. User id.</param>
/// <param name="parentId">Optional. Specify this to localize the search to a specific item or folder. Omit to use the root.</param>
/// <param name="includeItemTypes">Optional. If specified, results will be filtered based on item type. This allows multiple, comma delimited.</param>
/// <param name="isAiring">Optional. Is item airing.</param>
/// <param name="isMovie">Optional. Is item movie.</param>
/// <param name="isSports">Optional. Is item sports.</param>
/// <param name="isKids">Optional. Is item kids.</param>
/// <param name="isNews">Optional. Is item news.</param>
/// <param name="isSeries">Optional. Is item series.</param>
/// <param name="recursive">Optional. Search recursive.</param>
2020-06-08 13:37:16 +00:00
/// <response code="200">Filters retrieved.</response>
2020-06-08 04:04:59 +00:00
/// <returns>Query filters.</returns>
2020-08-04 18:48:53 +00:00
[HttpGet("Items/Filters2")]
2020-06-08 04:04:59 +00:00
[ProducesResponseType(StatusCodes.Status200OK)]
public ActionResult<QueryFilters> GetQueryFilters(
[FromQuery] Guid? userId,
[FromQuery] Guid? parentId,
2021-02-11 00:09:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,
2020-06-08 04:04:59 +00:00
[FromQuery] bool? isAiring,
[FromQuery] bool? isMovie,
[FromQuery] bool? isSports,
[FromQuery] bool? isKids,
[FromQuery] bool? isNews,
[FromQuery] bool? isSeries,
[FromQuery] bool? recursive)
{
var user = userId.HasValue && !userId.Equals(Guid.Empty)
? _userManager.GetUserById(userId.Value)
: null;
2020-06-08 04:04:59 +00:00
2020-12-02 14:11:42 +00:00
BaseItem? parentItem = null;
2020-11-17 03:37:43 +00:00
if (includeItemTypes.Length == 1
2021-02-11 00:09:23 +00:00
&& (includeItemTypes[0] == BaseItemKind.BoxSet
|| includeItemTypes[0] == BaseItemKind.Playlist
|| includeItemTypes[0] == BaseItemKind.Trailer
|| includeItemTypes[0] == BaseItemKind.Program))
2020-06-08 04:04:59 +00:00
{
parentItem = null;
2020-12-02 14:11:42 +00:00
}
else if (parentId.HasValue)
{
parentItem = _libraryManager.GetItemById(parentId.Value);
2020-06-08 04:04:59 +00:00
}
var filters = new QueryFilters();
var genreQuery = new InternalItemsQuery(user)
{
2021-02-11 00:09:23 +00:00
IncludeItemTypes = RequestHelpers.GetItemTypeStrings(includeItemTypes),
2020-06-08 04:04:59 +00:00
DtoOptions = new DtoOptions
{
Fields = Array.Empty<ItemFields>(),
EnableImages = false,
EnableUserData = false
},
IsAiring = isAiring,
IsMovie = isMovie,
IsSports = isSports,
IsKids = isKids,
IsNews = isNews,
IsSeries = isSeries
};
if ((recursive ?? true) || parentItem is UserView || parentItem is ICollectionFolder)
{
genreQuery.AncestorIds = parentItem == null ? Array.Empty<Guid>() : new[] { parentItem.Id };
}
else
{
genreQuery.Parent = parentItem;
}
2020-11-17 03:37:43 +00:00
if (includeItemTypes.Length == 1
2021-02-11 00:09:23 +00:00
&& (includeItemTypes[0] == BaseItemKind.MusicAlbum
|| includeItemTypes[0] == BaseItemKind.MusicVideo
|| includeItemTypes[0] == BaseItemKind.MusicArtist
|| includeItemTypes[0] == BaseItemKind.Audio))
2020-06-08 04:04:59 +00:00
{
filters.Genres = _libraryManager.GetMusicGenres(genreQuery).Items.Select(i => new NameGuidPair
{
Name = i.Item1.Name,
Id = i.Item1.Id
}).ToArray();
}
else
{
filters.Genres = _libraryManager.GetGenres(genreQuery).Items.Select(i => new NameGuidPair
{
Name = i.Item1.Name,
Id = i.Item1.Id
}).ToArray();
}
return filters;
}
}
}