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 importing and updating DATs /// /// private static void InitImport(bool ignore) { IImport imp = new ImportTwo(_datroot, _connectionString, _logger, ignore); imp.UpdateDatabase(); } /// /// 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, _connectionString, _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(_connectionString)) { 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 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 /// True to output to ClrMamePro format /// True to output to Missfile format /// True to output to RomCenter format /// True to output to SabreDAT format /// True to output to Logiqx XML format /// /* 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 /// 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) 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, bool outputCMP, bool outputMiss, bool outputRC, bool outputSD, bool outputXML, /* Missfile-specific DAT info */ bool usegame, string prefix, string postfix, bool quotes, string repext, string addext, bool datprefix, bool romba, bool? tsv, /* Merging and Diffing info */ bool merge, DiffMode diff, 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) { // 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 || diff != 0) { // Get the values that will be used if (date == "") { date = DateTime.Now.ToString("yyyy-MM-dd"); } if (name == "") { name = (diff != 0 ? "DiffDAT" : "MergeDAT") + (superdat ? "-SuperDAT" : "") + (dedup ? "-deduped" : ""); } if (description == "") { description = (diff != 0 ? "DiffDAT" : "MergeDAT") + (superdat ? "-SuperDAT" : "") + (dedup ? " - deduped" : ""); if (!bare) { description += " (" + date + ")"; } } if (category == "" && diff != 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, UseGame = usegame, Prefix = prefix, Postfix = postfix, Quotes = quotes, RepExt = repext, AddExt = addext, GameName = datprefix, Romba = romba, XSV = tsv, }; if (outputCMP) { userInputDat.OutputFormat = OutputFormat.ClrMamePro; DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } if (outputMiss || tsv != null) { userInputDat.OutputFormat = OutputFormat.MissFile; DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } if (outputRC) { userInputDat.OutputFormat = OutputFormat.RomCenter; DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } if (outputSD) { userInputDat.OutputFormat = OutputFormat.SabreDat; DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } if (outputXML) { userInputDat.OutputFormat = OutputFormat.Xml; DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } if (!outputCMP && !(outputMiss || tsv != null) && !outputRC && !outputSD && !outputXML) { DatTools.Update(inputs, userInputDat, outdir, merge, diff, cascade, inplace, skip, bare, clean, softlist, gamename, romname, romtype, sgt, slt, seq, crc, md5, sha1, nodump, trim, single, root, _logger); } } /// /// Wrap creating a DAT file from files or a directory /// /// List of innput filenames /// New filename /// New name /// New description /// New category /// New version /// New author /// True to set forcepacking="unzip" on the created file, false otherwise /// True to output in CMP format, false to output in Logiqx XML /// 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 private static void InitDatFromDir(List inputs, string filename, string name, string description, string category, string version, string author, bool forceunpack, bool old, bool romba, bool superdat, bool noMD5, bool noSHA1, bool bare, bool archivesAsFiles, bool enableGzip, string tempDir) { // Create a new DATFromDir object and process the inputs Dat datdata = 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 = (old ? OutputFormat.ClrMamePro : OutputFormat.Xml), Romba = romba, Type = (superdat ? "SuperDAT" : ""), Files = new Dictionary>(), }; DATFromDir dfd = new DATFromDir(inputs, datdata, noMD5, noSHA1, bare, archivesAsFiles, enableGzip, tempDir, _logger); bool success = dfd.Start(); /* // For DFDParallel only DatTools.WriteDatfile(dfd.DatData, "", _logger); */ // If we failed, show the help if (!success) { 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) { // Verify the input files foreach (string input in inputs) { if (!File.Exists(input) && !Directory.Exists(input)) { _logger.Error(input + " is not a valid file or folder!"); Console.WriteLine(); Build.Help(); return; } } // Convert comma-separated strings to list List extaList = exta.Split(',').ToList(); List extbList = extb.Split(',').ToList(); Split es = new Split(inputs, extaList, extbList, outdir, _logger); es.Process(); return; } /// /// 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) { // Verify the input files foreach (string input in inputs) { if (!File.Exists(input) && !Directory.Exists(input)) { _logger.Error(input + " is not a valid file or folder!"); Console.WriteLine(); Build.Help(); return; } } // If so, run the program Split hs = new Split(inputs, outdir, _logger); hs.Process(); } /// /// Wrap creating an Offline merged DAT set /// /// Old-current DAT with merged and deduped values /// Old-current missing DAT with merged and deduped values /// New-current DAT with merged and deduped values /// True if all values should be replaced with default 0-byte values, false otherwise private static void InitOfflineMerge(string currentAllMerged, string currentMissingMerged, string currentNewMerged, bool fake) { OfflineMerge om = new OfflineMerge(currentAllMerged, currentMissingMerged, currentNewMerged, fake, _logger); bool success = om.Process(); if (!success) { _logger.Warning("At least one complete DAT and the fixdat is needed to run!"); } } /// /// 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 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, _connectionString)) { _logger.Log("Source " + name + " added!"); } else { _logger.Error("Source " + name + " could not be added!"); } } /// /// 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, _connectionString)) { _logger.Log("Source '" + srcid + "' removed!"); } else { _logger.Error("Source with id '" + srcid + "' could not be removed."); } } else { _logger.Error("Invalid input"); } } /// /// 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, _connectionString)) { _logger.Log("System " + manufacturer + " - " + system + " added!"); } else { _logger.Error("System " + manufacturer + " - " + system + " could not be added!"); } } /// /// 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, _connectionString)) { _logger.Log("System '" + sysid + "' removed!"); } else { _logger.Error("System with id '" + sysid + "' could not be removed."); } } else { _logger.Error("Invalid input"); } } #endregion } }