using System.Threading.Tasks;
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Net;
using MediaBrowser.Controller.Session;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
using ServiceStack.Text.Controller;
using ServiceStack.Web;
using System;
using System.Collections.Generic;
using System.Linq;
namespace MediaBrowser.Api
{
    /// 
    /// Class BaseApiService
    /// 
    public class BaseApiService : IHasResultFactory, IRestfulService, IHasSession
    {
        /// 
        /// Gets or sets the logger.
        /// 
        /// The logger.
        public ILogger Logger { get; set; }
        /// 
        /// Gets or sets the HTTP result factory.
        /// 
        /// The HTTP result factory.
        public IHttpResultFactory ResultFactory { get; set; }
        /// 
        /// Gets or sets the request context.
        /// 
        /// The request context.
        public IRequest Request { get; set; }
        public ISessionContext SessionContext { get; set; }
        public IAuthorizationContext AuthorizationContext { get; set; }
        public string GetHeader(string name)
        {
            return Request.Headers[name];
        }
        /// 
        /// To the optimized result.
        /// 
        /// 
        /// The result.
        /// System.Object.
        protected object ToOptimizedResult(T result)
            where T : class
        {
            return ResultFactory.GetOptimizedResult(Request, result);
        }
        /// 
        /// To the optimized result using cache.
        /// 
        /// 
        /// The cache key.
        /// The last date modified.
        /// Duration of the cache.
        /// The factory function.
        /// System.Object.
        protected object ToOptimizedResultUsingCache(Guid cacheKey, DateTime? lastDateModified, TimeSpan? cacheDuration, Func factoryFn)
           where T : class
        {
            return ResultFactory.GetOptimizedResultUsingCache(Request, cacheKey, lastDateModified, cacheDuration, factoryFn);
        }
        protected void AssertCanUpdateUser(IUserManager userManager, string userId)
        {
            var auth = AuthorizationContext.GetAuthorizationInfo(Request);
            var authenticatedUser = userManager.GetUserById(auth.UserId);
            // If they're going to update the record of another user, they must be an administrator
            if (!string.Equals(userId, auth.UserId, StringComparison.OrdinalIgnoreCase))
            {
                if (!authenticatedUser.Policy.IsAdministrator)
                {
                    throw new SecurityException("Unauthorized access.");
                }
            }
            else
            {
                if (!authenticatedUser.Policy.EnableUserPreferenceAccess)
                {
                    throw new SecurityException("Unauthorized access.");
                }
            }
        }
        
        /// 
        /// To the optimized serialized result using cache.
        /// 
        /// 
        /// The result.
        /// System.Object.
        protected object ToOptimizedSerializedResultUsingCache(T result)
           where T : class
        {
            return ToOptimizedResult(result);
        }
        /// 
        /// Gets the session.
        /// 
        /// SessionInfo.
        protected async Task GetSession()
        {
            var session = await SessionContext.GetSession(Request).ConfigureAwait(false);
            if (session == null)
            {
                throw new ArgumentException("Session not found.");
            }
            return session;
        }
        /// 
        /// To the static file result.
        /// 
        /// The path.
        /// System.Object.
        protected object ToStaticFileResult(string path)
        {
            return ResultFactory.GetStaticFileResult(Request, path);
        }
        private readonly char[] _dashReplaceChars = { '?', '/', '&' };
        private const char SlugChar = '-';
        protected DtoOptions GetDtoOptions(object request)
        {
            var options = new DtoOptions();
            options.DeviceId = AuthorizationContext.GetAuthorizationInfo(Request).DeviceId;
            var hasFields = request as IHasItemFields;
            if (hasFields != null)
            {
                options.Fields = hasFields.GetItemFields().ToList();
            }
            var hasDtoOptions = request as IHasDtoOptions;
            if (hasDtoOptions != null)
            {
                options.EnableImages = hasDtoOptions.EnableImages ?? true;
                if (hasDtoOptions.ImageTypeLimit.HasValue)
                {
                    options.ImageTypeLimit = hasDtoOptions.ImageTypeLimit.Value;
                }
                if (!string.IsNullOrWhiteSpace(hasDtoOptions.EnableImageTypes))
                {
                    options.ImageTypes = (hasDtoOptions.EnableImageTypes ?? string.Empty).Split(',').Where(i => !string.IsNullOrWhiteSpace(i)).Select(v => (ImageType)Enum.Parse(typeof(ImageType), v, true)).ToList();
                }
            }
            return options;
        }
        protected MusicArtist GetArtist(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetArtist(DeSlugArtistName(name, libraryManager));
        }
        protected Studio GetStudio(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetStudio(DeSlugStudioName(name, libraryManager));
        }
        protected Genre GetGenre(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetGenre(DeSlugGenreName(name, libraryManager));
        }
        protected MusicGenre GetMusicGenre(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetMusicGenre(DeSlugGenreName(name, libraryManager));
        }
        protected GameGenre GetGameGenre(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetGameGenre(DeSlugGameGenreName(name, libraryManager));
        }
        protected Person GetPerson(string name, ILibraryManager libraryManager)
        {
            return libraryManager.GetPerson(DeSlugPersonName(name, libraryManager));
        }
        protected IList GetAllLibraryItems(Guid? userId, IUserManager userManager, ILibraryManager libraryManager, string parentId, Func filter)
        {
            if (!string.IsNullOrEmpty(parentId))
            {
                var folder = (Folder)libraryManager.GetItemById(new Guid(parentId));
                if (userId.HasValue)
                {
                    var user = userManager.GetUserById(userId.Value);
                    if (user == null)
                    {
                        throw new ArgumentException("User not found");
                    }
                    return folder
                        .GetRecursiveChildren(user, filter)
                        .ToList();
                }
                return folder
                    .GetRecursiveChildren(filter);
            }
            if (userId.HasValue)
            {
                var user = userManager.GetUserById(userId.Value);
                if (user == null)
                {
                    throw new ArgumentException("User not found");
                }
                return userManager
                    .GetUserById(userId.Value)
                    .RootFolder
                    .GetRecursiveChildren(user, filter)
                    .ToList();
            }
            return libraryManager
                .RootFolder
                .GetRecursiveChildren(filter);
        }
        /// 
        /// Deslugs an artist name by finding the correct entry in the library
        /// 
        /// 
        /// 
        /// 
        protected string DeSlugArtistName(string name, ILibraryManager libraryManager)
        {
            if (name.IndexOf(SlugChar) == -1)
            {
                return name;
            }
            return libraryManager.RootFolder
                .GetRecursiveChildren(i => i is IHasArtist)
                .Cast()
                .SelectMany(i => i.AllArtists)
                .Distinct(StringComparer.OrdinalIgnoreCase)
                .FirstOrDefault(i =>
                {
                    i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
                    return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
                }) ?? name;
        }
        /// 
        /// Deslugs a genre name by finding the correct entry in the library
        /// 
        protected string DeSlugGenreName(string name, ILibraryManager libraryManager)
        {
            if (name.IndexOf(SlugChar) == -1)
            {
                return name;
            }
            return libraryManager.RootFolder.GetRecursiveChildren()
                .SelectMany(i => i.Genres)
                .Distinct(StringComparer.OrdinalIgnoreCase)
                .FirstOrDefault(i =>
                {
                    i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
                    return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
                }) ?? name;
        }
        protected string DeSlugGameGenreName(string name, ILibraryManager libraryManager)
        {
            if (name.IndexOf(SlugChar) == -1)
            {
                return name;
            }
            return libraryManager.RootFolder
                .GetRecursiveChildren(i => i is Game)
                .SelectMany(i => i.Genres)
                .Distinct(StringComparer.OrdinalIgnoreCase)
                .FirstOrDefault(i =>
                {
                    i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
                    return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
                }) ?? name;
        }
        /// 
        /// Deslugs a studio name by finding the correct entry in the library
        /// 
        protected string DeSlugStudioName(string name, ILibraryManager libraryManager)
        {
            if (name.IndexOf(SlugChar) == -1)
            {
                return name;
            }
            return libraryManager.RootFolder
                .GetRecursiveChildren()
                .SelectMany(i => i.Studios)
                .Distinct(StringComparer.OrdinalIgnoreCase)
                .FirstOrDefault(i =>
                {
                    i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
                    return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
                }) ?? name;
        }
        /// 
        /// Deslugs a person name by finding the correct entry in the library
        /// 
        protected string DeSlugPersonName(string name, ILibraryManager libraryManager)
        {
            if (name.IndexOf(SlugChar) == -1)
            {
                return name;
            }
            return libraryManager.RootFolder
                .GetRecursiveChildren()
                .SelectMany(i => i.People)
                .Select(i => i.Name)
                .Distinct(StringComparer.OrdinalIgnoreCase)
                .FirstOrDefault(i =>
                {
                    i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
                    return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
                }) ?? name;
        }
        protected string GetPathValue(int index)
        {
            var pathInfo = PathInfo.Parse(Request.PathInfo);
            var first = pathInfo.GetArgumentValue(0);
            // backwards compatibility
            if (string.Equals(first, "mediabrowser", StringComparison.OrdinalIgnoreCase))
            {
                index++;
            }
            return pathInfo.GetArgumentValue(index);
        }
        /// 
        /// Gets the name of the item by.
        /// 
        /// The name.
        /// The type.
        /// The library manager.
        /// Task{BaseItem}.
        protected BaseItem GetItemByName(string name, string type, ILibraryManager libraryManager)
        {
            BaseItem item;
            if (type.IndexOf("Person", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetPerson(name, libraryManager);
            }
            else if (type.IndexOf("Artist", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetArtist(name, libraryManager);
            }
            else if (type.IndexOf("Genre", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetGenre(name, libraryManager);
            }
            else if (type.IndexOf("MusicGenre", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetMusicGenre(name, libraryManager);
            }
            else if (type.IndexOf("GameGenre", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetGameGenre(name, libraryManager);
            }
            else if (type.IndexOf("Studio", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = GetStudio(name, libraryManager);
            }
            else if (type.IndexOf("Year", StringComparison.OrdinalIgnoreCase) == 0)
            {
                item = libraryManager.GetYear(int.Parse(name));
            }
            else
            {
                throw new ArgumentException();
            }
            return item;
        }
    }
}