using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Logging;
using System;
using System.Collections.Generic;
using CommonIO;
namespace MediaBrowser.Controller.IO
{
    /// 
    /// Provides low level File access that is much faster than the File/Directory api's
    /// 
    public static class FileData
    {
        /// 
        /// Gets the filtered file system entries.
        /// 
        /// The directory service.
        /// The path.
        /// The file system.
        /// The logger.
        /// The args.
        /// The flatten folder depth.
        /// if set to true [resolve shortcuts].
        /// Dictionary{System.StringFileSystemInfo}.
        /// path
        public static Dictionary GetFilteredFileSystemEntries(IDirectoryService directoryService,
            string path,
            IFileSystem fileSystem,
            ILogger logger,
            ItemResolveArgs args,
            int flattenFolderDepth = 0,
            bool resolveShortcuts = true)
        {
            if (string.IsNullOrEmpty(path))
            {
                throw new ArgumentNullException("path");
            }
            if (args == null)
            {
                throw new ArgumentNullException("args");
            }
            if (!resolveShortcuts && flattenFolderDepth == 0)
            {
                return directoryService.GetFileSystemDictionary(path);
            }
            var entries = directoryService.GetFileSystemEntries(path);
            var dict = new Dictionary(StringComparer.OrdinalIgnoreCase);
            foreach (var entry in entries)
            {
                var isDirectory = entry.IsDirectory;
                var fullName = entry.FullName;
                if (resolveShortcuts && fileSystem.IsShortcut(fullName))
                {
                    try
                    {
                        var newPath = fileSystem.ResolveShortcut(fullName);
                        if (string.IsNullOrWhiteSpace(newPath))
                        {
                            //invalid shortcut - could be old or target could just be unavailable
                            logger.Warn("Encountered invalid shortcut: " + fullName);
                            continue;
                        }
                        // Don't check if it exists here because that could return false for network shares.
                        var data = fileSystem.GetDirectoryInfo(newPath);
                        // add to our physical locations
                        args.AddAdditionalLocation(newPath);
                        dict[newPath] = data;
                    }
                    catch (Exception ex)
                    {
                        logger.ErrorException("Error resolving shortcut from {0}", ex, fullName);
                    }
                }
                else if (flattenFolderDepth > 0 && isDirectory)
                {
                    foreach (var child in GetFilteredFileSystemEntries(directoryService, fullName, fileSystem, logger, args, flattenFolderDepth: flattenFolderDepth - 1, resolveShortcuts: resolveShortcuts))
                    {
                        dict[child.Key] = child.Value;
                    }
                }
                else
                {
                    dict[fullName] = entry;
                }
            }
            return dict;
        }
    }
}