chore: added a generic converter
This commit is contained in:
parent
9c129cefe2
commit
f51d5f342e
@ -1,28 +0,0 @@
|
|||||||
using Syntriax.Engine.Core;
|
|
||||||
using Syntriax.Engine.Core.Factory;
|
|
||||||
|
|
||||||
using YamlDotNet.Core;
|
|
||||||
using YamlDotNet.Serialization;
|
|
||||||
|
|
||||||
namespace Syntriax.Engine.Serialization;
|
|
||||||
|
|
||||||
public class EntityConverter : IEngineTypeYamlConverter
|
|
||||||
{
|
|
||||||
public bool Accepts(Type type) => typeof(IEntity).IsAssignableFrom(type);
|
|
||||||
|
|
||||||
public object? ReadYaml(IParser parser, Type type, ObjectDeserializer rootDeserializer)
|
|
||||||
{
|
|
||||||
EntityReference entityReference = (EntityReference)rootDeserializer(typeof(EntityReference))!;
|
|
||||||
|
|
||||||
IEntity entity = (IEntity)TypeFactory.Get(entityReference.TypeContainer.Type);
|
|
||||||
|
|
||||||
return entity;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void WriteYaml(IEmitter emitter, object? value, Type type, ObjectSerializer serializer)
|
|
||||||
{
|
|
||||||
IEntity? entity = (IEntity)value!;
|
|
||||||
|
|
||||||
serializer(new EntityReference(entity), typeof(EntityReference));
|
|
||||||
}
|
|
||||||
}
|
|
111
Engine.Serialization/Converters/InstanceConverter.cs
Normal file
111
Engine.Serialization/Converters/InstanceConverter.cs
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
using System.Collections;
|
||||||
|
using System.Reflection;
|
||||||
|
|
||||||
|
using Syntriax.Engine.Core.Factory;
|
||||||
|
|
||||||
|
using YamlDotNet.Core;
|
||||||
|
using YamlDotNet.Core.Events;
|
||||||
|
using YamlDotNet.Serialization;
|
||||||
|
|
||||||
|
namespace Syntriax.Engine.Serialization;
|
||||||
|
|
||||||
|
public class InstanceConverter : IEngineTypeYamlConverter
|
||||||
|
{
|
||||||
|
public bool Accepts(Type type) => !type.IsPrimitive && type != typeof(string);
|
||||||
|
|
||||||
|
public object? ReadYaml(IParser parser, Type type, ObjectDeserializer rootDeserializer)
|
||||||
|
{
|
||||||
|
parser.Consume<MappingStart>();
|
||||||
|
|
||||||
|
TypeContainer typeContainer = (TypeContainer)rootDeserializer(typeof(TypeContainer))!;
|
||||||
|
|
||||||
|
object instance = TypeFactory.Get(typeContainer.Type);
|
||||||
|
|
||||||
|
while (!parser.TryConsume<MappingEnd>(out _))
|
||||||
|
{
|
||||||
|
string key = parser.Consume<Scalar>().Value;
|
||||||
|
|
||||||
|
if (type.GetField(key, BindingFlags.Instance | BindingFlags.NonPublic) is FieldInfo fieldInfo)
|
||||||
|
{
|
||||||
|
object? fieldValue = rootDeserializer(fieldInfo.FieldType);
|
||||||
|
fieldInfo.SetValue(instance, fieldValue);
|
||||||
|
}
|
||||||
|
else if (type.GetProperty(key) is PropertyInfo propertyInfo)
|
||||||
|
{
|
||||||
|
object? propertyValue = rootDeserializer(propertyInfo.PropertyType);
|
||||||
|
propertyInfo.SetValue(instance, propertyValue);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
parser.SkipThisAndNestedEvents();
|
||||||
|
}
|
||||||
|
|
||||||
|
parser.TryConsume<MappingEnd>(out _);
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void WriteYaml(IEmitter emitter, object? value, Type type, ObjectSerializer serializer)
|
||||||
|
{
|
||||||
|
emitter.Emit(new MappingStart());
|
||||||
|
serializer(new TypeContainer(value), typeof(TypeContainer));
|
||||||
|
|
||||||
|
TypeData typeData = Utils.GetTypeData(type);
|
||||||
|
|
||||||
|
foreach (PropertyInfo propertyInfo in typeData.Properties)
|
||||||
|
{
|
||||||
|
if (propertyInfo.GetCustomAttribute<IgnoreSerializationAttribute>() is not null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if ((propertyInfo.PropertyType.IsClass && propertyInfo.PropertyType != typeof(string)) || propertyInfo.PropertyType.IsAbstract)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
emitter.Emit(new Scalar(propertyInfo.Name));
|
||||||
|
object? propertyValue = propertyInfo.GetValue(value);
|
||||||
|
|
||||||
|
EmitValue(propertyValue, propertyInfo.PropertyType, emitter, serializer);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (FieldInfo fieldInfo in typeData.Fields)
|
||||||
|
{
|
||||||
|
if (fieldInfo.GetCustomAttribute<System.Runtime.CompilerServices.CompilerGeneratedAttribute>() is not null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (fieldInfo.GetCustomAttribute<SerializeAttribute>() is null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if ((fieldInfo.FieldType.IsClass && fieldInfo.FieldType != typeof(string)) || fieldInfo.FieldType.IsAbstract)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
emitter.Emit(new Scalar(fieldInfo.Name));
|
||||||
|
object? fieldValue = fieldInfo.GetValue(value);
|
||||||
|
|
||||||
|
EmitValue(fieldValue, fieldInfo.FieldType, emitter, serializer);
|
||||||
|
}
|
||||||
|
|
||||||
|
emitter.Emit(new MappingEnd());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void EmitValue(object? value, Type declaredType, IEmitter emitter, ObjectSerializer serializer)
|
||||||
|
{
|
||||||
|
if (value is null)
|
||||||
|
{
|
||||||
|
emitter.Emit(new Scalar(""));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isSequence = Utils.IsEnumerable(declaredType);
|
||||||
|
|
||||||
|
if (!isSequence)
|
||||||
|
{
|
||||||
|
serializer(value);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
IEnumerable sequence = (IEnumerable)value;
|
||||||
|
|
||||||
|
emitter.Emit(new SequenceStart(null, null, false, SequenceStyle.Block));
|
||||||
|
foreach (object? item in sequence)
|
||||||
|
serializer(item);
|
||||||
|
emitter.Emit(new SequenceEnd());
|
||||||
|
}
|
||||||
|
}
|
@ -1,46 +0,0 @@
|
|||||||
using Syntriax.Engine.Core;
|
|
||||||
using Syntriax.Engine.Core.Factory;
|
|
||||||
|
|
||||||
using YamlDotNet.Core;
|
|
||||||
using YamlDotNet.Core.Events;
|
|
||||||
using YamlDotNet.Serialization;
|
|
||||||
|
|
||||||
namespace Syntriax.Engine.Serialization;
|
|
||||||
|
|
||||||
public class StateEnableConverter : IEngineTypeYamlConverter
|
|
||||||
{
|
|
||||||
public bool Accepts(Type type) => typeof(IStateEnable).IsAssignableFrom(type);
|
|
||||||
|
|
||||||
public object? ReadYaml(IParser parser, Type type, ObjectDeserializer rootDeserializer)
|
|
||||||
{
|
|
||||||
parser.Consume<MappingStart>();
|
|
||||||
|
|
||||||
TypeContainer typeContainer = (TypeContainer)rootDeserializer(typeof(TypeContainer))!;
|
|
||||||
EntityReference entityReference = (EntityReference)rootDeserializer(typeof(EntityReference))!;
|
|
||||||
|
|
||||||
if (parser.Consume<Scalar>().Value.CompareTo(nameof(IStateEnable.Enabled)) != 0)
|
|
||||||
throw new ArgumentException($"{nameof(IStateEnable)} mapping must have a {nameof(IStateEnable.Enabled)}");
|
|
||||||
bool enabled = bool.Parse(parser.Consume<Scalar>().Value);
|
|
||||||
|
|
||||||
parser.Consume<MappingEnd>();
|
|
||||||
|
|
||||||
IStateEnable stateEnable = (IStateEnable)TypeFactory.Get(typeContainer.Type);
|
|
||||||
stateEnable.Enabled = enabled;
|
|
||||||
stateEnable.Assign((IEntity)TypeFactory.Get(entityReference.TypeContainer.Type));
|
|
||||||
|
|
||||||
return stateEnable;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void WriteYaml(IEmitter emitter, object? value, Type type, ObjectSerializer serializer)
|
|
||||||
{
|
|
||||||
IStateEnable stateEnable = (IStateEnable)value!;
|
|
||||||
|
|
||||||
emitter.Emit(new MappingStart());
|
|
||||||
serializer(new TypeContainer(stateEnable), typeof(TypeContainer));
|
|
||||||
emitter.Emit(new Scalar(nameof(StateEnable.Entity)));
|
|
||||||
serializer(new EntityReference(stateEnable.Entity), typeof(EntityReference));
|
|
||||||
emitter.Emit(new Scalar(nameof(IStateEnable.Enabled)));
|
|
||||||
emitter.Emit(new Scalar(stateEnable.Enabled.ToString()));
|
|
||||||
emitter.Emit(new MappingEnd());
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +1,4 @@
|
|||||||
namespace Syntriax.Engine.Serialization;
|
namespace Syntriax.Engine.Serialization;
|
||||||
|
|
||||||
public class IgnoreSerializationAttribute : Attribute
|
[AttributeUsage(AttributeTargets.Property | AttributeTargets.Class)]
|
||||||
{
|
public class IgnoreSerializationAttribute : Attribute;
|
||||||
}
|
|
||||||
|
4
Engine.Serialization/SerializeAttribute.cs
Normal file
4
Engine.Serialization/SerializeAttribute.cs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
namespace Syntriax.Engine.Serialization;
|
||||||
|
|
||||||
|
[AttributeUsage(AttributeTargets.Field)]
|
||||||
|
public class SerializeAttribute : Attribute;
|
@ -4,28 +4,38 @@ namespace Syntriax.Engine.Serialization;
|
|||||||
|
|
||||||
internal static class Utils
|
internal static class Utils
|
||||||
{
|
{
|
||||||
internal static TypeData GetTypeData(this Type type)
|
internal static bool IsEnumerable(this Type type) => typeof(System.Collections.IEnumerable).IsAssignableFrom(type) && type != typeof(string);
|
||||||
|
|
||||||
|
internal static TypeData GetTypeData(this Type objectType)
|
||||||
{
|
{
|
||||||
IEnumerable<EventInfo> eventInfos = type.GetEvents(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)
|
// IEnumerable<EventInfo> eventInfos = objectType.GetEvents(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)
|
||||||
.OrderBy(ei => ei.Name)
|
// .OrderBy(ei => ei.Name)
|
||||||
.AsEnumerable();
|
// .ToList();
|
||||||
IEnumerable<FieldInfo> fieldInfos = type.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)
|
List<FieldInfo> fieldInfos = objectType.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)
|
||||||
.Where(fi => !eventInfos.Any(ei => fi.Name.CompareTo(ei.Name) == 0))
|
|
||||||
.OrderBy(ei => ei.Name) //ei => ei.FieldType.IsPrimitive || ei.FieldType == typeof(string))
|
.OrderBy(ei => ei.Name) //ei => ei.FieldType.IsPrimitive || ei.FieldType == typeof(string))
|
||||||
// .ThenByDescending(ei => ei.Name)
|
// .ThenByDescending(ei => ei.Name)
|
||||||
.AsEnumerable();
|
.ToList();
|
||||||
IEnumerable<PropertyInfo> propertyInfos = type.GetProperties(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)
|
|
||||||
|
List<PropertyInfo> propertyInfos = objectType.GetProperties(BindingFlags.Instance | BindingFlags.Public)
|
||||||
.Where(pi => pi.SetMethod is not null)
|
.Where(pi => pi.SetMethod is not null)
|
||||||
.OrderBy(ei => ei.Name)// ei => ei.PropertyType.IsPrimitive || ei.PropertyType == typeof(string))
|
.OrderBy(ei => ei.Name)// ei => ei.PropertyType.IsPrimitive || ei.PropertyType == typeof(string))
|
||||||
// .ThenByDescending(ei => ei.Name)
|
// .ThenByDescending(ei => ei.Name)
|
||||||
.AsEnumerable();
|
.ToList();
|
||||||
|
|
||||||
return new TypeData(eventInfos, fieldInfos, propertyInfos);
|
propertyInfos.AddRange(
|
||||||
|
objectType.GetProperties(BindingFlags.Instance | BindingFlags.NonPublic)
|
||||||
|
.Where(pi => pi.SetMethod is not null && pi.GetCustomAttribute<SerializeAttribute>() is not null)
|
||||||
|
.OrderBy(ei => ei.Name)// ei => ei.PropertyType.IsPrimitive || ei.PropertyType == typeof(string))
|
||||||
|
// .ThenByDescending(ei => ei.Name)
|
||||||
|
.ToList()
|
||||||
|
);
|
||||||
|
|
||||||
|
return new TypeData(fieldInfos, propertyInfos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal record struct TypeData(IEnumerable<EventInfo> Events, IEnumerable<FieldInfo> Fields, IEnumerable<PropertyInfo> Properties)
|
internal record struct TypeData(IEnumerable<FieldInfo> Fields, IEnumerable<PropertyInfo> Properties)
|
||||||
{
|
{
|
||||||
public static implicit operator (IEnumerable<EventInfo> events, IEnumerable<FieldInfo> fields, IEnumerable<PropertyInfo> properties)(TypeData value) => (value.Events, value.Fields, value.Properties);
|
public static implicit operator (IEnumerable<FieldInfo> fields, IEnumerable<PropertyInfo> properties)(TypeData value) => (value.Fields, value.Properties);
|
||||||
public static implicit operator TypeData((IEnumerable<EventInfo> events, IEnumerable<FieldInfo> fields, IEnumerable<PropertyInfo> properties) value) => new TypeData(value.events, value.fields, value.properties);
|
public static implicit operator TypeData((IEnumerable<FieldInfo> fields, IEnumerable<PropertyInfo> properties) value) => new(value.fields, value.properties);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user