/* * 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.paging; import javax.jms.Connection; import javax.jms.ConnectionFactory; import javax.jms.JMSException; import javax.jms.Message; import javax.jms.MessageConsumer; import javax.jms.MessageProducer; import javax.jms.Queue; import javax.jms.Session; import java.util.Arrays; import java.util.Collections; import java.util.HashSet; import java.util.Set; import java.util.concurrent.CountDownLatch; import java.util.concurrent.CyclicBarrier; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicLong; import org.apache.activemq.artemis.api.core.TransportConfiguration; import org.apache.activemq.artemis.core.config.Configuration; import org.apache.activemq.artemis.core.remoting.impl.netty.NettyAcceptorFactory; import org.apache.activemq.artemis.core.remoting.impl.netty.NettyConnectorFactory; import org.apache.activemq.artemis.core.settings.impl.AddressFullMessagePolicy; import org.apache.activemq.artemis.core.settings.impl.AddressSettings; import org.apache.activemq.artemis.jms.server.config.JMSConfiguration; import org.apache.activemq.artemis.jms.server.config.impl.ConnectionFactoryConfigurationImpl; import org.apache.activemq.artemis.jms.server.config.impl.JMSConfigurationImpl; import org.apache.activemq.artemis.jms.server.config.impl.JMSQueueConfigurationImpl; import org.apache.activemq.artemis.jms.server.embedded.EmbeddedJMS; import org.apache.activemq.artemis.tests.util.ActiveMQTestBase; import org.apache.activemq.artemis.utils.ActiveMQThreadFactory; import org.junit.After; import org.junit.Assert; import org.junit.Before; import org.junit.Test; public class MultipleProducersPagingTest extends ActiveMQTestBase { private static final int CONSUMER_WAIT_TIME_MS = 250; private static final int PRODUCERS = 5; private static final long MESSAGES_PER_PRODUCER = 2000; private static final long TOTAL_MSG = MESSAGES_PER_PRODUCER * PRODUCERS; private ExecutorService executor; private CountDownLatch runnersLatch; private CyclicBarrier barrierLatch; private AtomicLong msgReceived; private AtomicLong msgSent; private final Set<Connection> connections = new HashSet<>(); private EmbeddedJMS jmsServer; private ConnectionFactory cf; private Queue queue; @Override @Before public void setUp() throws Exception { super.setUp(); executor = Executors.newCachedThreadPool(ActiveMQThreadFactory.defaultThreadFactory()); AddressSettings addressSettings = new AddressSettings().setAddressFullMessagePolicy(AddressFullMessagePolicy.PAGE).setPageSizeBytes(50000).setMaxSizeBytes(404850); Configuration config = createBasicConfig().setPersistenceEnabled(false).setAddressesSettings(Collections.singletonMap("#", addressSettings)).setAcceptorConfigurations(Collections.singleton(new TransportConfiguration(NettyAcceptorFactory.class.getName()))).setConnectorConfigurations(Collections.singletonMap("netty", new TransportConfiguration(NettyConnectorFactory.class.getName()))); final JMSConfiguration jmsConfig = new JMSConfigurationImpl(); jmsConfig.getConnectionFactoryConfigurations().add(new ConnectionFactoryConfigurationImpl().setName("cf").setConnectorNames(Arrays.asList("netty")).setBindings("/cf")); jmsConfig.getQueueConfigurations().add(new JMSQueueConfigurationImpl().setName("simple").setSelector("").setDurable(false).setBindings("/queue/simple")); jmsServer = new EmbeddedJMS(); jmsServer.setConfiguration(config); jmsServer.setJmsConfiguration(jmsConfig); jmsServer.start(); cf = (ConnectionFactory) jmsServer.lookup("/cf"); queue = (Queue) jmsServer.lookup("/queue/simple"); barrierLatch = new CyclicBarrier(PRODUCERS + 1); runnersLatch = new CountDownLatch(PRODUCERS + 1); msgReceived = new AtomicLong(0); msgSent = new AtomicLong(0); } @Test public void testQueue() throws InterruptedException { executor.execute(new ConsumerRun()); for (int i = 0; i < PRODUCERS; i++) { executor.execute(new ProducerRun()); } Assert.assertTrue("must take less than a minute to run", runnersLatch.await(1, TimeUnit.MINUTES)); Assert.assertEquals("number sent", TOTAL_MSG, msgSent.longValue()); Assert.assertEquals("number received", TOTAL_MSG, msgReceived.longValue()); } private synchronized Session createSession() throws JMSException { Connection connection = cf.createConnection(); connections.add(connection); connection.start(); return connection.createSession(false, Session.AUTO_ACKNOWLEDGE); } final class ConsumerRun implements Runnable { @Override public void run() { try { Session session = createSession(); MessageConsumer consumer = session.createConsumer(queue); barrierLatch.await(); while (true) { Message msg = consumer.receive(CONSUMER_WAIT_TIME_MS); if (msg == null) break; msgReceived.incrementAndGet(); } } catch (Exception e) { throw new RuntimeException(e); } finally { runnersLatch.countDown(); } } } final class ProducerRun implements Runnable { @Override public void run() { try { Session session = createSession(); MessageProducer producer = session.createProducer(queue); barrierLatch.await(); for (int i = 0; i < MESSAGES_PER_PRODUCER; i++) { producer.send(session.createTextMessage(this.hashCode() + " counter " + i)); msgSent.incrementAndGet(); } } catch (Exception cause) { throw new RuntimeException(cause); } finally { runnersLatch.countDown(); } } } @Override @After public void tearDown() throws Exception { executor.shutdown(); for (Connection conn : connections) { conn.close(); } connections.clear(); if (jmsServer != null) jmsServer.stop(); super.tearDown(); } }