This closes #2299
This commit is contained in:
commit
0f8942b66a
|
@ -18,6 +18,7 @@
|
||||||
package org.apache.activemq.artemis.protocol.amqp.converter;
|
package org.apache.activemq.artemis.protocol.amqp.converter;
|
||||||
|
|
||||||
import static org.apache.activemq.artemis.api.core.FilterConstants.NATIVE_MESSAGE_ID;
|
import static org.apache.activemq.artemis.api.core.FilterConstants.NATIVE_MESSAGE_ID;
|
||||||
|
import static org.apache.activemq.artemis.api.core.Message.EMBEDDED_TYPE;
|
||||||
import static org.apache.activemq.artemis.api.core.Message.HDR_SCHEDULED_DELIVERY_TIME;
|
import static org.apache.activemq.artemis.api.core.Message.HDR_SCHEDULED_DELIVERY_TIME;
|
||||||
import static org.apache.activemq.artemis.protocol.amqp.converter.AMQPMessageSupport.AMQP_DATA;
|
import static org.apache.activemq.artemis.protocol.amqp.converter.AMQPMessageSupport.AMQP_DATA;
|
||||||
import static org.apache.activemq.artemis.protocol.amqp.converter.AMQPMessageSupport.AMQP_NULL;
|
import static org.apache.activemq.artemis.protocol.amqp.converter.AMQPMessageSupport.AMQP_NULL;
|
||||||
|
@ -82,6 +83,7 @@ import org.apache.activemq.artemis.protocol.amqp.exceptions.ActiveMQAMQPIllegalS
|
||||||
import org.apache.activemq.artemis.protocol.amqp.util.NettyWritable;
|
import org.apache.activemq.artemis.protocol.amqp.util.NettyWritable;
|
||||||
import org.apache.activemq.artemis.protocol.amqp.util.TLSEncode;
|
import org.apache.activemq.artemis.protocol.amqp.util.TLSEncode;
|
||||||
import org.apache.activemq.artemis.reader.MessageUtil;
|
import org.apache.activemq.artemis.reader.MessageUtil;
|
||||||
|
import org.apache.activemq.artemis.spi.core.protocol.EmbedMessageUtil;
|
||||||
import org.apache.qpid.proton.amqp.Binary;
|
import org.apache.qpid.proton.amqp.Binary;
|
||||||
import org.apache.qpid.proton.amqp.Symbol;
|
import org.apache.qpid.proton.amqp.Symbol;
|
||||||
import org.apache.qpid.proton.amqp.UnsignedByte;
|
import org.apache.qpid.proton.amqp.UnsignedByte;
|
||||||
|
@ -120,7 +122,13 @@ public class CoreAmqpConverter {
|
||||||
if (coreMessage == null) {
|
if (coreMessage == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
if (coreMessage.isServerMessage() && coreMessage.isLargeMessage() && coreMessage.getType() == EMBEDDED_TYPE) {
|
||||||
|
//large AMQP messages received across cluster nodes
|
||||||
|
final Message message = EmbedMessageUtil.extractEmbedded(coreMessage);
|
||||||
|
if (message instanceof AMQPMessage) {
|
||||||
|
return (AMQPMessage) message;
|
||||||
|
}
|
||||||
|
}
|
||||||
ServerJMSMessage message = ServerJMSMessage.wrapCoreMessage(coreMessage);
|
ServerJMSMessage message = ServerJMSMessage.wrapCoreMessage(coreMessage);
|
||||||
message.decode();
|
message.decode();
|
||||||
|
|
||||||
|
|
|
@ -16,8 +16,10 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.artemis.core.persistence.impl.nullpm;
|
package org.apache.activemq.artemis.core.persistence.impl.nullpm;
|
||||||
|
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQBuffers;
|
import io.netty.buffer.Unpooled;
|
||||||
|
import org.apache.activemq.artemis.api.core.ActiveMQBuffer;
|
||||||
import org.apache.activemq.artemis.api.core.Message;
|
import org.apache.activemq.artemis.api.core.Message;
|
||||||
|
import org.apache.activemq.artemis.core.buffers.impl.ChannelBufferWrapper;
|
||||||
import org.apache.activemq.artemis.core.io.SequentialFile;
|
import org.apache.activemq.artemis.core.io.SequentialFile;
|
||||||
import org.apache.activemq.artemis.core.message.impl.CoreMessage;
|
import org.apache.activemq.artemis.core.message.impl.CoreMessage;
|
||||||
import org.apache.activemq.artemis.core.server.LargeServerMessage;
|
import org.apache.activemq.artemis.core.server.LargeServerMessage;
|
||||||
|
@ -39,13 +41,18 @@ class NullStorageLargeServerMessage extends CoreMessage implements LargeServerMe
|
||||||
@Override
|
@Override
|
||||||
public synchronized void addBytes(final byte[] bytes) {
|
public synchronized void addBytes(final byte[] bytes) {
|
||||||
if (buffer == null) {
|
if (buffer == null) {
|
||||||
buffer = ActiveMQBuffers.dynamicBuffer(bytes.length).byteBuf();
|
buffer = Unpooled.buffer(bytes.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
// expand the buffer
|
// expand the buffer
|
||||||
buffer.writeBytes(bytes);
|
buffer.writeBytes(bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized ActiveMQBuffer getReadOnlyBodyBuffer() {
|
||||||
|
return new ChannelBufferWrapper(buffer.slice(0, buffer.writerIndex()).asReadOnly());
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void deleteFile() throws Exception {
|
public void deleteFile() throws Exception {
|
||||||
// nothing to be done here.. we don really have a file on this Storage
|
// nothing to be done here.. we don really have a file on this Storage
|
||||||
|
|
|
@ -51,9 +51,7 @@ public class EmbedMessageUtil {
|
||||||
ActiveMQBuffer buffer = message.getReadOnlyBodyBuffer();
|
ActiveMQBuffer buffer = message.getReadOnlyBodyBuffer();
|
||||||
|
|
||||||
if (buffer.readableBytes() < signature.length || !checkSignature(buffer)) {
|
if (buffer.readableBytes() < signature.length || !checkSignature(buffer)) {
|
||||||
if (!logger.isTraceEnabled()) {
|
logger.tracef("Message type %d was used for something other than embed messages, ignoring content and treating as a regular message", Message.EMBEDDED_TYPE);
|
||||||
logger.trace("Message type " + Message.EMBEDDED_TYPE + " was used for something other than embed messages, ignoring content and treating as a regular message");
|
|
||||||
}
|
|
||||||
return message;
|
return message;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,165 @@
|
||||||
|
/**
|
||||||
|
* 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.artemis.tests.integration.cluster.distribution;
|
||||||
|
|
||||||
|
import javax.jms.BytesMessage;
|
||||||
|
import javax.jms.Connection;
|
||||||
|
import javax.jms.DeliveryMode;
|
||||||
|
import javax.jms.Destination;
|
||||||
|
import javax.jms.Message;
|
||||||
|
import javax.jms.MessageConsumer;
|
||||||
|
import javax.jms.MessageProducer;
|
||||||
|
import javax.jms.Session;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
|
|
||||||
|
import org.apache.activemq.artemis.api.core.RoutingType;
|
||||||
|
import org.apache.activemq.artemis.core.server.cluster.impl.MessageLoadBalancingType;
|
||||||
|
import org.apache.activemq.artemis.core.settings.impl.AddressSettings;
|
||||||
|
import org.apache.activemq.artemis.protocol.amqp.broker.ProtonProtocolManagerFactory;
|
||||||
|
import org.apache.qpid.jms.JmsConnectionFactory;
|
||||||
|
import org.hamcrest.core.IsInstanceOf;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.Parameterized;
|
||||||
|
|
||||||
|
@RunWith(value = Parameterized.class)
|
||||||
|
public class AMQPLargeMessageClusterTest extends ClusterTestBase {
|
||||||
|
|
||||||
|
private static final int RECEIVE_TIMEOUT_MILLIS = 20_000;
|
||||||
|
private static final int MESSAGE_SIZE = 1024 * 1024;
|
||||||
|
private static final int MESSAGES = 1;
|
||||||
|
|
||||||
|
private final boolean persistenceEnabled;
|
||||||
|
private final boolean compressLargeMessages;
|
||||||
|
|
||||||
|
@Parameterized.Parameters(name = "persistenceEnabled = {0}, compressLargeMessages = {1}")
|
||||||
|
public static Iterable<? extends Object> persistenceEnabled() {
|
||||||
|
return Arrays.asList(new Object[][]{{true, false}, {false, false}, {true, true}, {false, true}});
|
||||||
|
}
|
||||||
|
|
||||||
|
public AMQPLargeMessageClusterTest(boolean persistenceEnabled, boolean compressLargeMessages) {
|
||||||
|
this.persistenceEnabled = persistenceEnabled;
|
||||||
|
this.compressLargeMessages = compressLargeMessages;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
super.setUp();
|
||||||
|
|
||||||
|
start();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void start() throws Exception {
|
||||||
|
setupServers();
|
||||||
|
|
||||||
|
setRedistributionDelay(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected boolean isNetty() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = RECEIVE_TIMEOUT_MILLIS * (MESSAGES + 1))
|
||||||
|
public void testSendReceiveLargeMessage() throws Exception {
|
||||||
|
setupCluster(MessageLoadBalancingType.ON_DEMAND);
|
||||||
|
|
||||||
|
startServers(0, 1);
|
||||||
|
|
||||||
|
setupSessionFactory(0, isNetty());
|
||||||
|
setupSessionFactory(1, isNetty());
|
||||||
|
final String queueName = "queues.0";
|
||||||
|
|
||||||
|
createQueue(0, queueName, queueName, null, false, null, null, RoutingType.ANYCAST);
|
||||||
|
createQueue(1, queueName, queueName, null, false, null, null, RoutingType.ANYCAST);
|
||||||
|
|
||||||
|
waitForBindings(0, queueName, 1, 0, true);
|
||||||
|
waitForBindings(1, queueName, 1, 0, true);
|
||||||
|
|
||||||
|
waitForBindings(0, queueName, /**/1, 0, false);
|
||||||
|
waitForBindings(1, queueName, 1, 0, false);
|
||||||
|
String producerUri = "amqp://localhost:61616";
|
||||||
|
if (compressLargeMessages) {
|
||||||
|
producerUri = producerUri + "?compressLargeMessages=true";
|
||||||
|
}
|
||||||
|
final JmsConnectionFactory producerFactory = new JmsConnectionFactory(producerUri);
|
||||||
|
try (Connection producerConnection = producerFactory.createConnection(); Session producerSession = producerConnection.createSession(false, Session.AUTO_ACKNOWLEDGE)) {
|
||||||
|
producerConnection.start();
|
||||||
|
final Destination queue = producerSession.createQueue(queueName);
|
||||||
|
String consumerUri = "amqp://localhost:61617";
|
||||||
|
if (compressLargeMessages) {
|
||||||
|
consumerUri = consumerUri + "?compressLargeMessages=true";
|
||||||
|
}
|
||||||
|
final JmsConnectionFactory consumerConnectionFactory = new JmsConnectionFactory(consumerUri);
|
||||||
|
try (Connection consumerConnection = consumerConnectionFactory.createConnection(); Session consumerSession = consumerConnection.createSession(false, Session.AUTO_ACKNOWLEDGE); MessageConsumer consumer = consumerSession.createConsumer(queue); MessageProducer producer = producerSession.createProducer(queue)) {
|
||||||
|
producer.setDeliveryMode(DeliveryMode.NON_PERSISTENT);
|
||||||
|
consumerConnection.start();
|
||||||
|
final byte[] largeMessageContent = new byte[MESSAGE_SIZE];
|
||||||
|
final byte[] receivedContent = new byte[largeMessageContent.length];
|
||||||
|
ThreadLocalRandom.current().nextBytes(largeMessageContent);
|
||||||
|
for (int i = 0; i < MESSAGES; i++) {
|
||||||
|
final BytesMessage sentMessage = producerSession.createBytesMessage();
|
||||||
|
sentMessage.writeBytes(largeMessageContent);
|
||||||
|
producer.send(sentMessage);
|
||||||
|
final Message receivedMessage = consumer.receive(RECEIVE_TIMEOUT_MILLIS);
|
||||||
|
Assert.assertNotNull("A message should be received in " + RECEIVE_TIMEOUT_MILLIS + " ms", receivedMessage);
|
||||||
|
Assert.assertThat(receivedMessage, IsInstanceOf.instanceOf(sentMessage.getClass()));
|
||||||
|
Assert.assertEquals(largeMessageContent.length, ((BytesMessage) receivedMessage).readBytes(receivedContent));
|
||||||
|
Assert.assertArrayEquals(largeMessageContent, receivedContent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setupCluster(final MessageLoadBalancingType messageLoadBalancingType) throws Exception {
|
||||||
|
setupClusterConnection("cluster0", "queues", messageLoadBalancingType, 1, isNetty(), 0, 1);
|
||||||
|
setupClusterConnection("cluster1", "queues", messageLoadBalancingType, 1, isNetty(), 1, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setRedistributionDelay(final long delay) {
|
||||||
|
AddressSettings as = new AddressSettings().setRedistributionDelay(delay);
|
||||||
|
|
||||||
|
getServer(0).getAddressSettingsRepository().addMatch("queues.*", as);
|
||||||
|
getServer(1).getAddressSettingsRepository().addMatch("queues.*", as);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setupServers() throws Exception {
|
||||||
|
setupServer(0, persistenceEnabled, isNetty());
|
||||||
|
setupServer(1, persistenceEnabled, isNetty());
|
||||||
|
|
||||||
|
servers[0].addProtocolManagerFactory(new ProtonProtocolManagerFactory());
|
||||||
|
servers[1].addProtocolManagerFactory(new ProtonProtocolManagerFactory());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void stopServers() throws Exception {
|
||||||
|
closeAllConsumers();
|
||||||
|
|
||||||
|
closeAllSessionFactories();
|
||||||
|
|
||||||
|
closeAllServerLocatorsFactories();
|
||||||
|
|
||||||
|
stopServers(0, 1);
|
||||||
|
|
||||||
|
clearServer(0, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue