mirror of https://github.com/apache/activemq.git
Updating MemoryTopicMessageStore to decrement store statistics on cache eviction. Updating KahaDBMessageStoreSizeStatTest to account for the fact that a LRU cache is used so the last 100 messages are kept in memroy.
This commit is contained in:
parent
a3c8bee1f0
commit
a49d46e3ca
|
@ -57,8 +57,7 @@ public class MemoryMessageStore extends AbstractMessageStore {
|
||||||
public synchronized void addMessage(ConnectionContext context, Message message) throws IOException {
|
public synchronized void addMessage(ConnectionContext context, Message message) throws IOException {
|
||||||
synchronized (messageTable) {
|
synchronized (messageTable) {
|
||||||
messageTable.put(message.getMessageId(), message);
|
messageTable.put(message.getMessageId(), message);
|
||||||
getMessageStoreStatistics().getMessageCount().increment();
|
incMessageStoreStatistics(message);
|
||||||
getMessageStoreStatistics().getMessageSize().addSize(message.getSize());
|
|
||||||
}
|
}
|
||||||
message.incrementReferenceCount();
|
message.incrementReferenceCount();
|
||||||
message.getMessageId().setFutureOrSequenceLong(sequenceId++);
|
message.getMessageId().setFutureOrSequenceLong(sequenceId++);
|
||||||
|
@ -94,8 +93,7 @@ public class MemoryMessageStore extends AbstractMessageStore {
|
||||||
Message removed = messageTable.remove(msgId);
|
Message removed = messageTable.remove(msgId);
|
||||||
if( removed !=null ) {
|
if( removed !=null ) {
|
||||||
removed.decrementReferenceCount();
|
removed.decrementReferenceCount();
|
||||||
getMessageStoreStatistics().getMessageCount().decrement();
|
decMessageStoreStatistics(removed);
|
||||||
getMessageStoreStatistics().getMessageSize().addSize(-removed.getSize());
|
|
||||||
}
|
}
|
||||||
if ((lastBatchId != null && lastBatchId.equals(msgId)) || messageTable.isEmpty()) {
|
if ((lastBatchId != null && lastBatchId.equals(msgId)) || messageTable.isEmpty()) {
|
||||||
lastBatchId = null;
|
lastBatchId = null;
|
||||||
|
@ -200,4 +198,14 @@ public class MemoryMessageStore extends AbstractMessageStore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected final void incMessageStoreStatistics(Message message) {
|
||||||
|
getMessageStoreStatistics().getMessageCount().increment();
|
||||||
|
getMessageStoreStatistics().getMessageSize().addSize(message.getSize());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final void decMessageStoreStatistics(Message message) {
|
||||||
|
getMessageStoreStatistics().getMessageCount().decrement();
|
||||||
|
getMessageStoreStatistics().getMessageSize().addSize(-message.getSize());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,7 @@ import java.util.HashMap;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
|
|
||||||
import org.apache.activemq.broker.ConnectionContext;
|
import org.apache.activemq.broker.ConnectionContext;
|
||||||
import org.apache.activemq.command.ActiveMQDestination;
|
import org.apache.activemq.command.ActiveMQDestination;
|
||||||
import org.apache.activemq.command.Message;
|
import org.apache.activemq.command.Message;
|
||||||
|
@ -29,36 +30,47 @@ import org.apache.activemq.command.MessageAck;
|
||||||
import org.apache.activemq.command.MessageId;
|
import org.apache.activemq.command.MessageId;
|
||||||
import org.apache.activemq.command.SubscriptionInfo;
|
import org.apache.activemq.command.SubscriptionInfo;
|
||||||
import org.apache.activemq.store.MessageRecoveryListener;
|
import org.apache.activemq.store.MessageRecoveryListener;
|
||||||
|
import org.apache.activemq.store.MessageStoreStatistics;
|
||||||
import org.apache.activemq.store.TopicMessageStore;
|
import org.apache.activemq.store.TopicMessageStore;
|
||||||
import org.apache.activemq.util.LRUCache;
|
import org.apache.activemq.util.LRUCache;
|
||||||
import org.apache.activemq.util.SubscriptionKey;
|
import org.apache.activemq.util.SubscriptionKey;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class MemoryTopicMessageStore extends MemoryMessageStore implements TopicMessageStore {
|
public class MemoryTopicMessageStore extends MemoryMessageStore implements TopicMessageStore {
|
||||||
|
|
||||||
private Map<SubscriptionKey, SubscriptionInfo> subscriberDatabase;
|
private Map<SubscriptionKey, SubscriptionInfo> subscriberDatabase;
|
||||||
private Map<SubscriptionKey, MemoryTopicSub> topicSubMap;
|
private Map<SubscriptionKey, MemoryTopicSub> topicSubMap;
|
||||||
|
private final Map<MessageId, Message> originalMessageTable;
|
||||||
|
|
||||||
public MemoryTopicMessageStore(ActiveMQDestination destination) {
|
public MemoryTopicMessageStore(ActiveMQDestination destination) {
|
||||||
this(destination, new LRUCache<MessageId, Message>(100, 100, 0.75f, false), makeSubscriptionInfoMap());
|
this(destination, new MemoryTopicMessageStoreLRUCache(100, 100, 0.75f, false), makeSubscriptionInfoMap());
|
||||||
|
|
||||||
|
//Set the messageStoreStatistics after the super class is initialized so that the stats can be
|
||||||
|
//properly updated on cache eviction
|
||||||
|
MemoryTopicMessageStoreLRUCache cache = (MemoryTopicMessageStoreLRUCache) originalMessageTable;
|
||||||
|
cache.setMessageStoreStatistics(messageStoreStatistics);
|
||||||
}
|
}
|
||||||
|
|
||||||
public MemoryTopicMessageStore(ActiveMQDestination destination, Map<MessageId, Message> messageTable, Map<SubscriptionKey, SubscriptionInfo> subscriberDatabase) {
|
public MemoryTopicMessageStore(ActiveMQDestination destination, Map<MessageId, Message> messageTable, Map<SubscriptionKey, SubscriptionInfo> subscriberDatabase) {
|
||||||
super(destination, messageTable);
|
super(destination, messageTable);
|
||||||
this.subscriberDatabase = subscriberDatabase;
|
this.subscriberDatabase = subscriberDatabase;
|
||||||
this.topicSubMap = makeSubMap();
|
this.topicSubMap = makeSubMap();
|
||||||
|
//this is only necessary so that messageStoreStatistics can be set if necessary
|
||||||
|
//We need the original reference since messageTable is wrapped in a synchronized map in the parent class
|
||||||
|
this.originalMessageTable = messageTable;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static Map<SubscriptionKey, SubscriptionInfo> makeSubscriptionInfoMap() {
|
protected static Map<SubscriptionKey, SubscriptionInfo> makeSubscriptionInfoMap() {
|
||||||
return Collections.synchronizedMap(new HashMap<SubscriptionKey, SubscriptionInfo>());
|
return Collections.synchronizedMap(new HashMap<SubscriptionKey, SubscriptionInfo>());
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static Map<SubscriptionKey, MemoryTopicSub> makeSubMap() {
|
protected static Map<SubscriptionKey, MemoryTopicSub> makeSubMap() {
|
||||||
return Collections.synchronizedMap(new HashMap<SubscriptionKey, MemoryTopicSub>());
|
return Collections.synchronizedMap(new HashMap<SubscriptionKey, MemoryTopicSub>());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void addMessage(ConnectionContext context, Message message) throws IOException {
|
public synchronized void addMessage(ConnectionContext context, Message message) throws IOException {
|
||||||
super.addMessage(context, message);
|
super.addMessage(context, message);
|
||||||
for (Iterator<MemoryTopicSub> i = topicSubMap.values().iterator(); i.hasNext();) {
|
for (Iterator<MemoryTopicSub> i = topicSubMap.values().iterator(); i.hasNext();) {
|
||||||
|
@ -67,6 +79,7 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void acknowledge(ConnectionContext context, String clientId, String subscriptionName,
|
public synchronized void acknowledge(ConnectionContext context, String clientId, String subscriptionName,
|
||||||
MessageId messageId, MessageAck ack) throws IOException {
|
MessageId messageId, MessageAck ack) throws IOException {
|
||||||
SubscriptionKey key = new SubscriptionKey(clientId, subscriptionName);
|
SubscriptionKey key = new SubscriptionKey(clientId, subscriptionName);
|
||||||
|
@ -76,10 +89,12 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized SubscriptionInfo lookupSubscription(String clientId, String subscriptionName) throws IOException {
|
public synchronized SubscriptionInfo lookupSubscription(String clientId, String subscriptionName) throws IOException {
|
||||||
return subscriberDatabase.get(new SubscriptionKey(clientId, subscriptionName));
|
return subscriberDatabase.get(new SubscriptionKey(clientId, subscriptionName));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void addSubscription(SubscriptionInfo info, boolean retroactive) throws IOException {
|
public synchronized void addSubscription(SubscriptionInfo info, boolean retroactive) throws IOException {
|
||||||
SubscriptionKey key = new SubscriptionKey(info);
|
SubscriptionKey key = new SubscriptionKey(info);
|
||||||
MemoryTopicSub sub = new MemoryTopicSub();
|
MemoryTopicSub sub = new MemoryTopicSub();
|
||||||
|
@ -93,12 +108,14 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
subscriberDatabase.put(key, info);
|
subscriberDatabase.put(key, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void deleteSubscription(String clientId, String subscriptionName) {
|
public synchronized void deleteSubscription(String clientId, String subscriptionName) {
|
||||||
org.apache.activemq.util.SubscriptionKey key = new SubscriptionKey(clientId, subscriptionName);
|
org.apache.activemq.util.SubscriptionKey key = new SubscriptionKey(clientId, subscriptionName);
|
||||||
subscriberDatabase.remove(key);
|
subscriberDatabase.remove(key);
|
||||||
topicSubMap.remove(key);
|
topicSubMap.remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void recoverSubscription(String clientId, String subscriptionName, MessageRecoveryListener listener) throws Exception {
|
public synchronized void recoverSubscription(String clientId, String subscriptionName, MessageRecoveryListener listener) throws Exception {
|
||||||
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
||||||
if (sub != null) {
|
if (sub != null) {
|
||||||
|
@ -106,16 +123,19 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void delete() {
|
public synchronized void delete() {
|
||||||
super.delete();
|
super.delete();
|
||||||
subscriberDatabase.clear();
|
subscriberDatabase.clear();
|
||||||
topicSubMap.clear();
|
topicSubMap.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public SubscriptionInfo[] getAllSubscriptions() throws IOException {
|
public SubscriptionInfo[] getAllSubscriptions() throws IOException {
|
||||||
return subscriberDatabase.values().toArray(new SubscriptionInfo[subscriberDatabase.size()]);
|
return subscriberDatabase.values().toArray(new SubscriptionInfo[subscriberDatabase.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized int getMessageCount(String clientId, String subscriberName) throws IOException {
|
public synchronized int getMessageCount(String clientId, String subscriberName) throws IOException {
|
||||||
int result = 0;
|
int result = 0;
|
||||||
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriberName));
|
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriberName));
|
||||||
|
@ -125,6 +145,7 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public synchronized void recoverNextMessages(String clientId, String subscriptionName, int maxReturned, MessageRecoveryListener listener) throws Exception {
|
public synchronized void recoverNextMessages(String clientId, String subscriptionName, int maxReturned, MessageRecoveryListener listener) throws Exception {
|
||||||
MemoryTopicSub sub = this.topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
MemoryTopicSub sub = this.topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
||||||
if (sub != null) {
|
if (sub != null) {
|
||||||
|
@ -132,10 +153,39 @@ public class MemoryTopicMessageStore extends MemoryMessageStore implements Topic
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public void resetBatching(String clientId, String subscriptionName) {
|
public void resetBatching(String clientId, String subscriptionName) {
|
||||||
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
MemoryTopicSub sub = topicSubMap.get(new SubscriptionKey(clientId, subscriptionName));
|
||||||
if (sub != null) {
|
if (sub != null) {
|
||||||
sub.resetBatching();
|
sub.resetBatching();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Since we initialize the store with a LRUCache in some cases, we need to account for cache evictions
|
||||||
|
* when computing the message store statistics.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
private static class MemoryTopicMessageStoreLRUCache extends LRUCache<MessageId, Message> {
|
||||||
|
private static final long serialVersionUID = -342098639681884413L;
|
||||||
|
private MessageStoreStatistics messageStoreStatistics;
|
||||||
|
|
||||||
|
public MemoryTopicMessageStoreLRUCache(int initialCapacity, int maximumCacheSize,
|
||||||
|
float loadFactor, boolean accessOrder) {
|
||||||
|
super(initialCapacity, maximumCacheSize, loadFactor, accessOrder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMessageStoreStatistics(
|
||||||
|
MessageStoreStatistics messageStoreStatistics) {
|
||||||
|
this.messageStoreStatistics = messageStoreStatistics;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCacheEviction(Map.Entry<MessageId, Message> eldest) {
|
||||||
|
if (messageStoreStatistics != null) {
|
||||||
|
messageStoreStatistics.getMessageCount().decrement();
|
||||||
|
messageStoreStatistics.getMessageSize().addSize(-eldest.getValue().getSize());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,7 @@ import javax.jms.QueueSession;
|
||||||
import javax.jms.Session;
|
import javax.jms.Session;
|
||||||
import javax.jms.Topic;
|
import javax.jms.Topic;
|
||||||
import javax.jms.TopicSession;
|
import javax.jms.TopicSession;
|
||||||
|
import javax.jms.TopicSubscriber;
|
||||||
import javax.management.ObjectName;
|
import javax.management.ObjectName;
|
||||||
import javax.management.openmbean.CompositeData;
|
import javax.management.openmbean.CompositeData;
|
||||||
import javax.management.openmbean.TabularData;
|
import javax.management.openmbean.TabularData;
|
||||||
|
@ -47,6 +48,8 @@ import org.apache.activemq.broker.region.Destination;
|
||||||
import org.apache.activemq.command.ActiveMQDestination;
|
import org.apache.activemq.command.ActiveMQDestination;
|
||||||
import org.apache.activemq.command.ActiveMQQueue;
|
import org.apache.activemq.command.ActiveMQQueue;
|
||||||
import org.apache.activemq.command.ActiveMQTopic;
|
import org.apache.activemq.command.ActiveMQTopic;
|
||||||
|
import org.apache.activemq.util.Wait;
|
||||||
|
import org.apache.activemq.util.Wait.Condition;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -67,6 +70,7 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
protected BrokerService broker;
|
protected BrokerService broker;
|
||||||
protected URI brokerConnectURI;
|
protected URI brokerConnectURI;
|
||||||
protected String defaultQueueName = "test.queue";
|
protected String defaultQueueName = "test.queue";
|
||||||
|
protected String defaultTopicName = "test.topic";
|
||||||
protected static int messageSize = 1000;
|
protected static int messageSize = 1000;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
|
@ -100,34 +104,67 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSize() throws Exception {
|
public void testMessageSize() throws Exception {
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeAfterConsumption() throws Exception {
|
public void testMessageSizeAfterConsumption() throws Exception {
|
||||||
|
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
|
||||||
consumeTestMessages();
|
consumeTestQueueMessages();
|
||||||
Thread.sleep(3000);
|
|
||||||
verifyStats(dest, 0, 0);
|
verifyStats(dest, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeDurable() throws Exception {
|
public void testMessageSizeOneDurable() throws Exception {
|
||||||
|
|
||||||
Destination dest = publishTestMessagesDurable();
|
Connection connection = new ActiveMQConnectionFactory(brokerConnectURI).createConnection();
|
||||||
|
connection.setClientID("clientId");
|
||||||
|
connection.start();
|
||||||
|
|
||||||
|
Destination dest = publishTestMessagesDurable(connection, new String[] {"sub1"}, 200, 200);
|
||||||
|
|
||||||
//verify the count and size
|
//verify the count and size
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
|
||||||
|
//consume all messages
|
||||||
|
consumeDurableTestMessages(connection, "sub1", 200);
|
||||||
|
|
||||||
|
//All messages should now be gone
|
||||||
|
verifyStats(dest, 0, 0);
|
||||||
|
|
||||||
|
connection.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout=10000)
|
||||||
|
public void testMessageSizeTwoDurables() throws Exception {
|
||||||
|
|
||||||
|
Connection connection = new ActiveMQConnectionFactory(brokerConnectURI).createConnection();
|
||||||
|
connection.setClientID("clientId");
|
||||||
|
connection.start();
|
||||||
|
|
||||||
|
Destination dest = publishTestMessagesDurable(connection, new String[] {"sub1", "sub2"}, 200, 200);
|
||||||
|
|
||||||
|
//verify the count and size
|
||||||
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
|
||||||
|
//consume messages just for sub1
|
||||||
|
consumeDurableTestMessages(connection, "sub1", 200);
|
||||||
|
|
||||||
|
//There is still a durable that hasn't consumed so the messages should exist
|
||||||
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
|
||||||
|
connection.stop();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeAfterDestinationDeletion() throws Exception {
|
public void testMessageSizeAfterDestinationDeletion() throws Exception {
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
|
||||||
//check that the size is 0 after deletion
|
//check that the size is 0 after deletion
|
||||||
|
@ -135,18 +172,34 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
verifyStats(dest, 0, 0);
|
verifyStats(dest, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void verifyStats(Destination dest, int count, long minimumSize) throws Exception {
|
protected void verifyStats(Destination dest, final int count, final long minimumSize) throws Exception {
|
||||||
MessageStore messageStore = dest.getMessageStore();
|
final MessageStore messageStore = dest.getMessageStore();
|
||||||
MessageStoreStatistics storeStats = dest.getMessageStore().getMessageStoreStatistics();
|
final MessageStoreStatistics storeStats = dest.getMessageStore().getMessageStoreStatistics();
|
||||||
assertEquals(messageStore.getMessageCount(), count);
|
|
||||||
assertEquals(messageStore.getMessageCount(),
|
Wait.waitFor(new Condition() {
|
||||||
storeStats.getMessageCount().getCount());
|
@Override
|
||||||
assertEquals(messageStore.getMessageSize(),
|
public boolean isSatisified() throws Exception {
|
||||||
|
return (count == messageStore.getMessageCount()) && (messageStore.getMessageCount() ==
|
||||||
|
storeStats.getMessageCount().getCount()) && (messageStore.getMessageSize() ==
|
||||||
messageStore.getMessageStoreStatistics().getMessageSize().getTotalSize());
|
messageStore.getMessageStoreStatistics().getMessageSize().getTotalSize());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
if (count > 0) {
|
if (count > 0) {
|
||||||
assertTrue(storeStats.getMessageSize().getTotalSize() > minimumSize);
|
assertTrue(storeStats.getMessageSize().getTotalSize() > minimumSize);
|
||||||
|
Wait.waitFor(new Condition() {
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return storeStats.getMessageSize().getTotalSize() > minimumSize;
|
||||||
|
}
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
assertEquals(storeStats.getMessageSize().getTotalSize(), 0);
|
Wait.waitFor(new Condition() {
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return storeStats.getMessageSize().getTotalSize() == 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,11 +219,11 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected Destination publishTestMessages(int count) throws Exception {
|
protected Destination publishTestQueueMessages(int count) throws Exception {
|
||||||
return publishTestMessages(count, defaultQueueName);
|
return publishTestQueueMessages(count, defaultQueueName);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Destination publishTestMessages(int count, String queueName) throws Exception {
|
protected Destination publishTestQueueMessages(int count, String queueName) throws Exception {
|
||||||
// create a new queue
|
// create a new queue
|
||||||
final ActiveMQDestination activeMqQueue = new ActiveMQQueue(
|
final ActiveMQDestination activeMqQueue = new ActiveMQQueue(
|
||||||
queueName);
|
queueName);
|
||||||
|
@ -196,17 +249,21 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
connection.stop();
|
connection.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
return dest;
|
return dest;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Destination consumeTestMessages() throws Exception {
|
protected Destination consumeTestQueueMessages() throws Exception {
|
||||||
return consumeTestMessages(defaultQueueName);
|
return consumeTestQueueMessages(defaultQueueName);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Destination consumeTestMessages(String queueName) throws Exception {
|
protected Destination consumeDurableTestMessages(Connection connection, String sub, int size) throws Exception {
|
||||||
|
return consumeDurableTestMessages(connection, sub, size, defaultTopicName);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Destination consumeTestQueueMessages(String queueName) throws Exception {
|
||||||
// create a new queue
|
// create a new queue
|
||||||
final ActiveMQDestination activeMqQueue = new ActiveMQQueue(
|
final ActiveMQDestination activeMqQueue = new ActiveMQQueue(
|
||||||
queueName);
|
queueName);
|
||||||
|
@ -235,22 +292,45 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
return dest;
|
return dest;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Destination publishTestMessagesDurable() throws Exception {
|
protected Destination consumeDurableTestMessages(Connection connection, String sub, int size, String topicName) throws Exception {
|
||||||
// create a new queue
|
// create a new queue
|
||||||
final ActiveMQDestination activeMqTopic = new ActiveMQTopic(
|
final ActiveMQDestination activeMqTopic = new ActiveMQTopic(
|
||||||
"test.topic");
|
topicName);
|
||||||
|
|
||||||
|
Destination dest = broker.getDestination(activeMqTopic);
|
||||||
|
|
||||||
|
Session session = connection.createSession(false,
|
||||||
|
QueueSession.AUTO_ACKNOWLEDGE);
|
||||||
|
Topic topic = session.createTopic(topicName);
|
||||||
|
|
||||||
|
try {
|
||||||
|
TopicSubscriber consumer = session.createDurableSubscriber(topic, sub);
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
consumer.receive();
|
||||||
|
}
|
||||||
|
|
||||||
|
} finally {
|
||||||
|
session.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
return dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Destination publishTestMessagesDurable(Connection connection, String[] subNames, int publishSize, int expectedSize) throws Exception {
|
||||||
|
// create a new queue
|
||||||
|
final ActiveMQDestination activeMqTopic = new ActiveMQTopic(
|
||||||
|
defaultTopicName);
|
||||||
|
|
||||||
Destination dest = broker.getDestination(activeMqTopic);
|
Destination dest = broker.getDestination(activeMqTopic);
|
||||||
|
|
||||||
// Start the connection
|
// Start the connection
|
||||||
Connection connection = new ActiveMQConnectionFactory(brokerConnectURI)
|
|
||||||
.createConnection();
|
|
||||||
connection.setClientID("clientId");
|
|
||||||
connection.start();
|
|
||||||
Session session = connection.createSession(false,
|
Session session = connection.createSession(false,
|
||||||
TopicSession.AUTO_ACKNOWLEDGE);
|
TopicSession.AUTO_ACKNOWLEDGE);
|
||||||
Topic topic = session.createTopic("test.topic");
|
Topic topic = session.createTopic(defaultTopicName);
|
||||||
session.createDurableSubscriber(topic, "sub1");
|
for (String subName : subNames) {
|
||||||
|
session.createDurableSubscriber(topic, subName);
|
||||||
|
}
|
||||||
|
|
||||||
// browse the durable sub - this test is to verify that browsing (which calls createTopicMessageStore)
|
// browse the durable sub - this test is to verify that browsing (which calls createTopicMessageStore)
|
||||||
//in KahaDBStore will not create a brand new store (ie uses the cache) If the cache is not used,
|
//in KahaDBStore will not create a brand new store (ie uses the cache) If the cache is not used,
|
||||||
|
@ -263,21 +343,21 @@ public abstract class AbstractMessageStoreSizeStatTest {
|
||||||
// store
|
// store
|
||||||
MessageProducer prod = session.createProducer(topic);
|
MessageProducer prod = session.createProducer(topic);
|
||||||
prod.setDeliveryMode(DeliveryMode.PERSISTENT);
|
prod.setDeliveryMode(DeliveryMode.PERSISTENT);
|
||||||
for (int i = 0; i < 200; i++) {
|
for (int i = 0; i < publishSize; i++) {
|
||||||
prod.send(createMessage(session));
|
prod.send(createMessage(session));
|
||||||
}
|
}
|
||||||
|
|
||||||
//verify the view has 200 messages
|
//verify the view has expected messages
|
||||||
assertEquals(1, subs.length);
|
assertEquals(subNames.length, subs.length);
|
||||||
ObjectName subName = subs[0];
|
ObjectName subName = subs[0];
|
||||||
DurableSubscriptionViewMBean sub = (DurableSubscriptionViewMBean)
|
DurableSubscriptionViewMBean sub = (DurableSubscriptionViewMBean)
|
||||||
broker.getManagementContext().newProxyInstance(subName, DurableSubscriptionViewMBean.class, true);
|
broker.getManagementContext().newProxyInstance(subName, DurableSubscriptionViewMBean.class, true);
|
||||||
CompositeData[] data = sub.browse();
|
CompositeData[] data = sub.browse();
|
||||||
assertNotNull(data);
|
assertNotNull(data);
|
||||||
assertEquals(200, data.length);
|
assertEquals(expectedSize, data.length);
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
connection.stop();
|
session.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
return dest;
|
return dest;
|
||||||
|
|
|
@ -64,7 +64,7 @@ public class KahaDBMessageStoreSizeStatTest extends
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeAfterRestartAndPublish() throws Exception {
|
public void testMessageSizeAfterRestartAndPublish() throws Exception {
|
||||||
|
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
@ -72,7 +72,7 @@ public class KahaDBMessageStoreSizeStatTest extends
|
||||||
// stop, restart broker and publish more messages
|
// stop, restart broker and publish more messages
|
||||||
stopBroker();
|
stopBroker();
|
||||||
this.setUpBroker(false);
|
this.setUpBroker(false);
|
||||||
dest = publishTestMessages(200);
|
dest = publishTestQueueMessages(200);
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest, 400, 400 * messageSize);
|
verifyStats(dest, 400, 400 * messageSize);
|
||||||
|
|
|
@ -84,7 +84,7 @@ public class MultiKahaDBMessageStoreSizeStatTest extends
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeAfterRestartAndPublish() throws Exception {
|
public void testMessageSizeAfterRestartAndPublish() throws Exception {
|
||||||
|
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
|
@ -92,7 +92,7 @@ public class MultiKahaDBMessageStoreSizeStatTest extends
|
||||||
// stop, restart broker and publish more messages
|
// stop, restart broker and publish more messages
|
||||||
stopBroker();
|
stopBroker();
|
||||||
this.setUpBroker(false);
|
this.setUpBroker(false);
|
||||||
dest = publishTestMessages(200);
|
dest = publishTestQueueMessages(200);
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest, 400, 400 * messageSize);
|
verifyStats(dest, 400, 400 * messageSize);
|
||||||
|
@ -102,13 +102,13 @@ public class MultiKahaDBMessageStoreSizeStatTest extends
|
||||||
@Test
|
@Test
|
||||||
public void testMessageSizeAfterRestartAndPublishMultiQueue() throws Exception {
|
public void testMessageSizeAfterRestartAndPublishMultiQueue() throws Exception {
|
||||||
|
|
||||||
Destination dest = publishTestMessages(200);
|
Destination dest = publishTestQueueMessages(200);
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest, 200, 200 * messageSize);
|
verifyStats(dest, 200, 200 * messageSize);
|
||||||
assertTrue(broker.getPersistenceAdapter().size() > 200 * messageSize);
|
assertTrue(broker.getPersistenceAdapter().size() > 200 * messageSize);
|
||||||
|
|
||||||
Destination dest2 = publishTestMessages(200, "test.queue2");
|
Destination dest2 = publishTestQueueMessages(200, "test.queue2");
|
||||||
|
|
||||||
// verify the count and size
|
// verify the count and size
|
||||||
verifyStats(dest2, 200, 200 * messageSize);
|
verifyStats(dest2, 200, 200 * messageSize);
|
||||||
|
@ -117,8 +117,8 @@ public class MultiKahaDBMessageStoreSizeStatTest extends
|
||||||
// stop, restart broker and publish more messages
|
// stop, restart broker and publish more messages
|
||||||
stopBroker();
|
stopBroker();
|
||||||
this.setUpBroker(false);
|
this.setUpBroker(false);
|
||||||
dest = publishTestMessages(200);
|
dest = publishTestQueueMessages(200);
|
||||||
dest2 = publishTestMessages(200, "test.queue2");
|
dest2 = publishTestQueueMessages(200, "test.queue2");
|
||||||
|
|
||||||
// verify the count and size after publishing messages
|
// verify the count and size after publishing messages
|
||||||
verifyStats(dest, 400, 400 * messageSize);
|
verifyStats(dest, 400, 400 * messageSize);
|
||||||
|
|
|
@ -18,16 +18,21 @@ package org.apache.activemq.store.memory;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.jms.Connection;
|
||||||
|
|
||||||
|
import org.apache.activemq.ActiveMQConnectionFactory;
|
||||||
import org.apache.activemq.broker.BrokerService;
|
import org.apache.activemq.broker.BrokerService;
|
||||||
|
import org.apache.activemq.broker.region.Destination;
|
||||||
import org.apache.activemq.store.AbstractMessageStoreSizeStatTest;
|
import org.apache.activemq.store.AbstractMessageStoreSizeStatTest;
|
||||||
|
import org.junit.Test;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This test checks that KahaDB properly sets the new storeMessageSize statistic.
|
* This test checks that KahaDB properly sets the new storeMessageSize statistic.
|
||||||
*
|
*
|
||||||
* AMQ-5748
|
* AMQ-5748
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class MemoryMessageStoreSizeStatTest extends AbstractMessageStoreSizeStatTest {
|
public class MemoryMessageStoreSizeStatTest extends AbstractMessageStoreSizeStatTest {
|
||||||
protected static final Logger LOG = LoggerFactory
|
protected static final Logger LOG = LoggerFactory
|
||||||
|
@ -39,7 +44,53 @@ public class MemoryMessageStoreSizeStatTest extends AbstractMessageStoreSizeStat
|
||||||
broker.setPersistenceAdapter(new MemoryPersistenceAdapter());
|
broker.setPersistenceAdapter(new MemoryPersistenceAdapter());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Test(timeout=10000)
|
||||||
|
public void testMessageSizeOneDurable() throws Exception {
|
||||||
|
Connection connection = new ActiveMQConnectionFactory(brokerConnectURI).createConnection();
|
||||||
|
connection.setClientID("clientId");
|
||||||
|
connection.start();
|
||||||
|
|
||||||
|
//The expected value is only 100 because for durables a LRUCache is being used
|
||||||
|
//with a max size of 100
|
||||||
|
Destination dest = publishTestMessagesDurable(connection, new String[] {"sub1"}, 200, 100);
|
||||||
|
|
||||||
|
//verify the count and size, should be 100 because of the LRUCache
|
||||||
|
verifyStats(dest, 100, 100 * messageSize);
|
||||||
|
|
||||||
|
consumeDurableTestMessages(connection, "sub1", 100);
|
||||||
|
|
||||||
|
//Since an LRU cache is used and messages are kept in memory, this should be 100 still
|
||||||
|
verifyStats(dest, 100, 100 * messageSize);
|
||||||
|
|
||||||
|
connection.stop();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Test(timeout=10000)
|
||||||
|
public void testMessageSizeTwoDurables() throws Exception {
|
||||||
|
Connection connection = new ActiveMQConnectionFactory(brokerConnectURI).createConnection();
|
||||||
|
connection.setClientID("clientId");
|
||||||
|
connection.start();
|
||||||
|
|
||||||
|
//The expected value is only 100 because for durables a LRUCache is being used
|
||||||
|
//with a max size of 100, so only 100 messages are kept
|
||||||
|
Destination dest = publishTestMessagesDurable(connection, new String[] {"sub1", "sub2"}, 200, 100);
|
||||||
|
|
||||||
|
//verify the count and size
|
||||||
|
verifyStats(dest, 100, 100 * messageSize);
|
||||||
|
|
||||||
|
//consume for sub1
|
||||||
|
consumeDurableTestMessages(connection, "sub1", 100);
|
||||||
|
|
||||||
|
//Should be 100 messages still
|
||||||
|
verifyStats(dest, 100, 100 * messageSize);
|
||||||
|
|
||||||
|
connection.stop();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue