} // ----------------------------------------- /// <summary> /// Take a crushed archive and extract only the info file, Returns a customized object with some info /// </summary> /// <param name="arcFile"></param> /// <param name="onComplete">(null) if error</param> public static bool loadQuickInfo(string arcFile, Action <Object> onComplete) { if (LOCKED) { ERROR = "LOCKED"; return(false); } if (!check_file_(arcFile, CDCRUSH_EXTENSION)) { return(false); } LOCKED = true; // Delete old files from previous quickInfos FileTools.tryDelete(Path.Combine(TEMP_FOLDER, CDCRUSH_SETTINGS)); FileTools.tryDelete(Path.Combine(TEMP_FOLDER, CDCRUSH_COVER)); var arc = new FreeArc(TOOLS_PATH); // -- arc.onComplete = (success) => { LOCKED = false; if (success) // OK { // Continue var cd = new CueReader(); if (!cd.loadJson(Path.Combine(TEMP_FOLDER, CDCRUSH_SETTINGS))) { ERROR = cd.ERROR; onComplete(null); return; } var info = new { title = cd.CD_TITLE, size0 = (int)new FileInfo(arcFile).Length, size1 = cd.CD_TOTAL_SIZE, audio = cd.CD_AUDIO_QUALITY, tracks = cd.tracks.Count, md5 = cd.getFirstDataTrackMD5(), cover = Path.Combine(TEMP_FOLDER, CDCRUSH_COVER) }; onComplete(info); } else { ERROR = arc.ERROR; onComplete(null); } }; // : Actually extract arc.extractFiles(arcFile, new[] { CDCRUSH_SETTINGS, CDCRUSH_COVER }, TEMP_FOLDER); return(true); } // -----------------------------------------
// -- public JobConvertCue(CrushParams p) : base("Convert CD") { // Check for input files // :: -------------------- if (!CDCRUSH.check_file_(p.inputFile, ".cue")) { fail(msg: CDCRUSH.ERROR); return; } if (string.IsNullOrEmpty(p.outputDir)) { p.outputDir = Path.GetDirectoryName(p.inputFile); } // : NEW : // : ALWAYS Create a subfolder to avoid overwriting the source files try { p.outputDir = Path.Combine(p.outputDir, p.cdTitle + FOLDER_SUFFIX); } catch (ArgumentException) { fail("Output Dir Error " + p.outputDir); return; } if (!FileTools.createDirectory(p.outputDir)) { fail(msg: "Can't create Output Dir " + p.outputDir); return; } p.tempDir = Path.Combine(CDCRUSH.TEMP_FOLDER, Guid.NewGuid().ToString().Substring(0, 12)); if (!FileTools.createDirectory(p.tempDir)) { fail(msg: "Can't create TEMP dir"); return; } // IMPORTANT!! sharedData gets set by value, NOT A POINTER, do not make changes to p after this jobData = p; // -- // - Read the CUE file :: add(new CTask((t) => { var cd = new CueReader(); jobData.cd = cd; if (!cd.load(p.inputFile)) { t.fail(msg: cd.ERROR); return; } // Post CD CUE load :: // In case user named the CD, otherwise it's going to be the same if (!string.IsNullOrWhiteSpace(p.cdTitle)) { cd.CD_TITLE = FileTools.sanitizeFilename(p.cdTitle); } // Real quality to string name cd.CD_AUDIO_QUALITY = CDCRUSH.getAudioQualityString(p.audioQuality); // This flag notes that all files will go to the TEMP folder jobData.workFromTemp = !cd.MULTIFILE; t.complete(); }, "Reading", true)); // - Cut tracks // --------------------------- add(new TaskCutTrackFiles()); // - Compress tracks // --------------------- add(new CTask((t) => { // Only encode the audio tracks foreach (CueTrack tr in (jobData.cd as CueReader).tracks) { if (!tr.isData) { addNextAsync(new TaskCompressTrack(tr)); } } t.complete(); }, "Preparing")); // - Create new CUE file // -------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // DEV: So far : // track.trackFile is UNSET. cd.saveCue needs it to be set. // track.workingFile points to a valid file, some might be in TEMP folder and some in input folder (data tracks) // -- Move files to output folder foreach (var track in cd.tracks) { if (!cd.MULTIFILE) { // Fix the index times to start with 00:00:00 track.setNewTimesReset(); } string ext = Path.GetExtension(track.workingFile); track.trackFile = $"{cd.CD_TITLE} (track {track.trackNo}){ext}"; // Data track was not cut or encoded. // It's in the input folder, don't move it if (track.isData && cd.MULTIFILE) { FileTools.tryCopy(track.workingFile, Path.Combine(p.outputDir, track.trackFile)); } else { FileTools.tryMove(track.workingFile, Path.Combine(p.outputDir, track.trackFile)); } } //-- Create the new CUE file if (!cd.saveCUE(Path.Combine(p.outputDir, cd.CD_TITLE + ".cue"))) { t.fail(cd.ERROR); return; } t.complete(); }, "Finalizing")); // -- COMPLETE -- }// -----------------------------------------
// -- public JobCrush(CrushParams p) : base("Compress CD") { // Check for input files // :: -------------------- if (!CDCRUSH.check_file_(p.inputFile, ".cue")) { fail(msg: CDCRUSH.ERROR); return; } if (string.IsNullOrEmpty(p.outputDir)) { p.outputDir = Path.GetDirectoryName(p.inputFile); } if (!FileTools.createDirectory(p.outputDir)) { fail(msg: "Can't create Output Dir " + p.outputDir); return; } p.tempDir = Path.Combine(CDCRUSH.TEMP_FOLDER, Guid.NewGuid().ToString().Substring(0, 12)); if (!FileTools.createDirectory(p.tempDir)) { fail(msg: "Can't create TEMP dir"); return; } // - p.flag_convert_only = false; // IMPORTANT!! sharedData gets set by value, NOT A POINTER, do not make changes to p after this jobData = p; // hack_setExpectedProgTracks(p.expectedTracks + 3); // -- // - Read the CUE file :: add(new CTask((t) => { var cd = new CueReader(); jobData.cd = cd; if (!cd.load(p.inputFile)) { t.fail(msg: cd.ERROR); return; } // Meaning the tracks are going to be extracted in the temp folder jobData.flag_sourceTracksOnTemp = (!cd.MULTIFILE && cd.tracks.Count > 1); // In case user named the CD, otherwise it's going to be the same if (!string.IsNullOrWhiteSpace(p.cdTitle)) { cd.CD_TITLE = FileTools.sanitizeFilename(p.cdTitle); } // Real quality to string name cd.CD_AUDIO_QUALITY = CDCRUSH.getAudioQualityString(p.audioQuality); // Generate the final arc name now that I have the CD TITLE jobData.finalArcPath = Path.Combine(p.outputDir, cd.CD_TITLE + ".arc"); t.complete(); }, "-Reading")); // - Cut tracks // --------------------------- add(new TaskCutTrackFiles()); // - Compress tracks // --------------------- add(new CTask((t) => { CueReader cd = jobData.cd; foreach (CueTrack tr in cd.tracks) { addNextAsync(new TaskCompressTrack(tr)); } //-- t.complete(); }, "-Preparing")); // Create Archive // Add all tracks to the final archive // --------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // -- Get list of files:: System.Collections.ArrayList files = new System.Collections.ArrayList(); foreach (var tr in cd.tracks) { files.Add(tr.workingFile); // Working file is valid, was set earlier } // Compress all the track files var arc = new FreeArc(CDCRUSH.TOOLS_PATH); t.handleCliReport(arc); arc.compress((string[])files.ToArray(typeof(string)), jobData.finalArcPath, p.compressionLevel); arc.onProgress = (pr) => t.PROGRESS = pr; t.killExtra = () => arc.kill(); }, "Compressing")); // - Create CD SETTINGS and push it to the final archive // ( I am appending these files so that they can be quickly loaded later ) // -------------------- add(new CTask((t) => { CueReader cd = jobData.cd; #if DEBUG LOG.log(cd.getDetailedInfo()); #endif string path_settings = Path.Combine(p.tempDir, CDCRUSH.CDCRUSH_SETTINGS); if (!cd.saveJson(path_settings)) { t.fail(msg: cd.ERROR); return; } // - Cover Image Set? string path_cover; if (p.cover != null) { path_cover = Path.Combine(p.tempDir, CDCRUSH.CDCRUSH_COVER); File.Copy(p.cover, path_cover); } else { path_cover = null; } // - Append the file(s) var arc = new FreeArc(CDCRUSH.TOOLS_PATH); t.handleCliReport(arc); arc.appendFiles(new string[] { path_settings, path_cover }, jobData.finalArcPath); t.killExtra = () => arc.kill(); }, "Finalizing")); // - Get post data add(new CTask((t) => { var finfo = new FileInfo(jobData.finalArcPath); jobData.crushedSize = (int)finfo.Length; t.complete(); }, "-Finalizing")); // -- COMPLETE -- }// -----------------------------------------
// -- public JobCrush(CrushParams p) : base("Compress CD") { // Check for input files // :: -------------------- if (!CDCRUSH.check_file_(p.inputFile, ".cue")) { fail(msg: CDCRUSH.ERROR); return; } if (string.IsNullOrEmpty(p.outputDir)) { p.outputDir = Path.GetDirectoryName(p.inputFile); } if (!FileTools.createDirectory(p.outputDir)) { fail(msg: "Can't create Output Dir " + p.outputDir); return; } p.tempDir = Path.Combine(CDCRUSH.TEMP_FOLDER, Guid.NewGuid().ToString().Substring(0, 12)); if (!FileTools.createDirectory(p.tempDir)) { fail(msg: "Can't create TEMP dir"); return; } // IMPORTANT!! sharedData gets set by value, NOT A POINTER, do not make changes to p after this jobData = p; // -- // - Read the CUE file :: add(new CTask((t) => { var cd = new CueReader(); jobData.cd = cd; if (!cd.load(p.inputFile)) { t.fail(msg: cd.ERROR); return; } // Post CD CUE load :: if (!string.IsNullOrWhiteSpace(p.cdTitle)) // valid: { cd.CD_TITLE = FileTools.sanitizeFilename(p.cdTitle); } // Real quality to string name if (p.audioQuality == 0) { cd.CD_AUDIO_QUALITY = "FLAC"; } else { cd.CD_AUDIO_QUALITY = FFmpeg.QUALITY[(p.audioQuality - 1)].ToString() + "kbps"; } // This flag notes that all files will go to the TEMP folder jobData.workFromTemp = !cd.MULTIFILE; // Generate the final arc name now that I have the CD TITLE jobData.finalArcPath = Path.Combine(p.outputDir, cd.CD_TITLE + ".arc"); t.complete(); }, "Reading", true)); // - Cut tracks if it has to // --------------------------- add(new TaskCutTrackFiles()); // - Compress tracks // --------------------- add(new CTask((t) => { CueReader cd = jobData.cd; foreach (CueTrack tr in cd.tracks) { addNextAsync(new TaskCompressTrack(tr)); } //-- t.complete(); }, "Preparing")); // Create Archive // Add all tracks to the final archive // --------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // -- Get list of files:: System.Collections.ArrayList files = new System.Collections.ArrayList(); foreach (var tr in cd.tracks) { files.Add(tr.workingFile); // Working file is valid, was set earlier } var arc = new FreeArc(CDCRUSH.TOOLS_PATH); t.handleCliReport(arc); arc.compress((string[])files.ToArray(typeof(string)), jobData.finalArcPath); t.killExtra = () => arc.kill(); }, "Compressing")); // - Create CD SETTINGS and push it to the final archive // ( I am appending these files so that they can be quickly loaded later ) // -------------------- add(new CTask((t) => { CueReader cd = jobData.cd; #if DEBUG cd.debugInfo(); #endif string path_settings = Path.Combine(p.tempDir, CDCRUSH.CDCRUSH_SETTINGS); if (!cd.saveJson(path_settings)) { t.fail(msg: cd.ERROR); return; } // - Cover Image Set? string path_cover; if (p.cover != null) { path_cover = Path.Combine(p.tempDir, CDCRUSH.CDCRUSH_COVER); File.Copy(p.cover, path_cover); } else { path_cover = null; } // - Append the file(s) var arc = new FreeArc(CDCRUSH.TOOLS_PATH); t.handleCliReport(arc); arc.appendFiles(new string[] { path_settings, path_cover }, jobData.finalArcPath); t.killExtra = () => arc.kill(); }, "Finalizing")); // - Get post data add(new CTask((t) => { var finfo = new FileInfo(jobData.finalArcPath); jobData.crushedSize = (int)finfo.Length; t.complete(); }, "Finalizing")); // -- COMPLETE -- }// -----------------------------------------
// -- public JobRestore(RestoreParams p) : base("Restore CD") { // Check for input files // :: -------------------- if (!CDCRUSH.check_file_(p.inputFile, CDCRUSH.CDCRUSH_EXTENSION)) { fail(msg: CDCRUSH.ERROR); return; } if (string.IsNullOrEmpty(p.outputDir)) { p.outputDir = Path.GetDirectoryName(p.inputFile); } if (p.flag_folder) { try { p.outputDir = Path.Combine(p.outputDir, Path.GetFileNameWithoutExtension(p.inputFile)); } catch (ArgumentException) { fail("Output Dir Error " + p.outputDir); return; } } if (!FileTools.createDirectory(p.outputDir)) { fail(msg: "Can't create Output Dir " + p.outputDir); return; } p.tempDir = Path.Combine(CDCRUSH.TEMP_FOLDER, Guid.NewGuid().ToString().Substring(0, 12)); if (!FileTools.createDirectory(p.tempDir)) { fail(msg: "Can't create TEMP dir"); return; } // IMPORTANT!! sharedData gets set by value, NOT A POINTER, do not make changes to p after this jobData = p; // - Extract the Archive // ----------------------- add(new CTask((t) => { var arc = new FreeArc(CDCRUSH.TOOLS_PATH); t.handleCliReport(arc); arc.extractAll(p.inputFile, p.tempDir); // In case the operation is aborted t.killExtra = () => { arc.kill(); }; }, "Extracting", true)); // - Read JSON data // - Restore tracks // - JOIN if it has to // ----------------------- add(new CTask((t) => { var cd = new CueReader(); jobData.cd = cd; // This runs in sync: if (!cd.loadJson(Path.Combine(p.tempDir, CDCRUSH.CDCRUSH_SETTINGS))) { t.fail(msg: cd.ERROR); return; } //-- #if DEBUG cd.debugInfo(); #endif // - Push TASK RESTORE tasks right after this one foreach (CueTrack tr in cd.tracks) { addNextAsync(new TaskRestoreTrack(tr)); } //-- t.complete(); }, "Preparing to Restore")); // - Join Tracks // ----------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // -- Join tracks if (p.flag_forceSingle || !cd.MULTIFILE) { // The task will read data from the shared job data var // Will join all tracks in place into track01.bin // Note: Sets track.workingFile to null to moved track addNext(new TaskJoinTrackFiles()); } //-- t.complete(); }, "Preparing to Join")); // - Prepare tracks `trackfile` which is the track written to the CUE // - Convert tracks // - Move files to final destination // - Create CUE files // - Delete Temp Files // ----------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // -- foreach (var track in cd.tracks) { if (p.flag_forceSingle && cd.MULTIFILE) // :: CONVERT MULTI TO SINGLE { track.setNewTimesBasedOnSector(); } if (cd.MULTIFILE && !p.flag_forceSingle) { track.trackFile = cd.CD_TITLE + " " + track.getFilenameRaw(); } if (!cd.MULTIFILE || p.flag_forceSingle) { if (track.trackNo == 1) { track.trackFile = cd.CD_TITLE + ".bin"; } else { track.trackFile = null; } } // -- // Move ALL files to final output folder // NULL workingFile means that is has been deleted if (track.workingFile != null) { FileTools.tryMove(track.workingFile, Path.Combine(p.outputDir, track.trackFile)); } } // -- end track processing // -- // Create CUE File if (!cd.saveCUE(Path.Combine(p.outputDir, cd.CD_TITLE + ".cue"))) { t.fail(cd.ERROR); return; } t.complete(); }, "Moving, Finalizing")); // - Complete - }// -----------------------------------------
// -- public JobConvertCue(CrushParams p) : base("Convert CD") { // Check for input files // :: -------------------- if (!CDCRUSH.check_file_(p.inputFile, ".cue")) { fail(msg: CDCRUSH.ERROR); return; } if (string.IsNullOrEmpty(p.outputDir)) { p.outputDir = Path.GetDirectoryName(p.inputFile); } // : NEW : // : ALWAYS Create a subfolder to avoid overwriting the source files p.outputDir = CDCRUSH.checkCreateUniqueOutput(p.outputDir, p.cdTitle + CDCRUSH.RESTORED_FOLDER_SUFFIX); if (p.outputDir == null) { fail("Output Dir Error " + p.outputDir); return; } // - p.tempDir = Path.Combine(CDCRUSH.TEMP_FOLDER, Guid.NewGuid().ToString().Substring(0, 12)); if (!FileTools.createDirectory(p.tempDir)) { fail(msg: "Can't create TEMP dir"); return; } // Useful to know. p.flag_convert_only = true; // IMPORTANT!! sharedData gets set by value, NOT A POINTER, do not make changes to p after this jobData = p; hack_setExpectedProgTracks(p.expectedTracks + 2); // -- // - Read the CUE file :: add(new CTask((t) => { var cd = new CueReader(); jobData.cd = cd; if (!cd.load(p.inputFile)) { t.fail(msg: cd.ERROR); return; } // -- if (cd.tracks.Count == 1) { t.fail(msg: "No point in converting. No audio tracks on the cd."); return; } // Meaning the tracks are going to be extracted in the temp folder jobData.flag_sourceTracksOnTemp = (!cd.MULTIFILE && cd.tracks.Count > 1); // In case user named the CD, otherwise it's going to be the same if (!string.IsNullOrWhiteSpace(p.cdTitle)) { cd.CD_TITLE = FileTools.sanitizeFilename(p.cdTitle); } // Real quality to string name cd.CD_AUDIO_QUALITY = CDCRUSH.getAudioQualityString(p.audioQuality); t.complete(); }, "-Reading", "Reading CUE data and preparing")); // - Cut tracks // --------------------------- add(new TaskCutTrackFiles()); // - Compress tracks // --------------------- add(new CTask((t) => { // Only encode the audio tracks foreach (CueTrack tr in (jobData.cd as CueReader).tracks) { if (!tr.isData) { addNextAsync(new TaskCompressTrack(tr)); } } t.complete(); }, "-Preparing", "Preparing to compress tracks")); // - Create new CUE file // -------------------- add(new CTask((t) => { CueReader cd = jobData.cd; // DEV: So far : // track.trackFile is UNSET. cd.saveCue needs it to be set. // track.workingFile points to a valid file, some might be in TEMP folder and some in input folder (data tracks) int stepProgress = (int)Math.Round(100.0f / cd.tracks.Count); // -- Move files to output folder foreach (var track in cd.tracks) { if (!cd.MULTIFILE) { // Fix the index times to start with 00:00:00 track.setNewTimesReset(); } string ext = Path.GetExtension(track.workingFile); track.trackFile = $"{cd.CD_TITLE} (track {track.trackNo}){ext}"; // Data track was not cut or encoded. // It's in the input folder, don't move it if (track.isData && cd.MULTIFILE) { FileTools.tryCopy(track.workingFile, Path.Combine(p.outputDir, track.trackFile)); } else { // TaskCompress already put the audio files on the output folder // But it's no big deal calling it again // This is for the data tracks that are on the temp folder FileTools.tryMove(track.workingFile, Path.Combine(p.outputDir, track.trackFile)); } t.PROGRESS += stepProgress; } //-- Create the new CUE file if (!cd.saveCUE(Path.Combine(p.outputDir, cd.CD_TITLE + ".cue"))) { t.fail(cd.ERROR); return; } t.complete(); }, "Finalizing", "Calculating track data and creating .CUE")); // -- COMPLETE -- }// -----------------------------------------