Search code examples
groovyjmockhamcrest

Using JMock2 with Matchers in Groovy: problems with 'with' method


I have a following snippet with JMock expectations in my test method:

context.checking(new Expectations() {{
    allowing(listener).tableChanged(with(anyInsertionEvent()));
    oneOf(listener).tableChanged(with(aRowChangedEvent(0)));
}});

where anyInsertionEvent and aRowChangedEventAs return instances of Matcher<TableModelEvent>. This is taken from book "Growing Object-Oriented Software Guided by Tests" (p. 181).

I try to convert this test to Groovy, but the method I need:

org.jmock.Expectations.with(org.hamcrest.Matcher<T>)

is shadowed by:

org.codehaus.groovy.runtime.DefaultGroovyMethods.with(java.lang.Object, 
    groovy.lang.Closure<T>)

As a result I get an error during my tests like:

groovy.lang.MissingMethodException: No signature of method: $Proxy8.tableChanged() is     applicable for argument types: (java.lang.Boolean) values: [false]
Possible solutions: tableChanged(javax.swing.event.TableModelEvent)
    at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.unwrap(ScriptBytecodeAdapter.java:55)
    at org.codehaus.groovy.runtime.callsite.PojoMetaClassSite.call(PojoMetaClassSite.java:46)
    at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:45)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:108)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:116)
    at org.rorick.auctionsniper.ui.SnipersTableModelTest$1.<init>(SnipersTableModelTest.groovy:43)
    at org.rorick.auctionsniper.ui.SnipersTableModelTest.setSniperValuesInColumns(SnipersTableModelTest.groovy:42)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at org.junit.internal.runners.TestMethod.invoke(TestMethod.java:66)
    at org.jmock.integration.junit4.JMock$1.invoke(JMock.java:37)
    at org.junit.internal.runners.MethodRoadie.runTestMethod(MethodRoadie.java:105)
    at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.java:86)
    at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:94)
    at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:84)
    at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:49)
    at org.junit.internal.runners.JUnit4ClassRunner.invokeTestMethod(JUnit4ClassRunner.java:98)
    at org.junit.internal.runners.JUnit4ClassRunner.runMethods(JUnit4ClassRunner.java:61)
    at org.junit.internal.runners.JUnit4ClassRunner$1.run(JUnit4ClassRunner.java:54)
    at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:34)
    at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:44)
    at org.junit.internal.runners.JUnit4ClassRunner.run(JUnit4ClassRunner.java:52)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:157)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:76)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:195)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:63)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120)

How can I use appropriate with method? Or, please, advice any other way to resolve this issue.

UPDATE: it is not shadowing. The method actually called is Expectations.with(Matcher<Boolean>) hence false value in stack-trace. So, method is incorrectly dispatched. Any ideas what to do with that?

UPDATE: Matcher method are the following:

public Matcher<TableModelEvent> anyInsertionEvent() {
    Matcher<Integer> insertMatcher = equalTo(TableModelEvent.INSERT);
    return new FeatureMatcher<TableModelEvent, Integer>(insertMatcher, "is an insertion event", "event type") {
        @Override
        protected Integer featureValueOf(TableModelEvent actual) {
            return actual.getType();
        }
    };
}

private Matcher<TableModelEvent> aRowChangedEvent(int row) {
    return samePropertyValuesAs(new TableModelEvent(model, row));
}

Solution

  • Finally, I found a workaround for this. See the checking code below. new Expectations() {{}} Java syntax has gone and is replaced with closure:

    context.checking {
        allowing(listener).tableChanged(argThat(anyInsertionEvent()));
        oneOf(listener).tableChanged(argThat(aRowChangedEvent(0)));
    }
    

    Note that with is replaced with argThat (modeled after Mockito). context is not an instance of org.jmock.integration.junit4.JUnit4Mockery as it were, but is an instance of the following class:

    class JUnit4GroovyMockery extends JUnit4Mockery {
        public void checking(Closure c) {
             ClosureExpectations expectations = new ClosureExpectations();
             expectations.closureInit(c, expectations);
             super.checking(expectations);
        }
    
        private static class ClosureExpectations extends Expectations {
            void closureInit(Closure cl, Object delegate) {
                cl.setDelegate(delegate);
                cl.call();
            }
    
            public Object argThat(Matcher<?> matcher) {
                currentBuilder().addParameterMatcher(matcher);
                return null;
            }
        }
    }
    

    This is based on JUnit4GroovyMockery class from http://docs.codehaus.org/display/GROOVY/Using+JMock+with+Groovy. argThat is copy-pasted from Expectations.with(Matcher<T>) and Expectations.addParameterMatcher(Matcher<?>).

    This works for me, though I'm not very pleased with this solution; and I'd prefer to keep with name for the method.