using System;
using System.Linq;
using Jellyfin.Data.Entities;
using Jellyfin.Data.Enums;
using MediaBrowser.Common.Extensions;
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Net;
using MediaBrowser.Controller.Session;
using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Querying;
using Microsoft.AspNetCore.Http;
namespace Jellyfin.Api.Helpers
{
    /// 
    /// Request Extensions.
    /// 
    public static class RequestHelpers
    {
        /// 
        /// Get Order By.
        /// 
        /// Sort By. Comma delimited string.
        /// Sort Order. Comma delimited string.
        /// Order By.
        public static ValueTuple[] GetOrderBy(string? sortBy, string? requestedSortOrder)
        {
            var val = sortBy;
            if (string.IsNullOrEmpty(val))
            {
                return Array.Empty>();
            }
            var vals = val.Split(',');
            if (string.IsNullOrWhiteSpace(requestedSortOrder))
            {
                requestedSortOrder = "Ascending";
            }
            var sortOrders = requestedSortOrder.Split(',');
            var result = new ValueTuple[vals.Length];
            for (var i = 0; i < vals.Length; i++)
            {
                var sortOrderIndex = sortOrders.Length > i ? i : 0;
                var sortOrderValue = sortOrders.Length > sortOrderIndex ? sortOrders[sortOrderIndex] : null;
                var sortOrder = string.Equals(sortOrderValue, "Descending", StringComparison.OrdinalIgnoreCase)
                    ? SortOrder.Descending
                    : SortOrder.Ascending;
                result[i] = new ValueTuple(vals[i], sortOrder);
            }
            return result;
        }
        /// 
        /// Splits a string at a separating character into an array of substrings.
        /// 
        /// The string to split.
        /// The char that separates the substrings.
        /// Option to remove empty substrings from the array.
        /// An array of the substrings.
        internal static string[] Split(string? value, char separator, bool removeEmpty)
        {
            if (string.IsNullOrWhiteSpace(value))
            {
                return Array.Empty();
            }
            return removeEmpty
                ? value.Split(separator, StringSplitOptions.RemoveEmptyEntries)
                : value.Split(separator);
        }
        /// 
        /// Checks if the user can update an entry.
        /// 
        /// Instance of the  interface.
        /// The .
        /// The user id.
        /// Whether to restrict the user preferences.
        /// A  whether the user can update the entry.
        internal static bool AssertCanUpdateUser(IAuthorizationContext authContext, HttpRequest requestContext, Guid userId, bool restrictUserPreferences)
        {
            var auth = authContext.GetAuthorizationInfo(requestContext);
            var authenticatedUser = auth.User;
            // If they're going to update the record of another user, they must be an administrator
            if ((!userId.Equals(auth.UserId) && !authenticatedUser.HasPermission(PermissionKind.IsAdministrator))
                || (restrictUserPreferences && !authenticatedUser.EnableUserPreferenceAccess))
            {
                return false;
            }
            return true;
        }
        internal static SessionInfo GetSession(ISessionManager sessionManager, IAuthorizationContext authContext, HttpRequest request)
        {
            var authorization = authContext.GetAuthorizationInfo(request);
            var user = authorization.User;
            var session = sessionManager.LogSessionActivity(
                authorization.Client,
                authorization.Version,
                authorization.DeviceId,
                authorization.Device,
                request.HttpContext.GetNormalizedRemoteIp(),
                user);
            if (session == null)
            {
                throw new ArgumentException("Session not found.");
            }
            return session;
        }
        internal static QueryResult CreateQueryResult(
            QueryResult<(BaseItem, 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
            {
                Items = dtos.ToArray(),
                TotalRecordCount = result.TotalRecordCount
            };
        }
    }
}