mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 00:02:29 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			331 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			331 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Collections.Immutable;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using System.Text.RegularExpressions;
 | |
| using System.Threading.Tasks;
 | |
| using API.Interfaces.Services;
 | |
| using Microsoft.Extensions.Logging;
 | |
| 
 | |
| namespace API.Services
 | |
| {
 | |
|     public class DirectoryService : IDirectoryService
 | |
|     {
 | |
|        private readonly ILogger<DirectoryService> _logger;
 | |
|        private static readonly Regex ExcludeDirectories = new Regex(
 | |
|           @"@eaDir|\.DS_Store", 
 | |
|           RegexOptions.Compiled | RegexOptions.IgnoreCase);
 | |
| 
 | |
|        public DirectoryService(ILogger<DirectoryService> logger)
 | |
|        {
 | |
|           _logger = logger;
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Given a set of regex search criteria, get files in the given path. 
 | |
|        /// </summary>
 | |
|        /// <param name="path">Directory to search</param>
 | |
|        /// <param name="searchPatternExpression">Regex version of search pattern (ie \.mp3|\.mp4). Defaults to * meaning all files.</param>
 | |
|        /// <param name="searchOption">SearchOption to use, defaults to TopDirectoryOnly</param>
 | |
|        /// <returns>List of file paths</returns>
 | |
|        private static IEnumerable<string> GetFilesWithCertainExtensions(string path, 
 | |
|           string searchPatternExpression = "",
 | |
|           SearchOption searchOption = SearchOption.TopDirectoryOnly)
 | |
|        {
 | |
|           if (!Directory.Exists(path)) return ImmutableList<string>.Empty;
 | |
|           var reSearchPattern = new Regex(searchPatternExpression, RegexOptions.IgnoreCase);
 | |
|           return Directory.EnumerateFiles(path, "*", searchOption)
 | |
|              .Where(file =>
 | |
|                 reSearchPattern.IsMatch(Path.GetExtension(file)));
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Returns a list of folders from end of fullPath to rootPath.
 | |
|        ///
 | |
|        /// Example) (C:/Manga/, C:/Manga/Love Hina/Specials/Omake/) returns [Omake, Specials, Love Hina]
 | |
|        /// </summary>
 | |
|        /// <param name="rootPath"></param>
 | |
|        /// <param name="fullPath"></param>
 | |
|        /// <returns></returns>
 | |
|        public static IEnumerable<string> GetFoldersTillRoot(string rootPath, string fullPath)
 | |
|        {
 | |
|           var separator = Path.AltDirectorySeparatorChar;
 | |
|           if (fullPath.Contains(Path.DirectorySeparatorChar))
 | |
|           {
 | |
|              fullPath = fullPath.Replace(Path.DirectorySeparatorChar, Path.AltDirectorySeparatorChar);
 | |
|           }
 | |
|           
 | |
|           if (rootPath.Contains(Path.DirectorySeparatorChar))
 | |
|           {
 | |
|              rootPath = rootPath.Replace(Path.DirectorySeparatorChar, Path.AltDirectorySeparatorChar);
 | |
|           }
 | |
| 
 | |
|           var path = fullPath.EndsWith(separator) ? fullPath.Substring(0, fullPath.Length - 1) : fullPath;
 | |
|           var root = rootPath.EndsWith(separator) ? rootPath.Substring(0, rootPath.Length - 1) : rootPath;
 | |
|           var paths = new List<string>();
 | |
|           while (Path.GetDirectoryName(path) != Path.GetDirectoryName(root))
 | |
|           {
 | |
|              var folder = new DirectoryInfo(path).Name;
 | |
|              paths.Add(folder);
 | |
|              path = path.Replace(separator + folder, string.Empty);
 | |
|           }
 | |
| 
 | |
|           return paths;
 | |
|        }
 | |
| 
 | |
|        public bool Exists(string directory)
 | |
|        {
 | |
|           var di = new DirectoryInfo(directory);
 | |
|           return di.Exists;
 | |
|        }
 | |
| 
 | |
|        public IEnumerable<string> GetFiles(string path, string searchPatternExpression = "", 
 | |
|           SearchOption searchOption = SearchOption.TopDirectoryOnly)
 | |
|        {
 | |
|           if (searchPatternExpression != string.Empty)
 | |
|           {
 | |
|              if (!Directory.Exists(path)) return ImmutableList<string>.Empty;
 | |
|              var reSearchPattern = new Regex(searchPatternExpression, RegexOptions.IgnoreCase);
 | |
|              return Directory.EnumerateFiles(path, "*", searchOption)
 | |
|                 .Where(file =>
 | |
|                    reSearchPattern.IsMatch(file));
 | |
|           }
 | |
|           
 | |
|           return !Directory.Exists(path) ? Array.Empty<string>() : Directory.GetFiles(path);
 | |
|        }
 | |
| 
 | |
|        public string[] GetFilesWithExtension(string path, string searchPatternExpression = "")
 | |
|        {
 | |
|           if (searchPatternExpression != string.Empty)
 | |
|           {
 | |
|              return GetFilesWithCertainExtensions(path, searchPatternExpression).ToArray();
 | |
|           }
 | |
|           
 | |
|           return !Directory.Exists(path) ? Array.Empty<string>() : Directory.GetFiles(path);
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Returns the total number of bytes for a given set of full file paths
 | |
|        /// </summary>
 | |
|        /// <param name="paths"></param>
 | |
|        /// <returns>Total bytes</returns>
 | |
|        public static long GetTotalSize(IEnumerable<string> paths)
 | |
|        {
 | |
|           return paths.Sum(path => new FileInfo(path).Length);
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Returns true if the path exists and is a directory. If path does not exist, this will create it. Returns false in all fail cases.
 | |
|        /// </summary>
 | |
|        /// <param name="directoryPath"></param>
 | |
|        /// <returns></returns>
 | |
|        public static bool ExistOrCreate(string directoryPath)
 | |
|        {
 | |
|           var di = new DirectoryInfo(directoryPath);
 | |
|           if (di.Exists) return true;
 | |
|           try
 | |
|           {
 | |
|              Directory.CreateDirectory(directoryPath);
 | |
|           }
 | |
|           catch (Exception)
 | |
|           {
 | |
|              return false;
 | |
|           }
 | |
|           return true;
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Deletes all files within the directory, then the directory itself.
 | |
|        /// </summary>
 | |
|        /// <param name="directoryPath"></param>
 | |
|        public static void ClearAndDeleteDirectory(string directoryPath)
 | |
|        {
 | |
|           if (!Directory.Exists(directoryPath)) return;
 | |
|           
 | |
|           DirectoryInfo di = new DirectoryInfo(directoryPath);
 | |
| 
 | |
|           ClearDirectory(directoryPath);
 | |
|           
 | |
|           di.Delete(true);
 | |
|        }
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Deletes all files within the directory.
 | |
|        /// </summary>
 | |
|        /// <param name="directoryPath"></param>
 | |
|        /// <returns></returns>
 | |
|        public static void ClearDirectory(string directoryPath)
 | |
|        {
 | |
|           var di = new DirectoryInfo(directoryPath);
 | |
|           if (!di.Exists) return;
 | |
| 
 | |
|           foreach (var file in di.EnumerateFiles())
 | |
|           {
 | |
|              file.Delete(); 
 | |
|           }
 | |
|           foreach (var dir in di.EnumerateDirectories())
 | |
|           {
 | |
|              dir.Delete(true); 
 | |
|           }
 | |
|        }
 | |
| 
 | |
|        public bool CopyFilesToDirectory(IEnumerable<string> filePaths, string directoryPath)
 | |
|        {
 | |
|           string currentFile = null;
 | |
|           try
 | |
|           {
 | |
|              foreach (var file in filePaths)
 | |
|              {
 | |
|                 currentFile = file;
 | |
|                 var fileInfo = new FileInfo(file);
 | |
|                 if (fileInfo.Exists)
 | |
|                 {
 | |
|                    fileInfo.CopyTo(Path.Join(directoryPath, fileInfo.Name));   
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                    _logger.LogWarning("Tried to copy {File} but it doesn't exist", file);
 | |
|                 }
 | |
|                 
 | |
|              }
 | |
|           }
 | |
|           catch (Exception ex)
 | |
|           {
 | |
|              _logger.LogError(ex, "Unable to copy {File} to {DirectoryPath}", currentFile, directoryPath);
 | |
|              return false;
 | |
|           }
 | |
| 
 | |
|           return true;
 | |
|        }
 | |
| 
 | |
|        public IEnumerable<string> ListDirectory(string rootPath)
 | |
|         {
 | |
|            if (!Directory.Exists(rootPath)) return ImmutableList<string>.Empty;
 | |
|             
 | |
|             var di = new DirectoryInfo(rootPath);
 | |
|             var dirs = di.GetDirectories()
 | |
|                 .Where(dir => !(dir.Attributes.HasFlag(FileAttributes.Hidden) || dir.Attributes.HasFlag(FileAttributes.System)))
 | |
|                 .Select(d => d.Name).ToImmutableList();
 | |
|             
 | |
|             return dirs;
 | |
|         }
 | |
| 
 | |
|        public async Task<byte[]> ReadFileAsync(string path)
 | |
|        {
 | |
|           if (!File.Exists(path)) return Array.Empty<byte>();
 | |
|           return await File.ReadAllBytesAsync(path);
 | |
|        }
 | |
| 
 | |
| 
 | |
|        /// <summary>
 | |
|        /// Recursively scans files and applies an action on them. This uses as many cores the underlying PC has to speed
 | |
|        /// up processing.
 | |
|        /// </summary>
 | |
|        /// <param name="root">Directory to scan</param>
 | |
|        /// <param name="action">Action to apply on file path</param>
 | |
|        /// <param name="searchPattern">Regex pattern to search against</param>
 | |
|        /// <param name="logger"></param>
 | |
|        /// <exception cref="ArgumentException"></exception>
 | |
|        public static int TraverseTreeParallelForEach(string root, Action<string> action, string searchPattern, ILogger logger)
 | |
|        {
 | |
|           //Count of files traversed and timer for diagnostic output
 | |
|             var fileCount = 0;
 | |
| 
 | |
|             // Determine whether to parallelize file processing on each folder based on processor count.
 | |
|             var procCount = Environment.ProcessorCount;
 | |
| 
 | |
|             // Data structure to hold names of subfolders to be examined for files.
 | |
|             var dirs = new Stack<string>();
 | |
| 
 | |
|             if (!Directory.Exists(root)) {
 | |
|                    throw new ArgumentException("The directory doesn't exist");
 | |
|             }
 | |
|             dirs.Push(root);
 | |
| 
 | |
|             while (dirs.Count > 0) {
 | |
|                var currentDir = dirs.Pop();
 | |
|                IEnumerable<string> subDirs;
 | |
|                string[] files;
 | |
| 
 | |
|                try {
 | |
|                   subDirs = Directory.GetDirectories(currentDir).Where(path => ExcludeDirectories.Matches(path).Count == 0);
 | |
|                }
 | |
|                // Thrown if we do not have discovery permission on the directory.
 | |
|                catch (UnauthorizedAccessException e) {
 | |
|                   Console.WriteLine(e.Message);
 | |
|                   logger.LogError(e, "Unauthorized access on {Directory}", currentDir);
 | |
|                   continue;
 | |
|                }
 | |
|                // Thrown if another process has deleted the directory after we retrieved its name.
 | |
|                catch (DirectoryNotFoundException e) {
 | |
|                   Console.WriteLine(e.Message);
 | |
|                   logger.LogError(e, "Directory not found on {Directory}", currentDir);
 | |
|                   continue;
 | |
|                }
 | |
| 
 | |
|                try {
 | |
|                   files = GetFilesWithCertainExtensions(currentDir, searchPattern)
 | |
|                      .ToArray();
 | |
|                }
 | |
|                catch (UnauthorizedAccessException e) {
 | |
|                   Console.WriteLine(e.Message);
 | |
|                   continue;
 | |
|                }
 | |
|                catch (DirectoryNotFoundException e) {
 | |
|                   Console.WriteLine(e.Message);
 | |
|                   continue;
 | |
|                }
 | |
|                catch (IOException e) {
 | |
|                   Console.WriteLine(e.Message);
 | |
|                   continue;
 | |
|                }
 | |
| 
 | |
|                // Execute in parallel if there are enough files in the directory.
 | |
|                // Otherwise, execute sequentially. Files are opened and processed
 | |
|                // synchronously but this could be modified to perform async I/O.
 | |
|                try {
 | |
|                   // if (files.Length < procCount) {
 | |
|                   //    foreach (var file in files) {
 | |
|                   //       action(file);
 | |
|                   //       fileCount++;
 | |
|                   //    }
 | |
|                   // }
 | |
|                   // else {
 | |
|                   //    Parallel.ForEach(files, () => 0, (file, _, localCount) =>
 | |
|                   //                                 { action(file);
 | |
|                   //                                   return ++localCount;
 | |
|                   //                                 },
 | |
|                   //                     (c) => {
 | |
|                   //                        Interlocked.Add(ref fileCount, c);
 | |
|                   //                     });
 | |
|                   // }
 | |
|                   foreach (var file in files) {
 | |
|                      action(file);
 | |
|                      fileCount++;
 | |
|                   }
 | |
|                }
 | |
|                catch (AggregateException ae) {
 | |
|                   ae.Handle((ex) => {
 | |
|                                if (ex is UnauthorizedAccessException) {
 | |
|                                   // Here we just output a message and go on.
 | |
|                                   Console.WriteLine(ex.Message);
 | |
|                                   return true;
 | |
|                                }
 | |
|                                // Handle other exceptions here if necessary...
 | |
| 
 | |
|                                return false;
 | |
|                   });
 | |
|                }
 | |
| 
 | |
|                // Push the subdirectories onto the stack for traversal.
 | |
|                // This could also be done before handing the files.
 | |
|                foreach (var str in subDirs)
 | |
|                   dirs.Push(str);
 | |
|             }
 | |
| 
 | |
|             return fileCount;
 | |
|         }
 | |
|         
 | |
|     }
 | |
| } |