| | 1 | | using System; |
| | 2 | | using System.ComponentModel.DataAnnotations; |
| | 3 | | using Jellyfin.Api.Extensions; |
| | 4 | | using Jellyfin.Api.Helpers; |
| | 5 | | using Jellyfin.Api.ModelBinders; |
| | 6 | | using Jellyfin.Data.Enums; |
| | 7 | | using Jellyfin.Database.Implementations.Entities; |
| | 8 | | using Jellyfin.Database.Implementations.Enums; |
| | 9 | | using Jellyfin.Extensions; |
| | 10 | | using MediaBrowser.Controller.Dto; |
| | 11 | | using MediaBrowser.Controller.Entities; |
| | 12 | | using MediaBrowser.Controller.Library; |
| | 13 | | using MediaBrowser.Model.Dto; |
| | 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 | | /// The suggestions controller. |
| | 23 | | /// </summary> |
| | 24 | | [Route("")] |
| | 25 | | [Authorize] |
| | 26 | | public class SuggestionsController : BaseJellyfinApiController |
| | 27 | | { |
| | 28 | | private readonly IDtoService _dtoService; |
| | 29 | | private readonly IUserManager _userManager; |
| | 30 | | private readonly ILibraryManager _libraryManager; |
| | 31 | |
|
| | 32 | | /// <summary> |
| | 33 | | /// Initializes a new instance of the <see cref="SuggestionsController"/> class. |
| | 34 | | /// </summary> |
| | 35 | | /// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param> |
| | 36 | | /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param> |
| | 37 | | /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param> |
| 0 | 38 | | public SuggestionsController( |
| 0 | 39 | | IDtoService dtoService, |
| 0 | 40 | | IUserManager userManager, |
| 0 | 41 | | ILibraryManager libraryManager) |
| | 42 | | { |
| 0 | 43 | | _dtoService = dtoService; |
| 0 | 44 | | _userManager = userManager; |
| 0 | 45 | | _libraryManager = libraryManager; |
| 0 | 46 | | } |
| | 47 | |
|
| | 48 | | /// <summary> |
| | 49 | | /// Gets suggestions. |
| | 50 | | /// </summary> |
| | 51 | | /// <param name="userId">The user id.</param> |
| | 52 | | /// <param name="mediaType">The media types.</param> |
| | 53 | | /// <param name="type">The type.</param> |
| | 54 | | /// <param name="startIndex">Optional. The start index.</param> |
| | 55 | | /// <param name="limit">Optional. The limit.</param> |
| | 56 | | /// <param name="enableTotalRecordCount">Whether to enable the total record count.</param> |
| | 57 | | /// <response code="200">Suggestions returned.</response> |
| | 58 | | /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the suggestions.</returns> |
| | 59 | | [HttpGet("Items/Suggestions")] |
| | 60 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 61 | | public ActionResult<QueryResult<BaseItemDto>> GetSuggestions( |
| | 62 | | [FromQuery] Guid? userId, |
| | 63 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] MediaType[] mediaType, |
| | 64 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] BaseItemKind[] type, |
| | 65 | | [FromQuery] int? startIndex, |
| | 66 | | [FromQuery] int? limit, |
| | 67 | | [FromQuery] bool enableTotalRecordCount = false) |
| | 68 | | { |
| | 69 | | User? user; |
| 0 | 70 | | if (userId.IsNullOrEmpty()) |
| | 71 | | { |
| 0 | 72 | | user = null; |
| | 73 | | } |
| | 74 | | else |
| | 75 | | { |
| 0 | 76 | | var requestUserId = RequestHelpers.GetUserId(User, userId); |
| 0 | 77 | | user = _userManager.GetUserById(requestUserId); |
| | 78 | | } |
| | 79 | |
|
| 0 | 80 | | var dtoOptions = new DtoOptions().AddClientFields(User); |
| 0 | 81 | | var result = _libraryManager.GetItemsResult(new InternalItemsQuery(user) |
| 0 | 82 | | { |
| 0 | 83 | | OrderBy = new[] { (ItemSortBy.Random, SortOrder.Descending) }, |
| 0 | 84 | | MediaTypes = mediaType, |
| 0 | 85 | | IncludeItemTypes = type, |
| 0 | 86 | | IsVirtualItem = false, |
| 0 | 87 | | StartIndex = startIndex, |
| 0 | 88 | | Limit = limit, |
| 0 | 89 | | DtoOptions = dtoOptions, |
| 0 | 90 | | EnableTotalRecordCount = enableTotalRecordCount, |
| 0 | 91 | | Recursive = true |
| 0 | 92 | | }); |
| | 93 | |
|
| 0 | 94 | | var dtoList = _dtoService.GetBaseItemDtos(result.Items, dtoOptions, user); |
| | 95 | |
|
| 0 | 96 | | return new QueryResult<BaseItemDto>( |
| 0 | 97 | | startIndex, |
| 0 | 98 | | result.TotalRecordCount, |
| 0 | 99 | | dtoList); |
| | 100 | | } |
| | 101 | |
|
| | 102 | | /// <summary> |
| | 103 | | /// Gets suggestions. |
| | 104 | | /// </summary> |
| | 105 | | /// <param name="userId">The user id.</param> |
| | 106 | | /// <param name="mediaType">The media types.</param> |
| | 107 | | /// <param name="type">The type.</param> |
| | 108 | | /// <param name="startIndex">Optional. The start index.</param> |
| | 109 | | /// <param name="limit">Optional. The limit.</param> |
| | 110 | | /// <param name="enableTotalRecordCount">Whether to enable the total record count.</param> |
| | 111 | | /// <response code="200">Suggestions returned.</response> |
| | 112 | | /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the suggestions.</returns> |
| | 113 | | [HttpGet("Users/{userId}/Suggestions")] |
| | 114 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 115 | | [Obsolete("Kept for backwards compatibility")] |
| | 116 | | [ApiExplorerSettings(IgnoreApi = true)] |
| | 117 | | public ActionResult<QueryResult<BaseItemDto>> GetSuggestionsLegacy( |
| | 118 | | [FromRoute, Required] Guid userId, |
| | 119 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] MediaType[] mediaType, |
| | 120 | | [FromQuery, ModelBinder(typeof(CommaDelimitedCollectionModelBinder))] BaseItemKind[] type, |
| | 121 | | [FromQuery] int? startIndex, |
| | 122 | | [FromQuery] int? limit, |
| | 123 | | [FromQuery] bool enableTotalRecordCount = false) |
| | 124 | | => GetSuggestions(userId, mediaType, type, startIndex, limit, enableTotalRecordCount); |
| | 125 | | } |