Hiho, I'm trying to generate a class in base of a given xml, to do this I'm using Reflection.Emit and I have the following methods so far:
private static AssemblyBuilder GetAssemblyBuilder()
{
AssemblyBuilder assemblyBuilder;
AssemblyName assemblyName = new AssemblyName(Constants.ClassGenerator.ASSEMBLY_NAME);
if (Assemblies.TryGetValue(assemblyName, out assemblyBuilder))
{
return assemblyBuilder;
}
AppDomain currentDomain = AppDomain.CurrentDomain; // Thread.GetDomain();
assemblyBuilder = currentDomain
.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
if (!Assemblies.TryAdd(assemblyName, assemblyBuilder))
{
throw new Exceptions.ClassGeneratorException("The assembly cannot be added to the dictionary.");
}
return assemblyBuilder;
}
private static ModuleBuilder GetModuleBuilder(AssemblyBuilder inAssemblyBuilder)
{
ModuleBuilder module = inAssemblyBuilder.DefineDynamicModule(Constants.ClassGenerator.MODULE_NAME,
$"{Constants.ClassGenerator.MODULE_NAME}.dll");
return module;
}
public static FormatBase CreateType(XmlConfiguration inConfiguration)
{
ModuleBuilder module = GetModuleBuilder(GetAssemblyBuilder());
TypeBuilder typeBuilder = module.DefineType(inConfiguration.EdiFactType,
TypeAttributes.Public |
TypeAttributes.Class |
TypeAttributes.Serializable, typeof (FormatBase));
AddDataContractAttribute(typeBuilder);
//<Adding Properties>
MapProperties(typeBuilder, inConfiguration.XmlProperties);
//<Generate the type>
Type t = typeBuilder.CreateType();
return (FormatBase) Activator.CreateInstance(t);
}
public static void MapProperties(TypeBuilder inTypeBuilder, List<XmlProperty> inProperties)
{
foreach (XmlProperty xmlProperty in inProperties)
{
if (xmlProperty.Children.Any())
{
TypeBuilder newClass = CreateClass(inTypeBuilder, xmlProperty);
MapProperties(newClass, xmlProperty.Children);
newClass.CreateType();
CreateProperty(inTypeBuilder, newClass);
}
else
{
PropertyBuilder propertyElement = CreateProperty(inTypeBuilder, xmlProperty);
}
}
}
public static TypeBuilder CreateClass(TypeBuilder inParentBuilder, XmlProperty inXmlProperty)
{
TypeBuilder typeBuilder = inParentBuilder.DefineNestedType(inXmlProperty.PropertyName, TypeAttributes.Public |
TypeAttributes
.NestedPublic |
TypeAttributes
.Serializable);
AddDataContractAttribute(typeBuilder);
return typeBuilder;
}
So far everything is working, the problem comes when I try to add a newly created class as a property for the Type I'm generating:
private static PropertyBuilder CreateProperty(TypeBuilder inParentTypeBuilder, TypeBuilder inTypeBuilder)
{
Type propertyType = inTypeBuilder.CreateType();
PropertyBuilder propertyBuilder =
inParentTypeBuilder.DefineProperty(propertyType.Name,
PropertyAttributes.None,
propertyType,
new[] {propertyType});
FieldBuilder field = inParentTypeBuilder.DefineField($"_{propertyBuilder.Name.ToLower()}",
propertyType,
FieldAttributes.Private);
GetMethodBuilder(inParentTypeBuilder, propertyBuilder, field);
SetMethodBuilder(inParentTypeBuilder, propertyBuilder, field);
//SetAttribute(propertyBuilder, inXmlProperty);
return propertyBuilder;
}
That throws a System.TypeLoadException, saying my Type cannot be loaded and raises in the point where I call inParentTypeBuilder.DefineProperty.
Any good Samaritan-Reflection.emit-guru can help me out?
I "fixed" it making the classes not nested.
It is not a solution but it is working.
private TypeBuilder CreateClass(XmlProperty inXmlProperty = null)
{
if (inXmlProperty == null)
{
//It's root class (CONTRL; UTILMD; INVOC; ETC)
TypeBuilder rootTypeBuilder = ModuleBuilder.DefineType(Configuration.EdiFactType,
TypeAttributes.Public |
TypeAttributes.Class |
TypeAttributes.Serializable);
RootClass = rootTypeBuilder;
TypeController.SetClassAttributes(RootClass);
return rootTypeBuilder;
}
//Nested class (Nachricht, Vorgang, etc)
TypeBuilder typeBuilder = ModuleBuilder.DefineType(inXmlProperty.PropertyName, TypeAttributes.Public |
TypeAttributes.Class |
TypeAttributes.Serializable);
TypeController.SetClassAttributes(typeBuilder, inXmlProperty);
return typeBuilder;
}