Search code examples
c#reflection.emit

Create dynamic nested class and adding it as a property in a dynamic Type


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?


Solution

  • 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;
        }