using System; using System.Collections.Generic; using SabreTools.DatItems; using SabreTools.DatItems.Formats; using SabreTools.Hashing; namespace SabreTools.DatFiles.Formats { /// /// Represents a hashfile such as an SFV, MD5, or SHA-1 file /// internal abstract class Hashfile : SerializableDatFile { #region Fields /// public override ItemType[] SupportedTypes => [ ItemType.Disk, ItemType.Media, ItemType.Rom, ]; // Private instance variables specific to Hashfile DATs protected HashType _hash; #endregion /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Hashfile(DatFile? datFile) : base(datFile) { } /// public override void ParseFile(string filename, int indexId, bool keep, bool statsOnly = false, bool throwOnError = false) { try { // Deserialize the input file var hashfile = Serialization.Deserializers.Hashfile.DeserializeFile(filename, _hash); var metadata = new Serialization.CrossModel.Hashfile().Serialize(hashfile); // Convert to the internal format ConvertMetadata(metadata, filename, indexId, keep, statsOnly); } catch (Exception ex) when (!throwOnError) { string message = $"'{filename}' - An error occurred during parsing"; _logger.Error(ex, message); } } /// protected override List? GetMissingRequiredFields(DatItem datItem) { List missingFields = []; // Check item name if (string.IsNullOrEmpty(datItem.GetName())) missingFields.Add(Models.Metadata.Rom.NameKey); // Check hash linked to specific Hashfile type switch (_hash) { case HashType.CRC32: case HashType.CRC32_AIXM: case HashType.CRC32_AUTOSAR: case HashType.CRC32_BASE91D: case HashType.CRC32_BZIP2: case HashType.CRC32_CDROMEDC: case HashType.CRC32_CKSUM: case HashType.CRC32_ISCSI: case HashType.CRC32_ISOHDLC: case HashType.CRC32_JAMCRC: case HashType.CRC32_MEF: case HashType.CRC32_MPEG2: case HashType.CRC32_XFER: switch (datItem) { case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.CRCKey))) missingFields.Add(Models.Metadata.Rom.CRCKey); break; default: missingFields.Add(Models.Metadata.Rom.CRCKey); break; } break; case HashType.MD2: switch (datItem) { case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.MD2Key))) missingFields.Add(Models.Metadata.Rom.MD2Key); break; default: missingFields.Add(Models.Metadata.Rom.MD2Key); break; } break; case HashType.MD4: switch (datItem) { case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.MD4Key))) missingFields.Add(Models.Metadata.Rom.MD4Key); break; default: missingFields.Add(Models.Metadata.Rom.MD4Key); break; } break; case HashType.MD5: switch (datItem) { case Disk disk: if (string.IsNullOrEmpty(disk.GetStringFieldValue(Models.Metadata.Disk.MD5Key))) missingFields.Add(Models.Metadata.Disk.MD5Key); break; case Media medium: if (string.IsNullOrEmpty(medium.GetStringFieldValue(Models.Metadata.Media.MD5Key))) missingFields.Add(Models.Metadata.Media.MD5Key); break; case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.MD5Key))) missingFields.Add(Models.Metadata.Rom.MD5Key); break; default: missingFields.Add(Models.Metadata.Rom.MD5Key); break; } break; case HashType.SHA1: switch (datItem) { case Disk disk: if (string.IsNullOrEmpty(disk.GetStringFieldValue(Models.Metadata.Disk.SHA1Key))) missingFields.Add(Models.Metadata.Disk.SHA1Key); break; case Media medium: if (string.IsNullOrEmpty(medium.GetStringFieldValue(Models.Metadata.Media.SHA1Key))) missingFields.Add(Models.Metadata.Media.SHA1Key); break; case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.SHA1Key))) missingFields.Add(Models.Metadata.Rom.SHA1Key); break; default: missingFields.Add(Models.Metadata.Rom.SHA1Key); break; } break; case HashType.SHA256: switch (datItem) { case Media medium: if (string.IsNullOrEmpty(medium.GetStringFieldValue(Models.Metadata.Media.SHA256Key))) missingFields.Add(Models.Metadata.Media.SHA256Key); break; case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.SHA256Key))) missingFields.Add(Models.Metadata.Rom.SHA256Key); break; default: missingFields.Add(Models.Metadata.Rom.SHA256Key); break; } break; case HashType.SHA384: switch (datItem) { case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.SHA384Key))) missingFields.Add(Models.Metadata.Rom.SHA384Key); break; default: missingFields.Add(Models.Metadata.Rom.SHA384Key); break; } break; case HashType.SHA512: switch (datItem) { case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.SHA512Key))) missingFields.Add(Models.Metadata.Rom.SHA512Key); break; default: missingFields.Add(Models.Metadata.Rom.SHA512Key); break; } break; case HashType.SpamSum: switch (datItem) { case Media medium: if (string.IsNullOrEmpty(medium.GetStringFieldValue(Models.Metadata.Media.SpamSumKey))) missingFields.Add(Models.Metadata.Media.SpamSumKey); break; case Rom rom: if (string.IsNullOrEmpty(rom.GetStringFieldValue(Models.Metadata.Rom.SpamSumKey))) missingFields.Add(Models.Metadata.Rom.SpamSumKey); break; default: missingFields.Add(Models.Metadata.Rom.SpamSumKey); break; } break; } return missingFields; } /// public override bool WriteToFile(string outfile, bool ignoreblanks = false, bool throwOnError = false) { try { _logger.User($"Writing to '{outfile}'..."); // Serialize the input file var metadata = ConvertMetadata(ignoreblanks); var hashfile = new Serialization.CrossModel.Hashfile().Deserialize(metadata, _hash); if (!(Serialization.Serializers.Hashfile.SerializeFile(hashfile, outfile, _hash))) { _logger.Warning($"File '{outfile}' could not be written! See the log for more details."); return false; } } catch (Exception ex) when (!throwOnError) { _logger.Error(ex); return false; } _logger.User($"'{outfile}' written!{Environment.NewLine}"); return true; } } /// /// Represents an SFV (CRC-32) hashfile /// internal sealed class SfvFile : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public SfvFile(DatFile? datFile) : base(datFile) { _hash = HashType.CRC32; } } /// /// Represents an MD2 hashfile /// internal sealed class Md2File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Md2File(DatFile? datFile) : base(datFile) { _hash = HashType.MD2; } } /// /// Represents an MD4 hashfile /// internal sealed class Md4File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Md4File(DatFile? datFile) : base(datFile) { _hash = HashType.MD4; } } /// /// Represents an MD5 hashfile /// internal sealed class Md5File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Md5File(DatFile? datFile) : base(datFile) { _hash = HashType.MD5; } } /// /// Represents an SHA-1 hashfile /// internal sealed class Sha1File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Sha1File(DatFile? datFile) : base(datFile) { _hash = HashType.SHA1; } } /// /// Represents an SHA-256 hashfile /// internal sealed class Sha256File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Sha256File(DatFile? datFile) : base(datFile) { _hash = HashType.SHA256; } } /// /// Represents an SHA-384 hashfile /// internal sealed class Sha384File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Sha384File(DatFile? datFile) : base(datFile) { _hash = HashType.SHA384; } } /// /// Represents an SHA-512 hashfile /// internal sealed class Sha512File : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public Sha512File(DatFile? datFile) : base(datFile) { _hash = HashType.SHA512; } } /// /// Represents an SpamSum hashfile /// internal sealed class SpamSumFile : Hashfile { /// /// Constructor designed for casting a base DatFile /// /// Parent DatFile to copy from public SpamSumFile(DatFile? datFile) : base(datFile) { _hash = HashType.SpamSum; } } }