/*
* 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.activemq.transport.amqp;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import static org.junit.Assume.assumeFalse;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import javax.jms.BytesMessage;
import javax.jms.Connection;
import javax.jms.Destination;
import javax.jms.MapMessage;
import javax.jms.Message;
import javax.jms.MessageConsumer;
import javax.jms.MessageProducer;
import javax.jms.ObjectMessage;
import javax.jms.Session;
import javax.jms.TextMessage;
import org.apache.activemq.ActiveMQConnection;
import org.apache.qpid.proton.amqp.Binary;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Tests interoperability between OpenWire and AMQP
*/
@RunWith(Parameterized.class)
public class JMSInteroperabilityTest extends JMSClientTestSupport {
protected static final Logger LOG = LoggerFactory.getLogger(JMSInteroperabilityTest.class);
private final String transformer;
@Parameters(name="Transformer->{0}")
public static Collection<Object[]> data() {
return Arrays.asList(new Object[][] {
{"jms"},
{"native"},
{"raw"},
});
}
public JMSInteroperabilityTest(String transformer) {
this.transformer = transformer;
}
@Override
protected boolean isUseOpenWireConnector() {
return true;
}
@Override
protected String getAmqpTransformer() {
return transformer;
}
//----- Tests for property handling between protocols --------------------//
@SuppressWarnings("unchecked")
@Test(timeout = 60000)
public void testMessagePropertiesArePreservedOpenWireToAMQP() throws Exception {
boolean bool = true;
byte bValue = 127;
short nShort = 10;
int nInt = 5;
long nLong = 333;
float nFloat = 1;
double nDouble = 100;
Enumeration<String> propertyNames = null;
String testMessageBody = "Testing msgPropertyExistTest";
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
TextMessage outbound = openwireSession.createTextMessage();
outbound.setText(testMessageBody);
outbound.setBooleanProperty("Boolean", bool);
outbound.setByteProperty("Byte", bValue);
outbound.setShortProperty("Short", nShort);
outbound.setIntProperty("Integer", nInt);
outbound.setFloatProperty("Float", nFloat);
outbound.setDoubleProperty("Double", nDouble);
outbound.setStringProperty("String", "test");
outbound.setLongProperty("Long", nLong);
outbound.setObjectProperty("BooleanObject", Boolean.valueOf(bool));
openwireProducer.send(outbound);
Message inbound = amqpConsumer.receive(2500);
propertyNames = inbound.getPropertyNames();
int propertyCount = 0;
do {
String propertyName = propertyNames.nextElement();
if (propertyName.indexOf("JMS") != 0) {
propertyCount++;
if (propertyName.equals("Boolean") || propertyName.equals("Byte") ||
propertyName.equals("Integer") || propertyName.equals("Short") ||
propertyName.equals("Float") || propertyName.equals("Double") ||
propertyName.equals("String") || propertyName.equals("Long") ||
propertyName.equals("BooleanObject")) {
LOG.debug("Appclication Property set by client is: {}", propertyName);
if (!inbound.propertyExists(propertyName)) {
assertTrue(inbound.propertyExists(propertyName));
LOG.debug("Positive propertyExists test failed for {}", propertyName);
} else if (inbound.propertyExists(propertyName + "1")) {
LOG.debug("Negative propertyExists test failed for {} 1", propertyName);
fail("Negative propertyExists test failed for " + propertyName + "1");
}
} else {
LOG.debug("Appclication Property not set by client: {}", propertyName);
fail("Appclication Property not set by client: " + propertyName);
}
} else {
LOG.debug("JMSProperty Name is: {}", propertyName);
}
} while (propertyNames.hasMoreElements());
amqp.close();
openwire.close();
assertEquals("Unexpected number of properties in received message.", 9, propertyCount);
}
@SuppressWarnings("unchecked")
@Test(timeout = 60000)
public void testMessagePropertiesArePreservedAMQPToOpenWire() throws Exception {
// Raw Transformer doesn't expand message properties.
assumeFalse(transformer.equals("raw"));
boolean bool = true;
byte bValue = 127;
short nShort = 10;
int nInt = 5;
long nLong = 333;
float nFloat = 1;
double nDouble = 100;
Enumeration<String> propertyNames = null;
String testMessageBody = "Testing msgPropertyExistTest";
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer amqpProducer = amqpSession.createProducer(queue);
MessageConsumer openwireConsumer = openwireSession.createConsumer(queue);
TextMessage outbound = amqpSession.createTextMessage();
outbound.setText(testMessageBody);
outbound.setBooleanProperty("Boolean", bool);
outbound.setByteProperty("Byte", bValue);
outbound.setShortProperty("Short", nShort);
outbound.setIntProperty("Integer", nInt);
outbound.setFloatProperty("Float", nFloat);
outbound.setDoubleProperty("Double", nDouble);
outbound.setStringProperty("String", "test");
outbound.setLongProperty("Long", nLong);
outbound.setObjectProperty("BooleanObject", Boolean.valueOf(bool));
amqpProducer.send(outbound);
Message inbound = openwireConsumer.receive(2500);
propertyNames = inbound.getPropertyNames();
int propertyCount = 0;
do {
String propertyName = propertyNames.nextElement();
if (propertyName.indexOf("JMS") != 0) {
propertyCount++;
if (propertyName.equals("Boolean") || propertyName.equals("Byte") ||
propertyName.equals("Integer") || propertyName.equals("Short") ||
propertyName.equals("Float") || propertyName.equals("Double") ||
propertyName.equals("String") || propertyName.equals("Long") ||
propertyName.equals("BooleanObject")) {
LOG.debug("Appclication Property set by client is: {}", propertyName);
if (!inbound.propertyExists(propertyName)) {
assertTrue(inbound.propertyExists(propertyName));
LOG.debug("Positive propertyExists test failed for {}", propertyName);
} else if (inbound.propertyExists(propertyName + "1")) {
LOG.debug("Negative propertyExists test failed for {} 1", propertyName);
fail("Negative propertyExists test failed for " + propertyName + "1");
}
} else {
LOG.debug("Appclication Property not set by client: {}", propertyName);
fail("Appclication Property not set by client: " + propertyName);
}
} else {
LOG.debug("JMSProperty Name is: {}", propertyName);
}
} while (propertyNames.hasMoreElements());
amqp.close();
openwire.close();
assertEquals("Unexpected number of properties in received message.", 9, propertyCount);
}
//----- Tests for OpenWire to Qpid JMS using MapMessage ------------------//
@SuppressWarnings("unchecked")
@Test
public void testMapMessageUsingPrimitiveSettersSendReceive() throws Exception {
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
byte[] bytesValue = new byte[] { 1, 2, 3, 4, 5 };
// Create the Message
MapMessage outgoing = openwireSession.createMapMessage();
outgoing.setBoolean("boolean", true);
outgoing.setByte("byte", (byte) 10);
outgoing.setBytes("bytes", bytesValue);
outgoing.setChar("char", 'B');
outgoing.setDouble("double", 24.42);
outgoing.setFloat("float", 3.14159f);
outgoing.setInt("integer", 1024);
outgoing.setLong("long", 8096l);
outgoing.setShort("short", (short) 255);
openwireProducer.send(outgoing);
// Now consume the MapMessage
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
assertTrue("Expected MapMessage but got " + received, received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Map<String, Object> incomingMap = (Map<String, Object>) incoming.getObject();
assertEquals(true, incomingMap.get("boolean"));
assertEquals(10, (byte) incomingMap.get("byte"));
assertEquals('B', incomingMap.get("char"));
assertEquals(24.42, (double) incomingMap.get("double"), 0.5);
assertEquals(3.14159f, (float) incomingMap.get("float"), 0.5f);
assertEquals(1024, incomingMap.get("integer"));
assertEquals(8096l, incomingMap.get("long"));
assertEquals(255, (short) incomingMap.get("short"));
// Test for the byte array which will be in an AMQP Binary as this message
// is received as an ObjectMessage by Qpid JMS
Object incomingValue = incomingMap.get("bytes");
assertNotNull(incomingValue);
assertTrue(incomingValue instanceof Binary);
Binary incomingBinary = (Binary) incomingValue;
byte[] incomingBytes = Arrays.copyOfRange(incomingBinary.getArray(), incomingBinary.getArrayOffset(), incomingBinary.getLength());
assertTrue(Arrays.equals(bytesValue, incomingBytes));
amqp.close();
openwire.close();
}
//----- Tests for OpenWire <-> Qpid JMS using ObjectMessage --------------//
@SuppressWarnings("unchecked")
@Test
public void testMapInObjectMessageSendReceive() throws Exception {
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
// Create the Message
ObjectMessage outgoing = openwireSession.createObjectMessage();
HashMap<String, Object> outgoingMap = new HashMap<String, Object>();
outgoingMap.put("none", null);
outgoingMap.put("string", "test");
outgoingMap.put("long", 255L);
outgoingMap.put("empty-string", "");
outgoingMap.put("negative-int", -1);
outgoingMap.put("float", 0.12f);
outgoing.setObject(outgoingMap);
openwireProducer.send(outgoing);
// Now consume the ObjectMessage
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
assertTrue("Expected ObjectMessage but got " + received, received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Object incomingObject = incoming.getObject();
assertNotNull(incomingObject);
assertTrue(incomingObject instanceof Map);
Map<String, Object> incomingMap = (Map<String, Object>) incomingObject;
assertEquals(outgoingMap.size(), incomingMap.size());
amqp.close();
openwire.close();
}
@Test
public void testQpidToOpenWireObjectMessage() throws Exception {
// Raw Transformer doesn't expand message properties.
assumeFalse(!transformer.equals("jms"));
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer amqpProducer = amqpSession.createProducer(queue);
MessageConsumer openwireConsumer = openwireSession.createConsumer(queue);
// Create and send the Message
ObjectMessage outgoing = amqpSession.createObjectMessage();
outgoing.setObject(UUID.randomUUID());
amqpProducer.send(outgoing);
// Now consume the ObjectMessage
Message received = openwireConsumer.receive(2000);
assertNotNull(received);
LOG.info("Read new message: {}", received);
assertTrue(received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Object payload = incoming.getObject();
assertNotNull(payload);
assertTrue(payload instanceof UUID);
amqp.close();
openwire.close();
}
@Test
public void testOpenWireToQpidObjectMessage() throws Exception {
// Raw Transformer doesn't expand message properties.
assumeFalse(!transformer.equals("jms"));
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
// Create and send the Message
ObjectMessage outgoing = amqpSession.createObjectMessage();
outgoing.setObject(UUID.randomUUID());
openwireProducer.send(outgoing);
// Now consume the ObjectMessage
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
LOG.info("Read new message: {}", received);
assertTrue(received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Object payload = incoming.getObject();
assertNotNull(payload);
assertTrue(payload instanceof UUID);
amqp.close();
openwire.close();
}
@Test
public void testOpenWireToQpidObjectMessageWithOpenWireCompression() throws Exception {
// Raw Transformer doesn't expand message properties.
assumeFalse(!transformer.equals("jms"));
Connection openwire = createJMSConnection();
((ActiveMQConnection) openwire).setUseCompression(true);
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
// Create and send the Message
ObjectMessage outgoing = amqpSession.createObjectMessage();
outgoing.setObject(UUID.randomUUID());
openwireProducer.send(outgoing);
// Now consume the ObjectMessage
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
LOG.info("Read new message: {}", received);
assertTrue(received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Object payload = incoming.getObject();
assertNotNull(payload);
assertTrue(payload instanceof UUID);
amqp.close();
openwire.close();
}
@SuppressWarnings("unchecked")
@Test
public void testObjectMessageContainingList() throws Exception {
Connection openwire = createJMSConnection();
Connection amqp = createConnection();
openwire.start();
amqp.start();
Session openwireSession = openwire.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session amqpSession = amqp.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = openwireSession.createQueue(getDestinationName());
MessageProducer openwireProducer = openwireSession.createProducer(queue);
MessageConsumer amqpConsumer = amqpSession.createConsumer(queue);
// Create the Message
ObjectMessage outgoing = openwireSession.createObjectMessage();
ArrayList<Object> outgoingList = new ArrayList<Object>();
outgoingList.add(null);
outgoingList.add("test");
outgoingList.add(255L);
outgoingList.add("");
outgoingList.add(-1);
outgoingList.add(0.12f);
outgoing.setObject(outgoingList);
openwireProducer.send(outgoing);
// Now consume the ObjectMessage
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
assertTrue(received instanceof ObjectMessage);
ObjectMessage incoming = (ObjectMessage) received;
Object incomingObject = incoming.getObject();
assertNotNull(incomingObject);
assertTrue(incomingObject instanceof List);
List<Object> incomingList = (List<Object>) incomingObject;
assertEquals(outgoingList.size(), incomingList.size());
amqp.close();
openwire.close();
}
//----- Test Qpid JMS to Qpid JMS interop with transformers --------------//
@Test
public void testQpidJMSToQpidJMSMessageSendReceive() throws Exception {
final int SIZE = 1024;
final int NUM_MESSAGES = 100;
Connection amqpSend = createConnection("client-1");
Connection amqpReceive = createConnection("client-2");
amqpReceive.start();
Session senderSession = amqpSend.createSession(false, Session.AUTO_ACKNOWLEDGE);
Session receiverSession = amqpReceive.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination queue = senderSession.createQueue(getDestinationName());
MessageProducer amqpProducer = senderSession.createProducer(queue);
MessageConsumer amqpConsumer = receiverSession.createConsumer(queue);
byte[] payload = new byte[SIZE];
for (int i = 0; i < NUM_MESSAGES; ++i) {
BytesMessage outgoing = senderSession.createBytesMessage();
outgoing.setLongProperty("SendTime", System.currentTimeMillis());
outgoing.writeBytes(payload);
amqpProducer.send(outgoing);
}
// Now consume the message
for (int i = 0; i < NUM_MESSAGES; ++i) {
Message received = amqpConsumer.receive(2000);
assertNotNull(received);
assertTrue("Expected BytesMessage but got " + received, received instanceof BytesMessage);
BytesMessage incoming = (BytesMessage) received;
assertEquals(SIZE, incoming.getBodyLength());
}
amqpReceive.close();
amqpSend.close();
}
}