Files
SabreTools/SabreTools.Serialization/RomCenter.Deserializer.cs

350 lines
14 KiB
C#
Raw Normal View History

2023-08-10 11:35:32 -04:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using SabreTools.IO.Readers;
using SabreTools.Models.RomCenter;
namespace SabreTools.Serialization
{
/// <summary>
/// Deserializer for RomCenter INI files
/// </summary>
public partial class RomCenter
{
/// <summary>
/// Deserializes a RomCenter INI file to the defined type
/// </summary>
/// <param name="path">Path to the file to deserialize</param>
/// <returns>Deserialized data on success, null on failure</returns>
public static MetadataFile? Deserialize(string path)
{
2023-07-30 09:00:15 -04:00
using var stream = PathProcessor.OpenStream(path);
return Deserialize(stream);
}
/// <summary>
/// Deserializes a RomCenter INI file in a stream to the defined type
/// </summary>
/// <param name="stream">Stream to deserialize</param>
/// <returns>Deserialized data on success, null on failure</returns>
public static MetadataFile? Deserialize(Stream? stream)
{
2023-07-30 09:00:15 -04:00
// If the stream is null
if (stream == null)
return default;
// Setup the reader and output
var reader = new IniReader(stream, Encoding.UTF8)
{
2023-07-30 09:00:15 -04:00
ValidateRows = false,
};
var dat = new MetadataFile();
// Loop through and parse out the values
var roms = new List<Rom>();
var additional = new List<string>();
var creditsAdditional = new List<string>();
var datAdditional = new List<string>();
var emulatorAdditional = new List<string>();
var gamesAdditional = new List<string>();
while (!reader.EndOfStream)
{
// If we have no next line
if (!reader.ReadNextLine())
break;
2023-07-30 09:00:15 -04:00
// Ignore certain row types
switch (reader.RowType)
{
2023-07-30 09:00:15 -04:00
case IniRowType.None:
case IniRowType.Comment:
continue;
case IniRowType.SectionHeader:
2023-08-10 15:02:40 -04:00
switch (reader.Section?.ToLowerInvariant())
{
2023-07-30 09:00:15 -04:00
case "credits":
dat.Credits ??= new Credits();
break;
2023-07-30 09:00:15 -04:00
case "dat":
dat.Dat ??= new Dat();
break;
2023-07-30 09:00:15 -04:00
case "emulator":
dat.Emulator ??= new Emulator();
break;
2023-07-30 09:00:15 -04:00
case "games":
dat.Games ??= new Games();
break;
default:
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
additional.Add(reader.CurrentLine);
break;
}
2023-07-30 09:00:15 -04:00
continue;
}
2023-07-30 09:00:15 -04:00
// If we're in credits
2023-08-10 15:02:40 -04:00
if (reader.Section?.ToLowerInvariant() == "credits")
2023-07-30 09:00:15 -04:00
{
// Create the section if we haven't already
dat.Credits ??= new Credits();
2023-07-30 09:00:15 -04:00
switch (reader.KeyValuePair?.Key?.ToLowerInvariant())
{
case "author":
dat.Credits.Author = reader.KeyValuePair?.Value;
break;
case "version":
dat.Credits.Version = reader.KeyValuePair?.Value;
break;
case "email":
dat.Credits.Email = reader.KeyValuePair?.Value;
break;
case "homepage":
dat.Credits.Homepage = reader.KeyValuePair?.Value;
break;
case "url":
dat.Credits.Url = reader.KeyValuePair?.Value;
break;
case "date":
dat.Credits.Date = reader.KeyValuePair?.Value;
break;
case "comment":
dat.Credits.Comment = reader.KeyValuePair?.Value;
break;
default:
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
creditsAdditional.Add(reader.CurrentLine);
2023-07-30 09:00:15 -04:00
break;
}
2023-07-30 09:00:15 -04:00
}
2023-07-30 09:00:15 -04:00
// If we're in dat
2023-08-10 15:02:40 -04:00
else if (reader.Section?.ToLowerInvariant() == "dat")
2023-07-30 09:00:15 -04:00
{
// Create the section if we haven't already
dat.Dat ??= new Dat();
2023-07-30 09:00:15 -04:00
switch (reader.KeyValuePair?.Key?.ToLowerInvariant())
{
case "version":
dat.Dat.Version = reader.KeyValuePair?.Value;
break;
case "plugin":
dat.Dat.Plugin = reader.KeyValuePair?.Value;
break;
case "split":
dat.Dat.Split = reader.KeyValuePair?.Value;
break;
case "merge":
dat.Dat.Merge = reader.KeyValuePair?.Value;
break;
default:
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
datAdditional.Add(reader.CurrentLine);
2023-07-30 09:00:15 -04:00
break;
}
2023-07-30 09:00:15 -04:00
}
// If we're in emulator
2023-08-10 15:02:40 -04:00
else if (reader.Section?.ToLowerInvariant() == "emulator")
2023-07-30 09:00:15 -04:00
{
// Create the section if we haven't already
dat.Emulator ??= new Emulator();
2023-07-30 09:00:15 -04:00
switch (reader.KeyValuePair?.Key?.ToLowerInvariant())
{
2023-07-30 09:00:15 -04:00
case "refname":
dat.Emulator.RefName = reader.KeyValuePair?.Value;
break;
case "version":
dat.Emulator.Version = reader.KeyValuePair?.Value;
break;
default:
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
emulatorAdditional.Add(reader.CurrentLine);
2023-07-30 09:00:15 -04:00
break;
}
}
2023-07-30 09:00:15 -04:00
// If we're in games
2023-08-10 15:02:40 -04:00
else if (reader.Section?.ToLowerInvariant() == "games")
2023-07-30 09:00:15 -04:00
{
// Create the section if we haven't already
dat.Games ??= new Games();
2023-07-30 09:00:15 -04:00
// If the line doesn't contain the delimiter
2023-08-10 15:02:40 -04:00
if (!(reader.CurrentLine?.Contains('¬') ?? false))
2023-07-30 09:00:15 -04:00
{
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
gamesAdditional.Add(reader.CurrentLine);
2023-07-30 09:00:15 -04:00
continue;
}
2023-07-30 09:00:15 -04:00
// Otherwise, separate out the line
string[] splitLine = reader.CurrentLine.Split('¬');
var rom = new Rom
{
2023-07-30 09:00:15 -04:00
// EMPTY = splitLine[0]
ParentName = splitLine[1],
ParentDescription = splitLine[2],
GameName = splitLine[3],
GameDescription = splitLine[4],
RomName = splitLine[5],
RomCRC = splitLine[6],
RomSize = splitLine[7],
RomOf = splitLine[8],
MergeName = splitLine[9],
// EMPTY = splitLine[10]
};
if (splitLine.Length > 11)
rom.ADDITIONAL_ELEMENTS = splitLine.Skip(11).ToArray();
roms.Add(rom);
}
2023-07-30 09:00:15 -04:00
else
{
2023-08-10 15:02:40 -04:00
if (reader.CurrentLine != null)
additional.Add(reader.CurrentLine);
}
}
2023-07-30 09:00:15 -04:00
// Add extra pieces and return
2023-08-10 15:02:40 -04:00
dat.ADDITIONAL_ELEMENTS = additional.Where(s => s != null).ToArray();
2023-07-30 09:00:15 -04:00
if (dat.Credits != null)
2023-08-10 15:02:40 -04:00
dat.Credits.ADDITIONAL_ELEMENTS = creditsAdditional.Where(s => s != null).ToArray();
2023-07-30 09:00:15 -04:00
if (dat.Dat != null)
2023-08-10 15:02:40 -04:00
dat.Dat.ADDITIONAL_ELEMENTS = datAdditional.Where(s => s != null).ToArray();
2023-07-30 09:00:15 -04:00
if (dat.Emulator != null)
2023-08-10 15:02:40 -04:00
dat.Emulator.ADDITIONAL_ELEMENTS = emulatorAdditional.Where(s => s != null).ToArray();
2023-07-30 09:00:15 -04:00
if (dat.Games != null)
{
2023-07-30 09:00:15 -04:00
dat.Games.Rom = roms.ToArray();
2023-08-10 15:02:40 -04:00
dat.Games.ADDITIONAL_ELEMENTS = gamesAdditional.Where(s => s != null).Select(s => s).ToArray();
}
2023-07-30 09:00:15 -04:00
return dat;
}
2023-08-09 23:57:28 -04:00
#region Internal
/// <summary>
2023-09-04 23:51:37 -04:00
/// Convert from <cref="Models.Metadata.MetadataFile"/> to <cref="Models.RomCenter.MetadataFile"/>
/// </summary>
2023-09-04 23:51:37 -04:00
public static MetadataFile? ConvertFromInternalModel(Models.Metadata.MetadataFile? item)
{
if (item == null)
return null;
2023-09-04 23:51:37 -04:00
var header = item.Read<Models.Metadata.Header>(Models.Metadata.MetadataFile.HeaderKey);
var metadataFile = header != null ? ConvertHeaderFromInternalModel(header) : new MetadataFile();
2023-09-04 23:51:37 -04:00
var machines = item.Read<Models.Metadata.Machine[]>(Models.Metadata.MetadataFile.MachineKey);
if (machines != null && machines.Any())
2023-08-10 11:35:32 -04:00
{
metadataFile.Games = new Games
{
Rom = machines
.Where(m => m != null)
.SelectMany(ConvertMachineFromInternalModel)
.ToArray()
};
}
return metadataFile;
}
2023-08-09 23:57:28 -04:00
/// <summary>
2023-09-04 23:51:37 -04:00
/// Convert from <cref="Models.Metadata.Header"/> to <cref="Models.RomCenter.MetadataFile"/>
2023-08-09 23:57:28 -04:00
/// </summary>
2023-09-04 23:51:37 -04:00
private static MetadataFile ConvertHeaderFromInternalModel(Models.Metadata.Header item)
2023-08-09 23:57:28 -04:00
{
var metadataFile = new MetadataFile();
2023-09-04 23:51:37 -04:00
if (item.ContainsKey(Models.Metadata.Header.AuthorKey)
|| item.ContainsKey(Models.Metadata.Header.VersionKey)
|| item.ContainsKey(Models.Metadata.Header.EmailKey)
|| item.ContainsKey(Models.Metadata.Header.HomepageKey)
|| item.ContainsKey(Models.Metadata.Header.UrlKey)
|| item.ContainsKey(Models.Metadata.Header.DateKey)
|| item.ContainsKey(Models.Metadata.Header.CommentKey))
2023-08-09 23:57:28 -04:00
{
metadataFile.Credits = new Credits
{
2023-09-04 23:51:37 -04:00
Author = item.ReadString(Models.Metadata.Header.AuthorKey),
Version = item.ReadString(Models.Metadata.Header.VersionKey),
Email = item.ReadString(Models.Metadata.Header.EmailKey),
Homepage = item.ReadString(Models.Metadata.Header.HomepageKey),
Url = item.ReadString(Models.Metadata.Header.UrlKey),
Date = item.ReadString(Models.Metadata.Header.DateKey),
Comment = item.ReadString(Models.Metadata.Header.CommentKey),
2023-08-09 23:57:28 -04:00
};
}
2023-09-04 23:51:37 -04:00
if (item.ContainsKey(Models.Metadata.Header.DatVersionKey)
|| item.ContainsKey(Models.Metadata.Header.PluginKey)
|| item.ContainsKey(Models.Metadata.Header.ForceMergingKey))
2023-08-09 23:57:28 -04:00
{
metadataFile.Dat = new Dat
{
2023-09-04 23:51:37 -04:00
Version = item.ReadString(Models.Metadata.Header.DatVersionKey),
Plugin = item.ReadString(Models.Metadata.Header.PluginKey),
Split = item.ReadString(Models.Metadata.Header.ForceMergingKey) == "split" ? "yes" : "no",
Merge = item.ReadString(Models.Metadata.Header.ForceMergingKey) == "merge" ? "yes" : "no",
2023-08-09 23:57:28 -04:00
};
}
2023-09-04 23:51:37 -04:00
if (item.ContainsKey(Models.Metadata.Header.RefNameKey)
|| item.ContainsKey(Models.Metadata.Header.EmulatorVersionKey))
2023-08-09 23:57:28 -04:00
{
metadataFile.Emulator = new Emulator
{
2023-09-04 23:51:37 -04:00
RefName = item.ReadString(Models.Metadata.Header.RefNameKey),
Version = item.ReadString(Models.Metadata.Header.EmulatorVersionKey),
2023-08-09 23:57:28 -04:00
};
}
return metadataFile;
}
/// <summary>
2023-09-04 23:51:37 -04:00
/// Convert from <cref="Models.Metadata.Machine"/> to an array of <cref="Models.RomCenter.Rom"/>
2023-08-09 23:57:28 -04:00
/// </summary>
2023-09-04 23:51:37 -04:00
private static Rom[] ConvertMachineFromInternalModel(Models.Metadata.Machine item)
2023-08-09 23:57:28 -04:00
{
2023-09-04 23:51:37 -04:00
var roms = item.Read<Models.Metadata.Rom[]>(Models.Metadata.Machine.RomKey);
2023-08-10 11:35:32 -04:00
if (roms == null)
return Array.Empty<Rom>();
return roms
.Where(r => r != null)
.Select(rom => ConvertFromInternalModel(rom, item))
.ToArray();
2023-08-09 23:57:28 -04:00
}
/// <summary>
2023-09-04 23:51:37 -04:00
/// Convert from <cref="Models.Metadata.Rom"/> to <cref="Models.RomCenter.Rom"/>
2023-08-09 23:57:28 -04:00
/// </summary>
2023-09-04 23:51:37 -04:00
private static Rom ConvertFromInternalModel(Models.Metadata.Rom item, Models.Metadata.Machine parent)
2023-08-09 23:57:28 -04:00
{
var row = new Rom
{
2023-09-04 23:51:37 -04:00
RomName = item.ReadString(Models.Metadata.Rom.NameKey),
RomCRC = item.ReadString(Models.Metadata.Rom.CRCKey),
RomSize = item.ReadString(Models.Metadata.Rom.SizeKey),
MergeName = item.ReadString(Models.Metadata.Rom.MergeKey),
ParentName = parent.ReadString(Models.Metadata.Machine.RomOfKey),
//ParentDescription = parent.ReadString(Models.Metadata.Machine.ParentDescriptionKey), // This is unmappable
GameName = parent.ReadString(Models.Metadata.Machine.NameKey),
GameDescription = parent.ReadString(Models.Metadata.Machine.DescriptionKey),
2023-08-09 23:57:28 -04:00
};
return row;
}
#endregion
}
}