Как смоделировать DefaultMessageListenerContainer - PullRequest
4 голосов
/ 30 января 2012

Я борюсь с издевательством (используя Mockito) DefaultMessageListenerContainer (org.springframework.jms.listener.DefaultMessageListenerContainer).Вот мой код:

@Mock
private DefaultMessageListenerContainer defaultMessageListenerContainer;

@Before
public void init() {

    MockitoAnnotations.initMocks( this );
    incomingFeedController = new IncomingFeedControllerImpl();

}

@Test 
public void testHandleConnectionState() { 
    List< DefaultMessageListenerContainer > listeners = 
        new ArrayList< DefaultMessageListenerContainer >(); 
    listeners.add( defaultMessageListenerContainer ); 
    incomingFeedController.setContainers( listeners ); 
    when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );
}

Затем я хотел бы сделать несколько правильных тестов, таких как:

when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );

Но после запуска junit эта строка в итоге будет иметь вид:

java.lang.NullPointerException
    at org.springframework.jms.listener.AbstractJmsListeningContainer.isRunning(AbstractJmsListeningContainer.java:312)
    at com.source.etf.manager.integrationgateway.feedcontroller.IncomingFeedControllerTest.testHandleConnectionState(IncomingFeedControllerTest.java:37)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.junit.internal.runners.TestMethod.invoke(TestMethod.java:59)
    at org.junit.internal.runners.MethodRoadie.runTestMethod(MethodRoadie.java:98)
    at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.java:79)
    at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:87)
    at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:77)
    at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:42)
    at org.junit.internal.runners.JUnit4ClassRunner.invokeTestMethod(JUnit4ClassRunner.java:88)
    at org.junit.internal.runners.JUnit4ClassRunner.runMethods(JUnit4ClassRunner.java:51)
    at org.junit.internal.runners.JUnit4ClassRunner$1.run(JUnit4ClassRunner.java:44)
    at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:27)
    at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:37)
    at org.junit.internal.runners.JUnit4ClassRunner.run(JUnit4ClassRunner.java:42)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

Я также проверил AbstractJmsListeningContainer, и вот код, где происходит NPE:

public final boolean isRunning() {
    synchronized (this.lifecycleMonitor) {
        return (this.running && runningAllowed());
    }
}

Я обнаружил, что объект lifecycleMonitor не создается.Этот объект объявлен в AbstractJmsListeningContainer только в верхней части:

protected final Object lifecycleMonitor = new Object();

Есть идеи, как правильно издеваться DefaultMessageListenerContainer?

Ответы [ 4 ]

1 голос
/ 30 января 2012

Мокито не может издеваться над финальными классами или финальными методами. Эти ограничения применяются самой JVM. Пересмешивание такого кода потребовало бы переписать байт-код класса и загрузить его в другой загрузчик классов. Это приводит к очень сложному коду; PowerMock пошла в этом направлении, и код трудно поддерживать.

Также не смейтесь над типами, которыми вы не владеете, см. Первые 4-5 результатов на google . Зачем вам нужно издеваться над типами Spring в модульном тесте. Вы должны либо создать некоторую косвенность, чтобы избежать присоединения к Spring, либо написать интеграционные тесты; последнее кажется более подходящим, поскольку оно связано с JMS.

1 голос
/ 30 января 2012

Вам все еще нужно установить макет на объект, который вы инициализируете.

Я вижу, что вы тестируете IncomingFeedControllerImpl, и, вероятно, ваш смоделированный объект является членом экземпляра этого класса. Поскольку вы не устанавливаете свой смоделированный DefaultMessageListenerContainer на свой IncomingFeedControllerImpl явно, AbstractJmsListeningContainer (который, вероятно, @Autowired) все еще висит и не подвергается насмешкам.

Вам нужно будет либо внедрить его, используя метод установки или конструктор. (вы также можете @Autowired это)

0 голосов
/ 01 июля 2016
private DefaultMessageListenerContainer createFailingContainer() {
    DefaultMessageListenerContainer container = new DefaultMessageListenerContainer() {
        @Override
        public void start() throws JmsException {
            throw new MyJmsException("TEST");
        }

        @Override
        public void stop() throws JmsException {
            throw new MyJmsException("TEST");
        }
    };
    return container;
}

class MyJmsException extends JmsException {
    private static final long serialVersionUID = 1L;

    public MyJmsException(String msg) {
        super(msg);
    }
}
0 голосов
/ 31 января 2012

Вам нужно будет либо использовать PowerMock, либо выполнить интеграционный тест, как рекомендует @Brice. Вот как это сделать в PowerMock:

@RunWith(PowerMockRunner.class)
@PrepareForTest(DefaultMessageListenerContainer.class)
public class MyTestClass {

    // cannot use the @Mock annotation
    private DefaultMessageListenerContainer defaultMessageListenerContainer;

    @Before
    public void init() {
        // This should allow the mocking of final methods as well.
        defaultMessageListenerContainer = 
            PowerMockito.mock(DefaultMessageListenerContainer.class);
        incomingFeedController = new IncomingFeedControllerImpl();

    }

    @Test 
    public void testHandleConnectionState() { 
        List< DefaultMessageListenerContainer > listeners = 
            new ArrayList< DefaultMessageListenerContainer >(); 
        listeners.add( defaultMessageListenerContainer ); 
        incomingFeedController.setContainers( listeners ); 
        when( defaultMessageListenerContainer.isRunning() ).thenReturn( false );
    }

}
...