Clean Tests: Isolation with Fakes


Other posts in this series:

So far in this series, I’ve walked through different modes of isolation – from internal state using child containers and external state with database resets and Respawn. In my tests, I try to avoid fakes/mocks as much as possible. If I can control the state, isolating it, then I’ll leave the real implementations in my tests.

There are some edge cases in which there are dependencies that I can’t control – web services, message queues and so on. For these difficult to isolate dependencies, fakes are acceptable. We’re using AutoFixture to supply our mocks, and child containers to isolate any modifications. It should be fairly straightforward then to forward mocks in our container.

As far as mocking frameworks go, I try to pick the mocking framework with the simplest interface and the least amount of features. More features is more headache, as mocking frameworks go. For me, that would be FakeItEasy.

First, let’s look at a simple scenario of creating a mock and modifying our container.

Manual injection

We’ve got our libraries added, now we just need to add a way to create a fake and inject it into our child container. Since we’ve built an explicit fixture object, this is the perfect place to put our code:

public T Fake<T>()
{
var fake = A.Fake<T>();
Container.EjectAllInstancesOf<T>();
Container.Inject(typeof(T), fake);
return fake;
}
view raw Fake.cs hosted with ❤ by GitHub

We create the fake using FakeItEasy, then inject the instance into our child container. Because we might have some existing instances configured, I use “EjectAllInstancesOf” to purge any configured instances. Once we’ve injected our fake, we can now both configure the fake and use our container to build out an instance of a root component. The code we’re trying to test is:

public class InvoiceApprover : IInvoiceApprover
{
private readonly IApprovalService _approvalService;
public InvoiceApprover(IApprovalService approvalService)
{
_approvalService = approvalService;
}
public void Approve(Invoice invoice)
{
var canBeApproved = _approvalService.CheckApproval(invoice.Id);
if (canBeApproved)
{
invoice.Approve();
}
}
}

In our situation, the approval service is some web service that we can’t control and we’d like to stub that out. Our test now becomes:

public class InvoiceApprovalTests
{
private readonly Invoice _invoice;
public InvoiceApprovalTests(Invoice invoice,
SlowTestFixture fixture)
{
_invoice = invoice;
var mockService = fixture.Fake<IApprovalService>();
A.CallTo(() => mockService.CheckApproval(invoice.Id)).Returns(true);
var invoiceApprover = fixture.Container.GetInstance<IInvoiceApprover>();
invoiceApprover.Approve(invoice);
fixture.Save(invoice);
}
public void ShouldMarkInvoiceApproved()
{
_invoice.IsApproved.ShouldBe(true);
}
public void ShouldMarkInvoiceLocked()
{
_invoice.IsLocked.ShouldBe(true);
}
}

Instead of using FakeItEasy directly, we go through our fixture instead. Once our fixture creates the fake, we can use the fixture’s child container directly to build out our root component. We configured the child container to use our fake instead of the real web service – but this is encapsulated in our test. We just grab a fake and start going.

The manual injection works fine, but we can also instruct AutoFixture to handle this a little more intelligently.

Automatic injection

We’re trying to get out of creating the fake and root component ourselves – that’s what AutoFixture is supposed to take care of, creating our fixtures. We can instead create an attribute that AutoFixture can key into:

[AttributeUsage(AttributeTargets.Parameter)]
public sealed class FakeAttribute : Attribute { }

Instead of building out the fixture items ourselves, we go back to AutoFixture supplying them, but now with our new Fake attribute:

public InvoiceApprovalTests(Invoice invoice,
[Fake] IApprovalService mockService,
IInvoiceApprover invoiceApprover,
SlowTestFixture fixture)
{
_invoice = invoice;
A.CallTo(() => mockService.CheckApproval(invoice.Id)).Returns(true);
invoiceApprover.Approve(invoice);
fixture.Save(invoice);
}

In order to build out our fake instances, we need to create a specimen builder for AutoFixture:

public class FakeBuilder : ISpecimenBuilder
{
private readonly IContainer _container;
public FakeBuilder(IContainer container)
{
_container = container;
}
public object Create(object request, ISpecimenContext context)
{
var paramInfo = request as ParameterInfo;
if (paramInfo == null)
return new NoSpecimen(request);
var attr = paramInfo.GetCustomAttribute<FakeAttribute>();
if (attr == null)
return new NoSpecimen(request);
var method = typeof(A)
.GetMethod("Fake", Type.EmptyTypes)
.MakeGenericMethod(paramInfo.ParameterType);
var fake = method.Invoke(null, null);
_container.Configure(cfg => cfg.For(paramInfo.ParameterType).Use(fake));
return fake;
}
}
view raw FakeBuilder.cs hosted with ❤ by GitHub

It’s the same code as inside our context object’s “Fake” method, made a tiny bit more verbose since we’re dealing with type metadata. Finally, we need to register our specimen builder with AutoFixture:

public class SlowTestsCustomization : ICustomization
{
public void Customize(IFixture fixture)
{
var contextFixture = new SlowTestFixture();
fixture.Register(() => contextFixture);
fixture.Customizations.Add(new FakeBuilder(contextFixture.Container));
fixture.Customizations.Add(new ContainerBuilder(contextFixture.Container));
}
}

We now have two options when building out fakes – manually through our context object, or automatically through AutoFixture. Either way, our fakes are completely isolated from other tests but we still build out our root components we’re testing through our container. Building out through the container forces our test to match what we’d do in production as much as possible. This cuts down on false positives/negatives.

That’s it for this series on clean tests – we looked at isolating internal and external state, using Fixie to build out how we want to structure tests, and AutoFixture to supply our inputs. At one point, I wasn’t too interested in structuring and refactoring test code. But having been on projects with lots of tests, I’ve found that tests retain their value when we put thought into their design, favor composition over inheritance, and try to keep them as tightly focused as possible (just like production code).

Clean Tests: Isolating the Database