Make such unconstant children of ourselves"
-- Shakespeare, King John
Act III, Scene I, Lines 242-243
For C# development, I love to use AutoFac! I find that AutoFac allows me to change the design of my code without having to worry about decencies. The last thing I want to have to do while refactoring my design is fix a bunch of broken unit tests. Typically if find that I have to support hard coded test setup for the system under test, when I change the signature of the constructor I'll also need to change all the test setups related to the class (this is often the case in code in which I was not the original author). This can become very annoying, since the test cases often do not really care about the setup of the system under test.
If you are using a decency injection framework like AutoFac you really need to pair the loose coupling of your production code decencies with loose coupling of the setup of your system under test. Loose coupling of the setup of your system under test is exactly what a test fixture is for.
Personally I love AutoFixture for C# development! AutoFixture does all the hard work of creating a test fixture for you, all you have to do is walk up to it and ask it for what you want.
If you are using AutoFac or some other IoC framework in your system under test, more likely than not, you are using a mocking framework like Moq or Rhino Mocks. Well, AutoFixture has you covered by allowing you to use an auto mocking plugin for Moq, Rhino Mocks, or whatever. By using auto mocking along with AutoFixture you are able to create a system under test with all the decencies already mocked out.
With AutoFixture creating the system under test for you, you can freely change the signature of your constructor for your system under test without the need to change any of the setup for the test cases!
Say we have the following:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using System.Linq; | |
using System.Reflection; | |
using System.Text; | |
using System.Threading.Tasks; | |
using Autofac; | |
namespace AutoFixture | |
{ | |
public static class DependencyResolver | |
{ | |
public static IContainer Resolver() | |
{ | |
var builder = new ContainerBuilder(); | |
builder | |
.RegisterAssemblyTypes(Assembly.GetExecutingAssembly()) | |
.AsImplementedInterfaces(); | |
return builder.Build(); | |
} | |
} | |
public class Program | |
{ | |
public static void Main(string[] args) | |
{ | |
var resolver = DependencyResolver.Resolver(); | |
var system = resolver.Resolve<ISystem>(); | |
system.Do(); | |
} | |
} | |
public interface ISystem | |
{ | |
void Do(); | |
bool Status(); | |
} | |
public class System : ISystem | |
{ | |
/*private readonly ILog _logger;*/ | |
private readonly IMessager _messager; | |
private readonly IProcessor _processor; | |
public System(/*ILog logger,*/ IMessager messager, IProcessor processor) | |
{ | |
/*_logger = logger;*/ | |
_messager = messager; | |
_processor = processor; | |
} | |
public void Do() | |
{ | |
} | |
public bool Status() | |
{ | |
var messages = _messager.Pull(); | |
var value = _processor.Calculate(messages); | |
/*_logger.Info(value.ToString());*/ | |
return value > 0; | |
} | |
} | |
public interface IMessager | |
{ | |
IList<int> Pull(); | |
} | |
public interface ILog | |
{ | |
void Info(string message); | |
} | |
public interface IProcessor | |
{ | |
int Calculate(IList<int> messages); | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using Microsoft.VisualStudio.TestTools.UnitTesting; | |
using Moq; | |
using Ploeh.AutoFixture; | |
using Ploeh.AutoFixture.AutoMoq; | |
using Ploeh.AutoFixture.Kernel; | |
namespace AutoFixture.Tests | |
{ | |
// these will continue to work after the constructor's signature is changed | |
[TestClass] | |
public class SystemTestsWithAutoFixture | |
{ | |
private Fixture _fixture; | |
[TestInitialize] | |
public void BeforeEach() | |
{ | |
_fixture = new Fixture(); | |
_fixture | |
.Customize(new AutoMoqCustomization()); | |
} | |
[TestMethod] | |
public void GivenProcessorReturnsLessThanZero_StatusMustReturnFalse() | |
{ | |
_fixture.Customizations.Add( | |
new RandomNumericSequenceGenerator(-1000, -1)); | |
var processor = _fixture.Freeze<Mock<IProcessor>>(); | |
processor | |
.Setup(s => s.Calculate(It.IsAny<IList<int>>())) | |
.Returns(_fixture.Create<int>()); | |
var sut = _fixture.Create<System>(); | |
Assert.IsFalse(sut.Status()); | |
} | |
[TestMethod] | |
public void GivenProcessorReturnsGreaterThanZero_StatusMustReturnTrue() | |
{ | |
_fixture.Customizations.Add( | |
new RandomNumericSequenceGenerator(1, 1000)); | |
var processor = _fixture.Freeze<Mock<IProcessor>>(); | |
processor | |
.Setup(s => s.Calculate(It.IsAny<IList<int>>())) | |
.Returns(_fixture.Create<int>()); | |
var sut = _fixture.Create<System>(); | |
Assert.IsTrue(sut.Status()); | |
} | |
} | |
// these will stop working once the constructor's signature is changed | |
[TestClass] | |
public class ProgramTestsWithoutAutoFixture | |
{ | |
private Mock<IMessager> _mockMessager; | |
private Mock<IProcessor> _mockProcessor; | |
private System _sut; | |
[TestInitialize] | |
public void BeforeEach() | |
{ | |
_mockMessager = new Mock<IMessager>(); | |
_mockProcessor = new Mock<IProcessor>(); | |
_sut = new System( | |
_mockMessager.Object, | |
_mockProcessor.Object); | |
} | |
[TestMethod] | |
public void GivenProcessorReturnsLessThanZero_StatusMustReturnFalse() | |
{ | |
_mockProcessor | |
.Setup(s => s.Calculate(It.IsAny<IList<int>>())) | |
.Returns(-5); | |
Assert.IsFalse(_sut.Status()); | |
} | |
[TestMethod] | |
public void GivenProcessorReturnsGreaterThanZero_StatusMustReturnTrue() | |
{ | |
_mockProcessor | |
.Setup(s => s.Calculate(It.IsAny<IList<int>>())) | |
.Returns(5); | |
Assert.IsTrue(_sut.Status()); | |
} | |
} | |
} |
The above is a fairly common example of C# application that I work with.
We see that we are testing using two different test classes, one using AutoFixture and one not. If we uncomment out the ILog decency in the System class we would break the test setup in the test class ProgramTestsWithoutAutoFixture, but the test class SystemTestsWithAutoFixture would continue to work! For such a simple example this might not matter, but with larger code bases this can become an issue.
By using AutoFixture with Moq we are pairing the same loosely coupling of the creation of our system under test with the loosely coupling we get for our decencies in our production code using AutoFac. Truly one can say, AutoFac iff AutoFixture with Moq.