| | 1 | | using System; |
| | 2 | | using System.ComponentModel.DataAnnotations; |
| | 3 | | using System.Linq; |
| | 4 | | using Jellyfin.Api.Extensions; |
| | 5 | | using Jellyfin.Api.Helpers; |
| | 6 | | using Jellyfin.Api.ModelBinders; |
| | 7 | | using Jellyfin.Database.Implementations.Entities; |
| | 8 | | using Jellyfin.Extensions; |
| | 9 | | using MediaBrowser.Controller.Dto; |
| | 10 | | using MediaBrowser.Controller.Entities; |
| | 11 | | using MediaBrowser.Controller.Library; |
| | 12 | | using MediaBrowser.Model.Dto; |
| | 13 | | using MediaBrowser.Model.Entities; |
| | 14 | | using MediaBrowser.Model.Querying; |
| | 15 | | using Microsoft.AspNetCore.Authorization; |
| | 16 | | using Microsoft.AspNetCore.Http; |
| | 17 | | using Microsoft.AspNetCore.Mvc; |
| | 18 | |
|
| | 19 | | namespace Jellyfin.Api.Controllers; |
| | 20 | |
|
| | 21 | | /// <summary> |
| | 22 | | /// Persons controller. |
| | 23 | | /// </summary> |
| | 24 | | [Authorize] |
| | 25 | | public class PersonsController : BaseJellyfinApiController |
| | 26 | | { |
| | 27 | | private readonly ILibraryManager _libraryManager; |
| | 28 | | private readonly IDtoService _dtoService; |
| | 29 | | private readonly IUserManager _userManager; |
| | 30 | |
|
| | 31 | | /// <summary> |
| | 32 | | /// Initializes a new instance of the <see cref="PersonsController"/> class. |
| | 33 | | /// </summary> |
| | 34 | | /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param> |
| | 35 | | /// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param> |
| | 36 | | /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param> |
| 1 | 37 | | public PersonsController( |
| 1 | 38 | | ILibraryManager libraryManager, |
| 1 | 39 | | IDtoService dtoService, |
| 1 | 40 | | IUserManager userManager) |
| | 41 | | { |
| 1 | 42 | | _libraryManager = libraryManager; |
| 1 | 43 | | _dtoService = dtoService; |
| 1 | 44 | | _userManager = userManager; |
| 1 | 45 | | } |
| | 46 | |
|
| | 47 | | /// <summary> |
| | 48 | | /// Gets all persons. |
| | 49 | | /// </summary> |
| | 50 | | /// <param name="limit">Optional. The maximum number of records to return.</param> |
| | 51 | | /// <param name="searchTerm">The search term.</param> |
| | 52 | | /// <param name="fields">Optional. Specify additional fields of information to return in the output.</param> |
| | 53 | | /// <param name="filters">Optional. Specify additional filters to apply.</param> |
| | 54 | | /// <param name="isFavorite">Optional filter by items that are marked as favorite, or not. userId is required.</para |
| | 55 | | /// <param name="enableUserData">Optional, include user data.</param> |
| | 56 | | /// <param name="imageTypeLimit">Optional, the max number of images to return, per image type.</param> |
| | 57 | | /// <param name="enableImageTypes">Optional. The image types to include in the output.</param> |
| | 58 | | /// <param name="excludePersonTypes">Optional. If specified results will be filtered to exclude those containing the |
| | 59 | | /// <param name="personTypes">Optional. If specified results will be filtered to include only those containing the s |
| | 60 | | /// <param name="appearsInItemId">Optional. If specified, person results will be filtered on items related to said p |
| | 61 | | /// <param name="userId">User id.</param> |
| | 62 | | /// <param name="enableImages">Optional, include image information in output.</param> |
| | 63 | | /// <response code="200">Persons returned.</response> |
| | 64 | | /// <returns>An <see cref="OkResult"/> containing the queryresult of persons.</returns> |
| | 65 | | [HttpGet] |
| | 66 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 67 | | public ActionResult<QueryResult<BaseItemDto>> GetPersons( |
| | 68 | | [FromQuery] int? limit, |
| | 69 | | [FromQuery] string? searchTerm, |
| | 70 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] ItemFields[] fields, |
| | 71 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] ItemFilter[] filters, |
| | 72 | | [FromQuery] bool? isFavorite, |
| | 73 | | [FromQuery] bool? enableUserData, |
| | 74 | | [FromQuery] int? imageTypeLimit, |
| | 75 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] ImageType[] enableImageTypes, |
| | 76 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] string[] excludePersonTypes, |
| | 77 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] string[] personTypes, |
| | 78 | | [FromQuery] Guid? appearsInItemId, |
| | 79 | | [FromQuery] Guid? userId, |
| | 80 | | [FromQuery] bool? enableImages = true) |
| | 81 | | { |
| 0 | 82 | | userId = RequestHelpers.GetUserId(User, userId); |
| 0 | 83 | | var dtoOptions = new DtoOptions { Fields = fields } |
| 0 | 84 | | .AddClientFields(User) |
| 0 | 85 | | .AddAdditionalDtoOptions(enableImages, enableUserData, imageTypeLimit, enableImageTypes); |
| | 86 | |
|
| 0 | 87 | | User? user = userId.IsNullOrEmpty() |
| 0 | 88 | | ? null |
| 0 | 89 | | : _userManager.GetUserById(userId.Value); |
| | 90 | |
|
| 0 | 91 | | var isFavoriteInFilters = filters.Any(f => f == ItemFilter.IsFavorite); |
| 0 | 92 | | var peopleItems = _libraryManager.GetPeopleItems(new InternalPeopleQuery( |
| 0 | 93 | | personTypes, |
| 0 | 94 | | excludePersonTypes) |
| 0 | 95 | | { |
| 0 | 96 | | NameContains = searchTerm, |
| 0 | 97 | | User = user, |
| 0 | 98 | | IsFavorite = !isFavorite.HasValue && isFavoriteInFilters ? true : isFavorite, |
| 0 | 99 | | AppearsInItemId = appearsInItemId ?? Guid.Empty, |
| 0 | 100 | | Limit = limit ?? 0 |
| 0 | 101 | | }); |
| | 102 | |
|
| 0 | 103 | | return new QueryResult<BaseItemDto>( |
| 0 | 104 | | peopleItems |
| 0 | 105 | | .Select(person => _dtoService.GetItemByNameDto(person, dtoOptions, null, user)) |
| 0 | 106 | | .ToArray()); |
| | 107 | | } |
| | 108 | |
|
| | 109 | | /// <summary> |
| | 110 | | /// Get person by name. |
| | 111 | | /// </summary> |
| | 112 | | /// <param name="name">Person name.</param> |
| | 113 | | /// <param name="userId">Optional. Filter by user id, and attach user data.</param> |
| | 114 | | /// <response code="200">Person returned.</response> |
| | 115 | | /// <response code="404">Person not found.</response> |
| | 116 | | /// <returns>An <see cref="OkResult"/> containing the person on success, |
| | 117 | | /// or a <see cref="NotFoundResult"/> if person not found.</returns> |
| | 118 | | [HttpGet("{name}")] |
| | 119 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 120 | | [ProducesResponseType(StatusCodes.Status404NotFound)] |
| | 121 | | public ActionResult<BaseItemDto> GetPerson([FromRoute, Required] string name, [FromQuery] Guid? userId) |
| | 122 | | { |
| 1 | 123 | | userId = RequestHelpers.GetUserId(User, userId); |
| 1 | 124 | | var dtoOptions = new DtoOptions() |
| 1 | 125 | | .AddClientFields(User); |
| | 126 | |
|
| 1 | 127 | | var item = _libraryManager.GetPerson(name); |
| 1 | 128 | | if (item is null) |
| | 129 | | { |
| 1 | 130 | | return NotFound(); |
| | 131 | | } |
| | 132 | |
|
| 0 | 133 | | if (!userId.IsNullOrEmpty()) |
| | 134 | | { |
| 0 | 135 | | var user = _userManager.GetUserById(userId.Value); |
| 0 | 136 | | return _dtoService.GetBaseItemDto(item, dtoOptions, user); |
| | 137 | | } |
| | 138 | |
|
| 0 | 139 | | return _dtoService.GetBaseItemDto(item, dtoOptions); |
| | 140 | | } |
| | 141 | | } |