/** * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.apache.camel.component.jms; import java.util.HashMap; import java.util.Map; import java.util.concurrent.CountDownLatch; import java.util.concurrent.TimeUnit; import javax.jms.ConnectionFactory; import javax.jms.Destination; import org.apache.camel.CamelContext; import org.apache.camel.Exchange; import org.apache.camel.ExchangePattern; import org.apache.camel.Message; import org.apache.camel.Processor; import org.apache.camel.builder.RouteBuilder; import org.apache.camel.test.junit4.CamelTestSupport; import org.junit.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import static org.apache.camel.component.jms.JmsComponent.jmsComponentAutoAcknowledge; /** * A simple request / late reply test. */ public class JmsSimpleRequestLateReplyTest extends CamelTestSupport { private static final Logger LOG = LoggerFactory.getLogger(JmsSimpleRequestLateReplyTest.class); private static Destination replyDestination; private static String cid; protected String expectedBody = "Late Reply"; protected JmsComponent activeMQComponent; private final CountDownLatch latch = new CountDownLatch(1); @Test public void testRequestLateReplyUsingCustomDestinationHeaderForReply() throws Exception { doTest(new SendLateReply()); } protected void doTest(Runnable runnable) throws InterruptedException { // use another thread to send the late reply to simulate that we do it later, not // from the original route anyway new Thread(runnable).start(); getMockEndpoint("mock:result").expectedMessageCount(1); Object body = template.requestBody(getQueueEndpointName(), "Hello World"); assertMockEndpointsSatisfied(); assertEquals(expectedBody, body); } private class SendLateReply implements Runnable { public void run() { try { LOG.info("Waiting for latch"); latch.await(30, TimeUnit.SECONDS); // wait 1 sec after latch before sending he late replay Thread.sleep(1000); } catch (Exception e) { // ignore } LOG.info("Sending late reply"); // use some dummy queue as we override this with the property: JmsConstants.JMS_DESTINATION Map<String, Object> headers = new HashMap<String, Object>(); headers.put(JmsConstants.JMS_DESTINATION, replyDestination); headers.put("JMSCorrelationID", cid); template.sendBodyAndHeaders("activemq:dummy", expectedBody, headers); } } protected CamelContext createCamelContext() throws Exception { CamelContext camelContext = super.createCamelContext(); ConnectionFactory connectionFactory = CamelJmsTestHelper.createConnectionFactory(); camelContext.addComponent("activemq", jmsComponentAutoAcknowledge(connectionFactory)); activeMQComponent = camelContext.getComponent("activemq", JmsComponent.class); // as this is a unit test I dont want to wait 20 sec before timeout occurs, so we use 10 activeMQComponent.getConfiguration().setRequestTimeout(10000); return camelContext; } protected RouteBuilder createRouteBuilder() throws Exception { return new RouteBuilder() { public void configure() throws Exception { from(getQueueEndpointName()) .process(new Processor() { public void process(Exchange exchange) throws Exception { // set the MEP to InOnly as we are not able to send a reply right now but will do it later // from that other thread exchange.setPattern(ExchangePattern.InOnly); Message in = exchange.getIn(); assertEquals("Hello World", in.getBody()); replyDestination = in.getHeader("JMSReplyTo", Destination.class); cid = in.getHeader("JMSCorrelationID", String.class); LOG.info("ReplyDestination: " + replyDestination); LOG.info("JMSCorrelationID: " + cid); LOG.info("Ahh I cannot send a reply. Someone else must do it."); // signal to the other thread to send back the reply message latch.countDown(); } }) .to("mock:result"); } }; } protected static String getQueueEndpointName() { // need to use a fixed queue for reply as a temp queue may be deleted return "activemq:queue:hello.queue?replyTo=myReplyQueue"; } }