Thoughts on Software by Andrew Davey
| | Sun | Mon | Tue | Wed | Thu | Fri | Sat |
|---|
| 24 | 25 | 26 | 27 | 28 | 29 | 30 | | 1 | 2 | 3 | 4 | 5 | 6 | 7 | | 8 | 9 | 10 | 11 | 12 | 13 | 14 | | 15 | 16 | 17 | 18 | 19 | 20 | 21 | | 22 | 23 | 24 | 25 | 26 | 27 | 28 | | 29 | 30 | 31 | 1 | 2 | 3 | 4 |
Search
Navigation
Categories
Blogroll
|

Tuesday, July 31, 2007
C# Syntactic Macros
After a late night hacking session, I finally have a working syntactic macro system for C#! It is comprised of a VS custom tool that uses the CSparser. The end result allows me to do this:
using System;
[Macro(AutoClass)]
public interface IModel
{
string FirstName { get; set; }
string Surname { get; set; }
int ID { get; }
event EventHandler FirstNameChanged;
event EventHandler SurnameChanged;
}
The CustomTool property of the IModel.cs file is set to "MacroExpander" and the Build action is set to None. This then generates a new file "under" the source file, called for example "IModel.generated.cs". This generated file contains the original interface (minus the bogus attribute) and a new class called "Model" that provides a stanard implementation of IModel. It includes private fields, a constructor, properties, events and event raising methods. The class is partial so you can still create a file called "Model.cs" alongside the interface and implement any methods. In addition, if you enter implementations of the interface properties/events in Model.cs then the generated class will not contain them. So it is easy to generate the 80% of standard members and custom write the others as required.
The sharp-minded out there will note that "AutoClass" is a parameter passed to the Macro attribute. The expansion process is able to load any IMacro implementation I care to write. The macro implementation is handed the parsed source tree and allowed to modify it as required. After all macros have expanded, the source tree is converted back to CS source text and outputted from the VS custom tool. This generated file is compiled by VS as usual.
I'm going to start using this AutoClass macro to generate most of my Model code (in the Model-View-Presenter architecture). Anywhere else I need code generation at this level I can create more macros easily.
If people are interested in seeing this product released in some way please get in contact. I will perhaps get a screen cast made up to show it in action.
Tuesday, July 31, 2007 12:04:41 PM (GMT Standard Time, UTC+00:00)
.net | programming | c#

Thursday, July 26, 2007
C# Code Generation
I came across http://www.codeplex.com/csparser today. This parser takes C# source code and produces in memory object tree representing the code. This object tree can be modified and the resulting source code extracted back out as a string.
I have been thinking about ways to remove the mindless plumbing code I have to write when implementing a Model-View-Presenter architectural system. For example, I define an interface for my Model. I then make a Model class that implements the Model interface. The slow bit here is implementing all the properties with fields, defining events and the OnSomeEventHappened(Args) methods to safely raise them.
Whilst there are tools that will probably do this plumbing for me, they have one flaw. I don't want to even read, let alone write, that boiler-plate code! It does not help me understand what it going on. The interface contains all the important details. I basically want the compiler to generate it for me.
C# has no syntactic macro system (unlike Boo and Nemerle) so I'm stuck with my current XML/XSLT based code generator. Whilst this is usable, it really means I'm no longer coding in C#, but rather long-winded XML. The context switch there is a bit annoying.
With the CS Parser I will be able to the following:
Write a Visual Studio Custom Tool (like my XML/XSLT code generator) that takes C# code file of an interface and generates an implementing class. This is done by parsing the interface definition, then looping through members and creating a new class definition accordingly.
If the class already partially exists I can allow the user to type specific members they need to implement in a non-standard way and not generate these automatically.
The output of the Custom Tool (new C# code) is then compiled automatically as part of the Visual Studio build process.
For example:
public interface IModel
{
int Amount { get; set; }
event EventHandler AmountChanged;
void Load();
}
Generates into:
public partial class Model : IModel
{
int _amount;
public int Amount
{
get { return _amount; }
set
{
_amount = value;
OnAmountChanged();
}
}
public event EventHandler AmountChanged;
protected virtual void OnAmountChanged(EventArgs e)
{
EventHandler handler = AmountChanged;
if (handler != null)
handler(this, e);
}
}
Leaving the developer just to write the Load method.
I'm wondering about generalising this a bit. Using magic attributes that are picked up by a tool (much like syntactic macro attributes in Nemerle) to invoke specific actions.
[AutoImplement(IModel)]
public class Model : IModel
{
public void Load() { ... }
}
Fun times ahead!!
Thursday, July 26, 2007 4:47:58 PM (GMT Standard Time, UTC+00:00)

Monday, July 16, 2007
Moving to Bristol
Sorry, this not technically "about code", but I'm moving to Bristol on the 27th July. I'm moving into a new flat and am looking forward to working and playing in Bristol!
Hopefully I'll get lots of work done ;)
Monday, July 16, 2007 6:53:33 PM (GMT Standard Time, UTC+00:00)

Friday, July 13, 2007
Splitting Objects Across Tiers
Erik Meijer is involved in some very interesting research at Microsoft Research. This latest video on Channel 9 introduces the work his team are doing to make writing an application a single concept and then splitting it over tiers as required.
He talks about using a refactoring to split a class out into different parts for required tiers. For example, a model running on a client needs to talk to a database at some point. That part of the code needs to be running in a context that can access the database i.e. the server. The refactoring generates a service class that contains the database access code, and the client class is updated to call the service.
Apparently they are working to make this declarative. I imagine this looking something like:
[RunAtServer] void MyMethod(int id)
{
var people = from p in DB.GetPeople() where p.ID = id select p.Name;
// etc...
}
Of course, the "macro" siren went off in my head. I could probably do something like this today in Boo or Nemerle. For making close-coupled systems (where data types are shared between tiers) this could be amazing. All we need is an assembly level macro that scans class for certain attributes and auto generates the required services, proxies and proxy calls.
When you have .NET running everywhere (i.e. Silverlight in the browser) this kind of magic becomes more of a reality. I'm also very interested to see the MSIL to JavaScript compiler Erik is working on.
Friday, July 13, 2007 10:22:55 PM (GMT Standard Time, UTC+00:00)

Sunday, July 08, 2007
The Right Amount of Code Generation
David Hayen talks about the problem of using code generation at a large scale within software projects. The first use gives a lovely RAD hit, but over time you get more constrained and limited by the generated code. I always found this problem using any kind of data-n-drop data binding (although .NET 2.0 windows forms data binding seems almost flexible enough).
It is hard to get large scale code generation perfect - almost impossible in reality. I find using code generation at a smaller scale to actually raise the level of abstraction in code more useful. Languages such as Boo and Ruby have great syntax for defining mini domain specific languages to raise abstraction levels. Syntactic macros in Nemerle and Lisp are true compile-time code generators. The key to these methods is that the programmer has not lost the expressivity his language gives him. If a certain macro is not really fitting, you can always write out the code long-hand and refactor later if necessary.
Large scale code generation assumes software is smooth and regular. In the real-world we have to deal with special cases and, worse, changing requirements. Trying to make a code generation tool fit all of these requires it to be as expressive as a real programming language! Only now we are buried in stacks of XML config files and templates.
One area I do see a benefit for these quick-n-dirty code generators is in prototyping. You want to get something half-real in front of customers quickly to gather feedback. However, never be precious about your code. The best writing is re-writing! A lot of software hangs around for a lot longer than we expect, or even hope. It is probably best to make it as maintainable as humanly possible.
The fact that a magic tool generated thousands of lines of code for you doesn't stop them for existing. The more code that exists, more chance there is for something to be wrong. If your chosen programming language requires you to write reams of code to express something that you can conceptualise more simply, then I'm sorry you need a better language.
Sunday, July 08, 2007 12:03:08 AM (GMT Standard Time, UTC+00:00)
programming | thinking

Friday, July 06, 2007
Code for the Rhino Mock DSL
I have attached the Boo code containing the Rhino Mocks DSL I'm currently working on. This is very much a work in progress release, just to get it out there. Please have a play and tell me what you think. Happy Mocking!
Dsl.boo (3.4 KB)
To use the DSL, add the following in your code to import the static methods into scope.
import Rhino.Mocks.Dsl
Friday, July 06, 2007 12:19:48 PM (GMT Standard Time, UTC+00:00)
.net | boo | dsl | mock_objects | programming | rhino_mocks

Wednesday, July 04, 2007
Better Syntax for Mocking
I thought some more about the syntax of my Rhino Mock DSL. It can feel unnatural putting all the mock expectation code before the call to the object being tested. I came up with this working prototype instead:
[Test]
def Get_data_objects_for_nonexistent_company_throws():
with_mocks:
database = mocks.CreateMock[of IDatabase]()
userProvider = StubUserProvider("andrew", "bad corp")
userProvider.MakeCurrent()
execute:
uds = UserDataService(database, userProvider)
expect_throw FaultException[of GetDataObjectsFault]:
uds.GetDataObjects()
assert thrown_exception.Detail.Type.Equals(GetDataObjectsFaultType.InvalidCompany)
assuming:
database.GetUserID("andrew", "bad corp")
returned 1
assuming:
database.GetCompanyID("bad corp")
returned 0 # returning 0 from database implies nonexistent company.
The with_mocks method sets up the mock repository and a Store object. The execute and assumption methods then put their blocks into the Store. At the end of with_mocks I iterate through assumptions calling each to set up the Rhino Mock expectations. Following that is: mocks.ReplayAll(), call to the "execute" block, then mocks.VerifyAll().
The other clever bit in there is the expect_throw method. This runs the block inside a try...except and fails if no exception (or wrong exception type) is thrown. It puts the exception object into field that is readable using thrown_exception. This means we can then test assertions about the exception contents. I had to cheat a bit and declare thrown_exception as "duck" in Boo i.e. it is late bound. This is so we can access members on the actual object despite not really knowing about it at compile time.
I like the readability now. The outline is:
- Initialize mocks and data objects
- Call the object being tested
- Assert about the result
- State the assumptions about how dependencies are used
The key bit, I feel, is that the call to the object being tested is not buried down at the bottom of the method.
How does everyone else feel about this modified approach?
Wednesday, July 04, 2007 10:41:55 AM (GMT Standard Time, UTC+00:00)
.net | boo | dsl | mock_objects | programming | rhino_mocks

Tuesday, July 03, 2007
Microsoft and External .NET Languages
Why can't Microsoft do more to support external projects like Boo and Nemerle? It drives me insane seeing all the love Python and Ruby are getting from Microsoft, when there are languages out there that embraced the CLR from day one being ignored!
To be honest, by "love" I mean IDE love. Nemerle is a great language - light years ahead of C#. The same goes for Boo. But the tool support for these languages in poor. The Nemerle VSIP project barely usable for any serious work and for Boo the only real option is SharpDevelop.
I confess, I am a total Visual Studio junkie. I need my tools. Using emacs and cmd prompt for some project with hundreds of source, resource and data files is stupid. You shall take intellisense from my cold dead hands!
The problem it seems that developing a reliable, complete, VS integration package is very hard. Microsoft: please reach out to these pioneering developers and offer them what ever they need to get their awesome languages in Visual Studio. Just do it. In fact, go one better and release Boo Express and Nemerle Express for free! The .NET world will thank you.
Reading this kind of shit makes me so very sad.
Tuesday, July 03, 2007 8:40:46 AM (GMT Standard Time, UTC+00:00)