using System; using System.IO; #if NET40 using System.Threading; using System.Threading.Tasks; #endif using BinaryObjectScanner; using MPF.Frontend; using MPF.Frontend.Tools; using SabreTools.RedumpLib; using SabreTools.RedumpLib.Data; using SabreTools.RedumpLib.Web; namespace MPF.Check { public class Program { public static void Main(string[] args) { // Create a default options object var options = new Frontend.Options() { // Internal Program InternalProgram = InternalProgram.NONE, // Extra Dumping Options ScanForProtection = false, AddPlaceholders = true, PullAllInformation = false, AddFilenameSuffix = false, OutputSubmissionJSON = false, IncludeArtifacts = false, CompressLogFiles = false, DeleteUnnecessaryFiles = false, CreateIRDAfterDumping = false, // Protection Scanning Options ScanArchivesForProtection = true, ScanPackersForProtection = false, IncludeDebugProtectionInformation = false, HideDriveLetters = false, // Redump Login Information RedumpUsername = null, RedumpPassword = null, }; // Try processing the standalone arguments bool? standaloneProcessed = OptionsLoader.ProcessStandaloneArguments(args); if (standaloneProcessed != false) { if (standaloneProcessed == null) DisplayHelp(); return; } // Try processing the common arguments bool success = OptionsLoader.ProcessCommonArguments(args, out MediaType mediaType, out RedumpSystem? knownSystem, out var error); if (!success) { DisplayHelp(error); return; } // Loop through and process options int startIndex = 2; CommandOptions opts = LoadFromArguments(args, options, ref startIndex); if (options.InternalProgram == InternalProgram.NONE) { DisplayHelp("A program name needs to be provided"); return; } // Make new Progress objects var resultProgress = new Progress(); resultProgress.ProgressChanged += ConsoleLogger.ProgressUpdated; var protectionProgress = new Progress(); protectionProgress.ProgressChanged += ConsoleLogger.ProgressUpdated; // Validate the supplied credentials bool? validated = RedumpClient.ValidateCredentials(options.RedumpUsername ?? string.Empty, options.RedumpPassword ?? string.Empty).GetAwaiter().GetResult(); string message = validated switch { true => "Redump username and password accepted!", false => "Redump username and password denied!", null => "An error occurred validating your credentials!", }; if (!string.IsNullOrEmpty(message)) Console.WriteLine(message); // Loop through all the rest of the args for (int i = startIndex; i < args.Length; i++) { // Check for a file if (!File.Exists(args[i].Trim('"'))) { DisplayHelp($"{args[i].Trim('"')} does not exist"); return; } // Get the full file path string filepath = Path.GetFullPath(args[i].Trim('"')); // Now populate an environment Drive? drive = null; if (!string.IsNullOrEmpty(opts.DevicePath)) drive = Drive.Create(null, opts.DevicePath!); var env = new DumpEnvironment(options, filepath, drive, knownSystem, mediaType, internalProgram: null, parameters: null); // Finally, attempt to do the output dance var result = env.VerifyAndSaveDumpOutput(resultProgress, protectionProgress, seedInfo: opts.Seed).GetAwaiter().GetResult(); Console.WriteLine(result.Message); } } /// /// Display help for MPF.Check /// /// Error string to prefix the help text with private static void DisplayHelp(string? error = null) { if (error != null) Console.WriteLine(error); Console.WriteLine("Usage:"); Console.WriteLine("MPF.Check [options] ..."); Console.WriteLine(); Console.WriteLine("Standalone Options:"); Console.WriteLine("-h, -? Show this help text"); Console.WriteLine("-lc, --listcodes List supported comment/content site codes"); Console.WriteLine("-lm, --listmedia List supported media types"); Console.WriteLine("-ls, --listsystems List supported system types"); Console.WriteLine("-lp, --listprograms List supported dumping program outputs"); Console.WriteLine(); Console.WriteLine("Check Options:"); Console.WriteLine("-u, --use Dumping program output type [REQUIRED]"); Console.WriteLine(" --load-seed Load a seed submission JSON for user information"); Console.WriteLine(" --no-placeholders Disable placeholder values in submission info"); Console.WriteLine(" --create-ird Create IRD from output files (PS3 only)"); Console.WriteLine("-c, --credentials Redump username and password"); Console.WriteLine(" --pull-all Pull all information from Redump (requires --credentials)"); Console.WriteLine("-p, --path Physical drive path for additional checks"); Console.WriteLine("-s, --scan Enable copy protection scan (requires --path)"); Console.WriteLine(" --disable-archives Disable scanning archives (requires --scan)"); Console.WriteLine(" --enable-packers Enable scanning for packers (requires --scan)"); Console.WriteLine(" --enable-debug Enable debug protection information (requires --scan)"); Console.WriteLine(" --hide-drive-letters Hide drive letters from scan output (requires --scan)"); Console.WriteLine("-x, --suffix Enable adding filename suffix"); Console.WriteLine("-j, --json Enable submission JSON output"); Console.WriteLine(" --include-artifacts Include artifacts in JSON (requires --json)"); Console.WriteLine("-z, --zip Enable log file compression"); Console.WriteLine("-d, --delete Enable unnecessary file deletion"); Console.WriteLine(); } /// /// Load the current set of options from application arguments /// private static CommandOptions LoadFromArguments(string[] args, Frontend.Options options, ref int startIndex) { // Create return values var opts = new CommandOptions(); // These values require multiple parts to be active bool scan = false, enableArchives = true, enablePackers = false, enableDebug = false, hideDriveLetters = false; // If we have no arguments, just return if (args == null || args.Length == 0) { startIndex = 0; return opts; } // If we have an invalid start index, just return if (startIndex < 0 || startIndex >= args.Length) return opts; // Loop through the arguments and parse out values for (; startIndex < args.Length; startIndex++) { // Use specific program if (args[startIndex].StartsWith("-u=") || args[startIndex].StartsWith("--use=")) { string internalProgram = args[startIndex].Split('=')[1]; options.InternalProgram = Frontend.Options.ToInternalProgram(internalProgram); } else if (args[startIndex] == "-u" || args[startIndex] == "--use") { string internalProgram = args[startIndex + 1]; options.InternalProgram = Frontend.Options.ToInternalProgram(internalProgram); startIndex++; } // Include seed info file else if (args[startIndex].StartsWith("--load-seed=")) { string seedInfo = args[startIndex].Split('=')[1]; opts.Seed = Builder.CreateFromFile(seedInfo); } else if (args[startIndex] == "--load-seed") { string seedInfo = args[startIndex + 1]; opts.Seed = Builder.CreateFromFile(seedInfo); startIndex++; } // Disable placeholder values in submission info else if (args[startIndex].Equals("--no-placeholders")) { options.AddPlaceholders = false; } // Create IRD from output files (PS3 only) else if (args[startIndex].Equals("--create-ird")) { options.CreateIRDAfterDumping = true; } // Redump login else if (args[startIndex].StartsWith("-c=") || args[startIndex].StartsWith("--credentials=")) { string[] credentials = args[startIndex].Split('=')[1].Split(';'); options.RedumpUsername = credentials[0]; options.RedumpPassword = credentials[1]; } else if (args[startIndex] == "-c" || args[startIndex] == "--credentials") { options.RedumpUsername = args[startIndex + 1]; options.RedumpPassword = args[startIndex + 2]; startIndex += 2; } // Pull all information (requires Redump login) else if (args[startIndex].Equals("--pull-all")) { options.PullAllInformation = true; } // Use a device path for physical checks else if (args[startIndex].StartsWith("-p=") || args[startIndex].StartsWith("--path=")) { opts.DevicePath = args[startIndex].Split('=')[1]; } else if (args[startIndex] == "-p" || args[startIndex] == "--path") { opts.DevicePath = args[startIndex + 1]; startIndex++; } // Scan for protection (requires device path) else if (args[startIndex].Equals("-s") || args[startIndex].Equals("--scan")) { scan = true; } // Disable scanning archives (requires --scan) else if (args[startIndex].Equals("--disable-archives")) { enableArchives = false; } // Enable scanning for packers (requires --scan) else if (args[startIndex].Equals("--enable-packers")) { enablePackers = true; } // Enable debug protection information (requires --scan) else if (args[startIndex].Equals("--enable-debug")) { enableDebug = true; } // Hide drive letters from scan output (requires --scan) else if (args[startIndex].Equals("--hide-drive-letters")) { hideDriveLetters = true; } // Add filename suffix else if (args[startIndex].Equals("-x") || args[startIndex].Equals("--suffix")) { options.AddFilenameSuffix = true; } // Output submission JSON else if (args[startIndex].Equals("-j") || args[startIndex].Equals("--json")) { options.OutputSubmissionJSON = true; } // Output submission JSON else if (args[startIndex].Equals("--include-artifacts")) { options.IncludeArtifacts = true; } // Compress log and extraneous files else if (args[startIndex].Equals("-z") || args[startIndex].Equals("--zip")) { options.CompressLogFiles = true; } // Delete unnecessary files files else if (args[startIndex].Equals("-d") || args[startIndex].Equals("--delete")) { options.DeleteUnnecessaryFiles = true; } // Default, we fall out else { break; } } // Now deal with the complex options options.ScanForProtection = scan && !string.IsNullOrEmpty(opts.DevicePath); options.ScanArchivesForProtection = enableArchives && scan && !string.IsNullOrEmpty(opts.DevicePath); options.ScanPackersForProtection = enablePackers && scan && !string.IsNullOrEmpty(opts.DevicePath); options.IncludeDebugProtectionInformation = enableDebug && scan && !string.IsNullOrEmpty(opts.DevicePath); options.HideDriveLetters = hideDriveLetters && scan && !string.IsNullOrEmpty(opts.DevicePath); return opts; } /// /// Represents commandline options /// private class CommandOptions { /// /// Seed submission info from an input file /// public SubmissionInfo? Seed { get; set; } = null; /// /// Path to the device to scan /// public string? DevicePath { get; set; } = null; } } }