TotT: Use EasyMock
Thursday, January 08, 2009
Welcome back! We trust you all had a good holiday season and are ready for more TotTs -- Dave
Most of us are aware that mock and stub objects can make testing easier by isolating the class under test from external dependencies. This goes hand-in-hand with dependency injection. Writing all these classes can be a pain though.
EasyMock provides an alternative. It dynamically implements an interface which records and replays your desired behavior. Let's say you want to model an ATM interface:
It is pretty easy to mock this interface. Still, every mock has to implement all three methods, even if you only need one. You also need a separate mock for each set of inputs. With EasyMock, you can create mocks as you need them, recording and replaying your expectations:
We tell EasyMock to create a dynamic proxy implementing Atm (1), which starts in record mode. Then we record two method calls along with the expected results (2 and 3). The replay() call tells EasyMock to stop recording (4). After that, calls on the object return the set values. If it gets a call it does not expect, it throws an Exception to fail fast. Account now uses the mock as if it were the real thing (5). The verify() method checks to see if the mock actually received all the calls you expect (6). It really is that simple. If we want to simulate failure, we can set up another test to return false from one of the method calls.
EasyMock has lots more capabilities as well. It can throw exceptions. It also can record multiple calls to the same method returning the same or different results. You also can create stub expectations and nice mocks so you don't have to record every expected call. You also can create several mocks, and even nest them to test classes with complex dependencies. Beware, though, this often creates brittle tests, and is a sign the class under test needs refactoring.
Basic EasyMock only mocks interfaces, but there is an EasyMockClassExtension that mocks non-final classes when you really must. See the EasyMock documentation at the link below for details.
Most of us are aware that mock and stub objects can make testing easier by isolating the class under test from external dependencies. This goes hand-in-hand with dependency injection. Writing all these classes can be a pain though.
EasyMock provides an alternative. It dynamically implements an interface which records and replays your desired behavior. Let's say you want to model an ATM interface:
public interface Atm {
boolean enterAccount(String accountNumber);
boolean enterPin(String pin);
boolean enterWithdrawalAmount(int dollars);
}
boolean enterAccount(String accountNumber);
boolean enterPin(String pin);
boolean enterWithdrawalAmount(int dollars);
}
It is pretty easy to mock this interface. Still, every mock has to implement all three methods, even if you only need one. You also need a separate mock for each set of inputs. With EasyMock, you can create mocks as you need them, recording and replaying your expectations:
public void testAtmLogin() {
Atm mockAtm = createMock(Atm.class); // 1
EasyMock.expect(mockAtm.enterAccount("MyAccount")).andReturn(true); // 2
EasyMock.expect(mockAtm.enterPin("1234")).andReturn(true); // 3
EasyMock.replay(mockAtm); // 4
Account account = new Account();
account.login(mockAtm); // 5
assertTrue(account.isLoggedIn());
EasyMock.verify(mockAtm); // 6
}
Atm mockAtm = createMock(Atm.class); // 1
EasyMock.expect(mockAtm.enterAccount("MyAccount")).andReturn(true); // 2
EasyMock.expect(mockAtm.enterPin("1234")).andReturn(true); // 3
EasyMock.replay(mockAtm); // 4
Account account = new Account();
account.login(mockAtm); // 5
assertTrue(account.isLoggedIn());
EasyMock.verify(mockAtm); // 6
}
We tell EasyMock to create a dynamic proxy implementing Atm (1), which starts in record mode. Then we record two method calls along with the expected results (2 and 3). The replay() call tells EasyMock to stop recording (4). After that, calls on the object return the set values. If it gets a call it does not expect, it throws an Exception to fail fast. Account now uses the mock as if it were the real thing (5). The verify() method checks to see if the mock actually received all the calls you expect (6). It really is that simple. If we want to simulate failure, we can set up another test to return false from one of the method calls.
EasyMock has lots more capabilities as well. It can throw exceptions. It also can record multiple calls to the same method returning the same or different results. You also can create stub expectations and nice mocks so you don't have to record every expected call. You also can create several mocks, and even nest them to test classes with complex dependencies. Beware, though, this often creates brittle tests, and is a sign the class under test needs refactoring.
Basic EasyMock only mocks interfaces, but there is an EasyMockClassExtension that mocks non-final classes when you really must. See the EasyMock documentation at the link below for details.
A comparison between easymock and jmock 1/2 would be a good point for future articles from my point of view. I like the simplicity of jmock 1 and easymock, but decided to use jmock 2 at work instead. Though, it might all be personal reference.
ReplyDeleteNice introduction. I'll print it out and put it into our office tomorrow.
Markus, glad you enjoyed it. It's a lot for one page... hard to do much beyond raising awareness or wet the appetite.
ReplyDeletePersonally, I also prefer jMock style mocking.
-Dave
You should also take a look at Mockito.
ReplyDeletehttp://code.google.com/p/mockito/wiki/MockitoVSEasyMock
It removes the need to specify your expectations before running the test and so appears in a more sensible order
// Create the mock
MyClass myClass = mock(MyClass.class);
// Do something with the mock
result = doSomethingWith(myClass);
// Verify that the mock was called correctly
verify(mock).myMethod("Testing"); // The string comes from the doSomethingWith...
+1 for Andy
ReplyDeleteAfter having used JMock, EasyMock and Mockito my favorite Java mocking framework is Mockito: http://mockito.org/
I think Andy is correct. The re-ordering of your expectations is much clearer. I also think the re-ordering is really important for clarity and maintainability I have a few more examples at my blog: http://zdsbs.blogspot.com/2009/01/easymock-vs-mockito.html. But it seems to me Mockito's syntax makes it a better choice for mocking than EasyMock or jMock. But that's just my 2 cents.
ReplyDeleteRe: "See the EasyMock documentation at the link below for details"
ReplyDeleteThere was no link below.
(Disclaimer. I'm an author of JMock)
ReplyDeleteFirst, we (obviously) prefer JMock. The structure is a little arcane but we wrote it that way to make the relationships between the object under test and its neighbours as clear as declarative as possible. JMock is "opinionated", it's designed to push the code in the "right" direction.
EasyMock tends to bury this information in a list of method calls and I find that when working with EasyMock I have to put a lot of effort into making the tests expressive, and interpreting failures can be challenging. I also keep forgetting to call replayMocks().
Mockito, on the other hand, is an interesting challenge to us. It's very easy to use, which makes it a good introduction, but it's forgiving approach means that it doesn't push the design of the code as hard as I like. Talking to the author, it turned out that his initial motivation was to cope with the unpleasantnesses in the code base he was working with. We prefer to improve the code (and the skills of the team).