2020-12-10 11:28:11 -08:00
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2020-12-10 15:52:31 -08:00
|
|
|
using System.Linq;
|
2024-03-05 03:04:47 -05:00
|
|
|
#if NET40_OR_GREATER || NETCOREAPP
|
2020-12-10 11:28:11 -08:00
|
|
|
using System.Threading.Tasks;
|
2024-03-05 03:04:47 -05:00
|
|
|
#endif
|
2020-12-10 11:28:11 -08:00
|
|
|
using SabreTools.Core;
|
2020-12-10 23:24:09 -08:00
|
|
|
using SabreTools.DatFiles;
|
2020-12-14 15:43:01 -08:00
|
|
|
using SabreTools.DatItems;
|
2020-12-10 11:28:11 -08:00
|
|
|
using SabreTools.IO;
|
2020-12-10 14:03:07 -08:00
|
|
|
using SabreTools.Logging;
|
2020-12-11 10:10:56 -08:00
|
|
|
using SabreTools.Reports;
|
2020-12-10 11:28:11 -08:00
|
|
|
|
2020-12-10 23:24:09 -08:00
|
|
|
namespace SabreTools.DatTools
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
2020-12-21 11:38:56 -08:00
|
|
|
/// <summary>
|
|
|
|
|
/// Helper methods for writing from DatFiles
|
|
|
|
|
/// </summary>
|
2020-12-10 14:03:07 -08:00
|
|
|
public class Writer
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
2020-12-10 14:03:07 -08:00
|
|
|
#region Logging
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Logging object
|
|
|
|
|
/// </summary>
|
2023-04-19 16:39:58 -04:00
|
|
|
private static readonly Logger logger = new();
|
2020-12-10 14:03:07 -08:00
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2020-12-10 11:28:11 -08:00
|
|
|
/// <summary>
|
|
|
|
|
/// Create and open an output file for writing direct from a dictionary
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="datFile">Current DatFile object to write from</param>
|
|
|
|
|
/// <param name="outDir">Set the output directory (current directory on null)</param>
|
|
|
|
|
/// <param name="overwrite">True if files should be overwritten (default), false if they should be renamed instead</param>
|
|
|
|
|
/// <param name="ignoreblanks">True if blank roms should be skipped on output, false otherwise (default)</param>
|
|
|
|
|
/// <param name="quotes">True if quotes are assumed in supported types (default), false otherwise</param>
|
|
|
|
|
/// <param name="throwOnError">True if the error that is thrown should be thrown back to the caller, false otherwise</param>
|
|
|
|
|
/// <returns>True if the DAT was written correctly, false otherwise</returns>
|
2020-12-10 11:58:46 -08:00
|
|
|
public static bool Write(
|
2020-12-10 11:28:11 -08:00
|
|
|
DatFile datFile,
|
2024-03-06 00:53:32 -05:00
|
|
|
string ?outDir,
|
2020-12-10 11:28:11 -08:00
|
|
|
bool overwrite = true,
|
|
|
|
|
bool ignoreblanks = false,
|
|
|
|
|
bool quotes = true,
|
|
|
|
|
bool throwOnError = false)
|
|
|
|
|
{
|
|
|
|
|
// If we have nothing writable, abort
|
|
|
|
|
if (!HasWritable(datFile))
|
|
|
|
|
{
|
|
|
|
|
logger.User("There were no items to write out!");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Ensure the output directory is set and created
|
2024-03-06 00:53:32 -05:00
|
|
|
outDir = outDir?.Ensure(create: true);
|
2020-12-10 11:28:11 -08:00
|
|
|
|
2023-04-19 16:39:58 -04:00
|
|
|
InternalStopwatch watch = new($"Writing out internal dat to '{outDir}'");
|
2021-02-02 14:09:49 -08:00
|
|
|
|
2020-12-10 11:28:11 -08:00
|
|
|
// If the DAT has no output format, default to XML
|
|
|
|
|
if (datFile.Header.DatFormat == 0)
|
|
|
|
|
{
|
|
|
|
|
logger.Verbose("No DAT format defined, defaulting to XML");
|
|
|
|
|
datFile.Header.DatFormat = DatFormat.Logiqx;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure that the three essential fields are filled in
|
2020-12-10 11:58:46 -08:00
|
|
|
EnsureHeaderFields(datFile);
|
2020-12-10 11:28:11 -08:00
|
|
|
|
|
|
|
|
// Bucket roms by game name, if not already
|
2020-12-14 15:43:01 -08:00
|
|
|
datFile.Items.BucketBy(ItemKey.Machine, DedupeType.None);
|
2020-12-10 11:28:11 -08:00
|
|
|
|
|
|
|
|
// Output the number of items we're going to be writing
|
|
|
|
|
logger.User($"A total of {datFile.Items.TotalCount - datFile.Items.RemovedCount} items will be written out to '{datFile.Header.FileName}'");
|
|
|
|
|
|
|
|
|
|
// Get the outfile names
|
2024-03-06 00:53:32 -05:00
|
|
|
Dictionary<DatFormat, string> outfiles = datFile.Header.CreateOutFileNames(outDir!, overwrite);
|
2020-12-10 11:28:11 -08:00
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
// Write out all required formats
|
2024-02-28 22:54:56 -05:00
|
|
|
#if NET452_OR_GREATER || NETCOREAPP
|
2020-12-10 11:28:11 -08:00
|
|
|
Parallel.ForEach(outfiles.Keys, Globals.ParallelOptions, datFormat =>
|
2024-02-28 22:54:56 -05:00
|
|
|
#elif NET40_OR_GREATER
|
|
|
|
|
Parallel.ForEach(outfiles.Keys, datFormat =>
|
|
|
|
|
#else
|
|
|
|
|
foreach (var datFormat in outfiles.Keys)
|
|
|
|
|
#endif
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
|
|
|
|
string outfile = outfiles[datFormat];
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
DatFile.Create(datFormat, datFile, quotes)?.WriteToFile(outfile, ignoreblanks, throwOnError);
|
|
|
|
|
}
|
2021-01-12 15:54:14 -08:00
|
|
|
catch (Exception ex) when (!throwOnError)
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
2021-02-03 11:24:03 -08:00
|
|
|
logger.Error(ex, $"Datfile '{outfile}' could not be written out");
|
2020-12-10 11:28:11 -08:00
|
|
|
}
|
2024-02-28 21:59:13 -05:00
|
|
|
#if NET40_OR_GREATER || NETCOREAPP
|
2020-12-10 11:28:11 -08:00
|
|
|
});
|
2024-02-28 21:59:13 -05:00
|
|
|
#else
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2020-12-10 11:28:11 -08:00
|
|
|
}
|
2021-01-12 15:54:14 -08:00
|
|
|
catch (Exception ex) when (!throwOnError)
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
|
|
|
|
logger.Error(ex);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2021-02-02 14:09:49 -08:00
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
watch.Stop();
|
|
|
|
|
}
|
2020-12-10 11:28:11 -08:00
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-10 15:52:31 -08:00
|
|
|
/// <summary>
|
|
|
|
|
/// Write the stats out to console for the current DatFile
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="datFile">Current DatFile object to write from</param>
|
|
|
|
|
public static void WriteStatsToConsole(DatFile datFile)
|
|
|
|
|
{
|
2024-03-04 22:52:03 -05:00
|
|
|
long diskCount = datFile.Items.GetItemCount(ItemType.Disk);
|
|
|
|
|
long mediaCount = datFile.Items.GetItemCount(ItemType.Media);
|
|
|
|
|
long romCount = datFile.Items.GetItemCount(ItemType.Rom);
|
|
|
|
|
|
|
|
|
|
if (diskCount + mediaCount + romCount == 0)
|
2020-12-10 15:52:31 -08:00
|
|
|
datFile.Items.RecalculateStats();
|
|
|
|
|
|
2020-12-14 15:43:01 -08:00
|
|
|
datFile.Items.BucketBy(ItemKey.Machine, DedupeType.None, norename: true);
|
2020-12-10 15:52:31 -08:00
|
|
|
|
2021-02-18 11:13:11 -08:00
|
|
|
var statsList = new List<DatStatistics>
|
|
|
|
|
{
|
2024-02-28 22:54:56 -05:00
|
|
|
new()
|
2021-02-18 11:13:11 -08:00
|
|
|
{
|
|
|
|
|
Statistics = datFile.Items,
|
|
|
|
|
DisplayName = datFile.Header.FileName,
|
2023-04-19 16:39:58 -04:00
|
|
|
MachineCount = datFile.Items.Keys.Count,
|
2021-02-18 11:13:11 -08:00
|
|
|
IsDirectory = false,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
var consoleOutput = BaseReport.Create(StatReportFormat.None, statsList);
|
2024-02-28 19:19:50 -05:00
|
|
|
consoleOutput!.WriteToFile(null, true, true);
|
2020-12-10 15:52:31 -08:00
|
|
|
}
|
|
|
|
|
|
2020-12-10 11:28:11 -08:00
|
|
|
/// <summary>
|
|
|
|
|
/// Ensure that FileName, Name, and Description are filled with some value
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="datFile">Current DatFile object to write from</param>
|
2020-12-10 11:58:46 -08:00
|
|
|
private static void EnsureHeaderFields(DatFile datFile)
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
|
|
|
|
// Empty FileName
|
2024-02-29 00:14:16 -05:00
|
|
|
if (string.IsNullOrEmpty(datFile.Header.FileName))
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
2024-02-29 00:14:16 -05:00
|
|
|
if (string.IsNullOrEmpty(datFile.Header.Name) && string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.FileName = datFile.Header.Name = datFile.Header.Description = "Default";
|
|
|
|
|
|
2024-02-29 00:14:16 -05:00
|
|
|
else if (string.IsNullOrEmpty(datFile.Header.Name) && !string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.FileName = datFile.Header.Name = datFile.Header.Description;
|
|
|
|
|
|
2024-02-29 00:14:16 -05:00
|
|
|
else if (!string.IsNullOrEmpty(datFile.Header.Name) && string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.FileName = datFile.Header.Description = datFile.Header.Name;
|
|
|
|
|
|
2024-02-29 00:14:16 -05:00
|
|
|
else if (!string.IsNullOrEmpty(datFile.Header.Name) && !string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.FileName = datFile.Header.Description;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Filled FileName
|
|
|
|
|
else
|
|
|
|
|
{
|
2024-02-29 00:14:16 -05:00
|
|
|
if (string.IsNullOrEmpty(datFile.Header.Name) && string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.Name = datFile.Header.Description = datFile.Header.FileName;
|
|
|
|
|
|
2024-02-29 00:14:16 -05:00
|
|
|
else if (string.IsNullOrEmpty(datFile.Header.Name) && !string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.Name = datFile.Header.Description;
|
|
|
|
|
|
2024-02-29 00:14:16 -05:00
|
|
|
else if (!string.IsNullOrEmpty(datFile.Header.Name) && string.IsNullOrEmpty(datFile.Header.Description))
|
2020-12-10 11:28:11 -08:00
|
|
|
datFile.Header.Description = datFile.Header.Name;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Get if the DatFile has any writable items
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="datFile">Current DatFile object to write from</param>
|
|
|
|
|
/// <returns>True if there are any writable items, false otherwise</returns>
|
2020-12-10 11:58:46 -08:00
|
|
|
private static bool HasWritable(DatFile datFile)
|
2020-12-10 11:28:11 -08:00
|
|
|
{
|
|
|
|
|
// Force a statistics recheck, just in case
|
|
|
|
|
datFile.Items.RecalculateStats();
|
|
|
|
|
|
|
|
|
|
// If there's nothing there, abort
|
|
|
|
|
if (datFile.Items.TotalCount == 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// If every item is removed, abort
|
|
|
|
|
if (datFile.Items.TotalCount == datFile.Items.RemovedCount)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return true;
|
2024-02-28 22:54:56 -05:00
|
|
|
}
|
2020-12-10 11:28:11 -08:00
|
|
|
}
|
|
|
|
|
}
|