Runtime resolution of JMS reply destination
Add JmsResponse that can be used as return type of any JMS listener method to indicate not only the response but also the actual destination to which the reply should be sent. Issue: SPR-13133
This commit is contained in:
@@ -0,0 +1,84 @@
|
||||
/*
|
||||
* Copyright 2002-2015 the original author or authors.
|
||||
*
|
||||
* Licensed 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.springframework.jms.listener.adapter;
|
||||
|
||||
import javax.jms.Destination;
|
||||
import javax.jms.JMSException;
|
||||
import javax.jms.Session;
|
||||
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.rules.ExpectedException;
|
||||
|
||||
import org.springframework.jms.support.destination.DestinationResolver;
|
||||
|
||||
import static org.junit.Assert.assertSame;
|
||||
import static org.mockito.BDDMockito.given;
|
||||
import static org.mockito.Mockito.mock;
|
||||
|
||||
/**
|
||||
* @author Stephane Nicoll
|
||||
*/
|
||||
public class JmsResponseTests {
|
||||
|
||||
@Rule
|
||||
public final ExpectedException thrown = ExpectedException.none();
|
||||
|
||||
@Test
|
||||
public void destinationDoesNotUseDestinationResolver() throws JMSException {
|
||||
Destination destination = mock(Destination.class);
|
||||
Destination actual = JmsResponse.forDestination("foo", destination).resolveDestination(null, null);
|
||||
assertSame(destination, actual);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void resolveDestinationForQueue() throws JMSException {
|
||||
Session session = mock(Session.class);
|
||||
DestinationResolver destinationResolver = mock(DestinationResolver.class);
|
||||
Destination destination = mock(Destination.class);
|
||||
|
||||
given(destinationResolver.resolveDestinationName(session, "myQueue", false)).willReturn(destination);
|
||||
JmsResponse jmsResponse = JmsResponse.forQueue("foo", "myQueue");
|
||||
Destination actual = jmsResponse.resolveDestination(destinationResolver, session);
|
||||
assertSame(destination, actual);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void createWithNulResponse() {
|
||||
thrown.expect(IllegalArgumentException.class);
|
||||
JmsResponse.forQueue(null, "myQueue");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void createWithNullQueueName() {
|
||||
thrown.expect(IllegalArgumentException.class);
|
||||
JmsResponse.forQueue("foo", null);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void createWithNullTopicName() {
|
||||
thrown.expect(IllegalArgumentException.class);
|
||||
JmsResponse.forTopic("foo", null);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void createWithNulDestination() {
|
||||
thrown.expect(IllegalArgumentException.class);
|
||||
JmsResponse.forDestination("foo", null);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -21,8 +21,11 @@ import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import javax.jms.Destination;
|
||||
import javax.jms.JMSException;
|
||||
import javax.jms.MessageProducer;
|
||||
import javax.jms.Queue;
|
||||
import javax.jms.Session;
|
||||
import javax.jms.TextMessage;
|
||||
import javax.jms.Topic;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
@@ -146,6 +149,97 @@ public class MessagingMessageListenerAdapterTests {
|
||||
assertEquals("Response", ((TextMessage) replyMessage).getText());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void replyPayloadToQueue() throws JMSException {
|
||||
Message<String> request = MessageBuilder.withPayload("Response").build();
|
||||
|
||||
Session session = mock(Session.class);
|
||||
Queue replyDestination = mock(Queue.class);
|
||||
given(session.createQueue("queueOut")).willReturn(replyDestination);
|
||||
|
||||
|
||||
MessageProducer messageProducer = mock(MessageProducer.class);
|
||||
TextMessage responseMessage = mock(TextMessage.class);
|
||||
given(session.createTextMessage("Response")).willReturn(responseMessage);
|
||||
given(session.createProducer(replyDestination)).willReturn(messageProducer);
|
||||
|
||||
MessagingMessageListenerAdapter listener = getPayloadInstance(request, "replyPayloadToQueue", Message.class);
|
||||
listener.onMessage(mock(javax.jms.Message.class), session);
|
||||
|
||||
|
||||
verify(session).createQueue("queueOut");
|
||||
verify(session).createTextMessage("Response");
|
||||
verify(messageProducer).send(responseMessage);
|
||||
verify(messageProducer).close();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void replyPayloadToTopic() throws JMSException {
|
||||
Message<String> request = MessageBuilder.withPayload("Response").build();
|
||||
|
||||
Session session = mock(Session.class);
|
||||
Topic replyDestination = mock(Topic.class);
|
||||
given(session.createTopic("topicOut")).willReturn(replyDestination);
|
||||
|
||||
|
||||
MessageProducer messageProducer = mock(MessageProducer.class);
|
||||
TextMessage responseMessage = mock(TextMessage.class);
|
||||
given(session.createTextMessage("Response")).willReturn(responseMessage);
|
||||
given(session.createProducer(replyDestination)).willReturn(messageProducer);
|
||||
|
||||
MessagingMessageListenerAdapter listener = getPayloadInstance(request, "replyPayloadToTopic", Message.class);
|
||||
listener.onMessage(mock(javax.jms.Message.class), session);
|
||||
|
||||
|
||||
verify(session).createTopic("topicOut");
|
||||
verify(session).createTextMessage("Response");
|
||||
verify(messageProducer).send(responseMessage);
|
||||
verify(messageProducer).close();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void replyPayloadToDestination() throws JMSException {
|
||||
Queue replyDestination = mock(Queue.class);
|
||||
Message<String> request = MessageBuilder.withPayload("Response")
|
||||
.setHeader("destination", replyDestination).build();
|
||||
|
||||
Session session = mock(Session.class);
|
||||
MessageProducer messageProducer = mock(MessageProducer.class);
|
||||
TextMessage responseMessage = mock(TextMessage.class);
|
||||
given(session.createTextMessage("Response")).willReturn(responseMessage);
|
||||
given(session.createProducer(replyDestination)).willReturn(messageProducer);
|
||||
|
||||
MessagingMessageListenerAdapter listener = getPayloadInstance(request, "replyPayloadToDestination", Message.class);
|
||||
listener.onMessage(mock(javax.jms.Message.class), session);
|
||||
|
||||
verify(session, times(0)).createQueue(anyString());
|
||||
verify(session).createTextMessage("Response");
|
||||
verify(messageProducer).send(responseMessage);
|
||||
verify(messageProducer).close();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void replyPayloadNoDestination() throws JMSException {
|
||||
Queue replyDestination = mock(Queue.class);
|
||||
Message<String> request = MessageBuilder.withPayload("Response").build();
|
||||
|
||||
Session session = mock(Session.class);
|
||||
MessageProducer messageProducer = mock(MessageProducer.class);
|
||||
TextMessage responseMessage = mock(TextMessage.class);
|
||||
given(session.createTextMessage("Response")).willReturn(responseMessage);
|
||||
given(session.createProducer(replyDestination)).willReturn(messageProducer);
|
||||
|
||||
MessagingMessageListenerAdapter listener =
|
||||
getPayloadInstance(request, "replyPayloadNoDestination", Message.class);
|
||||
listener.setDefaultResponseDestination(replyDestination);
|
||||
listener.onMessage(mock(javax.jms.Message.class), session);
|
||||
|
||||
verify(session, times(0)).createQueue(anyString());
|
||||
verify(session).createTextMessage("Response");
|
||||
verify(messageProducer).send(responseMessage);
|
||||
verify(messageProducer).close();
|
||||
}
|
||||
|
||||
protected MessagingMessageListenerAdapter getSimpleInstance(String methodName, Class... parameterTypes) {
|
||||
Method m = ReflectionUtils.findMethod(SampleBean.class, methodName, parameterTypes);
|
||||
return createInstance(m);
|
||||
@@ -157,6 +251,19 @@ public class MessagingMessageListenerAdapterTests {
|
||||
return adapter;
|
||||
}
|
||||
|
||||
protected MessagingMessageListenerAdapter getPayloadInstance(final Object payload,
|
||||
String methodName, Class... parameterTypes) {
|
||||
Method m = ReflectionUtils.findMethod(SampleBean.class, methodName, parameterTypes);
|
||||
MessagingMessageListenerAdapter adapter = new MessagingMessageListenerAdapter() {
|
||||
@Override
|
||||
protected Object extractMessage(javax.jms.Message message) {
|
||||
return payload;
|
||||
}
|
||||
};
|
||||
adapter.setHandlerMethod(factory.createInvocableHandlerMethod(sample, m));
|
||||
return adapter;
|
||||
}
|
||||
|
||||
private void initializeFactory(DefaultMessageHandlerMethodFactory factory) {
|
||||
factory.setBeanFactory(new StaticListableBeanFactory());
|
||||
factory.afterPropertiesSet();
|
||||
@@ -178,6 +285,23 @@ public class MessagingMessageListenerAdapterTests {
|
||||
.build();
|
||||
}
|
||||
|
||||
public JmsResponse replyPayloadToQueue(Message<String> input) {
|
||||
return JmsResponse.forQueue(input.getPayload(), "queueOut");
|
||||
}
|
||||
|
||||
public JmsResponse replyPayloadToTopic(Message<String> input) {
|
||||
return JmsResponse.forTopic(input.getPayload(), "topicOut");
|
||||
}
|
||||
|
||||
public JmsResponse replyPayloadToDestination(Message<String> input) {
|
||||
return JmsResponse.forDestination(input.getPayload(),
|
||||
input.getHeaders().get("destination", Destination.class));
|
||||
}
|
||||
|
||||
public JmsResponse replyPayloadNoDestination(Message<String> input) {
|
||||
return new JmsResponse(input.getPayload(), null);
|
||||
}
|
||||
|
||||
public void fail(String input) {
|
||||
throw new IllegalArgumentException("Expected test exception");
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user