在测试Camel路由时如何正确使用Mocks?

时间:2015-02-23 19:50:10

标签: apache-camel

我正在尝试编写一个Camel测试,检查以确保基于内容的路由器正确地路由XML文件。以下是我的blueprint.xml中的enpoints和路由:

<endpoint uri="activemq:queue:INPUTQUEUE" id="jms.queue.input" />
<endpoint uri="activemq:queue:QUEUE1" id="jms.queue.1" />
<endpoint uri="activemq:queue:QUEUE2" id="jms.queue.2" />

<route id="general-jms.to.specific-jms">
    <from ref="jms.queue.input" />
    <choice>
      <when>
        <xpath>//object-type = '1'</xpath>
        <log message="Sending message to queue: QUEUE1" />
        <to ref="jms.queue.1" />
      </when>
      <when>
        <xpath>//object-type = '2'</xpath>
        <log message="Sending message to queue: QUEUE2" />
        <to ref="jms.queue.2" />
      </when>
      <otherwise>
        <log message="No output was able to be determined based on the input." />
      </otherwise>
    </choice>
</route>

现在,我要做的就是发送一个<object-type>为1的示例源文件,并验证它是否路由到正确的队列(QUEUE1)并且是正确的数据(应该只是将整个XML文件发送到QUEUE1)。这是我的测试代码:

public class RouteTest extends CamelBlueprintTestSupport {

    @Override
    protected String getBlueprintDescriptor() {
        return "/OSGI-INF/blueprint/blueprint.xml";
    }

    @Override
    public String isMockEndpointsAndSkip() {
        return "activemq:queue:QUEUE1";
    }

    @Test
    public void testQueue1Route() throws Exception {

        getMockEndpoint("mock:activemq:queue:QUEUE1").expectedBodiesReceived(context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue1-test.xml")));

        template.sendBody("activemq:queue:INPUTQUEUE", context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue1-test.xml")));

        assertMockEndpointsSatisfied();
    }
}

当我运行此测试时,我看到我在路由定义中输入的日志消息表明它正在将其发送到QUEUE1,但是JUnit测试失败并显示以下错误消息: java.lang.AssertionError:mock :// activemq:queue:QUEUE1收到消息计数。预期:&lt; 1&gt;但是:&lt; 0&gt;

有人能帮我理解我做错了什么吗?

我的理解是Camel将自动模拟QUEUE1端点,因为我覆盖isMockEndpointsAndSkip()并提供了QUEUE1端点uri。我认为这意味着我应该能够在getMockEnpoint()方法中使用该端点,只需附加&#34; mock:&#34;到了uri的开头。然后我应该有一个模拟的端点,我可以设置预期(即必须有输入文件)。

如果我对某些事情不清楚,请告诉我,非常感谢任何帮助!

2 个答案:

答案 0 :(得分:8)

解决方案是使用CamelTestSupport.replaceRouteFromWith

这个方法完全没有任何文档,但是在调用它时它对我有用:

public class FooTest extends CamelTestSupport {

    @Override
    public void setUp() throws Exception {
        replaceRouteFromWith("route-id", "direct:route-input-replaced");
        super.setUp();
    }

    // other stuff ...

}

这也将阻止启动路线的from目的地的原始消费者。例如,这意味着当要测试具有activemq使用者的路由时,不再需要运行activemq实例。

答案 1 :(得分:0)

经过一段时间的努力之后,我提出的唯一解决方案就是在我的测试类中使用createRouteBuilder()方法在末尾添加一个到模拟端点的路由。我的blueprint.xml文件中定义的路由。然后我可以检查那个模拟端点以达到我的期望。下面是我测试类的最终代码。蓝图XML保持不变。

public class RouteTest extends CamelBlueprintTestSupport {

@Override
protected String getBlueprintDescriptor() {
    return "/OSGI-INF/blueprint/blueprint.xml";
}

@Test
public void testQueue1Route() throws Exception {

    getMockEndpoint("mock:QUEUE1").expectedBodiesReceived(context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue1-test.xml")));

    template.sendBody("activemq:queue:INPUTQUEUE", context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue1-test.xml")));

    assertMockEndpointsSatisfied();
}

@Test
public void testQueue2Route() throws Exception {

    getMockEndpoint("mock:QUEUE2").expectedBodiesReceived(context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue2-test.xml")));

    template.sendBody("activemq:queue:INPUTQUEUE", context.getTypeConverter().convertTo(String.class, new File("src/test/resources/queue2-test.xml")));

    assertMockEndpointsSatisfied();
}

@Override
protected RouteBuilder createRouteBuilder() throws Exception {
    return new RouteBuilder() {
        public void configure() throws Exception {
            from("activemq:queue:QUEUE1").to("mock:QUEUE1");
            from("activemq:queue:QUEUE2").to("mock:QUEUE2");
        }
    };
}

}

虽然此解决方案有效,但我并不完全理解为什么我不能仅使用isMockEndpointsAndSkip()而不必在现有blueprint.xml路由的末尾手动定义新路由。我的理解是,使用isMockEndpointsAndSkip()定义return "*";将为blueprint.xml文件中定义的所有端点注入模拟端点。然后,您可以检查那些模拟端点上的预期。但出于某种原因,这对我不起作用。