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 { // Private instance variables specific to Hashfile DATs protected HashType _hash; /// /// 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 ItemType[] GetSupportedTypes() { return [ ItemType.Disk, ItemType.Media, ItemType.Rom ]; } /// protected override List? GetMissingRequiredFields(DatItem datItem) { var missingFields = new List(); // 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_ISO: case HashType.CRC32_Naive: case HashType.CRC32_Optimized: case HashType.CRC32_Parallel: 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.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 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; } } }