using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using MediaBrowser.Common.Progress;
using MediaBrowser.Controller;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Net;
using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.IO;
using MediaBrowser.Model.Services;
namespace MediaBrowser.Api.Library
{
    /// 
    /// Class GetDefaultVirtualFolders
    /// 
    [Route("/Library/VirtualFolders", "GET")]
    public class GetVirtualFolders : IReturn>
    {
        /// 
        /// Gets or sets the user id.
        /// 
        /// The user id.
        public string UserId { get; set; }
    }
    [Route("/Library/VirtualFolders", "POST")]
    public class AddVirtualFolder : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        /// 
        /// Gets or sets the type of the collection.
        /// 
        /// The type of the collection.
        public string CollectionType { get; set; }
        /// 
        /// Gets or sets a value indicating whether [refresh library].
        /// 
        /// true if [refresh library]; otherwise, false.
        public bool RefreshLibrary { get; set; }
        /// 
        /// Gets or sets the path.
        /// 
        /// The path.
        public string[] Paths { get; set; }
        public LibraryOptions LibraryOptions { get; set; }
    }
    [Route("/Library/VirtualFolders", "DELETE")]
    public class RemoveVirtualFolder : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        /// 
        /// Gets or sets a value indicating whether [refresh library].
        /// 
        /// true if [refresh library]; otherwise, false.
        public bool RefreshLibrary { get; set; }
    }
    [Route("/Library/VirtualFolders/Name", "POST")]
    public class RenameVirtualFolder : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string NewName { get; set; }
        /// 
        /// Gets or sets a value indicating whether [refresh library].
        /// 
        /// true if [refresh library]; otherwise, false.
        public bool RefreshLibrary { get; set; }
    }
    [Route("/Library/VirtualFolders/Paths", "POST")]
    public class AddMediaPath : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Path { get; set; }
        public MediaPathInfo PathInfo { get; set; }
        /// 
        /// Gets or sets a value indicating whether [refresh library].
        /// 
        /// true if [refresh library]; otherwise, false.
        public bool RefreshLibrary { get; set; }
    }
    [Route("/Library/VirtualFolders/Paths/Update", "POST")]
    public class UpdateMediaPath : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        public MediaPathInfo PathInfo { get; set; }
    }
    [Route("/Library/VirtualFolders/Paths", "DELETE")]
    public class RemoveMediaPath : IReturnVoid
    {
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Name { get; set; }
        /// 
        /// Gets or sets the name.
        /// 
        /// The name.
        public string Path { get; set; }
        /// 
        /// Gets or sets a value indicating whether [refresh library].
        /// 
        /// true if [refresh library]; otherwise, false.
        public bool RefreshLibrary { get; set; }
    }
    [Route("/Library/VirtualFolders/LibraryOptions", "POST")]
    public class UpdateLibraryOptions : IReturnVoid
    {
        public string Id { get; set; }
        public LibraryOptions LibraryOptions { get; set; }
    }
    /// 
    /// Class LibraryStructureService
    /// 
    [Authenticated(Roles = "Admin", AllowBeforeStartupWizard = true)]
    public class LibraryStructureService : BaseApiService
    {
        /// 
        /// The _app paths
        /// 
        private readonly IServerApplicationPaths _appPaths;
        /// 
        /// The _library manager
        /// 
        private readonly ILibraryManager _libraryManager;
        private readonly ILibraryMonitor _libraryMonitor;
        private readonly IFileSystem _fileSystem;
        /// 
        /// Initializes a new instance of the  class.
        /// 
        public LibraryStructureService(IServerApplicationPaths appPaths, ILibraryManager libraryManager, ILibraryMonitor libraryMonitor, IFileSystem fileSystem)
        {
            if (appPaths == null)
            {
                throw new ArgumentNullException(nameof(appPaths));
            }
            _appPaths = appPaths;
            _libraryManager = libraryManager;
            _libraryMonitor = libraryMonitor;
            _fileSystem = fileSystem;
        }
        /// 
        /// Gets the specified request.
        /// 
        /// The request.
        /// System.Object.
        public object Get(GetVirtualFolders request)
        {
            var result = _libraryManager.GetVirtualFolders(true);
            return ToOptimizedResult(result);
        }
        public void Post(UpdateLibraryOptions request)
        {
            var collectionFolder = (CollectionFolder)_libraryManager.GetItemById(request.Id);
            collectionFolder.UpdateLibraryOptions(request.LibraryOptions);
        }
        /// 
        /// Posts the specified request.
        /// 
        /// The request.
        public Task Post(AddVirtualFolder request)
        {
            var libraryOptions = request.LibraryOptions ?? new LibraryOptions();
            if (request.Paths != null && request.Paths.Length > 0)
            {
                libraryOptions.PathInfos = request.Paths.Select(i => new MediaPathInfo { Path = i }).ToArray();
            }
            return _libraryManager.AddVirtualFolder(request.Name, request.CollectionType, libraryOptions, request.RefreshLibrary);
        }
        /// 
        /// Posts the specified request.
        /// 
        /// The request.
        public void Post(RenameVirtualFolder request)
        {
            if (string.IsNullOrWhiteSpace(request.Name))
            {
                throw new ArgumentNullException(nameof(request));
            }
            if (string.IsNullOrWhiteSpace(request.NewName))
            {
                throw new ArgumentNullException(nameof(request));
            }
            var rootFolderPath = _appPaths.DefaultUserViewsPath;
            var currentPath = Path.Combine(rootFolderPath, request.Name);
            var newPath = Path.Combine(rootFolderPath, request.NewName);
            if (!_fileSystem.DirectoryExists(currentPath))
            {
                throw new FileNotFoundException("The media collection does not exist");
            }
            if (!string.Equals(currentPath, newPath, StringComparison.OrdinalIgnoreCase) && _fileSystem.DirectoryExists(newPath))
            {
                throw new ArgumentException("Media library already exists at " + newPath + ".");
            }
            _libraryMonitor.Stop();
            try
            {
                // Changing capitalization. Handle windows case insensitivity
                if (string.Equals(currentPath, newPath, StringComparison.OrdinalIgnoreCase))
                {
                    var tempPath = Path.Combine(rootFolderPath, Guid.NewGuid().ToString("N"));
                    _fileSystem.MoveDirectory(currentPath, tempPath);
                    currentPath = tempPath;
                }
                _fileSystem.MoveDirectory(currentPath, newPath);
            }
            finally
            {
                CollectionFolder.OnCollectionFolderChange();
                Task.Run(() =>
                {
                    // No need to start if scanning the library because it will handle it
                    if (request.RefreshLibrary)
                    {
                        _libraryManager.ValidateMediaLibrary(new SimpleProgress(), CancellationToken.None);
                    }
                    else
                    {
                        // Need to add a delay here or directory watchers may still pick up the changes
                        var task = Task.Delay(1000);
                        // Have to block here to allow exceptions to bubble
                        Task.WaitAll(task);
                        _libraryMonitor.Start();
                    }
                });
            }
        }
        /// 
        /// Deletes the specified request.
        /// 
        /// The request.
        public Task Delete(RemoveVirtualFolder request)
        {
            return _libraryManager.RemoveVirtualFolder(request.Name, request.RefreshLibrary);
        }
        /// 
        /// Posts the specified request.
        /// 
        /// The request.
        public void Post(AddMediaPath request)
        {
            if (string.IsNullOrWhiteSpace(request.Name))
            {
                throw new ArgumentNullException(nameof(request));
            }
            _libraryMonitor.Stop();
            try
            {
                var mediaPath = request.PathInfo;
                if (mediaPath == null)
                {
                    mediaPath = new MediaPathInfo
                    {
                        Path = request.Path
                    };
                }
                _libraryManager.AddMediaPath(request.Name, mediaPath);
            }
            finally
            {
                Task.Run(() =>
                {
                    // No need to start if scanning the library because it will handle it
                    if (request.RefreshLibrary)
                    {
                        _libraryManager.ValidateMediaLibrary(new SimpleProgress(), CancellationToken.None);
                    }
                    else
                    {
                        // Need to add a delay here or directory watchers may still pick up the changes
                        var task = Task.Delay(1000);
                        // Have to block here to allow exceptions to bubble
                        Task.WaitAll(task);
                        _libraryMonitor.Start();
                    }
                });
            }
        }
        /// 
        /// Posts the specified request.
        /// 
        /// The request.
        public void Post(UpdateMediaPath request)
        {
            if (string.IsNullOrWhiteSpace(request.Name))
            {
                throw new ArgumentNullException(nameof(request));
            }
            _libraryManager.UpdateMediaPath(request.Name, request.PathInfo);
        }
        /// 
        /// Deletes the specified request.
        /// 
        /// The request.
        public void Delete(RemoveMediaPath request)
        {
            if (string.IsNullOrWhiteSpace(request.Name))
            {
                throw new ArgumentNullException(nameof(request));
            }
            _libraryMonitor.Stop();
            try
            {
                _libraryManager.RemoveMediaPath(request.Name, request.Path);
            }
            finally
            {
                Task.Run(() =>
                {
                    // No need to start if scanning the library because it will handle it
                    if (request.RefreshLibrary)
                    {
                        _libraryManager.ValidateMediaLibrary(new SimpleProgress(), CancellationToken.None);
                    }
                    else
                    {
                        // Need to add a delay here or directory watchers may still pick up the changes
                        var task = Task.Delay(1000);
                        // Have to block here to allow exceptions to bubble
                        Task.WaitAll(task);
                        _libraryMonitor.Start();
                    }
                });
            }
        }
    }
}