Browse Source
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-13133pull/816/merge
4 changed files with 370 additions and 7 deletions
@ -0,0 +1,140 @@
@@ -0,0 +1,140 @@
|
||||
/* |
||||
* 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.springframework.jms.support.destination.DestinationResolver; |
||||
import org.springframework.util.Assert; |
||||
|
||||
/** |
||||
* Return type of any JMS listener method used to indicate the actual response destination |
||||
* alongside the response itself. Typically used when said destination needs to be |
||||
* computed at runtime. |
||||
* <p> |
||||
* The example below sends a response with the content of the {@code result} argument to |
||||
* the {@code queueOut Queue}: |
||||
* |
||||
* <pre class="code"> |
||||
* package com.acme.foo; |
||||
* |
||||
* public class MyService { |
||||
* @JmsListener |
||||
* public JmsResponse process(String msg) { |
||||
* // process incoming message
|
||||
* return JmsResponse.forQueue(result, "queueOut"); |
||||
* } |
||||
* }</pre> |
||||
* |
||||
* If the destination does not need to be computed at runtime, |
||||
* {@link org.springframework.messaging.handler.annotation.SendTo @SendTo} is the |
||||
* recommended declarative approach. |
||||
* |
||||
* @author Stephane Nicoll |
||||
* @since 4.2 |
||||
* @see org.springframework.jms.annotation.JmsListener |
||||
* @see org.springframework.messaging.handler.annotation.SendTo |
||||
*/ |
||||
public class JmsResponse { |
||||
|
||||
private final Object response; |
||||
|
||||
private final Object destination; |
||||
|
||||
/** |
||||
* Create a new instance |
||||
* @param response the content of the result |
||||
* @param destination the destination |
||||
*/ |
||||
protected JmsResponse(Object response, Object destination) { |
||||
Assert.notNull(response, "Result must not be null"); |
||||
this.response = response; |
||||
this.destination = destination; |
||||
} |
||||
|
||||
/** |
||||
* Create a {@link JmsResponse} targeting the queue with the specified name. |
||||
*/ |
||||
public static JmsResponse forQueue(Object result, String queueName) { |
||||
Assert.notNull(queueName, "Queue name must not be null"); |
||||
return new JmsResponse(result, new DestinationNameHolder(queueName, false)); |
||||
} |
||||
|
||||
/** |
||||
* Create a {@link JmsResponse} targeting the topic with the specified name. |
||||
*/ |
||||
public static JmsResponse forTopic(Object result, String topicName) { |
||||
Assert.notNull(topicName, "Topic name must not be null"); |
||||
return new JmsResponse(result, new DestinationNameHolder(topicName, true)); |
||||
} |
||||
|
||||
/** |
||||
* Create a {@link JmsResponse} targeting the specified {@link Destination}. |
||||
*/ |
||||
public static JmsResponse forDestination(Object result, Destination destination) { |
||||
Assert.notNull(destination, "Destination must not be null"); |
||||
return new JmsResponse(result, destination); |
||||
} |
||||
|
||||
|
||||
public Object getResponse() { |
||||
return response; |
||||
} |
||||
|
||||
public Destination resolveDestination(DestinationResolver destinationResolver, Session session) |
||||
throws JMSException { |
||||
|
||||
if (this.destination instanceof Destination) { |
||||
return (Destination) this.destination; |
||||
} |
||||
if (this.destination instanceof DestinationNameHolder) { |
||||
DestinationNameHolder nameHolder = (DestinationNameHolder) this.destination; |
||||
return destinationResolver.resolveDestinationName(session, |
||||
nameHolder.destinationName, nameHolder.pubSubDomain); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
@Override |
||||
public String toString() { |
||||
return "JmsResponse{" + "response=" + this.response + ", destination=" + this.destination + '}'; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* Internal class combining a destination name |
||||
* and its target destination type (queue or topic). |
||||
*/ |
||||
protected static class DestinationNameHolder { |
||||
private final String destinationName; |
||||
|
||||
private final boolean pubSubDomain; |
||||
|
||||
public DestinationNameHolder(String destinationName, boolean pubSubDomain) { |
||||
this.destinationName = destinationName; |
||||
this.pubSubDomain = pubSubDomain; |
||||
} |
||||
|
||||
@Override |
||||
public String toString() { |
||||
return this.destinationName + "{" + "pubSubDomain=" + this.pubSubDomain + '}'; |
||||
} |
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,84 @@
@@ -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); |
||||
} |
||||
|
||||
} |
||||
Loading…
Reference in new issue