using System.IO; using System.Linq; using API.Entities.Enums; using API.Services; namespace API.Parser; /// /// This is an implementation of the Parser that is the basis for everything /// public class DefaultParser { private readonly IDirectoryService _directoryService; public DefaultParser(IDirectoryService directoryService) { _directoryService = directoryService; } /// /// Parses information out of a file path. Will fallback to using directory name if Series couldn't be parsed /// from filename. /// /// /// Root folder /// Defaults to Manga. Allows different Regex to be used for parsing. /// or null if Series was empty public ParserInfo Parse(string filePath, string rootPath, LibraryType type = LibraryType.Manga) { var fileName = _directoryService.FileSystem.Path.GetFileNameWithoutExtension(filePath); ParserInfo ret; if (Parser.IsEpub(filePath)) { ret = new ParserInfo() { Chapters = Parser.ParseChapter(fileName) ?? Parser.ParseComicChapter(fileName), Series = Parser.ParseSeries(fileName) ?? Parser.ParseComicSeries(fileName), Volumes = Parser.ParseVolume(fileName) ?? Parser.ParseComicVolume(fileName), Filename = Path.GetFileName(filePath), Format = Parser.ParseFormat(filePath), FullFilePath = filePath }; } else { ret = new ParserInfo() { Chapters = type == LibraryType.Manga ? Parser.ParseChapter(fileName) : Parser.ParseComicChapter(fileName), Series = type == LibraryType.Manga ? Parser.ParseSeries(fileName) : Parser.ParseComicSeries(fileName), Volumes = type == LibraryType.Manga ? Parser.ParseVolume(fileName) : Parser.ParseComicVolume(fileName), Filename = Path.GetFileName(filePath), Format = Parser.ParseFormat(filePath), Title = Path.GetFileNameWithoutExtension(fileName), FullFilePath = filePath }; } if (Parser.IsImage(filePath) && Parser.IsCoverImage(filePath)) return null; if (Parser.IsImage(filePath)) { // Reset Chapters, Volumes, and Series as images are not good to parse information out of. Better to use folders. ret.Volumes = Parser.DefaultVolume; ret.Chapters = Parser.DefaultChapter; ret.Series = string.Empty; } if (ret.Series == string.Empty || Parser.IsImage(filePath)) { // Try to parse information out of each folder all the way to rootPath ParseFromFallbackFolders(filePath, rootPath, type, ref ret); } var edition = Parser.ParseEdition(fileName); if (!string.IsNullOrEmpty(edition)) { ret.Series = Parser.CleanTitle(ret.Series.Replace(edition, ""), type is LibraryType.Comic); ret.Edition = edition; } var isSpecial = type == LibraryType.Comic ? Parser.ParseComicSpecial(fileName) : Parser.ParseMangaSpecial(fileName); // We must ensure that we can only parse a special out. As some files will have v20 c171-180+Omake and that // could cause a problem as Omake is a special term, but there is valid volume/chapter information. if (ret.Chapters == Parser.DefaultChapter && ret.Volumes == Parser.DefaultVolume && !string.IsNullOrEmpty(isSpecial)) { ret.IsSpecial = true; ParseFromFallbackFolders(filePath, rootPath, type, ref ret); } // If we are a special with marker, we need to ensure we use the correct series name. we can do this by falling back to Folder name if (Parser.HasSpecialMarker(fileName)) { ret.IsSpecial = true; ret.Chapters = Parser.DefaultChapter; ret.Volumes = Parser.DefaultVolume; ParseFromFallbackFolders(filePath, rootPath, type, ref ret); } if (string.IsNullOrEmpty(ret.Series)) { ret.Series = Parser.CleanTitle(fileName, type is LibraryType.Comic); } // Pdfs may have .pdf in the series name, remove that if (Parser.IsPdf(filePath) && ret.Series.ToLower().EndsWith(".pdf")) { ret.Series = ret.Series.Substring(0, ret.Series.Length - ".pdf".Length); } return ret.Series == string.Empty ? null : ret; } /// /// Fills out by trying to parse volume, chapters, and series from folders /// /// /// /// /// Expects a non-null ParserInfo which this method will populate public void ParseFromFallbackFolders(string filePath, string rootPath, LibraryType type, ref ParserInfo ret) { var fallbackFolders = _directoryService.GetFoldersTillRoot(rootPath, filePath).ToList(); for (var i = 0; i < fallbackFolders.Count; i++) { var folder = fallbackFolders[i]; if (!string.IsNullOrEmpty(Parser.ParseMangaSpecial(folder))) continue; var parsedVolume = type is LibraryType.Manga ? Parser.ParseVolume(folder) : Parser.ParseComicVolume(folder); var parsedChapter = type is LibraryType.Manga ? Parser.ParseChapter(folder) : Parser.ParseComicChapter(folder); if (!parsedVolume.Equals(Parser.DefaultVolume) || !parsedChapter.Equals(Parser.DefaultChapter)) { if ((string.IsNullOrEmpty(ret.Volumes) || ret.Volumes.Equals(Parser.DefaultVolume)) && !parsedVolume.Equals(Parser.DefaultVolume)) { ret.Volumes = parsedVolume; } if ((string.IsNullOrEmpty(ret.Chapters) || ret.Chapters.Equals(Parser.DefaultChapter)) && !parsedChapter.Equals(Parser.DefaultChapter)) { ret.Chapters = parsedChapter; } } // Generally users group in series folders. Let's try to parse series from the top folder if (!folder.Equals(ret.Series) && i == fallbackFolders.Count - 1) { var series = Parser.ParseSeries(folder); if (string.IsNullOrEmpty(series)) { ret.Series = Parser.CleanTitle(folder, type is LibraryType.Comic); break; } if (!string.IsNullOrEmpty(series) && (string.IsNullOrEmpty(ret.Series) || !folder.Contains(ret.Series))) { ret.Series = series; break; } } } } }