mirror of https://github.com/apache/activemq.git
add test variant that demonstrates the need to disable the audit on the dlq when there is a concurrent consumer with multiple durable subs, some duplicate sends will not be caught (if they are acked in the mean time) so that audit stop needs to be disabled such that they can be consumed. This points to the need for a dlq per durable sub. work around for: https://issues.apache.org/activemq/browse/AMQ-2584
git-svn-id: https://svn.apache.org/repos/asf/activemq/trunk@1027886 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
ef1a3c1f67
commit
c290a17460
|
@ -0,0 +1,223 @@
|
||||||
|
/**
|
||||||
|
* 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.bugs;
|
||||||
|
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import javax.jms.JMSException;
|
||||||
|
import javax.jms.Message;
|
||||||
|
import javax.jms.MessageConsumer;
|
||||||
|
import javax.jms.MessageListener;
|
||||||
|
import javax.jms.MessageProducer;
|
||||||
|
import javax.jms.Session;
|
||||||
|
|
||||||
|
import junit.framework.Test;
|
||||||
|
|
||||||
|
import org.apache.activemq.ActiveMQConnection;
|
||||||
|
import org.apache.activemq.ActiveMQConnectionFactory;
|
||||||
|
import org.apache.activemq.broker.BrokerService;
|
||||||
|
import org.apache.activemq.broker.jmx.BrokerView;
|
||||||
|
import org.apache.activemq.broker.region.policy.PolicyEntry;
|
||||||
|
import org.apache.activemq.broker.region.policy.PolicyMap;
|
||||||
|
import org.apache.activemq.command.ActiveMQQueue;
|
||||||
|
import org.apache.activemq.command.ActiveMQTopic;
|
||||||
|
import org.apache.activemq.store.PersistenceAdapter;
|
||||||
|
import org.apache.activemq.store.kahadb.KahaDBPersistenceAdapter;
|
||||||
|
import org.apache.activemq.util.IntrospectionSupport;
|
||||||
|
import org.apache.commons.logging.Log;
|
||||||
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
|
||||||
|
// variation on AMQ2584 where the DLQ consumer works in parallel to producer so
|
||||||
|
// that some dups are not suppressed as they are already acked by the consumer
|
||||||
|
// the audit needs to be disabled to allow these dupes to be consumed
|
||||||
|
public class AMQ2584ConcurrentDlqTest extends org.apache.activemq.TestSupport {
|
||||||
|
|
||||||
|
static final Log LOG = LogFactory.getLog(AMQ2584ConcurrentDlqTest.class);
|
||||||
|
BrokerService broker = null;
|
||||||
|
ActiveMQTopic topic;
|
||||||
|
|
||||||
|
ActiveMQConnection consumerConnection = null, producerConnection = null, dlqConnection = null;
|
||||||
|
Session producerSession;
|
||||||
|
MessageProducer producer;
|
||||||
|
final int minPercentUsageForStore = 10;
|
||||||
|
final int numMessages = 1000;
|
||||||
|
|
||||||
|
String data;
|
||||||
|
|
||||||
|
public void testSize() throws Exception {
|
||||||
|
CountDownLatch redeliveryConsumerLatch = new CountDownLatch(((2*numMessages) *3) -1);
|
||||||
|
CountDownLatch dlqConsumerLatch = new CountDownLatch((numMessages) -1);
|
||||||
|
openConsumer(redeliveryConsumerLatch);
|
||||||
|
openDlqConsumer(dlqConsumerLatch);
|
||||||
|
|
||||||
|
|
||||||
|
assertEquals(0, broker.getAdminView().getStorePercentUsage());
|
||||||
|
|
||||||
|
for (int i = 0; i < numMessages; i++) {
|
||||||
|
sendMessage(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
final BrokerView brokerView = broker.getAdminView();
|
||||||
|
|
||||||
|
broker.getSystemUsage().getStoreUsage().isFull();
|
||||||
|
LOG.info("store percent usage: "+brokerView.getStorePercentUsage());
|
||||||
|
//assertTrue("some store in use", broker.getAdminView().getStorePercentUsage() > minPercentUsageForStore);
|
||||||
|
assertTrue("redelivery consumer got all it needs", redeliveryConsumerLatch.await(60, TimeUnit.SECONDS));
|
||||||
|
assertTrue("dql consumer got all it needs", dlqConsumerLatch.await(60, TimeUnit.SECONDS));
|
||||||
|
closeConsumer();
|
||||||
|
|
||||||
|
LOG.info("Giving dlq a chance to clear down once topic consumer is closed");
|
||||||
|
//get broker a chance to clean obsolete messages, wait 2*cleanupInterval
|
||||||
|
Thread.sleep(5000);
|
||||||
|
|
||||||
|
// consumer some of the duplicates that arrived after the first ack
|
||||||
|
closeDlqConsumer();
|
||||||
|
int numFiles = ((KahaDBPersistenceAdapter)broker.getPersistenceAdapter()).getDirectory().list().length;
|
||||||
|
LOG.info("num files: " + numFiles);
|
||||||
|
assertTrue("kahaDB dir should contain few db files,but definitely less than 10, is: " + numFiles,10>numFiles);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void openConsumer(final CountDownLatch latch) throws Exception {
|
||||||
|
consumerConnection = (ActiveMQConnection) createConnection();
|
||||||
|
consumerConnection.setClientID("cliID");
|
||||||
|
consumerConnection.start();
|
||||||
|
final Session session = consumerConnection.createSession(false, Session.AUTO_ACKNOWLEDGE);
|
||||||
|
|
||||||
|
MessageListener listener = new MessageListener() {
|
||||||
|
public void onMessage(Message message) {
|
||||||
|
latch.countDown();
|
||||||
|
try {
|
||||||
|
session.recover();
|
||||||
|
} catch (Exception ignored) {
|
||||||
|
ignored.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
session.createDurableSubscriber(topic, "subName1").setMessageListener(listener);
|
||||||
|
session.createDurableSubscriber(topic, "subName2").setMessageListener(listener);
|
||||||
|
session.createDurableSubscriber(topic, "subName3").setMessageListener(listener);
|
||||||
|
}
|
||||||
|
private void openDlqConsumer(final CountDownLatch received)throws Exception{
|
||||||
|
|
||||||
|
dlqConnection = (ActiveMQConnection) createConnection();
|
||||||
|
Session dlqSession = dlqConnection .createSession(false, Session.AUTO_ACKNOWLEDGE);
|
||||||
|
MessageConsumer dlqConsumer = dlqSession.createConsumer(new ActiveMQQueue("ActiveMQ.DLQ"));
|
||||||
|
dlqConsumer.setMessageListener(new MessageListener() {
|
||||||
|
public void onMessage(Message message) {
|
||||||
|
if (received.getCount() % 200 == 0) {
|
||||||
|
LOG.info("remaining on DLQ: " + received.getCount());
|
||||||
|
}
|
||||||
|
received.countDown();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
dlqConnection.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void closeConsumer() throws JMSException {
|
||||||
|
if (consumerConnection != null)
|
||||||
|
consumerConnection.close();
|
||||||
|
consumerConnection = null;
|
||||||
|
}
|
||||||
|
private void closeDlqConsumer() throws JMSException {
|
||||||
|
if (dlqConnection != null)
|
||||||
|
dlqConnection.close();
|
||||||
|
dlqConnection = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void sendMessage(boolean filter) throws Exception {
|
||||||
|
if (producerConnection == null) {
|
||||||
|
producerConnection = (ActiveMQConnection) createConnection();
|
||||||
|
producerConnection.start();
|
||||||
|
producerSession = producerConnection.createSession(false, Session.AUTO_ACKNOWLEDGE);
|
||||||
|
producer = producerSession.createProducer(topic);
|
||||||
|
}
|
||||||
|
|
||||||
|
Message message = producerSession.createMessage();
|
||||||
|
message.setStringProperty("data", data);
|
||||||
|
producer.send(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void startBroker(boolean deleteMessages) throws Exception {
|
||||||
|
broker = new BrokerService();
|
||||||
|
broker.setAdvisorySupport(false);
|
||||||
|
broker.setBrokerName("testStoreSize");
|
||||||
|
|
||||||
|
PolicyMap map = new PolicyMap();
|
||||||
|
PolicyEntry entry = new PolicyEntry();
|
||||||
|
entry.setEnableAudit(false);
|
||||||
|
map.setDefaultEntry(entry);
|
||||||
|
broker.setDestinationPolicy(map);
|
||||||
|
|
||||||
|
if (deleteMessages) {
|
||||||
|
broker.setDeleteAllMessagesOnStartup(true);
|
||||||
|
}
|
||||||
|
KahaDBPersistenceAdapter persistenceAdapter=new KahaDBPersistenceAdapter();
|
||||||
|
persistenceAdapter.setEnableJournalDiskSyncs(false);
|
||||||
|
|
||||||
|
broker.setPersistenceAdapter(persistenceAdapter);
|
||||||
|
configurePersistenceAdapter(broker.getPersistenceAdapter());
|
||||||
|
broker.getSystemUsage().getStoreUsage().setLimit(200 * 1000 * 1000);
|
||||||
|
broker.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void configurePersistenceAdapter(PersistenceAdapter persistenceAdapter) {
|
||||||
|
Properties properties = new Properties();
|
||||||
|
String maxFileLengthVal = String.valueOf(2 * 1024 * 1024);
|
||||||
|
properties.put("journalMaxFileLength", maxFileLengthVal);
|
||||||
|
properties.put("maxFileLength", maxFileLengthVal);
|
||||||
|
properties.put("cleanupInterval", "2000");
|
||||||
|
properties.put("checkpointInterval", "2000");
|
||||||
|
|
||||||
|
IntrospectionSupport.setProperties(persistenceAdapter, properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void stopBroker() throws Exception {
|
||||||
|
if (broker != null)
|
||||||
|
broker.stop();
|
||||||
|
broker = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected ActiveMQConnectionFactory createConnectionFactory() throws Exception {
|
||||||
|
return new ActiveMQConnectionFactory("vm://testStoreSize?jms.watchTopicAdvisories=false&jms.redeliveryPolicy.maximumRedeliveries=1&jms.redeliveryPolicy.initialRedeliveryDelay=0&waitForStart=5000&create=false");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void setUp() throws Exception {
|
||||||
|
super.setUp();
|
||||||
|
|
||||||
|
StringBuilder sb = new StringBuilder(5000);
|
||||||
|
for (int i = 0; i < 5000; i++) {
|
||||||
|
sb.append('a');
|
||||||
|
}
|
||||||
|
data = sb.toString();
|
||||||
|
|
||||||
|
startBroker(true);
|
||||||
|
topic = (ActiveMQTopic) createDestination();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void tearDown() throws Exception {
|
||||||
|
stopBroker();
|
||||||
|
super.tearDown();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue