WrapThat.System : Wrappers for system.io to simplify unit testing
By terje
Introduction
You have all seen it, you have all done it: Written code using the static methods from the System.IO namespace for handling directories and files. Or, you are maintaining legacy code where that usage is common in certain places. And now the code is huge, you have no unit tests, because unit testing classes that directly access the file system is just pain, and the cost of a rewrite is too big.
You have either realized you need to change this code before it grows even more, or you might have been given the task of encapsulating that code in unit tests. But, how do you proceed with that? Because these are static methods in static classes. And the amount of code is too much to rewrite into something more modern, using file providers or something similar.
You’re thinking of shims, and shudders when you remember the earlier attempts of that kind of mock frameworks. And you already are using modern mocking frameworks like Moq or NSubstitute. You want something more lightweight than reflection based shims.
The question then is: What is the minimal impact change you can do, in such a way that you don’t touch the underlying functionality, but still make the code testable.
This is where the wrapper technique comes in handy, combined with injection. And this is where the use of the WrapThat.System package can be very useful.
Note that the same principles as are being used in the WrapThat.System can be used with any other library or system you want to isolate for unit testing.
The works
The WrapThat.System package contains a 100% 1-to-1 wrapping of the System.IO and System.Console static methods, replacing the existing static classes with instance classes, and exposing them as injectable interfaces.
You need to do but a small change to your code, which will not change the functionality.
You have two scenarios:
1) Production scenario. The code should work as before.
2) Unit Test Scenario. The code should be testable, with the ability to mock the file system.
Let’s look at the unit test scenario first, since that is the objective for the change.
Using the WrapThat.System package, the unit test program pass in mocked instances of the interfaces to the software under test.
Common injection can be done either as constructor injection or as property injection. In the example we look at constructor injection.
This means that the constructor, or constructors, of the class under test must be enhanced with one more parameter, to the mocked interface.
public FileHandling() {}
Now let us start by adding in the injection, but so that it doesn’t get in the way for the old usage, we default it to null.
public FileHandling(ISystemIO systemIo=null) {}
The ISystemIO interface contains two properties, one is the Directory, and one is the File. These are the two main static classes in System.IO that are being wrapped now by the WrapThat.System package.
We will make two properties for these, named exactly like their static class they’re wrapping, and also redirect the usage of any subsequent use of these to the wrapper library.
using ISystemIO = WrapThat.SystemIO.ISystemIO;
// Redirect subsequent use
using Directory = WrapThat.SystemIO.Directory;
using File = WrapThat.SystemIO.File;
// Add the two properties public IDirectory Directory { get; } public IFile File { get; }
// Initialize it in the ctor public FileHandling(ISystemIO systemIo=null) { Directory = systemIo?.Directory ?? new Directory(); File = systemIo?.File ?? new File(); }
This is all we need to do to the “class under test”. Now, since the property names are identical to the old System.IO static classes, all the rest of the code in the class are completely unaware that these calls have been wrapped.
Note that the injected interface is deliberately defaulted to null, so that the constructor appears unchanged to the rest of the code. In that case, a new instance of Directory and File will be added to the properties, otherwise, they get their values from the injected interface.
Note again that the naming of the properties MUST match the original static class names, otherwise the rest of the code would be very confused.
The using statements also ensures that you can keep your old “using System.IO” statement.
Let us see how a small unit test may look like, for a very trivial example (too trivial to make any sense, but just so you get the idea). Let us just assume we have the following method in the class FileHandling:
With File being static, this can’t be mocked, but now that we have it wrapped, we can.
(The code above uses NSubstitute and NUnit 3)
Note that if your class is only using Directory or File, you can just pass in the interface to those instead of the SystemIO interface. The SystemIO interface is just to avoid having to pass in multiple interfaces if you use both.
System.Console wrapping
If you have code with a lot of Console handling, like Console writeline , it will be testable, but it will also be slow. Further, if you add in some ReadLine commands, then you’re stuck. With the System.Console wrapper you can unit test that stuff too.
Assume we have the following class, ready prepared with the injection
And the tests for these two methods then look like this:
The package
Download the package from Nuget: https://www.nuget.org/packages/WrapThat.System
The package contains wrappers for the System.IO classes Directory and File, and the System.Console class. It is currently based on .net 4.0.
You have to add the package to both the code that you’re wrapping, that is your production code, and to the test project where you are going to test your code.
How is it done
The wrappers are generated by reflecting over the types. This means all method wrappers are in principle the same. The parameters and return types are not changed.
The wrappers look like this:
As you can see they are simple one line wrappers that don’t change any behavior.
Source and example code
The source code for the wrappers, the generator and some examples of using them can be found in the git repository at https://github.com/wrapthat
What is coming
Wrappers for the instance classes in System.IO are next up. Stay tuned!