Rust Result
I've been toying with Rust because I realised all I can do is write CRUD in C#. They have a really awesome book that you can read while you wait for your hello world program to compile.
One sweet feature of rust is the way it handles error propagation, providing a handy ?
shortcut allowing you to give up when you encounter an error and let the calling code deal with all your problems. For example:
If File::open
encounters an error, it will automatically return the error and exit the method early.
Results are a first class feature of Rust, intrinsic to it's design. C# doesn't have results (we have exceptions...) but we can easily build our own!
C# Result
An incredibly basic result type:
Let's try it out.
In this slightly contrived example we are getting a number, which may fail, then operating on it if it succeeds, returning the result early if it fails. But look how long it is! I've often toyed with using results, options, maybe etc. in C#, but handling them is either so verbose or you start trying to write "functional C#", which quickly looks absolutely ridiculous and every other C# developer who looks at your 9 chain deep function wants to cry. Libraries like OneOf are interesting and I've had some success with them in carefully chosen use cases.
But what if we could have what Rust has?
The idea: happy path only
We don't care about error checking, we just want to code the happy path! If we can somehow detect when our results are being unwrapped in C# and check if there was an error, returning it if there was, carrying on if not but without having to write a single if
statement...
Introducing the new C# interceptor feature!
Wait wait, I didn't read enough before I started writing interceptor code (and trusted Claude who confidently told me it was possible...). Interceptors can intercept a method call and replace it with another method call. And that is it, it is a very narrow, specific use case. We cant start adding arbitrary if statements and extra returns, mad stuff like that.
Old is new again: Fody Weavers!
In my first job, fresh out of university, I had a senior developer who loved Fody Weavers and would tell anyone who listened how they were the future and everything would be sunshine and rainbows if we just could manipulate the IL a bit more. I had only learnt how to drink a lot at uni and didn't know what IL was, so I nodded sagely along and never thought about them again.
But... maybe we could manipulate the IL?
First let's create a new solution and a netstandard2.0
class library called ResultWeaver.Fody
. The .Fody
ending is important as it is how the weaver is loaded, apparently.
Each project can contain a single weaver, which we can define by creating a class that must be called ModuleWeaver
. Let's start with something basic that just adds some console writing.
Every time I made a change to my weaver I had to delete the sample project's bin
and obj
folders. This powershell one-liner was handy.
Now we can add a little command line project to test our weaver, loading the weaver dll. We do not make a reference to the weaver project.
Now we just need to tell our project to use the referenced weaver.
Now we can test it with this simple program:
Without the weaver it will just print "world", with it will print "Hello world".
If you every have any issues with your weaver not being loaded or being a general pain (which constantly happened to me...), try deleting all bin and obj folders and then rebuild the solution. Build order is important, you can right click the solution and make sure your weaver is built before any test project.
But what is it actually doing? You can grab ILSpy (which has a handy VS extension too) and view the C#, IL and best, the IL + C# generated by your project + weaver.
Weaving in error checks
So let's add a method OrReturn
to our result that will unwrap the value and allow us to detect where we want to start messing around.
Feeling confident, let's not even bother implementing anything! What we are aiming for is that this:
Will be weaved into this:
With a little bit of copy paste from various open source weavers, having to read what the hell a stack height is, all with too much trial and error, let's see what we've got.
Phew, that is a lot of code. It is actually pretty simple, a little verbose. But it works!
Given the following program:
We see the following printed in the console:
Starting
Got first value: 12
Error: result failure
And we can confirm it is working by peeking at the new C# our modified IL represents:
But what it demonstrates most clearly is that this is a rabbit hole of issues waiting to be found. I handled the scenario where the OrReturn
is in a try
. But what about a catch
? Nested try catch
? async
code. Expression-bodied members. Using statements and disposal patterns. Nested scopes (e.g. local functions). Pattern matching or switch statements. Generic methods and covariance/contravariance scenarios. What if they try to do GetOne().OrReturn() + GetTwo().OrReturn()
?! The list goes on and on...
And what if someone chains multiple methods together? We probably need an analyzer to go with this to stop naughty developers toppling our fody weaved tower of cards.
We'd also want to improve our Result
to be able to map one type of result to another, though at least we don't have to touch IL for that!
Okay this is all getting a bit out of hand. This was a fun if pointless experiment, but was a nice reminder of how mature tooling around C# is and the strange, wonderful things we can do with it.