Refactoring Test Code: Removing Constructor Dependency
Table of Contents
In the earlier post, Removing Unnecessary Dependencies, we saw how having an unnecessary dependency hinders testability. In this post we will see how the test code changed by the refactoring we did for removing the unnecessary dependency and explore ways to control these changes.
Impact on Tests by the Refactoring
The refactoring in the last post involved a change in the updating the constructor signature to take in a string value instead of an interface. This broke a lot of our tests and forced us to change all constructor usages with the below code.
var anonymousName = "Anonymous Name";
myService = new MyService(otherDependency, anonymousName);
When seen in isolation, this is not much a change, but as the number of tests grows it becomes a pain. This definitely does not feel right. Breaking tests forces us out of Test-First Development and reduces the confidence in the tests and in the code.
The idea behind TDD was Red-Green-Refactor. But if tests break when Refactoring, then why follow TDD at all?
Refactoring Tests
Ideally, we should write tests that do not break when we refactor, so that it helps us to use the same tests over the refactored code. Let's see how we can improve the test code to prevent tests from breaking, when we refactor to remove unnecessary dependency. Below is the original code (rewritten into xUnit and Moq, as I prefer that) with the dependency on IAppSettings (which we will change it to string later)
[Fact]
public void PerformOperationsShouldReturnTrue()
{
var otherDependency = new Mock<IMyOtherDependency>();
var appSettings = new Mock<IAppSettings>();
appSettings.Setup(a => a["app.name"]).Returns("My Test Application");
var myService = new MyService(otherDependency, appSettings);
var result = myService.PerformOperations();
otherDependency.Verify(a => a.UtilityMethod(), Times.Once());
Assert.True(result);
}
Let's analyze the test code for the dependencies that it has:
- UtilityMethod of IMyOtherDependency
- app.name configuration value from IAppSettings
- Constructor of System Under Test(SUT) - MyService
- PerformOperations of SUT which is getting tested
The test by itself verifies that calling PerformOperations returns true and UtilityMethod gets called once. It is not dependent on the value ('My Test Application') returned by appSettings. The only need is that it should return some (dummy) value when asked for 'app.name'. Assuming that there are multiple tests in this class that does the same setup of IAppSettings to return a dummy value you can start smelling Cut-and-Paste code reuse for fixture setup.
Rule of Three: “The first time you do something, you just do it. Second time you do something similar, you wince at the duplication, but you do the duplicate thing anyway. The third time you do something similar, you refactor.”
The [Rule of Three](https://en.wikipedia.org/wiki/Rule_of_three_(computer_programming) is applicable even when writing test code and we should always keep an eye for duplication. It is easy to get lost in the thought that it's just test code and does not hurt to copy paste. Code duplication in test code does hurt and it hurts the most when you refactor production code.
So lets Refactor applying the various techniques that we know of!
Extract Method
You have a code fragment that can be grouped together. Turn the fragment into a method whose name explains the purpose of the method.
Since we only depend on the IMyOtherDependency and the SUT instance instantiated with that, we can extract SUT creation with a given instance of IMyOtherDependency as below.
[Fact]
public void PerformOperationsShouldReturnTrue()
{
var otherDependency = new Mock<IMyOtherDependency>();
var myService = GetMyServiceWithMyOtherDependency(otherDependency);
var result = myService.PerformOperations();
otherDependency.Verify(a => a.UtilityMethod(), Times.Once());
Assert.True(result);
}
private MyService GetMyServiceWithMyOtherDependency(Mock<IMyOtherDependency> otherDependency)
{
var appSettings = new Mock<IAppSettings>();
appSettings.Setup(a => a["app.name"]).Returns("My Test Application");
var myService = new MyService(otherDependency, appSettings);
return myService;
}
This starts taking us towards Object Mother Pattern. It looks good to start with and might work well if all we have is the same fixture setup. But if we have a different kind of fixture setup, with more dependency and combinations of setup, we will soon have a lot of similar creational methods with different combinations of parameters - GetMyServiceWithMyOtherDependencyAndAppSettings,GetMyServiceWithAppSettings etc. The problem with having different methods is that all of them are dependent on the SUT constructor and set the same properties, leading to code duplication again.
Extract Class
You have one class doing work that should be done by two. Create a new class and move the relevant fields and methods from the old class into the new class.
With these new creational methods the test class is having more responsibility than it should actually have, so let's extract these creation methods into MyServiceBuilder class to see if we can further solve the problem.
public class MyServiceBuilder
{
public IAppSettings AppSettings { get; private set; }
public IMyOtherDependency OtherDependency { get; private set; }
public MyServiceBuilder()
{
var appsettingsMock = new Mock<IAppSettings>();
appsettingsMock.Setup(a => a["app.name"]).Returns("My Test Application");
AppSettings = appsettingsMock.Object;
OtherDependency = new Mock<IMyOtherDependency>().Object;
}
public MyService Build()
{
return new MyService(OtherDependency, AppSettings);
}
public MyServiceBuilder WithAppSettings(IAppSettings appSettings)
{
AppSettings = appSettings;
return this;
}
public MyServiceBuilder WithOtherDependency(IMyOtherDependency otherDependency)
{
OtherDependency = otherDependency;
return this;
}
}
This takes us to Test Data Builder Pattern and as we notice we have reduced the dependency on the MyService constructor to just one and only place where we need to change if the constructor signature changes. Since the Cyclomatic Complexity of MyServiceBuilder is one it is fine not to write tests for it . Using the new builder class our original test case now looks like below.
[Fact]
public void PerformOperationsShouldReturnTrue()
{
var otherDependency = new Mock<IMyOtherDependency>();
var myService = new MyServiceBuilder().WithOtherDependency(otherDependency.Object).Build();
var result = myService.PerformOperations();
otherDependency.Verify(a => a.UtilityMethod(), Times.Once());
Assert.True(result);
}
Now the test is just dependent on the objects that it needs. If all the test use MyServiceBuilder, we can now easily refactor to Remove the Unnecessary Dependency on IAppSettings, by just changing the MyServiceBuilder to use a string property. We will also need to change tests that use the WithAppSettings method which is expected, as those tests are dependent on the app settings value in the first place and so the tests definitely need to be re-written.
Generic Test Data Builder
We could have essentially stopped at the above step, but then we realize that it is too much work to create a Test Data Builder class for each of the production code classes that we have. It takes a lot out of the finite number of keystrokes left in your hands and you definitely don't want to waste that in typing redundant code. This is where we can use AutoFixture, that is an open source library for .NET that helps reduce the Setup/Arrange phase. Using AutoData Theories with AutoFixture our test case now looks like below.
[Theory, AutoMoqData]
public void PerformOperationsShouldReturnTrue(
[Frozen]Mock<IMyOtherDependency> otherDependency,
MyService myService)
{
var result = myService.PerformOperations();
otherDependency.Verify(a => a.UtilityMethod(), Times.Once());
Assert.True(result);
}
The test code does not have any dependency on the constructor of the SUT and a change in constructor signature does not affect our tests at all. We can refactor MyService and use the same tests as long as the functionality served by the class remains the same. Constructors are implementation details and it's better to keep tests independent of it. This keeps our test code clean and more robust!
Rahul Nath Newsletter
Join the newsletter to receive the latest updates in your inbox.