using Mono.Data.Sqlite; using SabreTools.Helper; using System; using System.Collections.Generic; using System.IO; using System.Linq; namespace SabreTools { public partial class SabreTools { #region Init Methods /// /// Wrap adding a new source to the database /// /// Source name /// Source URL(s) private static void InitAddSource(string name, string url) { if (DBTools.AddSource(name, url, Constants.DATabaseConnectionString)) { _logger.Log("Source " + name + " added!"); } else { _logger.Error("Source " + name + " could not be added!"); } } /// /// Wrap adding a new system to the database /// /// Manufacturer name /// System name private static void InitAddSystem(string manufacturer, string system) { if (DBTools.AddSystem(manufacturer, system, Constants.DATabaseConnectionString)) { _logger.Log("System " + manufacturer + " - " + system + " added!"); } else { _logger.Error("System " + manufacturer + " - " + system + " could not be added!"); } } /// /// Wrap sorting files using an input DAT /// /// List of all inputted files and folders /// Output directory (empty for default directory) /// Temporary directory for archive extraction /// True if input files should be deleted, false otherwise /// True if files should be output in Romba depot folders, false otherwise /// Integer representing the archive handling level for 7z /// Integer representing the archive handling level for GZip /// Integer representing the archive handling level for RAR /// Integer representing the archive handling level for Zip /// Logger object for file and console output public static bool InitConvertFolderTGZ(List inputs, string outdir, string tempdir, bool delete, bool romba, int sevenzip, int gz, int rar, int zip, Logger logger) { // Get all individual files from the inputs List newinputs = new List(); foreach (string input in inputs) { if (File.Exists(input)) { newinputs.Add(Path.GetFullPath(input)); } else if (Directory.Exists(input)) { foreach (string file in Directory.EnumerateFiles(input, "*", SearchOption.AllDirectories)) { newinputs.Add(Path.GetFullPath(file)); } } } SimpleSort ss = new SimpleSort(new Dat(), newinputs, outdir, tempdir, false, false, false, delete, true, romba, sevenzip, gz, rar, zip, false, logger); return ss.Convert(); } /// /// Wrap creating a DAT file from files or a directory in parallel /// /// List of input filenames /// New filename /// New name /// New description /// New category /// New version /// New author /// True to set forcepacking="unzip" on the created file, false otherwise /// OutputFormat to be used for outputting the DAT /// True to enable reading a directory like a Romba depot, false otherwise /// True to enable SuperDAT-style reading, false otherwise /// True to disable getting MD5 hash, false otherwise /// True to disable getting SHA-1 hash, false otherwise /// True if the date should be omitted from the DAT, false otherwise /// True if archives should be treated as files, false otherwise /// True if GZIP archives should be treated as files, false otherwise /// Name of the directory to create a temp folder in (blank is current directory /// Integer representing the maximum amount of parallelization to be used private static void InitDatFromDir(List inputs, string filename, string name, string description, string category, string version, string author, bool forceunpack, OutputFormat outputFormat, bool romba, bool superdat, bool noMD5, bool noSHA1, bool bare, bool archivesAsFiles, bool enableGzip, string tempDir, int maxDegreeOfParallelism) { // Create a new DATFromDir object and process the inputs Dat basedat = new Dat { FileName = filename, Name = name, Description = description, Category = category, Version = version, Date = DateTime.Now.ToString("yyyy-MM-dd"), Author = author, ForcePacking = (forceunpack ? ForcePacking.Unzip : ForcePacking.None), OutputFormat = outputFormat, Romba = romba, Type = (superdat ? "SuperDAT" : ""), }; // If the user has only set a single thread, use the original version if (maxDegreeOfParallelism == 1) { DATFromDir dfd = new DATFromDir(inputs, basedat, noMD5, noSHA1, bare, archivesAsFiles, enableGzip, tempDir, _logger); bool success = dfd.Start(); // If we failed, show the help if (!success) { Console.WriteLine(); Build.Help(); } } // Otherwise, make full use of the threads else { // For each input directory, create a DAT foreach (string path in inputs) { if (Directory.Exists(path)) { // Clone the base Dat for information Dat datdata = (Dat)basedat.Clone(); datdata.Files = new Dictionary>(); string basePath = Path.GetFullPath(path); DATFromDirParallel dfd = new DATFromDirParallel(basePath, datdata, noMD5, noSHA1, bare, archivesAsFiles, enableGzip, tempDir, maxDegreeOfParallelism, _logger); bool success = dfd.Start(); // If it was a success, write the DAT out if (success) { DatTools.WriteDatfile(dfd.DatData, "", _logger); } // Otherwise, show the help else { Console.WriteLine(); Build.Help(); } } } } } /// /// Wrap splitting a DAT by 2 extensions /// /// Input files or folders to be split /// First extension to split on /// Second extension to split on /// Output directory for the split files private static void InitExtSplit(List inputs, string exta, string extb, string outdir) { // Convert comma-separated strings to list List extaList = exta.Split(',').ToList(); List extbList = extb.Split(',').ToList(); // Loop over the input files foreach (string input in inputs) { if (File.Exists(input)) { DatTools.SplitByExt(Path.GetFullPath(input), outdir, Path.GetDirectoryName(input), extaList, extbList, _logger); } else if (Directory.Exists(input)) { foreach (string file in Directory.EnumerateFiles(input, "*", SearchOption.AllDirectories)) { DatTools.SplitByExt(file, outdir, (input.EndsWith(Path.DirectorySeparatorChar.ToString()) ? input : input + Path.DirectorySeparatorChar), extaList, extbList, _logger); } } else { _logger.Error(input + " is not a valid file or folder!"); Console.WriteLine(); Build.Help(); return; } } } /// /// Wrap generating a DAT from the library /// /// System ID to be used in the DAT (blank means all) /// True if files should not be renamed with system and/or source in merged mode (default false) /// True if the output file should be in ClrMamePro format (default false) private static void InitGenerate(string systemid, bool norename, bool old) { IGenerate gen = new GenerateTwo(systemid, "" /* sourceid */, _datroot, _outroot, Constants.DATabaseConnectionString, _logger, norename, old); gen.Export(); } /// /// Wrap generating all standard DATs from the library /// private static void InitGenerateAll(bool norename, bool old) { List systems = new List(); using (SqliteConnection dbc = new SqliteConnection(Constants.DATabaseConnectionString)) { dbc.Open(); string query = "SELECT id FROM system"; using (SqliteCommand slc = new SqliteCommand(query, dbc)) { using (SqliteDataReader sldr = slc.ExecuteReader()) { // If nothing is found, tell the user and exit if (!sldr.HasRows) { _logger.Warning("No systems found! Please add a system and then try again."); return; } while (sldr.Read()) { systems.Add(sldr.GetInt32(0).ToString()); } } } // Loop through the inputs foreach (string system in systems) { _logger.User("Generating DAT for system id " + system); InitGenerate(system, norename, old); } } } /// /// Wrap splitting a DAT by best available hashes /// /// List of inputs to be used /// Output directory for the split files private static void InitHashSplit(List inputs, string outdir) { // Loop over the input files foreach (string input in inputs) { if (File.Exists(input)) { DatTools.SplitByHash(Path.GetFullPath(input), outdir, Path.GetDirectoryName(input), _logger); } else if (Directory.Exists(input)) { foreach (string file in Directory.EnumerateFiles(input, "*", SearchOption.AllDirectories)) { DatTools.SplitByHash(file, outdir, (input.EndsWith(Path.DirectorySeparatorChar.ToString()) ? input : input + Path.DirectorySeparatorChar), _logger); } } else { _logger.Error(input + " is not a valid file or folder!"); Console.WriteLine(); Build.Help(); return; } } } /// /// Wrap extracting and replacing headers /// /// Input file or folder names /// False if we're extracting headers (default), true if we're restoring them /// Output directory to write new files to, blank defaults to rom folder /// Logger object for file and console output private static void InitHeaderer(List inputs, bool restore, string outdir, Logger logger) { Headerer headerer = new Headerer(inputs, restore, outdir, logger); headerer.Process(); } /// /// Wrap importing and updating DATs /// /// private static void InitImport(bool ignore) { IImport imp = new ImportTwo(_datroot, Constants.DATabaseConnectionString, _logger, ignore); imp.UpdateDatabase(); } /// /// Wrap removing an existing source from the database /// /// Source ID to be removed from the database private static void InitRemoveSource(string sourceid) { int srcid = -1; if (Int32.TryParse(sourceid, out srcid)) { if (DBTools.RemoveSource(srcid, Constants.DATabaseConnectionString)) { _logger.Log("Source '" + srcid + "' removed!"); } else { _logger.Error("Source with id '" + srcid + "' could not be removed."); } } else { _logger.Error("Invalid input"); } } /// /// Wrap removing an existing system from the database /// /// System ID to be removed from the database private static void InitRemoveSystem(string systemid) { int sysid = -1; if (Int32.TryParse(systemid, out sysid)) { if (DBTools.RemoveSystem(sysid, Constants.DATabaseConnectionString)) { _logger.Log("System '" + sysid + "' removed!"); } else { _logger.Error("System with id '" + sysid + "' could not be removed."); } } else { _logger.Error("Invalid input"); } } /// /// Wrap sorting files using an input DAT /// /// Names of the DATs to compare against /// List of input files/folders to check /// Output directory to use to build to /// Temporary directory for archive extraction /// True to enable external scanning of archives, false otherwise /// Integer representing the archive handling level for 7z /// True if files should be output to folder, false otherwise /// True if output directory should be checked instead of rebuilt to, false otherwise /// True if input files should be deleted, false otherwise /// True if files should be output in TorrentGZ format, false for standard zip /// True if files should be output in Romba depot folders, false otherwise /// Integer representing the archive handling level for GZip /// Integer representing the archive handling level for RAR /// Integer representing the archive handling level for Zip /// True if the updated DAT should be output, false otherwise /// Logger object for file and console output private static void InitSortVerify(List datfiles, List inputs, string outdir, string tempdir, bool quickScan, bool toFolder, bool verify, bool delete, bool tgz, bool romba, int sevenzip, int gz, int rar, int zip, bool updateDat, Logger logger) { // Add all of the input DATs into one huge internal DAT Dat datdata = new Dat(); foreach (string datfile in datfiles) { datdata = DatTools.Parse(datfile, 99, 99, datdata, logger); } SimpleSort ss = new SimpleSort(datdata, inputs, outdir, tempdir, quickScan, toFolder, verify, delete, tgz, romba, sevenzip, gz, rar, zip, updateDat, logger); ss.StartProcessing(); } /// /// Wrap getting statistics on a DAT or folder of DATs /// /// List of inputs to be used /// True to show individual DAT statistics, false otherwise private static void InitStats(List inputs, bool single) { List newinputs = new List(); foreach (string input in inputs) { if (File.Exists(input)) { newinputs.Add(input); } if (Directory.Exists(input)) { foreach (string file in Directory.GetFiles(input, "*", SearchOption.AllDirectories)) { newinputs.Add(file); } } } Logger statlog = new Logger(true, "stats.txt"); statlog.Start(); Stats stats = new Stats(newinputs, single, statlog); stats.Process(); statlog.Close(true); } /// /// Wrap splitting a DAT by item type /// /// List of inputs to be used /// Output directory for the split files private static void InitTypeSplit(List inputs, string outdir) { // Loop over the input files foreach (string input in inputs) { if (File.Exists(input)) { DatTools.SplitByType(Path.GetFullPath(input), outdir, Path.GetFullPath(Path.GetDirectoryName(input)), _logger); } else if (Directory.Exists(input)) { foreach (string file in Directory.EnumerateFiles(input, "*", SearchOption.AllDirectories)) { DatTools.SplitByType(file, outdir, Path.GetFullPath((input.EndsWith(Path.DirectorySeparatorChar.ToString()) ? input : input + Path.DirectorySeparatorChar)), _logger); } } else { _logger.Error(input + " is not a valid file or folder!"); Console.WriteLine(); Build.Help(); return; } } } /// /// Wrap converting and updating DAT file from any format to any format /// /// List of input filenames /// /* Normal DAT header info */ /// New filename /// New name /// New description /// New rootdir /// New category /// New version /// New date /// New author /// New email /// New homepage /// New URL /// New comment /// New header /// True to set SuperDAT type, false otherwise /// None, Split, Full /// None, Obsolete, Required, Ignore /// None, Zip, Unzip /// Non-zero flag for output format, zero otherwise for default /// /* Missfile-specific DAT info */ /// True if games are to be used in output, false if roms are /// Generic prefix to be added to each line /// Generic postfix to be added to each line /// Add quotes to each item /// Replace all extensions with another /// Add an extension to all items /// Remove all extensions /// Add the dat name as a directory prefix /// Output files in romba format /// True to output files in TSV format, false to output files in CSV format, null otherwise /// /* Merging and Diffing info */ /// True if input files should be merged into a single file, false otherwise /// Non-zero flag for diffing mode, zero otherwise /// True if the diffed files should be cascade diffed, false if diffed files should be reverse cascaded, null otherwise /// True if the cascade-diffed files should overwrite their inputs, false otherwise /// True if the first cascaded diff file should be skipped on output, false otherwise /// True if the date should not be appended to the default name, false otherwise [OBSOLETE] /// /* Filtering info */ /// Name of the game to match (can use asterisk-partials) /// Name of the rom to match (can use asterisk-partials) /// Type of the rom to match /// Find roms greater than or equal to this size /// Find roms less than or equal to this size /// Find roms equal to this size /// CRC of the rom to match (can use asterisk-partials) /// MD5 of the rom to match (can use asterisk-partials) /// SHA-1 of the rom to match (can use asterisk-partials) /// Select roms with nodump status as follows: null (match all), true (match Nodump only), false (exclude Nodump) /// /* Trimming info */ /// True if we are supposed to trim names to NTFS length, false otherwise /// True if all games should be replaced by '!', false otherwise /// String representing root directory to compare against for length calculation /// /* Output DAT info */ /// Optional param for output directory /// True to clean the game names to WoD standard, false otherwise (default) /// True to allow SL DATs to have game names used instead of descriptions, false otherwise (default) /// True to dedupe the roms in the DAT, false otherwise (default) /// /* Multithreading info */ /// Integer representing the maximum amount of parallelization to be used private static void InitUpdate(List inputs, /* Normal DAT header info */ string filename, string name, string description, string rootdir, string category, string version, string date, string author, string email, string homepage, string url, string comment, string header, bool superdat, string forcemerge, string forcend, string forcepack, OutputFormat outputFormat, /* Missfile-specific DAT info */ bool usegame, string prefix, string postfix, bool quotes, string repext, string addext, bool remext, bool datprefix, bool romba, bool? tsv, /* Merging and Diffing info */ bool merge, DiffMode diffMode, bool? cascade, bool inplace, bool skip, bool bare, /* Filtering info */ string gamename, string romname, string romtype, long sgt, long slt, long seq, string crc, string md5, string sha1, bool? nodump, /* Trimming info */ bool trim, bool single, string root, /* Output DAT info */ string outdir, bool clean, bool softlist, bool dedup, /* Multithreading info */ int maxDegreeOfParallelism) { // Set the special flags ForceMerging fm = ForceMerging.None; switch (forcemerge.ToLowerInvariant()) { case "none": default: fm = ForceMerging.None; break; case "split": fm = ForceMerging.Split; break; case "full": fm = ForceMerging.Full; break; } ForceNodump fn = ForceNodump.None; switch (forcend.ToLowerInvariant()) { case "none": default: fn = ForceNodump.None; break; case "obsolete": fn = ForceNodump.Obsolete; break; case "required": fn = ForceNodump.Required; break; case "ignore": fn = ForceNodump.Ignore; break; } ForcePacking fp = ForcePacking.None; switch (forcepack.ToLowerInvariant()) { case "none": default: fp = ForcePacking.None; break; case "zip": fp = ForcePacking.Zip; break; case "unzip": fp = ForcePacking.Unzip; break; } // Normalize the extensions addext = (addext == "" || addext.StartsWith(".") ? addext : "." + addext); repext = (repext == "" || repext.StartsWith(".") ? repext : "." + repext); // If we're in merge or diff mode and the names aren't set, set defaults if (merge || diffMode != 0) { // Get the values that will be used if (date == "") { date = DateTime.Now.ToString("yyyy-MM-dd"); } if (name == "") { name = (diffMode != 0 ? "DiffDAT" : "MergeDAT") + (superdat ? "-SuperDAT" : "") + (dedup ? "-deduped" : ""); } if (description == "") { description = (diffMode != 0 ? "DiffDAT" : "MergeDAT") + (superdat ? "-SuperDAT" : "") + (dedup ? " - deduped" : ""); if (!bare) { description += " (" + date + ")"; } } if (category == "" && diffMode != 0) { category = "DiffDAT"; } if (author == "") { author = "SabreTools"; } } // Populate the DatData object Dat userInputDat = new Dat { FileName = filename, Name = name, Description = description, RootDir = rootdir, Category = category, Version = version, Date = date, Author = author, Email = email, Homepage = homepage, Url = url, Comment = comment, Header = header, Type = (superdat ? "SuperDAT" : null), ForceMerging = fm, ForceNodump = fn, ForcePacking = fp, MergeRoms = dedup, OutputFormat = outputFormat, UseGame = usegame, Prefix = prefix, Postfix = postfix, Quotes = quotes, RepExt = repext, AddExt = addext, RemExt = remext, GameName = datprefix, Romba = romba, XSV = tsv, }; DatTools.Update(inputs, userInputDat, outputFormat, outdir, merge, diffMode, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, maxDegreeOfParallelism, _logger); } #endregion } }