TotT: Partial Mocks using Forwarding Objects
For example, when writing an Olympic swimming event for ducks, you could create a simple forwarding object to be used by multiple tests:
interface Duck {
Point getLocation();
void quack();
void swimTo(Point p);
}
class ForwardingDuck implements Duck {
private final Duck d;
ForwardingDuck(Duck delegate) {
this.d = delegate;
}
public Point getLocation() {
return d.getLocation();
}
public void quack() {
d.quack();
}
public void swimTo(Point p) {
d.swimTo(p);
}
}
And then create a test that uses all of the real OlympicDuck class's behavior except quacking.
public void testDuckCrossesPoolAndQuacks() {
final Duck mock = EasyMock.createStrictMock(Duck.class);
mock.swimTo(FAR_SIDE);
mock.quack(); // quack after the race
EasyMock.replay(mock);
Duck duck = OlympicDuck.createInstance();
Duck partialDuck = new ForwardingDuck(duck) {
@Override public void quack() {
mock.quack();
}
@Override public void swimTo(Point p) {
mock.swimTo(p);
super.swimTo(p);
}
// no need to @Override “Point getLocation()”
}
OlympicSwimmingEvent.createEventForDucks()
.withDistance(ONE_LENGTH)
.sponsoredBy(QUACKERS_CRACKERS)
.addParticipant(partialDuck)
.doRace();
MatcherAssert.assertThat(duck.getLocation(), is(FAR_SIDE));
EasyMock.verify(mock);
partialDuck is a complex example of a partial mock – it combines real and mock objects in three different ways:
- quack() calls the mock object. It verifies that the duck doesn't promote the sponsor (by quacking) until after the race. (We skip the real quack() method so that our continuous build doesn't drive us crazy.)
- getLocation() calls the real object. It allows us to use the OlympicDuck's location logic instead of rewriting/simulating the logic from that implementation.
- swimTo(point) calls both objects. It allows us to verify the call to the real duck before executing it.
There is some debate about whether you should forward to the real or mock Duck by default. If you use the mock duck by default, any new calls to the mock will break the test, making them brittle. If you use the real duck, some very sensitive calls like submitToDrugTest() might get called by your test if your duck happens to win.
Consider using a Partial Mock in tests when you need to leverage the implementation of the real object, but want to limit, simulate or verify method calls using the power of a mock object.
This comment has been removed by the author.
ReplyDeleteCool, but I'd rather stick with simple & readable tests which is exact opposite of what you're showing :). Simple tests push simple implementation.
ReplyDeleteEasyMock's Class Extension gives you all that for free.
ReplyDeletecode:
see http://www.easymock.org/EasyMock2_4_ClassExtension_Documentation.html
the signature is: createMock(java.lang.Class<T> toMock, java.lang.reflect.Method... mockedMethods)
while this achieves your goal, without need to write additional delegates, I think that in some cases partial mocking is an indicator you need to refactor.
why would you need partial mocking in a real world scenario? that means you want to test your unit in sub-isolation, and that means you can probably split or refactor and reorganize the code.
I think partial mocks should have a huge health warning on them. They imply that there's a role in there that's implicit and that ought to be brought out, and they make refactoring harder. Repeat after me, "Composition over inheritance".
ReplyDeleteThe only time I consider using them is to override a factory method that creates an object that's used internally -- and then only until I can figure out a better plan.
Here is my predicament...
ReplyDeleteI have read a lot of the papers/docs out right now about using partial mocks, etc. They usually indicate that if you are using them really look at the design of your software. I agree and disagree leading to my predicament:)
I can easily make my object under test a JavaBean style object so that an object that would normally be encapsulated via a "new SomeObject()" can be directly set on the object under test. This is my predicament...usually encapsulation is a good thing making an API easier to program against and less error prone, however when using Mocks (EasyMock in my case) and you want to test, for example a "void" method and ensure that "fileInputStream.close()" gets called in the implementation code, do you expose a "setFileInputStream()" method on the interface or use a partial mock and keep this functionality encapsulated inside a method called, let's say "createInputStream()" which can be partially mocked???
One of the things I really like about following a TDD paradigm with a mock framework such as EasyMock is that it lets you easily test "void" methods and ensure important functionality is actually implemented (for instance inputStream.close()). I just wanted to get others thoughts on "encapsulation vs. ease of testing"...
Karim,
ReplyDeleteI guess all we want to say is that partial mock is a warning that the design could be better. Your example describes inputStream.close() as something that can be nicely tested with partial mocks. Say I really need to make sure I never forget to call close(). I would prefer to design the code in a way I cannot forget it. For example (not sure if it makes sense because I'm just making it up :)
interface InputStreamHandler {
Object handle(InputStream is);
}
class ClosingHandler implements InputStreamHandler {
public ClosingHandler(InputStreamHandler ish) {}
public Object handle(InputStream is) {
//ish.handle & is.close
}
}