I have a Java class:
import java.util.List;
public class Service
{
public List<Object> someMethod(final List<Object> list) {
return null;
}
}
And a Spock test where I've defined a custom matcher:
import org.mockito.ArgumentMatcher import spock.lang.Specification
import static org.mockito.Mockito.*
class InstantBookingInitialDecisionTest extends Specification {
def mock = mock(Service.class)
def setup() {
when(mock.someMethod(argThat(hasSize(2)))).thenReturn([])
when(mock.someMethod(argThat(hasSize(3)))).thenReturn([])
}
def 'Minimum hunger requirements do not apply to schedulable pros'() {
when:
'something'
then:
'something else'
}
// Damn, there's a Hamcrest matcher for this, but it's not in the jar that the javadocs say it is, so making my own
static def hasSize(size) {
new ArgumentMatcher<List>() {
@Override
boolean matches(Object o) {
List list = (List) o
return list.size() == size
}
}
}
}
As is, this test gives me the following error:
java.lang.NullPointerException: Cannot invoke method size() on null object
If I remove either one of the when
's, I get no error. So what it doesn't like is the stubbing portion of the test, and the fact that I used the custom matcher twice.
Notes:
- I've tried declaring a separate class for each list size, as in mockito anyList of a given size and the Mockito documentation. I get the same error.
- I've tried to use the Hamcrest matcher that looks like this, but despite the fact that the 1.3 Javadocs lists a Matchers.hasSize() method, my imported 1.3 jar does not include Matchers. (But even if I got the dependency resolved, I would still like to understand the problem.)
Please do not ask why I am using Mockito instead of Spock Mocks - I have my reasons. ;)
Thank you
The root cause is that your custom matcher can throw exceptions, which doesn't comply with Matcher's general contract. You're running into it in
when
because of Mockito's internals.Matcher's contract states that
matches(Object)
can accept any Object and return true or false. This means that in every single Matcher implementation, you should make no assumptions about the type of the object passed in, or whether the object is non-null; after all,isNull()
is a perfectly valid and useful matcher. If you want your Matcher to returnfalse
for any null or non-List argument, you should check that manually, or extendTypeSafeMatcher<List>
instead ofBaseMatcher
so that Hamcrest can returnfalse
for you in those cases. Otherwise, you risk an uncaught ClassCastException or NullPointerException, which is what you're getting here. That's the only real problem here, and fixing it will solve your trouble.This is a good moment, though, to explain Mockito's syntax. You don't have trouble with the first line, so why would the second fail? The answer is that when your second line runs:
...then Java evaluates the call to
when
, so it runs:...and then
when
can detectsomeMethod
as the last method called and start its stubbing. Like for all other Mockito matchers, the call toargThat
returnsnull
(keeping its side effect in a stack Mockito can analyze when Java callswhen
later), butsomeMethod
has to have a return value and Mockito can't detect that you're about to callwhen
. This means checking for existing stubs, so it pipes thenull
fromargThat
into your Matcher to see if it should apply your first stub, which causes theNullPointerException
. (I put a little more aboutargThat
's return value and Mockito's evaluation order in another SO answer.)You'll want to fix up your Matcher anyway, but it's also possible for you to have rephrased the second line as follows:
...because the call to
when
beforesomeMethod
means that Mockito can temporarily disarm your stubbing. As long as the first line doesn't throw an exception or call real implementations, though, there's no harm in leaving your syntax aswhen
, and Mockito will handle verification gracefully.