using MediaBrowser.Common.Configuration;
using MediaBrowser.Controller.IO;
using MediaBrowser.Controller.Library;
using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Serialization;
using System;
using System.IO;
using System.Runtime.Serialization;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Controller.Entities
{
    /// 
    /// Class User
    /// 
    public class User : BaseItem
    {
        public static IUserManager UserManager { get; set; }
        public static IXmlSerializer XmlSerializer { get; set; }
        /// 
        /// Gets the root folder path.
        /// 
        /// The root folder path.
        [IgnoreDataMember]
        public string RootFolderPath
        {
            get
            {
                var path = Configuration.UseCustomLibrary ? GetRootFolderPath(Name) : ConfigurationManager.ApplicationPaths.DefaultUserViewsPath;
                if (!Directory.Exists(path))
                {
                    Directory.CreateDirectory(path);
                }
                return path;
            }
        }
        /// 
        /// Gets the root folder path based on a given username
        /// 
        /// The username.
        /// System.String.
        private string GetRootFolderPath(string username)
        {
            var safeFolderName = FileSystem.GetValidFilename(username);
            return System.IO.Path.Combine(ConfigurationManager.ApplicationPaths.RootFolderPath, safeFolderName);
        }
        /// 
        /// Gets or sets the password.
        /// 
        /// The password.
        public string Password { get; set; }
        /// 
        /// Gets or sets the path.
        /// 
        /// The path.
        public override string Path
        {
            get
            {
                // Return this so that metadata providers will look in here
                return ConfigurationDirectoryPath;
            }
            set
            {
                base.Path = value;
            }
        }
        /// 
        /// The _root folder
        /// 
        private UserRootFolder _rootFolder;
        /// 
        /// The _user root folder initialized
        /// 
        private bool _userRootFolderInitialized;
        /// 
        /// The _user root folder sync lock
        /// 
        private object _userRootFolderSyncLock = new object();
        /// 
        /// Gets the root folder.
        /// 
        /// The root folder.
        [IgnoreDataMember]
        public UserRootFolder RootFolder
        {
            get
            {
                LazyInitializer.EnsureInitialized(ref _rootFolder, ref _userRootFolderInitialized, ref _userRootFolderSyncLock, () => LibraryManager.GetUserRootFolder(RootFolderPath));
                return _rootFolder;
            }
            private set
            {
                _rootFolder = value;
                if (_rootFolder == null)
                {
                    _userRootFolderInitialized = false;
                }
            }
        }
        /// 
        /// Gets or sets the last login date.
        /// 
        /// The last login date.
        public DateTime? LastLoginDate { get; set; }
        /// 
        /// Gets or sets the last activity date.
        /// 
        /// The last activity date.
        public DateTime? LastActivityDate { get; set; }
        /// 
        /// The _configuration
        /// 
        private UserConfiguration _configuration;
        /// 
        /// The _configuration initialized
        /// 
        private bool _configurationInitialized;
        /// 
        /// The _configuration sync lock
        /// 
        private object _configurationSyncLock = new object();
        /// 
        /// Gets the user's configuration
        /// 
        /// The configuration.
        [IgnoreDataMember]
        public UserConfiguration Configuration
        {
            get
            {
                // Lazy load
                LazyInitializer.EnsureInitialized(ref _configuration, ref _configurationInitialized, ref _configurationSyncLock, () => (UserConfiguration)ConfigurationHelper.GetXmlConfiguration(typeof(UserConfiguration), ConfigurationFilePath, XmlSerializer));
                return _configuration;
            }
            private set
            {
                _configuration = value;
                if (value == null)
                {
                    _configurationInitialized = false;
                }
            }
        }
        /// 
        /// Gets the last date modified of the configuration
        /// 
        /// The configuration date last modified.
        [IgnoreDataMember]
        public DateTime ConfigurationDateLastModified
        {
            get
            {
                // Ensure it's been lazy loaded
                var config = Configuration;
                return File.GetLastWriteTimeUtc(ConfigurationFilePath);
            }
        }
        /// 
        /// Reloads the root media folder
        /// 
        /// The cancellation token.
        /// The progress.
        /// Task.
        public async Task ValidateMediaLibrary(IProgress progress, CancellationToken cancellationToken)
        {
            Logger.Info("Validating media library for {0}", Name);
            await RootFolder.RefreshMetadata(cancellationToken).ConfigureAwait(false);
            cancellationToken.ThrowIfCancellationRequested();
            await RootFolder.ValidateChildren(progress, cancellationToken).ConfigureAwait(false);
        }
        /// 
        /// Renames the user.
        /// 
        /// The new name.
        /// Task.
        /// 
        public Task Rename(string newName)
        {
            if (string.IsNullOrEmpty(newName))
            {
                throw new ArgumentNullException();
            }
            // If only the casing is changing, leave the file system alone
            if (!newName.Equals(Name, StringComparison.OrdinalIgnoreCase))
            {
                // Move configuration
                var newConfigDirectory = GetConfigurationDirectoryPath(newName);
                // Exceptions will be thrown if these paths already exist
                if (Directory.Exists(newConfigDirectory))
                {
                    Directory.Delete(newConfigDirectory, true);
                }
                Directory.Move(ConfigurationDirectoryPath, newConfigDirectory);
                var customLibraryPath = GetRootFolderPath(Name);
                // Move the root folder path if using a custom library
                if (Directory.Exists(customLibraryPath))
                {
                    var newRootFolderPath = GetRootFolderPath(newName);
                    if (Directory.Exists(newRootFolderPath))
                    {
                        Directory.Delete(newRootFolderPath, true);
                    }
                    Directory.Move(customLibraryPath, newRootFolderPath);
                }
            }
            Name = newName;
            // Force these to be lazy loaded again
            _configurationDirectoryPath = null;
            RootFolder = null;
            // Kick off a task to validate the media library
            Task.Run(() => ValidateMediaLibrary(new Progress(), CancellationToken.None));
            return RefreshMetadata(CancellationToken.None, forceSave: true, forceRefresh: true);
        }
        /// 
        /// The _configuration directory path
        /// 
        private string _configurationDirectoryPath;
        /// 
        /// Gets the path to the user's configuration directory
        /// 
        /// The configuration directory path.
        private string ConfigurationDirectoryPath
        {
            get
            {
                if (_configurationDirectoryPath == null)
                {
                    _configurationDirectoryPath = GetConfigurationDirectoryPath(Name);
                    if (!Directory.Exists(_configurationDirectoryPath))
                    {
                        Directory.CreateDirectory(_configurationDirectoryPath);
                    }
                }
                return _configurationDirectoryPath;
            }
        }
        /// 
        /// Gets the configuration directory path.
        /// 
        /// The username.
        /// System.String.
        private string GetConfigurationDirectoryPath(string username)
        {
            var safeFolderName = FileSystem.GetValidFilename(username);
            return System.IO.Path.Combine(ConfigurationManager.ApplicationPaths.UserConfigurationDirectoryPath, safeFolderName);
        }
        /// 
        /// Gets the path to the user's configuration file
        /// 
        /// The configuration file path.
        public string ConfigurationFilePath
        {
            get
            {
                return System.IO.Path.Combine(ConfigurationDirectoryPath, "config.xml");
            }
        }
        /// 
        /// Saves the current configuration to the file system
        /// 
        public void SaveConfiguration(IXmlSerializer serializer)
        {
            serializer.SerializeToFile(Configuration, ConfigurationFilePath);
        }
        /// 
        /// Refresh metadata on us by execution our provider chain
        /// The item will be persisted if a change is made by a provider, or if it's new or changed.
        /// 
        /// The cancellation token.
        /// if set to true [is new item].
        /// if set to true [force].
        /// if set to true [allow slow providers].
        /// true if a provider reports we changed
        public override async Task RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
        {
            if (resetResolveArgs)
            {
                // Reload this
                ResolveArgs = null;
            }
            var updateReason = await ProviderManager.ExecuteMetadataProviders(this, cancellationToken, forceRefresh, allowSlowProviders).ConfigureAwait(false);
            var changed = updateReason.HasValue;
            if (changed || forceSave)
            {
                cancellationToken.ThrowIfCancellationRequested();
                await UserManager.UpdateUser(this).ConfigureAwait(false);
            }
            return changed;
        }
        /// 
        /// Updates the configuration.
        /// 
        /// The config.
        /// The serializer.
        /// config
        public void UpdateConfiguration(UserConfiguration config, IXmlSerializer serializer)
        {
            if (config == null)
            {
                throw new ArgumentNullException("config");
            }
            var customLibraryChanged = config.UseCustomLibrary != Configuration.UseCustomLibrary;
            Configuration = config;
            SaveConfiguration(serializer);
            // Force these to be lazy loaded again
            if (customLibraryChanged)
            {
                RootFolder = null;
            }
        }
    }
}