mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 00:02:29 -04:00 
			
		
		
		
	Co-authored-by: Christopher <39032787+MrRobotjs@users.noreply.github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com>
		
			
				
	
	
		
			842 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			842 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Concurrent;
 | |
| using System.Collections.Generic;
 | |
| using System.Diagnostics;
 | |
| using System.Globalization;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using System.Threading.Tasks;
 | |
| using API.Entities;
 | |
| using API.Entities.Enums;
 | |
| using API.Extensions;
 | |
| using API.Services.Tasks.Scanner.Parser;
 | |
| using API.SignalR;
 | |
| using ExCSS;
 | |
| using Kavita.Common.Helpers;
 | |
| using Microsoft.Extensions.Logging;
 | |
| 
 | |
| namespace API.Services.Tasks.Scanner;
 | |
| #nullable enable
 | |
| 
 | |
| public class ParsedSeries
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Name of the Series
 | |
|     /// </summary>
 | |
|     public required string Name { get; init; }
 | |
|     /// <summary>
 | |
|     /// Normalized Name of the Series
 | |
|     /// </summary>
 | |
|     public required string NormalizedName { get; init; }
 | |
|     /// <summary>
 | |
|     /// Format of the Series
 | |
|     /// </summary>
 | |
|     public required MangaFormat Format { get; init; }
 | |
| }
 | |
| 
 | |
| public class ScanResult
 | |
| {
 | |
|     /// <summary>
 | |
|     /// A list of files in the Folder. Empty if HasChanged = false
 | |
|     /// </summary>
 | |
|     public IList<string> Files { get; set; }
 | |
|     /// <summary>
 | |
|     /// A nested folder from Library Root (at any level)
 | |
|     /// </summary>
 | |
|     public string Folder { get; set; }
 | |
|     /// <summary>
 | |
|     /// The library root
 | |
|     /// </summary>
 | |
|     public string LibraryRoot { get; set; }
 | |
|     /// <summary>
 | |
|     /// Was the Folder scanned or not. If not modified since last scan, this will be false and Files empty
 | |
|     /// </summary>
 | |
|     public bool HasChanged { get; set; }
 | |
|     /// <summary>
 | |
|     /// Set in Stage 2: Parsed Info from the Files
 | |
|     /// </summary>
 | |
|     public IList<ParserInfo> ParserInfos { get; set; }
 | |
| }
 | |
| 
 | |
| /// <summary>
 | |
| /// The final product of ParseScannedFiles. This has all the processed parserInfo and is ready for tracking/processing into entities
 | |
| /// </summary>
 | |
| public class ScannedSeriesResult
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Was the Folder scanned or not. If not modified since last scan, this will be false and indicates that upstream should count this as skipped
 | |
|     /// </summary>
 | |
|     public bool HasChanged { get; set; }
 | |
|     /// <summary>
 | |
|     /// The Parsed Series information used for tracking
 | |
|     /// </summary>
 | |
|     public ParsedSeries ParsedSeries { get; set; }
 | |
|     /// <summary>
 | |
|     /// Parsed files
 | |
|     /// </summary>
 | |
|     public IList<ParserInfo> ParsedInfos { get; set; }
 | |
| }
 | |
| 
 | |
| public class SeriesModified
 | |
| {
 | |
|     public required string? FolderPath { get; set; }
 | |
|     public required string? LowestFolderPath { get; set; }
 | |
|     public required string SeriesName { get; set; }
 | |
|     public DateTime LastScanned { get; set; }
 | |
|     public MangaFormat Format { get; set; }
 | |
|     public IEnumerable<string> LibraryRoots { get; set; } = ArraySegment<string>.Empty;
 | |
| }
 | |
| 
 | |
| /// <summary>
 | |
| /// Responsible for taking parsed info from ReadingItemService and DirectoryService and combining them to emit DB work
 | |
| /// on a series by series.
 | |
| /// </summary>
 | |
| public class ParseScannedFiles
 | |
| {
 | |
|     private readonly ILogger _logger;
 | |
|     private readonly IDirectoryService _directoryService;
 | |
|     private readonly IReadingItemService _readingItemService;
 | |
|     private readonly IEventHub _eventHub;
 | |
| 
 | |
|     /// <summary>
 | |
|     /// An instance of a pipeline for processing files and returning a Map of Series -> ParserInfos.
 | |
|     /// Each instance is separate from other threads, allowing for no cross over.
 | |
|     /// </summary>
 | |
|     /// <param name="logger">Logger of the parent class that invokes this</param>
 | |
|     /// <param name="directoryService">Directory Service</param>
 | |
|     /// <param name="readingItemService">ReadingItemService Service for extracting information on a number of formats</param>
 | |
|     /// <param name="eventHub">For firing off SignalR events</param>
 | |
|     public ParseScannedFiles(ILogger logger, IDirectoryService directoryService,
 | |
|         IReadingItemService readingItemService, IEventHub eventHub)
 | |
|     {
 | |
|         _logger = logger;
 | |
|         _directoryService = directoryService;
 | |
|         _readingItemService = readingItemService;
 | |
|         _eventHub = eventHub;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// This will Scan all files in a folder path. For each folder within the folderPath, FolderAction will be invoked for all files contained
 | |
|     /// </summary>
 | |
|     /// <param name="scanDirectoryByDirectory">Scan directory by directory and for each, call folderAction</param>
 | |
|     /// <param name="seriesPaths">A dictionary mapping a normalized path to a list of <see cref="SeriesModified"/> to help scanner skip I/O</param>
 | |
|     /// <param name="folderPath">A library folder or series folder</param>
 | |
|     /// <param name="forceCheck">If we should bypass any folder last write time checks on the scan and force I/O</param>
 | |
|     public async Task<IList<ScanResult>> ScanFiles(string folderPath, bool scanDirectoryByDirectory,
 | |
|         IDictionary<string, IList<SeriesModified>> seriesPaths, Library library, bool forceCheck = false)
 | |
|     {
 | |
|         var fileExtensions = string.Join("|", library.LibraryFileTypes.Select(l => l.FileTypeGroup.GetRegex()));
 | |
| 
 | |
|         // If there are no library file types, skip scanning entirely
 | |
|         if (string.IsNullOrWhiteSpace(fileExtensions))
 | |
|         {
 | |
|             return ArraySegment<ScanResult>.Empty;
 | |
|         }
 | |
| 
 | |
|         var matcher = BuildMatcher(library);
 | |
| 
 | |
|         var result = new List<ScanResult>();
 | |
| 
 | |
|         // Not to self: this whole thing can be parallelized because we don't deal with any DB or global state
 | |
|         if (scanDirectoryByDirectory)
 | |
|         {
 | |
|             return await ScanDirectories(folderPath, seriesPaths, library, forceCheck, matcher, result, fileExtensions);
 | |
|         }
 | |
| 
 | |
|         return await ScanSingleDirectory(folderPath, seriesPaths, library, forceCheck, result, fileExtensions, matcher);
 | |
|     }
 | |
| 
 | |
|     private async Task<IList<ScanResult>> ScanDirectories(string folderPath, IDictionary<string, IList<SeriesModified>> seriesPaths,
 | |
|         Library library, bool forceCheck, GlobMatcher matcher, List<ScanResult> result, string fileExtensions)
 | |
|     {
 | |
|         var allDirectories = _directoryService.GetAllDirectories(folderPath, matcher)
 | |
|             .Select(Parser.Parser.NormalizePath)
 | |
|             .OrderByDescending(d => d.Length)
 | |
|             .ToList();
 | |
| 
 | |
|         var processedDirs = new HashSet<string>();
 | |
| 
 | |
|         _logger.LogDebug("[ScannerService] Step 1.C Found {DirectoryCount} directories to process for {FolderPath}", allDirectories.Count, folderPath);
 | |
|         foreach (var directory in allDirectories)
 | |
|         {
 | |
|             // Don't process any folders where we've already scanned everything below
 | |
|             if (processedDirs.Any(d => d.StartsWith(directory + Path.AltDirectorySeparatorChar) || d.Equals(directory)))
 | |
|             {
 | |
|                 // Skip this directory as we've already processed a parent unless there are loose files at that directory
 | |
|                 CheckSurfaceFiles(result, directory, folderPath, fileExtensions, matcher);
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             // Skip directories ending with "Specials", let the parent handle it
 | |
|             if (directory.EndsWith("Specials", StringComparison.OrdinalIgnoreCase))
 | |
|             {
 | |
|                 // Log or handle that we are skipping this directory
 | |
|                 _logger.LogDebug("Skipping {Directory} as it ends with 'Specials'", directory);
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|                 MessageFactory.FileScanProgressEvent(directory, library.Name, ProgressEventType.Updated));
 | |
| 
 | |
|             if (HasSeriesFolderNotChangedSinceLastScan(seriesPaths, directory, forceCheck))
 | |
|             {
 | |
|                 HandleUnchangedFolder(result, folderPath, directory);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 PerformFullScan(result, directory, folderPath, fileExtensions, matcher);
 | |
|             }
 | |
| 
 | |
|             processedDirs.Add(directory);
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Checks against all folder paths on file if the last scanned is >= the directory's last write time, down to the second
 | |
|     /// </summary>
 | |
|     /// <param name="seriesPaths"></param>
 | |
|     /// <param name="directory">This should be normalized</param>
 | |
|     /// <param name="forceCheck"></param>
 | |
|     /// <returns></returns>
 | |
|     private bool HasSeriesFolderNotChangedSinceLastScan(IDictionary<string, IList<SeriesModified>> seriesPaths, string directory, bool forceCheck)
 | |
|     {
 | |
|         // With the bottom-up approach, this can report a false positive where a nested folder will get scanned even though a parent is the series
 | |
|         // This can't really be avoided. This is more likely to happen on Image chapter folder library layouts.
 | |
|         if (forceCheck || !seriesPaths.TryGetValue(directory, out var seriesList))
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         foreach (var series in seriesList)
 | |
|         {
 | |
|             var lastWriteTime = _directoryService.GetLastWriteTime(series.LowestFolderPath!).Truncate(TimeSpan.TicksPerSecond);
 | |
|             var seriesLastScanned = series.LastScanned.Truncate(TimeSpan.TicksPerSecond);
 | |
|             if (seriesLastScanned < lastWriteTime)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Handles directories that haven't changed since the last scan.
 | |
|     /// </summary>
 | |
|     private void HandleUnchangedFolder(List<ScanResult> result, string folderPath, string directory)
 | |
|     {
 | |
|         if (result.Exists(r => r.Folder == directory))
 | |
|         {
 | |
|             _logger.LogDebug("[ProcessFiles] Skipping adding {Directory} as it's already added, this indicates a bad layout issue", directory);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             _logger.LogDebug("[ProcessFiles] Skipping {Directory} as it hasn't changed since last scan", directory);
 | |
|             result.Add(CreateScanResult(directory, folderPath, false, ArraySegment<string>.Empty));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Performs a full scan of the directory and adds it to the result.
 | |
|     /// </summary>
 | |
|     private void PerformFullScan(List<ScanResult> result, string directory, string folderPath, string fileExtensions, GlobMatcher matcher)
 | |
|     {
 | |
|         _logger.LogDebug("[ProcessFiles] Performing full scan on {Directory}", directory);
 | |
|         var files = _directoryService.ScanFiles(directory, fileExtensions, matcher);
 | |
|         if (files.Count == 0)
 | |
|         {
 | |
|             _logger.LogDebug("[ProcessFiles] Empty directory: {Directory}. Keeping empty will cause Kavita to scan this each time", directory);
 | |
|         }
 | |
|         result.Add(CreateScanResult(directory, folderPath, true, files));
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Performs a full scan of the directory and adds it to the result.
 | |
|     /// </summary>
 | |
|     private void CheckSurfaceFiles(List<ScanResult> result, string directory, string folderPath, string fileExtensions, GlobMatcher matcher)
 | |
|     {
 | |
|         var files = _directoryService.ScanFiles(directory, fileExtensions, matcher, SearchOption.TopDirectoryOnly);
 | |
|         if (files.Count == 0)
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         result.Add(CreateScanResult(directory, folderPath, true, files));
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Scans a single directory and processes the scan result.
 | |
|     /// </summary>
 | |
|     private async Task<IList<ScanResult>> ScanSingleDirectory(string folderPath, IDictionary<string, IList<SeriesModified>> seriesPaths, Library library, bool forceCheck, List<ScanResult> result,
 | |
|         string fileExtensions, GlobMatcher matcher)
 | |
|     {
 | |
|         var normalizedPath = Parser.Parser.NormalizePath(folderPath);
 | |
|         var libraryRoot =
 | |
|             library.Folders.FirstOrDefault(f =>
 | |
|                 normalizedPath.Contains(Parser.Parser.NormalizePath(f.Path)))?.Path ??
 | |
|             folderPath;
 | |
| 
 | |
|         await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|             MessageFactory.FileScanProgressEvent(normalizedPath, library.Name, ProgressEventType.Updated));
 | |
| 
 | |
|         if (HasSeriesFolderNotChangedSinceLastScan(seriesPaths, normalizedPath, forceCheck))
 | |
|         {
 | |
|             result.Add(CreateScanResult(folderPath, libraryRoot, false, ArraySegment<string>.Empty));
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             result.Add(CreateScanResult(folderPath, libraryRoot, true,
 | |
|                 _directoryService.ScanFiles(folderPath, fileExtensions, matcher)));
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     private static GlobMatcher BuildMatcher(Library library)
 | |
|     {
 | |
|         var matcher = new GlobMatcher();
 | |
|         foreach (var pattern in library.LibraryExcludePatterns.Where(p => !string.IsNullOrEmpty(p.Pattern)))
 | |
|         {
 | |
|             matcher.AddExclude(pattern.Pattern);
 | |
|         }
 | |
| 
 | |
|         return matcher;
 | |
|     }
 | |
| 
 | |
|     private static ScanResult CreateScanResult(string folderPath, string libraryRoot, bool hasChanged,
 | |
|         IList<string> files)
 | |
|     {
 | |
|         return new ScanResult()
 | |
|         {
 | |
|             Files = files,
 | |
|             Folder = Parser.Parser.NormalizePath(folderPath),
 | |
|             LibraryRoot = libraryRoot,
 | |
|             HasChanged = hasChanged
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Processes scanResults to track all series across the combined results.
 | |
|     /// Ensures series are correctly grouped even if they span multiple folders.
 | |
|     /// </summary>
 | |
|     /// <param name="scanResults">A collection of scan results</param>
 | |
|     /// <param name="scannedSeries">A concurrent dictionary to store the tracked series</param>
 | |
|     private void TrackSeriesAcrossScanResults(IList<ScanResult> scanResults, ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries)
 | |
|     {
 | |
|         // Flatten all ParserInfos from scanResults
 | |
|         var allInfos = scanResults.SelectMany(sr => sr.ParserInfos).ToList();
 | |
| 
 | |
|         // Iterate through each ParserInfo and track the series
 | |
|         foreach (var info in allInfos)
 | |
|         {
 | |
|             if (info == null) continue;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 TrackSeries(scannedSeries, info);
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 _logger.LogError(ex, "[ScannerService] Exception occurred during tracking {FilePath}. Skipping this file", info?.FullFilePath);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Attempts to either add a new instance of a series mapping to the _scannedSeries bag or adds to an existing.
 | |
|     /// This will check if the name matches an existing series name (multiple fields) <see cref="MergeName"/>
 | |
|     /// </summary>
 | |
|     /// <param name="scannedSeries">A localized list of a series' parsed infos</param>
 | |
|     /// <param name="info"></param>
 | |
|     private void TrackSeries(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries, ParserInfo? info)
 | |
|     {
 | |
|         if (info == null || info.Series == string.Empty) return;
 | |
| 
 | |
|         // Check if normalized info.Series already exists and if so, update info to use that name instead
 | |
|         info.Series = MergeName(scannedSeries, info);
 | |
| 
 | |
|         // BUG: This will fail for Solo Leveling & Solo Leveling (Manga)
 | |
| 
 | |
|         var normalizedSeries = info.Series.ToNormalized();
 | |
|         var normalizedSortSeries = info.SeriesSort.ToNormalized();
 | |
|         var normalizedLocalizedSeries = info.LocalizedSeries.ToNormalized();
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             var existingKey = scannedSeries.Keys.SingleOrDefault(ps =>
 | |
|                 ps.Format == info.Format && (ps.NormalizedName.Equals(normalizedSeries)
 | |
|                                              || ps.NormalizedName.Equals(normalizedLocalizedSeries)
 | |
|                                              || ps.NormalizedName.Equals(normalizedSortSeries)));
 | |
|             existingKey ??= new ParsedSeries()
 | |
|             {
 | |
|                 Format = info.Format,
 | |
|                 Name = info.Series,
 | |
|                 NormalizedName = normalizedSeries
 | |
|             };
 | |
| 
 | |
|             scannedSeries.AddOrUpdate(existingKey, [info], (_, oldValue) =>
 | |
|             {
 | |
|                 oldValue ??= new List<ParserInfo>();
 | |
|                 if (!oldValue.Contains(info))
 | |
|                 {
 | |
|                     oldValue.Add(info);
 | |
|                 }
 | |
| 
 | |
|                 return oldValue;
 | |
|             });
 | |
|         }
 | |
|         catch (Exception ex)
 | |
|         {
 | |
|             _logger.LogCritical("[ScannerService] {SeriesName} matches against multiple series in the parsed series. This indicates a critical kavita issue. Key will be skipped", info.Series);
 | |
|             foreach (var seriesKey in scannedSeries.Keys.Where(ps =>
 | |
|                          ps.Format == info.Format && (ps.NormalizedName.Equals(normalizedSeries)
 | |
|                                                       || ps.NormalizedName.Equals(normalizedLocalizedSeries)
 | |
|                                                       || ps.NormalizedName.Equals(normalizedSortSeries))))
 | |
|             {
 | |
|                 _logger.LogCritical("[ScannerService] Matches: '{SeriesName}' matches on '{SeriesKey}'", info.Series, seriesKey.Name);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Using a normalized name from the passed ParserInfo, this checks against all found series so far and if an existing one exists with
 | |
|     /// same normalized name, it merges into the existing one. This is important as some manga may have a slight difference with punctuation or capitalization.
 | |
|     /// </summary>
 | |
|     /// <param name="scannedSeries"></param>
 | |
|     /// <param name="info"></param>
 | |
|     /// <returns>Series Name to group this info into</returns>
 | |
|     private string MergeName(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries, ParserInfo info)
 | |
|     {
 | |
|         var normalizedSeries = info.Series.ToNormalized();
 | |
|         var normalizedLocalSeries = info.LocalizedSeries.ToNormalized();
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             var existingName =
 | |
|                 scannedSeries.SingleOrDefault(p =>
 | |
|                         (p.Key.NormalizedName.ToNormalized().Equals(normalizedSeries) ||
 | |
|                          p.Key.NormalizedName.ToNormalized().Equals(normalizedLocalSeries)) &&
 | |
|                         p.Key.Format == info.Format)
 | |
|                     .Key;
 | |
| 
 | |
|             if (existingName == null)
 | |
|             {
 | |
|                 return info.Series;
 | |
|             }
 | |
| 
 | |
|             if (!string.IsNullOrEmpty(existingName.Name))
 | |
|             {
 | |
|                 return existingName.Name;
 | |
|             }
 | |
|         }
 | |
|         catch (Exception ex)
 | |
|         {
 | |
|             _logger.LogCritical("[ScannerService] Multiple series detected for {SeriesName} ({File})! This is critical to fix! There should only be 1", info.Series, info.FullFilePath);
 | |
|             var values = scannedSeries.Where(p =>
 | |
|                 (p.Key.NormalizedName.ToNormalized() == normalizedSeries ||
 | |
|                  p.Key.NormalizedName.ToNormalized() == normalizedLocalSeries) &&
 | |
|                 p.Key.Format == info.Format);
 | |
| 
 | |
|             foreach (var pair in values)
 | |
|             {
 | |
|                 _logger.LogCritical("[ScannerService] Duplicate Series in DB matches with {SeriesName}: {DuplicateName}", info.Series, pair.Key.Name);
 | |
|             }
 | |
| 
 | |
|         }
 | |
| 
 | |
|         return info.Series;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// This will process series by folder groups. This is used solely by ScanSeries
 | |
|     /// </summary>
 | |
|     /// <param name="library">This should have the FileTypes included</param>
 | |
|     /// <param name="folders"></param>
 | |
|     /// <param name="isLibraryScan">If true, does a directory scan first (resulting in folders being tackled in parallel), else does an immediate scan files</param>
 | |
|     /// <param name="seriesPaths">A map of Series names -> existing folder paths to handle skipping folders</param>
 | |
|     /// <param name="forceCheck">Defaults to false</param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IList<ScannedSeriesResult>> ScanLibrariesForSeries(Library library,
 | |
|         IList<string> folders, bool isLibraryScan,
 | |
|         IDictionary<string, IList<SeriesModified>> seriesPaths, bool forceCheck = false)
 | |
|     {
 | |
|         await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|             MessageFactory.FileScanProgressEvent("File Scan Starting", library.Name, ProgressEventType.Started));
 | |
| 
 | |
|         _logger.LogDebug("[ScannerService] Library {LibraryName} Step 1.A: Process {FolderCount} folders", library.Name, folders.Count);
 | |
|         var processedScannedSeries = new ConcurrentBag<ScannedSeriesResult>();
 | |
| 
 | |
|         foreach (var folder in folders)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 await ScanAndParseFolder(folder, library, isLibraryScan, seriesPaths, processedScannedSeries, forceCheck);
 | |
|             }
 | |
|             catch (ArgumentException ex)
 | |
|             {
 | |
|                 _logger.LogError(ex, "[ScannerService] The directory '{FolderPath}' does not exist", folder);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|             MessageFactory.FileScanProgressEvent("File Scan Done", library.Name, ProgressEventType.Ended));
 | |
| 
 | |
|         return processedScannedSeries.ToList();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Helper method to scan and parse a folder
 | |
|     /// </summary>
 | |
|     /// <param name="folderPath"></param>
 | |
|     /// <param name="library"></param>
 | |
|     /// <param name="isLibraryScan"></param>
 | |
|     /// <param name="seriesPaths"></param>
 | |
|     /// <param name="processedScannedSeries"></param>
 | |
|     /// <param name="forceCheck"></param>
 | |
|     private async Task ScanAndParseFolder(string folderPath, Library library,
 | |
|         bool isLibraryScan, IDictionary<string, IList<SeriesModified>> seriesPaths,
 | |
|         ConcurrentBag<ScannedSeriesResult> processedScannedSeries, bool forceCheck)
 | |
|     {
 | |
|         _logger.LogDebug("\t[ScannerService] Library {LibraryName} Step 1.B: Scan files in {Folder}", library.Name, folderPath);
 | |
|         var scanResults = await ScanFiles(folderPath, isLibraryScan, seriesPaths, library, forceCheck);
 | |
| 
 | |
|         // Aggregate the scanned series across all scanResults
 | |
|         var scannedSeries = new ConcurrentDictionary<ParsedSeries, List<ParserInfo>>();
 | |
| 
 | |
|         _logger.LogDebug("\t[ScannerService] Library {LibraryName} Step 1.C: Process files in {Folder}", library.Name, folderPath);
 | |
|         foreach (var scanResult in scanResults)
 | |
|         {
 | |
|             await ParseFiles(scanResult, seriesPaths, library);
 | |
|         }
 | |
| 
 | |
|         _logger.LogDebug("\t[ScannerService] Library {LibraryName} Step 1.D: Merge any localized series with series {Folder}", library.Name, folderPath);
 | |
|         scanResults = MergeLocalizedSeriesAcrossScanResults(scanResults);
 | |
| 
 | |
|         _logger.LogDebug("\t[ScannerService] Library {LibraryName} Step 1.E: Group all parsed data into logical Series", library.Name);
 | |
|         TrackSeriesAcrossScanResults(scanResults, scannedSeries);
 | |
| 
 | |
| 
 | |
|         // Now transform and add to processedScannedSeries AFTER everything is processed
 | |
|         _logger.LogDebug("\t[ScannerService] Library {LibraryName} Step 1.F: Generate Sort Order for Series and Finalize", library.Name);
 | |
|         GenerateProcessedScannedSeries(scannedSeries, scanResults, processedScannedSeries);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Processes and generates the final results for processedScannedSeries after updating sort order.
 | |
|     /// </summary>
 | |
|     /// <param name="scannedSeries">A concurrent dictionary of tracked series and their parsed infos</param>
 | |
|     /// <param name="scanResults">List of all scan results, used to determine if any series has changed</param>
 | |
|     /// <param name="processedScannedSeries">A thread-safe concurrent bag of processed series results</param>
 | |
|     private void GenerateProcessedScannedSeries(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries, IList<ScanResult> scanResults, ConcurrentBag<ScannedSeriesResult> processedScannedSeries)
 | |
|     {
 | |
|         // First, update the sort order for all series
 | |
|         UpdateSeriesSortOrder(scannedSeries);
 | |
| 
 | |
|         // Now, generate the final processed scanned series results
 | |
|         CreateFinalSeriesResults(scannedSeries, scanResults, processedScannedSeries);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Updates the sort order for all series in the scannedSeries dictionary.
 | |
|     /// </summary>
 | |
|     /// <param name="scannedSeries">A concurrent dictionary of tracked series and their parsed infos</param>
 | |
|     private void UpdateSeriesSortOrder(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries)
 | |
|     {
 | |
|         foreach (var series in scannedSeries.Keys)
 | |
|         {
 | |
|             if (scannedSeries[series].Count <= 0) continue;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 UpdateSortOrder(scannedSeries, series);  // Call to method that updates sort order
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 _logger.LogError(ex, "[ScannerService] Issue occurred while setting IssueOrder for series {SeriesName}", series.Name);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Generates the final processed scanned series results after processing the sort order.
 | |
|     /// </summary>
 | |
|     /// <param name="scannedSeries">A concurrent dictionary of tracked series and their parsed infos</param>
 | |
|     /// <param name="scanResults">List of all scan results, used to determine if any series has changed</param>
 | |
|     /// <param name="processedScannedSeries">The list where processed results will be added</param>
 | |
|     private static void CreateFinalSeriesResults(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries,
 | |
|         IList<ScanResult> scanResults, ConcurrentBag<ScannedSeriesResult> processedScannedSeries)
 | |
|     {
 | |
|         foreach (var series in scannedSeries.Keys)
 | |
|         {
 | |
|             if (scannedSeries[series].Count <= 0) continue;
 | |
| 
 | |
|             processedScannedSeries.Add(new ScannedSeriesResult
 | |
|             {
 | |
|                 HasChanged = scanResults.Any(sr => sr.HasChanged),  // Combine HasChanged flag across all scanResults
 | |
|                 ParsedSeries = series,
 | |
|                 ParsedInfos = scannedSeries[series]
 | |
|             });
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Merges localized series with the series field across all scan results.
 | |
|     /// Combines ParserInfos from all scanResults and processes them collectively
 | |
|     /// to ensure consistent series names.
 | |
|     /// </summary>
 | |
|     /// <example>
 | |
|     /// Accel World v01.cbz has Series "Accel World" and Localized Series "World of Acceleration"
 | |
|     /// World of Acceleration v02.cbz has Series "World of Acceleration"
 | |
|     /// After running this code, we'd have:
 | |
|     /// World of Acceleration v02.cbz having Series "Accel World" and Localized Series of "World of Acceleration"
 | |
|     /// </example>
 | |
|     /// <param name="scanResults">A collection of scan results</param>
 | |
|     /// <returns>A new list of scan results with merged series</returns>
 | |
|     private IList<ScanResult> MergeLocalizedSeriesAcrossScanResults(IList<ScanResult> scanResults)
 | |
|     {
 | |
|         // Flatten all ParserInfos across scanResults
 | |
|         var allInfos = scanResults.SelectMany(sr => sr.ParserInfos).ToList();
 | |
| 
 | |
|         // Filter relevant infos (non-special and with localized series)
 | |
|         var relevantInfos = GetRelevantInfos(allInfos);
 | |
| 
 | |
|         if (relevantInfos.Count == 0) return scanResults;
 | |
| 
 | |
|         // Get distinct localized series and process each one
 | |
|         var distinctLocalizedSeries = relevantInfos
 | |
|             .Select(i => i.LocalizedSeries)
 | |
|             .Distinct()
 | |
|             .ToList();
 | |
| 
 | |
|         foreach (var localizedSeries in distinctLocalizedSeries)
 | |
|         {
 | |
|             if (string.IsNullOrEmpty(localizedSeries)) continue;
 | |
| 
 | |
|             // Process the localized series for merging
 | |
|             ProcessLocalizedSeries(scanResults, allInfos, relevantInfos, localizedSeries);
 | |
|         }
 | |
| 
 | |
|         // Remove or clear any scan results that now have no ParserInfos after merging
 | |
|         return scanResults.Where(sr => sr.ParserInfos.Count > 0).ToList();
 | |
|     }
 | |
| 
 | |
|     private static List<ParserInfo> GetRelevantInfos(List<ParserInfo> allInfos)
 | |
|     {
 | |
|         return allInfos
 | |
|             .Where(i => !i.IsSpecial && !string.IsNullOrEmpty(i.LocalizedSeries))
 | |
|             .GroupBy(i => i.Format)
 | |
|             .SelectMany(g => g.ToList())
 | |
|             .ToList();
 | |
|     }
 | |
| 
 | |
|     private void ProcessLocalizedSeries(IList<ScanResult> scanResults, List<ParserInfo> allInfos, List<ParserInfo> relevantInfos, string localizedSeries)
 | |
|     {
 | |
|         var seriesForLocalized = GetSeriesForLocalized(relevantInfos, localizedSeries);
 | |
|         if (seriesForLocalized.Count == 0) return;
 | |
| 
 | |
|         var nonLocalizedSeries = GetNonLocalizedSeries(seriesForLocalized, localizedSeries);
 | |
|         if (nonLocalizedSeries == null) return;
 | |
| 
 | |
|         // Remap and update relevant ParserInfos
 | |
|         RemapSeries(scanResults, allInfos, localizedSeries, nonLocalizedSeries);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private static List<string> GetSeriesForLocalized(List<ParserInfo> relevantInfos, string localizedSeries)
 | |
|     {
 | |
|         return relevantInfos
 | |
|             .Where(i => i.LocalizedSeries == localizedSeries)
 | |
|             .DistinctBy(r => r.Series)
 | |
|             .Select(r => r.Series)
 | |
|             .ToList();
 | |
|     }
 | |
| 
 | |
|     private string? GetNonLocalizedSeries(List<string> seriesForLocalized, string localizedSeries)
 | |
|     {
 | |
|         switch (seriesForLocalized.Count)
 | |
|         {
 | |
|             case 1:
 | |
|                 return seriesForLocalized[0];
 | |
|             case <= 2:
 | |
|                 return seriesForLocalized.FirstOrDefault(s => !s.Equals(Parser.Parser.Normalize(localizedSeries)));
 | |
|             default:
 | |
|                 _logger.LogError(
 | |
|                     "[ScannerService] Multiple series detected across scan results that contain localized series. " +
 | |
|                     "This will cause them to group incorrectly. Please separate series into their own dedicated folder: {LocalizedSeries}",
 | |
|                     string.Join(", ", seriesForLocalized)
 | |
|                 );
 | |
|                 return null;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private static void RemapSeries(IList<ScanResult> scanResults, List<ParserInfo> allInfos, string localizedSeries, string nonLocalizedSeries)
 | |
|     {
 | |
|         // Find all infos that need to be remapped from the localized series to the non-localized series
 | |
|         var normalizedLocalizedSeries = localizedSeries.ToNormalized();
 | |
|         var seriesToBeRemapped = allInfos.Where(i => i.Series.ToNormalized().Equals(normalizedLocalizedSeries)).ToList();
 | |
| 
 | |
|         foreach (var infoNeedingMapping in seriesToBeRemapped)
 | |
|         {
 | |
|             infoNeedingMapping.Series = nonLocalizedSeries;
 | |
| 
 | |
|             // Find the scan result containing the localized info
 | |
|             var localizedScanResult = scanResults.FirstOrDefault(sr => sr.ParserInfos.Contains(infoNeedingMapping));
 | |
|             if (localizedScanResult == null) continue;
 | |
| 
 | |
|             // Remove the localized series from this scan result
 | |
|             localizedScanResult.ParserInfos.Remove(infoNeedingMapping);
 | |
| 
 | |
|             // Find the scan result that should be merged with
 | |
|             var nonLocalizedScanResult = scanResults.FirstOrDefault(sr => sr.ParserInfos.Any(pi => pi.Series == nonLocalizedSeries));
 | |
| 
 | |
|             if (nonLocalizedScanResult == null) continue;
 | |
| 
 | |
|             // Add the remapped info to the non-localized scan result
 | |
|             nonLocalizedScanResult.ParserInfos.Add(infoNeedingMapping);
 | |
| 
 | |
|             // Assign the higher folder path (i.e., the one closer to the root)
 | |
|             //nonLocalizedScanResult.Folder = DirectoryService.GetDeepestCommonPath(localizedScanResult.Folder, nonLocalizedScanResult.Folder);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// For a given ScanResult, sets the ParserInfos on the result
 | |
|     /// </summary>
 | |
|     /// <param name="result"></param>
 | |
|     /// <param name="seriesPaths"></param>
 | |
|     /// <param name="library"></param>
 | |
|     private async Task ParseFiles(ScanResult result, IDictionary<string, IList<SeriesModified>> seriesPaths, Library library)
 | |
|     {
 | |
|         var normalizedFolder = Parser.Parser.NormalizePath(result.Folder);
 | |
| 
 | |
|         // If folder hasn't changed, generate fake ParserInfos
 | |
|         if (!result.HasChanged)
 | |
|         {
 | |
|             result.ParserInfos = seriesPaths[normalizedFolder]
 | |
|                 .Select(fp => new ParserInfo { Series = fp.SeriesName, Format = fp.Format })
 | |
|                 .ToList();
 | |
| 
 | |
|             _logger.LogDebug("[ScannerService] Skipped File Scan for {Folder} as it hasn't changed", normalizedFolder);
 | |
|             await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|                 MessageFactory.FileScanProgressEvent($"Skipped {normalizedFolder}", library.Name, ProgressEventType.Updated));
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         var files = result.Files;
 | |
|         var fileCount = files.Count;
 | |
| 
 | |
|         if (fileCount == 0)
 | |
|         {
 | |
|             _logger.LogInformation("[ScannerService] {Folder} is empty or has no matching file types", normalizedFolder);
 | |
|             result.ParserInfos = ArraySegment<ParserInfo>.Empty;
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         _logger.LogDebug("[ScannerService] Found {Count} files for {Folder}", files.Count, normalizedFolder);
 | |
|         await _eventHub.SendMessageAsync(MessageFactory.NotificationProgress,
 | |
|             MessageFactory.FileScanProgressEvent($"{fileCount} files in {normalizedFolder}", library.Name, ProgressEventType.Updated));
 | |
| 
 | |
|         // Parse files into ParserInfos
 | |
|         if (fileCount < 100)
 | |
|         {
 | |
|             // Process files sequentially
 | |
|             result.ParserInfos = files
 | |
|                 .Select(file => _readingItemService.ParseFile(file, normalizedFolder, result.LibraryRoot, library.Type))
 | |
|                 .Where(info => info != null)
 | |
|                 .ToList()!;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // Process files in parallel
 | |
|             var tasks = files.Select(file => Task.Run(() =>
 | |
|                 _readingItemService.ParseFile(file, normalizedFolder, result.LibraryRoot, library.Type)));
 | |
| 
 | |
|             var infos = await Task.WhenAll(tasks);
 | |
|             result.ParserInfos = infos.Where(info => info != null).ToList()!;
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     private static void UpdateSortOrder(ConcurrentDictionary<ParsedSeries, List<ParserInfo>> scannedSeries, ParsedSeries series)
 | |
|     {
 | |
|         // Set the Sort order per Volume
 | |
|         var volumes = scannedSeries[series].GroupBy(info => info.Volumes);
 | |
|         foreach (var volume in volumes)
 | |
|         {
 | |
|             var infos = scannedSeries[series].Where(info => info.Volumes == volume.Key).ToList();
 | |
|             IList<ParserInfo> chapters;
 | |
|             var specialTreatment = infos.TrueForAll(info => info.IsSpecial);
 | |
|             var hasAnySpMarker = infos.Exists(info => info.SpecialIndex > 0);
 | |
|             var counter = 0f;
 | |
| 
 | |
|             // Handle specials with SpecialIndex
 | |
|             if (specialTreatment && hasAnySpMarker)
 | |
|             {
 | |
|                 chapters = infos
 | |
|                     .OrderBy(info => info.SpecialIndex)
 | |
|                     .ToList();
 | |
| 
 | |
|                 foreach (var chapter in chapters)
 | |
|                 {
 | |
|                     chapter.IssueOrder = counter;
 | |
|                     counter++;
 | |
|                 }
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             // Handle specials without SpecialIndex (natural order)
 | |
|             if (specialTreatment)
 | |
|             {
 | |
|                 chapters = infos
 | |
|                     .OrderByNatural(info => Parser.Parser.RemoveExtensionIfSupported(info.Filename)!)
 | |
|                     .ToList();
 | |
| 
 | |
|                 foreach (var chapter in chapters)
 | |
|                 {
 | |
|                     chapter.IssueOrder = counter;
 | |
|                     counter++;
 | |
|                 }
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             // Ensure chapters are sorted numerically when possible, otherwise push unparseable to the end
 | |
|             chapters = infos
 | |
|                 .OrderBy(info => float.TryParse(info.Chapters, NumberStyles.Any, CultureInfo.InvariantCulture, out var val) ? val : float.MaxValue)
 | |
|                 .ToList();
 | |
| 
 | |
|             counter = 0f;
 | |
|             var prevIssue = string.Empty;
 | |
|             foreach (var chapter in chapters)
 | |
|             {
 | |
|                 if (float.TryParse(chapter.Chapters, NumberStyles.Any, CultureInfo.InvariantCulture, out var parsedChapter))
 | |
|                 {
 | |
|                     // Parsed successfully, use the numeric value
 | |
|                     counter = parsedChapter;
 | |
|                     chapter.IssueOrder = counter;
 | |
| 
 | |
|                     // Increment for next chapter (unless the next has a similar value, then add 0.1)
 | |
|                     if (!string.IsNullOrEmpty(prevIssue) && float.TryParse(prevIssue, CultureInfo.InvariantCulture, out var prevIssueFloat) && parsedChapter.Is(prevIssueFloat))
 | |
|                     {
 | |
|                         counter += 0.1f; // bump if same value as the previous issue
 | |
|                     }
 | |
|                     prevIssue = $"{parsedChapter}";
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     // Unparsed chapters: use the current counter and bump for the next
 | |
|                     if (!string.IsNullOrEmpty(prevIssue) && prevIssue == counter.ToString(CultureInfo.InvariantCulture))
 | |
|                     {
 | |
|                         counter += 0.1f; // bump if same value as the previous issue
 | |
|                     }
 | |
|                     chapter.IssueOrder = counter;
 | |
|                     counter++;
 | |
|                     prevIssue = chapter.Chapters;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |