123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Security.Claims;
- using System.Threading.Tasks;
- using Jellyfin.Api.Constants;
- using Jellyfin.Api.Extensions;
- using Jellyfin.Data.Enums;
- using Jellyfin.Database.Implementations.Entities;
- using Jellyfin.Database.Implementations.Enums;
- using Jellyfin.Extensions;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Controller.Dto;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Library;
- using MediaBrowser.Controller.Net;
- using MediaBrowser.Controller.Session;
- using MediaBrowser.Model.Dto;
- using MediaBrowser.Model.Querying;
- using Microsoft.AspNetCore.Http;
- namespace Jellyfin.Api.Helpers;
- /// <summary>
- /// Request Extensions.
- /// </summary>
- public static class RequestHelpers
- {
- /// <summary>
- /// Get Order By.
- /// </summary>
- /// <param name="sortBy">Sort By. Comma delimited string.</param>
- /// <param name="requestedSortOrder">Sort Order. Comma delimited string.</param>
- /// <returns>Order By.</returns>
- public static (ItemSortBy, SortOrder)[] GetOrderBy(IReadOnlyList<ItemSortBy> sortBy, IReadOnlyList<SortOrder> requestedSortOrder)
- {
- if (sortBy.Count == 0)
- {
- return Array.Empty<(ItemSortBy, SortOrder)>();
- }
- var result = new (ItemSortBy, SortOrder)[sortBy.Count];
- var i = 0;
- // Add elements which have a SortOrder specified
- for (; i < requestedSortOrder.Count; i++)
- {
- result[i] = (sortBy[i], requestedSortOrder[i]);
- }
- // Add remaining elements with the first specified SortOrder
- // or the default one if no SortOrders are specified
- var order = requestedSortOrder.Count > 0 ? requestedSortOrder[0] : SortOrder.Ascending;
- for (; i < sortBy.Count; i++)
- {
- result[i] = (sortBy[i], order);
- }
- return result;
- }
- /// <summary>
- /// Checks if the user can access a user.
- /// </summary>
- /// <param name="claimsPrincipal">The <see cref="ClaimsPrincipal"/> for the current request.</param>
- /// <param name="userId">The user id.</param>
- /// <returns>A <see cref="bool"/> whether the user can access the user.</returns>
- internal static Guid GetUserId(ClaimsPrincipal claimsPrincipal, Guid? userId)
- {
- var authenticatedUserId = claimsPrincipal.GetUserId();
- // UserId not provided, fall back to authenticated user id.
- if (userId.IsNullOrEmpty())
- {
- return authenticatedUserId;
- }
- // User must be administrator to access another user.
- var isAdministrator = claimsPrincipal.IsInRole(UserRoles.Administrator);
- if (!userId.Value.Equals(authenticatedUserId) && !isAdministrator)
- {
- throw new SecurityException("Forbidden");
- }
- return userId.Value;
- }
- /// <summary>
- /// Checks if the user can update an entry.
- /// </summary>
- /// <param name="claimsPrincipal">The <see cref="ClaimsPrincipal"/> for the current request.</param>
- /// <param name="user">The user id.</param>
- /// <param name="restrictUserPreferences">Whether to restrict the user preferences.</param>
- /// <returns>A <see cref="bool"/> whether the user can update the entry.</returns>
- internal static bool AssertCanUpdateUser(ClaimsPrincipal claimsPrincipal, User user, bool restrictUserPreferences)
- {
- var authenticatedUserId = claimsPrincipal.GetUserId();
- var isAdministrator = claimsPrincipal.IsInRole(UserRoles.Administrator);
- // If they're going to update the record of another user, they must be an administrator
- if (!user.Id.Equals(authenticatedUserId) && !isAdministrator)
- {
- return false;
- }
- // TODO the EnableUserPreferenceAccess policy does not seem to be used elsewhere
- if (!restrictUserPreferences || isAdministrator)
- {
- return true;
- }
- return user.EnableUserPreferenceAccess;
- }
- internal static async Task<SessionInfo> GetSession(ISessionManager sessionManager, IUserManager userManager, HttpContext httpContext, Guid? userId = null)
- {
- userId ??= httpContext.User.GetUserId();
- User? user = null;
- if (!userId.IsNullOrEmpty())
- {
- user = userManager.GetUserById(userId.Value);
- }
- var session = await sessionManager.LogSessionActivity(
- httpContext.User.GetClient(),
- httpContext.User.GetVersion(),
- httpContext.User.GetDeviceId(),
- httpContext.User.GetDevice(),
- httpContext.GetNormalizedRemoteIP().ToString(),
- user).ConfigureAwait(false);
- if (session is null)
- {
- throw new ResourceNotFoundException("Session not found.");
- }
- return session;
- }
- internal static async Task<string> GetSessionId(ISessionManager sessionManager, IUserManager userManager, HttpContext httpContext)
- {
- var session = await GetSession(sessionManager, userManager, httpContext).ConfigureAwait(false);
- return session.Id;
- }
- internal static QueryResult<BaseItemDto> CreateQueryResult(
- QueryResult<(BaseItem Item, ItemCounts ItemCounts)> result,
- DtoOptions dtoOptions,
- IDtoService dtoService,
- bool includeItemTypes,
- User? user)
- {
- var dtos = result.Items.Select(i =>
- {
- var (baseItem, counts) = i;
- var dto = dtoService.GetItemByNameDto(baseItem, dtoOptions, null, user);
- if (includeItemTypes)
- {
- dto.ChildCount = counts.ItemCount;
- dto.ProgramCount = counts.ProgramCount;
- dto.SeriesCount = counts.SeriesCount;
- dto.EpisodeCount = counts.EpisodeCount;
- dto.MovieCount = counts.MovieCount;
- dto.TrailerCount = counts.TrailerCount;
- dto.AlbumCount = counts.AlbumCount;
- dto.SongCount = counts.SongCount;
- dto.ArtistCount = counts.ArtistCount;
- }
- return dto;
- });
- return new QueryResult<BaseItemDto>(
- result.StartIndex,
- result.TotalRecordCount,
- dtos.ToArray());
- }
- }
|