Archive for July, 2013

Wednesday, July 17th, 2013

Clockwork (stock photo by xtrapink)

This picks up with the code generation series. This is planned to be the last post in this mini series. See past posts:

This post will go over some basics using the CodeDom. The code samples used in this post come from a project I wrote that allowed me to copy/paste parts of a customer spec and generate classes that could be used for serialization.

CodeDom is flexible enough for a wide spectrum of code generation needs. As with all things, the power that comes with that generating code with CodeDom is balanced by the complexity of using CodeDom.

Let’s dive right into some code:

// Generate the container unit
CodeCompileUnit program = new CodeCompileUnit();

// Generate the namespace
CodeNamespace ns = new CodeNamespace("Net.Sirchristian");

// Add the required imports
ns.Imports.Add(new CodeNamespaceImport("System"));
ns.Imports.Add(new CodeNamespaceImport("System.Xml.Serialization"));

CodeDom works off of CodeCompileUnits. A CodeCompileUnit can be thought of as a file to be generated. Every file to be generated will construct a CodeCompileUnit. CodeNamespace is equivalent to a namespace block in C#.

namespace Net.Sirchristian { }

To the namespace we add CodeNamespaceImport which are the ‘using’ statements.

namespace Net.Sirchristian 
    using System;
    using System.Xml.Serialization;

So far fairly straight forward. Pretty much a one -> one CodeDom object to code structure. One thing to note is we have not yet added anything to our CodeCompileUnit. We have to fully construct the objects that will go inside the CodeCompileUnit then we can add namespace to it. This is true with any container object when using CodeDom. The children must be fully populated before getting put into the container. Next we will generate our class container.

// Declare the class
CodeTypeDeclaration recordClass = new CodeTypeDeclaration()
    Name = "Record", 
    IsClass = true

A class is represented by a CodeTypeDeclaration. A CodeTypeDeclaration can be any user definable types allowed by the CLR, currently that means class, interface, enum, struct, or delegate (although to generate a delegate you have to use CodeTypeDelegate which inherits from CodeTypeDeclaration). Now we have to generate objects to add to the class.

string anyString = "property of chris";

// Make a nice property name by making it title case, and no spaces
string propertyName = CultureInfo.CurrentCulture.TextInfo.ToTitleCase(anyString.ToLower());
propertyName = Regex.Replace(propertyName, @"\W|\s", "");

// Make the private file starting with 2 underscores and an all lower name
string privateFieldName = "__" + propertyName.ToLower();

// Generate the private field
CodeMemberField field = new CodeMemberField()
	Name = privateFieldName,
	Type = new CodeTypeReference(typeof(string)),
	Attributes = MemberAttributes.Private

// Generate the property
CodeMemberProperty property = new CodeMemberProperty()
	Name = propertyName,
	Type = new CodeTypeReference(typeof(string)),
	Attributes = MemberAttributes.Public | MemberAttributes.Final,
	HasGet = true,
	HasSet = true

// Add the return field statement to the property
property.GetStatements.Add(new CodeMethodReturnStatement(
	new CodeFieldReferenceExpression(
		new CodeThisReferenceExpression(), privateFieldName)));

// Add the set field to value to the property
	new CodeAssignStatement(
		new CodeFieldReferenceExpression(
			new CodeThisReferenceExpression(), privateFieldName),
		new CodePropertySetValueReferenceExpression()));

// Add a comment
string comment = "Remember children: ALWAYS COMMENT YOUR CODE.";
property.Comments.Add(new CodeCommentStatement(comment, true));

// Add the XmlElement Attribute
CodeAttributeDeclaration attribute = new CodeAttributeDeclaration(
	new CodeTypeReference(typeof(XmlElementAttribute)),
		new CodeAttributeArgument("Type", new CodeTypeOfExpression(typeof(string))),
		new CodeAttributeArgument("ElementName", new CodePrimitiveExpression(propertyName)),
		new CodeAttributeArgument("Namespace", new CodePrimitiveExpression("")))

What the above code does is add a public property with a getter and setter that will get and set the private field. The public property is decorated with an XmlElement attribute. This is where you can start to notice some of the power/flexibility trade offs of using CodeDom. Conceptually we will be building out something that looks like.

// Remember children: ALWAYS COMMENT YOUR CODE.
public string PropertyOfChris
        return this.__propertyofchris;
        this.__propertyofchris = value;
private string __propertyofchris;

To generate that we use the following objects from CodeDom:

I won’t go over all the objects in details, but I want to illustrate that everything that needs to be generated will have an object associated with it. To see more of the CodeDom objects look at the CodeDom namespace on MDSN.

There are patterns on how the CodeDom object model is structured. Mostly this can be inferred by the object name. There are CodeObjects , CodeExpressions, CodeStatements, all of which need to be combined just like you would need to type out tokens when editing a source file. The big benefit to CodeDom is that once the structures are built out with CodeDom objects it is easy to add additional logic around them. This is the flexibility of CodeDom.

So we’ve pretty much constructed all the objects we need to have the code generated for us, the last step is to add the constructed CodeDom objects to the appropriate parent objects.

// add property to the class

// add the field to the cass

// Add record class to the namespace


We finally added the members to the class, the class to the namespace, and the namespace to the CodeCompileUnit. We have a whole source file finally constructed. The last step is to render the CodeCompileUnit into a C# source file. We do that by created a CodeProvider. Specifically a CSharpCodeProvider (although as long as the CodeDom objects don’t contain any C# specific features all the C# code used above could generate a VB file just by changing CSharpCodeProvider to a VBCodeProvider).

using (var outFile = File.Open("out.cs", FileMode.Create))
using (var fileWriter = new StreamWriter(outFile))
using (var indentedTextWriter = new IndentedTextWriter(fileWriter, "    "))
	// Generate source code using the code provider.
	var provider = new Microsoft.CSharp.CSharpCodeProvider();
		new CodeGeneratorOptions() { BracingStyle = "C" });

One more cool thing…creating an assembly at runtime!

Another cool thing about CodeDom is that an Assembly can actually be compiled and used at runtime.

// Build the parameters for compilation
CompilerParameters cp = new CompilerParameters();

// Add references

// Save the assembly in memory
cp.GenerateInMemory = true;

// Invoke compilation.
CompilerResults cr = provider.CompileAssemblyFromDom(cp, program);

if (cr.Errors.Count > 0)
    // Build an exception
    ApplicationException exception = new ApplicationException("Error building assembly");
    StringBuilder errorBuilder = new StringBuilder();
    foreach (CompilerError ce in cr.Errors)
        exception.Data.Add(ce.ToString(), ce);

    throw exception;

Assembly assembly = cr.CompiledAssembly;

CompilerParameters and CompilerResults are also part of the CodeDom namespace. Feed the CodeProvider with the CompilerParameters and the CodeCompileUnit and the provider can compile the generated code for you, without even needed to generate code!

This is the last entry I planned for the code generation series. Want more, or have questions, leave a comment, or ping me on twitter @sirchristian.