mirror of
https://github.com/claunia/SabreTools.git
synced 2025-12-16 19:14:27 +00:00
Add full-file deserialization across the board
This commit is contained in:
@@ -19,16 +19,8 @@ namespace SabreTools.Serialization
|
||||
/// <returns>Deserialized data on success, null on failure</returns>
|
||||
public static MetadataFile? Deserialize(string path)
|
||||
{
|
||||
try
|
||||
{
|
||||
using var stream = PathProcessor.OpenStream(path);
|
||||
return Deserialize(stream);
|
||||
}
|
||||
catch
|
||||
{
|
||||
// TODO: Handle logging the exception
|
||||
return default;
|
||||
}
|
||||
using var stream = PathProcessor.OpenStream(path);
|
||||
return Deserialize(stream);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -38,163 +30,155 @@ namespace SabreTools.Serialization
|
||||
/// <returns>Deserialized data on success, null on failure</returns>
|
||||
public static MetadataFile? Deserialize(Stream? stream)
|
||||
{
|
||||
try
|
||||
// If the stream is null
|
||||
if (stream == null)
|
||||
return default;
|
||||
|
||||
// Setup the reader and output
|
||||
var reader = new ClrMameProReader(stream, Encoding.UTF8) { DosCenter = true };
|
||||
var dat = new MetadataFile();
|
||||
|
||||
// Loop through and parse out the values
|
||||
string lastTopLevel = reader.TopLevel;
|
||||
|
||||
Game? game = null;
|
||||
var games = new List<Game?>();
|
||||
var files = new List<Models.DosCenter.File?>();
|
||||
|
||||
var additional = new List<string>();
|
||||
var headerAdditional = new List<string>();
|
||||
var gameAdditional = new List<string>();
|
||||
var fileAdditional = new List<string>();
|
||||
while (!reader.EndOfStream)
|
||||
{
|
||||
// If the stream is null
|
||||
if (stream == null)
|
||||
return default;
|
||||
// If we have no next line
|
||||
if (!reader.ReadNextLine())
|
||||
break;
|
||||
|
||||
// Setup the reader and output
|
||||
var reader = new ClrMameProReader(stream, Encoding.UTF8) { DosCenter = true };
|
||||
var dat = new MetadataFile();
|
||||
|
||||
// Loop through and parse out the values
|
||||
string lastTopLevel = reader.TopLevel;
|
||||
|
||||
Game? game = null;
|
||||
var games = new List<Game?>();
|
||||
var files = new List<Models.DosCenter.File?>();
|
||||
|
||||
var additional = new List<string>();
|
||||
var headerAdditional = new List<string>();
|
||||
var gameAdditional = new List<string>();
|
||||
var fileAdditional = new List<string>();
|
||||
while (!reader.EndOfStream)
|
||||
// Ignore certain row types
|
||||
switch (reader.RowType)
|
||||
{
|
||||
// If we have no next line
|
||||
if (!reader.ReadNextLine())
|
||||
break;
|
||||
|
||||
// Ignore certain row types
|
||||
switch (reader.RowType)
|
||||
{
|
||||
case CmpRowType.None:
|
||||
case CmpRowType.Comment:
|
||||
continue;
|
||||
case CmpRowType.EndTopLevel:
|
||||
switch (lastTopLevel)
|
||||
{
|
||||
case "doscenter":
|
||||
dat.DosCenter.ADDITIONAL_ELEMENTS = headerAdditional.ToArray();
|
||||
headerAdditional.Clear();
|
||||
break;
|
||||
case "game":
|
||||
game.File = files.ToArray();
|
||||
game.ADDITIONAL_ELEMENTS = gameAdditional.ToArray();
|
||||
games.Add(game);
|
||||
|
||||
game = null;
|
||||
files.Clear();
|
||||
gameAdditional.Clear();
|
||||
break;
|
||||
default:
|
||||
// No-op
|
||||
break;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// If we're at the root
|
||||
if (reader.RowType == CmpRowType.TopLevel)
|
||||
{
|
||||
lastTopLevel = reader.TopLevel;
|
||||
switch (reader.TopLevel)
|
||||
case CmpRowType.None:
|
||||
case CmpRowType.Comment:
|
||||
continue;
|
||||
case CmpRowType.EndTopLevel:
|
||||
switch (lastTopLevel)
|
||||
{
|
||||
case "doscenter":
|
||||
dat.DosCenter = new Models.DosCenter.DosCenter();
|
||||
dat.DosCenter.ADDITIONAL_ELEMENTS = headerAdditional.ToArray();
|
||||
headerAdditional.Clear();
|
||||
break;
|
||||
case "game":
|
||||
game = new Game();
|
||||
game.File = files.ToArray();
|
||||
game.ADDITIONAL_ELEMENTS = gameAdditional.ToArray();
|
||||
games.Add(game);
|
||||
|
||||
game = null;
|
||||
files.Clear();
|
||||
gameAdditional.Clear();
|
||||
break;
|
||||
default:
|
||||
additional.Add(reader.CurrentLine);
|
||||
// No-op
|
||||
break;
|
||||
}
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// If we're in the doscenter block
|
||||
else if (reader.TopLevel == "doscenter" && reader.RowType == CmpRowType.Standalone)
|
||||
// If we're at the root
|
||||
if (reader.RowType == CmpRowType.TopLevel)
|
||||
{
|
||||
lastTopLevel = reader.TopLevel;
|
||||
switch (reader.TopLevel)
|
||||
{
|
||||
// Create the block if we haven't already
|
||||
dat.DosCenter ??= new Models.DosCenter.DosCenter();
|
||||
|
||||
switch (reader.Standalone?.Key?.ToLowerInvariant())
|
||||
{
|
||||
case "name:":
|
||||
dat.DosCenter.Name = reader.Standalone?.Value;
|
||||
break;
|
||||
case "description:":
|
||||
dat.DosCenter.Description = reader.Standalone?.Value;
|
||||
break;
|
||||
case "version:":
|
||||
dat.DosCenter.Version = reader.Standalone?.Value;
|
||||
break;
|
||||
case "date:":
|
||||
dat.DosCenter.Date = reader.Standalone?.Value;
|
||||
break;
|
||||
case "author:":
|
||||
dat.DosCenter.Author = reader.Standalone?.Value;
|
||||
break;
|
||||
case "homepage:":
|
||||
dat.DosCenter.Homepage = reader.Standalone?.Value;
|
||||
break;
|
||||
case "comment:":
|
||||
dat.DosCenter.Comment = reader.Standalone?.Value;
|
||||
break;
|
||||
default:
|
||||
headerAdditional.Add(item: reader.CurrentLine);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// If we're in a game block
|
||||
else if (reader.TopLevel == "game" && reader.RowType == CmpRowType.Standalone)
|
||||
{
|
||||
// Create the block if we haven't already
|
||||
game ??= new Game();
|
||||
|
||||
switch (reader.Standalone?.Key?.ToLowerInvariant())
|
||||
{
|
||||
case "name":
|
||||
game.Name = reader.Standalone?.Value;
|
||||
break;
|
||||
default:
|
||||
gameAdditional.Add(item: reader.CurrentLine);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// If we're in a file block
|
||||
else if (reader.TopLevel == "game" && reader.RowType == CmpRowType.Internal)
|
||||
{
|
||||
// If we have an unknown type, log it
|
||||
if (reader.InternalName != "file")
|
||||
{
|
||||
gameAdditional.Add(reader.CurrentLine);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Create the file and add to the list
|
||||
var file = CreateFile(reader);
|
||||
files.Add(file);
|
||||
}
|
||||
|
||||
else
|
||||
{
|
||||
additional.Add(item: reader.CurrentLine);
|
||||
case "doscenter":
|
||||
dat.DosCenter = new Models.DosCenter.DosCenter();
|
||||
break;
|
||||
case "game":
|
||||
game = new Game();
|
||||
break;
|
||||
default:
|
||||
additional.Add(reader.CurrentLine);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Add extra pieces and return
|
||||
dat.Game = games.ToArray();
|
||||
dat.ADDITIONAL_ELEMENTS = additional.ToArray();
|
||||
return dat;
|
||||
}
|
||||
catch
|
||||
{
|
||||
// TODO: Handle logging the exception
|
||||
return default;
|
||||
// If we're in the doscenter block
|
||||
else if (reader.TopLevel == "doscenter" && reader.RowType == CmpRowType.Standalone)
|
||||
{
|
||||
// Create the block if we haven't already
|
||||
dat.DosCenter ??= new Models.DosCenter.DosCenter();
|
||||
|
||||
switch (reader.Standalone?.Key?.ToLowerInvariant())
|
||||
{
|
||||
case "name:":
|
||||
dat.DosCenter.Name = reader.Standalone?.Value;
|
||||
break;
|
||||
case "description:":
|
||||
dat.DosCenter.Description = reader.Standalone?.Value;
|
||||
break;
|
||||
case "version:":
|
||||
dat.DosCenter.Version = reader.Standalone?.Value;
|
||||
break;
|
||||
case "date:":
|
||||
dat.DosCenter.Date = reader.Standalone?.Value;
|
||||
break;
|
||||
case "author:":
|
||||
dat.DosCenter.Author = reader.Standalone?.Value;
|
||||
break;
|
||||
case "homepage:":
|
||||
dat.DosCenter.Homepage = reader.Standalone?.Value;
|
||||
break;
|
||||
case "comment:":
|
||||
dat.DosCenter.Comment = reader.Standalone?.Value;
|
||||
break;
|
||||
default:
|
||||
headerAdditional.Add(item: reader.CurrentLine);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// If we're in a game block
|
||||
else if (reader.TopLevel == "game" && reader.RowType == CmpRowType.Standalone)
|
||||
{
|
||||
// Create the block if we haven't already
|
||||
game ??= new Game();
|
||||
|
||||
switch (reader.Standalone?.Key?.ToLowerInvariant())
|
||||
{
|
||||
case "name":
|
||||
game.Name = reader.Standalone?.Value;
|
||||
break;
|
||||
default:
|
||||
gameAdditional.Add(item: reader.CurrentLine);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// If we're in a file block
|
||||
else if (reader.TopLevel == "game" && reader.RowType == CmpRowType.Internal)
|
||||
{
|
||||
// If we have an unknown type, log it
|
||||
if (reader.InternalName != "file")
|
||||
{
|
||||
gameAdditional.Add(reader.CurrentLine);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Create the file and add to the list
|
||||
var file = CreateFile(reader);
|
||||
files.Add(file);
|
||||
}
|
||||
|
||||
else
|
||||
{
|
||||
additional.Add(item: reader.CurrentLine);
|
||||
}
|
||||
}
|
||||
|
||||
// Add extra pieces and return
|
||||
dat.Game = games.ToArray();
|
||||
dat.ADDITIONAL_ELEMENTS = additional.ToArray();
|
||||
return dat;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -231,14 +215,34 @@ namespace SabreTools.Serialization
|
||||
file.ADDITIONAL_ELEMENTS = itemAdditional.ToArray();
|
||||
return file;
|
||||
}
|
||||
|
||||
// TODO: Add deserialization of entire MetadataFile
|
||||
|
||||
#region Internal
|
||||
|
||||
/// <summary>
|
||||
/// Convert from <cref="Models.Internal.MetadataFile"/> to <cref="Models.DosCenter.MetadataFile"/>
|
||||
/// </summary>
|
||||
public static MetadataFile? ConvertFromInternalModel(Models.Internal.MetadataFile? item)
|
||||
{
|
||||
if (item == null)
|
||||
return null;
|
||||
|
||||
var metadataFile = new MetadataFile();
|
||||
|
||||
var header = item.Read<Models.Internal.Header>(Models.Internal.MetadataFile.HeaderKey);
|
||||
if (header != null)
|
||||
metadataFile.DosCenter = ConvertHeaderFromInternalModel(header);
|
||||
|
||||
var machines = item.Read<Models.Internal.Machine[]>(Models.Internal.MetadataFile.MachineKey);
|
||||
if (machines != null && machines.Any())
|
||||
metadataFile.Game = machines.Select(ConvertMachineFromInternalModel).ToArray();
|
||||
|
||||
return metadataFile;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert from <cref="Models.Internal.Header"/> to <cref="Models.DosCenter.DosCenter"/>
|
||||
/// </summary>
|
||||
public static Models.DosCenter.DosCenter? ConvertHeaderFromInternalModel(Models.Internal.Header? item)
|
||||
private static Models.DosCenter.DosCenter? ConvertHeaderFromInternalModel(Models.Internal.Header? item)
|
||||
{
|
||||
if (item == null)
|
||||
return null;
|
||||
@@ -259,7 +263,7 @@ namespace SabreTools.Serialization
|
||||
/// <summary>
|
||||
/// Convert from <cref="Models.Internal.Machine"/> to <cref="Models.DosCenter.Game"/>
|
||||
/// </summary>
|
||||
public static Game? ConvertMachineFromInternalModel(Models.Internal.Machine? item)
|
||||
private static Game? ConvertMachineFromInternalModel(Models.Internal.Machine? item)
|
||||
{
|
||||
if (item == null)
|
||||
return null;
|
||||
|
||||
Reference in New Issue
Block a user