class TradeValueCalculator {
// ..
// ..
public BigDecimal calculateTradeValue(final Trade trade, ..) {
// ..
BigDecimal tax = TradeUtils.calculateTax(trade);
BigDecimal commission = TradeUtils.calculateCommission(trade);
// .. other business logic to compute net value
}
// ..
// ..
}
What is the problem with the above two innocuous looking Java lines of code ? The answer is very simple - Unit Testability of the POJO class
TradeValueCalculator
! Yes, this post is about unit testability and some tips that we can follow to design classes that can be easily unit tested. I encountered many of these problems while doing code review of a live Java project in recent times.Avoid Statics
When it comes to testability, statics are definitely not your friends. In the above code snippet, the class
TradeValueCalculator
depends on the implementation of the static methods like TradeUtils.calculateTax(..)
and TradeUtils.calculateCommission(..)
. Any change in these static methods can lead to failures of unit tests of class TradeValueCalculator
. Hence statics introduce unwanted coupling between classes, thereby violating the principle of easy unit-testability of POJOs. Avoid them, if you can, and use standard design idioms like composition-with-dependency-injection instead. And while using composition with service components, make sure they are powered by interfaces. Interfaces provide the right level of abstraction for multiple implementations and are much easier to mock while testing. Let us refactor the above snippet to compose using service components for calculating tax and commission :class TradeValueCalculator {
// .. to be dependency injected
private ITaxCalculator taxCalculator;
private ICommissionCalculator commissionCalculator;
// ..
// ..
public BigDecimal calculateTradeValue(final Trade trade, ..) {
// ..
BigDecimal tax = taxCalculator.calculateTax(trade);
BigDecimal commission = commissionCalculator.calculateCommission(trade);
// .. other business logic to compute net value
}
// ..
// ..
}
interface ITaxCalculator {
BigDecimal calculateTax(..);
}
interface ICommissionCalculator {
BigDecimal calculateCommission(..);
}
We can then have concrete instances of these service contracts and inject them into the POJO
TradeValueCalculator
:class DefaultTaxCalculator implements ITaxCalculator {
// ..
}
class DefaultCommissionCalculator implements ICommissionCalculator {
// ..
}
Using standard IoC containers like Guice or Spring, we can inject concrete implementations into our POJO non-invasively through configuration code. In Guice we can define Modules that bind interfaces to concrete implementations and use Java 5 annotation to inject those bindings in appropriate places.
// define module to configure bindings
class TradeModule extends AbstractModule {
@Override
protected void configure() {
bind(ITaxCalculator .class)
.to(DefaultTaxCalculator .class)
.in(Scopes.SINGLETON);
bind(ICommissionCalculator .class)
.to(DefaultCommissionCalculator .class)
.in(Scopes.SINGLETON);
}
}
and then inject ..
class TradeValueCalculator {
// ..
@Inject private ITaxCalculator taxCalculator;
@Inject private ICommissionCalculator commissionCalculator;
// ..
// ..
}
How does this improve testability of our class
TradeValueCalculator
?Just replace the defined
Module
by another one for unit testing :// define module to configure bindings
class TestTradeModule extends AbstractModule {
@Override
protected void configure() {
bind(ITaxCalculator .class)
.to(MockTaxCalculator .class)
.in(Scopes.SINGLETON);
bind(ICommissionCalculator .class)
.to(MockCommissionCalculator .class)
.in(Scopes.SINGLETON);
}
}
What we have done just now is mocked out the service interfaces for tax and commission calculation. And that too without a single line of code being changed in the actual class!
TradeValueCalculator
can now be unit-tested without having any dependency on other classes.Extreme Encapsulation
I have come across many abuses of FluentInterfaces, where developers use chained method invocations involving multiple classes. Take this example from this Mock Objects paper, which discusses this same problem :
dog.getBody().getTail().wag();
The problem here is that the main class Dog is indirectly coupled with multiple classes, thereby violating the Law of Demeter and making it totally unsuitable for unit testing. The situation is typically called "The Train Wreck" and has been discussed extensively in the said paper. The takeway from this situation is to minimize coupling with neighbouring classes - couple only with the class directly associated with you. Think in terms of abstracting the behavior *only* with respect to the class with which you collaborate directly - leave implementation of the rest of the behavior to the latter.
Privates also need to be Unit-Tested
There is a school of thought which espouses the policy that *only* public api s need to be unit-tested. This is not true - I firmly believe that all your methods and behaviors need unit testing. Strive to achieve the maximum coverage of unit testing in your classes. Roy Osherove thinks that we may have to bend some of the rules of pure OOD to make our design implementations more testable e.g. by exposing or replacing private instances of objects using interfaces, injection patterns, public setters etc. Or by discouraging default sealing of classes allowing overriding in unit tests. Or by allowing singletons to be replaced in tests to break dependencies. I think, I agree to many of these policies.
Fortunately Java provides a useful access specifier that comes in handy here - the package private scope of access. Instead of making your implementation members *private*, make them *package private* and implement unit test classes in the same package. Doing this, you do not expose the private parts to the public, while allowing access to all unit test classes. Crazy Bob has more details on this. Another useful trick to this may be usage of AOP. As part of unit test classes, you can introduce additional getters through AOP to access the implementation artifacts of your class. This can be done through inter-type declarations, and the test classes can access all private data at gay abandon.
Look out for Instantiations
There are many cases where the class that is being unit tested needs to create / instantiate objects of the collaborating class. e.g.
class TradeController {
// ..
// ..
public void doTrade(TradeDTO dto, ..) {
Trade trade = new Trade(dto);
// .. logic for trade
}
// ..
}
Increase the testability of the class
TradeController
by separating out all creation into appropriate factory methods. These methods can then be overridden in test cases to inject creation of Mock objects.class TradeController {
TradeDTO dto;
// ..
// ..
public void doTrade() {
Trade trade = createTrade(dto);
// .. logic for trade
}
// ..
protected Trade createTrade(TradeDTO dto) {
return new Trade(dto);
}
}
and create
MockTrade
in test cases ..class TradeControllerTest extends TestCase {
// ..
public void testTradeController(..) {
TradeController tc = new TradeController() {
protected Trade createTrade(TradeDTO dto) {
return new MockTrade(dto);
}
}
tc.doTrade();
}
}
The Factory Method pattern proves quite helpful in such circumstances. However, there are some design patterns like The Abstract Factory, which can potentially introduce unwanted coupling between classes, thereby making them difficult to unit-test. Most of the design patterns in GOF are built on composition - try implementing them using Interfaces in Java, so that they can be easily mocked out. Another difficult pattern is the Singleton - I usually employ the IoC container to manage and unit-test classes that collaborate with Singletons. Apart from static methods, which I have already mentioned above, static members are also problematic cases for unit testing. In many applicatiosn they are used for caching (e.g. ORMs) - hence an obvious problem child for unit testing.