所以我试图对基于Spring(v4.1.6)的JMS处理代码进行一些集成测试。
这是一个非常标准的Spring设置,使用@JmsListener
注释方法和DefaultMessageListenerContainer
,concurrency
设置为1
,因此只允许1个监听线程。
现在,我利用ActiveMQ的嵌入式代理不依赖任何外部jms代理来随时随地运行测试(我应该在市场营销中工作)。
所以这一切都很好,然后我有我的JUnit测试:
@Test
public void test() {
sendSomeMessage();
//how to wait here for the @JMSListener method to complete
verify();
}
我发送消息,但后来我需要以某种方式等待@JMSListener
注释方法完成。我该怎么做呢?
好吧,我希望我能以某种方式进入Message Driven Pojos生命周期来实现这一目标,但是通过其他关于异步代码的SO问题,我想出了一个基于CountDownLatch
的解决方案
@JMSListener
countDown()
此解决方案的缺点是您必须实际更改您的@JmsListener(destination = "dest", containerFactory = "cf")
public void processMessage(TextMessage message) throws JMSException {
//do the actual processing
actualProcessing(message);
//if there's countDownLatch call the countdown.
if(countDownLatch != null) {
countDownLatch.countDown();
}
}
注释方法专门用于集成测试
为了避免必须更改您的实际@JmsListener方法,您可以尝试在测试中使用AOP ...
首先创建一个像这样的方面类:
@Test
public void test() throws InterruptedException {
//initialize the countDownLatch and set in on the processing class
CountDownLatch countDownLatch = new CountDownLatch(1);
messageProcessor.setCountDownLatch(countDownLatch);
//sendthemessage
sendSomeMessage();
//wait for the processing method to call countdown()
countDownLatch.await();
verify();
}
然后将其添加到您用于测试的应用程序上下文配置中,如下所示:
@JMSListener
如果一切按计划进行,JmsListenerInterceptor将倒计时,您不必更改实际代码。
重要提示:我刚刚发现使用AOP和Mockito来验证@JmsListener中的某些方法是否被调用是一个糟糕的组合。原因似乎是CGLib类的额外包装导致调用错误/实际的目标实例而不是Mockito代理。
在我的测试中,我有一个@Autowired,@ InjectMocks Listener对象和一个@Mock Facade对象,我想验证它是否已经调用了某个方法。
使用AOP:
没有AOP:
这表明你需要按照我尝试的方式注意使用AOP,因为你可能最终在两个线程中都有不同的实例......
如果要将记录添加到@JmsListener带注释的方法,则可以在测试类中执行类似的操作
@Aspect
public static class JmsListenerInterceptor {
@org.aspectj.lang.annotation.After("@annotation(org.springframework.jms.annotation.JmsListener)")
public void afterOnMessage(JoinPoint jp) {
// Do countdown latch stuff...
}
}