added better MBeans for topic subscriptions so that they can see the discarded and matched statistics. Also added a configurable MessageEvictionStrategy

git-svn-id: https://svn.apache.org/repos/asf/incubator/activemq/trunk@382816 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
James Strachan 2006-03-03 13:56:29 +00:00
parent 4dad3fc833
commit 062dc5681c
7 changed files with 232 additions and 4 deletions

View File

@ -43,6 +43,7 @@ import org.apache.activemq.broker.region.Region;
import org.apache.activemq.broker.region.RegionBroker; import org.apache.activemq.broker.region.RegionBroker;
import org.apache.activemq.broker.region.Subscription; import org.apache.activemq.broker.region.Subscription;
import org.apache.activemq.broker.region.Topic; import org.apache.activemq.broker.region.Topic;
import org.apache.activemq.broker.region.TopicSubscription;
import org.apache.activemq.broker.region.policy.PolicyMap; import org.apache.activemq.broker.region.policy.PolicyMap;
import org.apache.activemq.command.ActiveMQDestination; import org.apache.activemq.command.ActiveMQDestination;
import org.apache.activemq.command.ActiveMQMessage; import org.apache.activemq.command.ActiveMQMessage;
@ -159,7 +160,12 @@ public class ManagedRegionBroker extends RegionBroker{
if(sub.getConsumerInfo().isDurable()){ if(sub.getConsumerInfo().isDurable()){
view=new DurableSubscriptionView(this,context.getClientId(),sub); view=new DurableSubscriptionView(this,context.getClientId(),sub);
}else{ }else{
view=new SubscriptionView(context.getClientId(),sub); if (sub instanceof TopicSubscription) {
view = new TopicSubscriptionView(context.getClientId(),(TopicSubscription) sub);
}
else {
view=new SubscriptionView(context.getClientId(),sub);
}
} }
subscriptionMap.put(sub,objectName); subscriptionMap.put(sub,objectName);
registerSubscription(objectName,sub.getConsumerInfo(),key,view); registerSubscription(objectName,sub.getConsumerInfo(),key,view);

View File

@ -0,0 +1,52 @@
/**
*
* Copyright 2005-2006 The Apache Software Foundation
*
* Licensed 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.broker.jmx;
import org.apache.activemq.broker.region.TopicSubscription;
/**
*
* @version $Revision$
*/
public class TopicSubscriptionView extends SubscriptionView implements TopicSubscriptionViewMBean {
public TopicSubscriptionView(String clientId, TopicSubscription subs) {
super(clientId, subs);
}
protected TopicSubscription getTopicSubscription() {
return (TopicSubscription) subscription;
}
/**
* @return the number of messages discarded due to being a slow consumer
*/
public int getDiscarded() {
TopicSubscription topicSubscription = getTopicSubscription();
return topicSubscription != null ? topicSubscription.discarded() : 0;
}
/**
* @return the number of matched messages (messages targeted for the
* subscription but not yet able to be dispatched due to the
* prefetch buffer being full).
*/
public int getMatched() {
TopicSubscription topicSubscription = getTopicSubscription();
return topicSubscription != null ? topicSubscription.matched() : 0;
}
}

View File

@ -0,0 +1,36 @@
/**
*
* Copyright 2005-2006 The Apache Software Foundation
*
* Licensed 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.broker.jmx;
/**
*
* @version $Revision$
*/
public interface TopicSubscriptionViewMBean extends SubscriptionViewMBean {
/**
* @return the number of messages discarded due to being a slow consumer
*/
public int getDiscarded();
/**
* @return the number of matched messages (messages targeted for the subscription but not
* yet able to be dispatched due to the prefetch buffer being full).
*/
public int getMatched();
}

View File

@ -20,6 +20,8 @@ import javax.jms.InvalidSelectorException;
import javax.jms.JMSException; import javax.jms.JMSException;
import org.apache.activemq.broker.Broker; import org.apache.activemq.broker.Broker;
import org.apache.activemq.broker.ConnectionContext; import org.apache.activemq.broker.ConnectionContext;
import org.apache.activemq.broker.region.policy.MessageEvictionStrategy;
import org.apache.activemq.broker.region.policy.OldestMessageEvictionStrategy;
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.ConsumerInfo; import org.apache.activemq.command.ConsumerInfo;
@ -42,6 +44,8 @@ public class TopicSubscription extends AbstractSubscription{
protected AtomicInteger dispatched=new AtomicInteger(); protected AtomicInteger dispatched=new AtomicInteger();
protected AtomicInteger delivered=new AtomicInteger(); protected AtomicInteger delivered=new AtomicInteger();
private int maximumPendingMessages=-1; private int maximumPendingMessages=-1;
private MessageEvictionStrategy messageEvictionStrategy = new OldestMessageEvictionStrategy();
private int discarded = 0;
private final Object matchedListMutex=new Object(); private final Object matchedListMutex=new Object();
public TopicSubscription(Broker broker,ConnectionContext context,ConsumerInfo info,UsageManager usageManager) public TopicSubscription(Broker broker,ConnectionContext context,ConsumerInfo info,UsageManager usageManager)
@ -64,8 +68,9 @@ public class TopicSubscription extends AbstractSubscription{
if(maximumPendingMessages>0){ if(maximumPendingMessages>0){
// lets discard old messages as we are a slow consumer // lets discard old messages as we are a slow consumer
while(!matched.isEmpty()&&matched.size()>maximumPendingMessages){ while(!matched.isEmpty()&&matched.size()>maximumPendingMessages){
MessageReference oldMessage=(MessageReference) matched.removeFirst(); MessageReference oldMessage=messageEvictionStrategy.evictMessage(matched);
oldMessage.decrementReferenceCount(); oldMessage.decrementReferenceCount();
discarded++;
if (log.isDebugEnabled()){ if (log.isDebugEnabled()){
log.debug("Discarding message " + oldMessage); log.debug("Discarding message " + oldMessage);
} }
@ -122,7 +127,7 @@ public class TopicSubscription extends AbstractSubscription{
} }
public int pending(){ public int pending(){
return matched.size()-dispatched.get(); return matched()-dispatched();
} }
public int dispatched(){ public int dispatched(){
@ -137,6 +142,26 @@ public class TopicSubscription extends AbstractSubscription{
return maximumPendingMessages; return maximumPendingMessages;
} }
/**
* @return the number of messages discarded due to being a slow consumer
*/
public int discarded() {
synchronized(matchedListMutex) {
return discarded;
}
}
/**
* @return the number of matched messages (messages targeted for the subscription but not
* yet able to be dispatched due to the prefetch buffer being full).
*/
public int matched() {
synchronized(matchedListMutex) {
return matched.size();
}
}
/** /**
* Sets the maximum number of pending messages that can be matched against this consumer before old messages are * Sets the maximum number of pending messages that can be matched against this consumer before old messages are
* discarded. * discarded.
@ -145,6 +170,22 @@ public class TopicSubscription extends AbstractSubscription{
this.maximumPendingMessages=maximumPendingMessages; this.maximumPendingMessages=maximumPendingMessages;
} }
public MessageEvictionStrategy getMessageEvictionStrategy() {
return messageEvictionStrategy;
}
/**
* Sets the eviction strategy used to decide which message to evict when the slow consumer
* needs to discard messages
*/
public void setMessageEvictionStrategy(MessageEvictionStrategy messageEvictionStrategy) {
this.messageEvictionStrategy = messageEvictionStrategy;
}
// Implementation methods
// -------------------------------------------------------------------------
private boolean isFull(){ private boolean isFull(){
return dispatched.get()-delivered.get()>=info.getPrefetchSize(); return dispatched.get()-delivered.get()>=info.getPrefetchSize();
} }
@ -182,6 +223,6 @@ public class TopicSubscription extends AbstractSubscription{
public String toString(){ public String toString(){
return "TopicSubscription:"+" consumer="+info.getConsumerId()+", destinations="+destinations.size() return "TopicSubscription:"+" consumer="+info.getConsumerId()+", destinations="+destinations.size()
+", dispatched="+dispatched+", delivered="+this.delivered+", matched="+this.matched.size(); +", dispatched="+dispatched()+", delivered="+delivered()+", matched="+matched()+", discarded="+discarded();
} }
} }

View File

@ -0,0 +1,40 @@
/**
*
* Copyright 2005-2006 The Apache Software Foundation
*
* Licensed 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.broker.region.policy;
import org.apache.activemq.broker.region.MessageReference;
import java.io.IOException;
import java.util.LinkedList;
/**
* A strategy for evicting messages from slow consumers.
*
* @version $Revision$
*/
public interface MessageEvictionStrategy {
/**
* Find the message reference in the given list with oldest messages at the front and newer messages at the end
*
* @return the message that has been evicted.
* @throws IOException if an exception occurs such as reading a message content (but should not ever happen
* as usually all the messages will be in RAM when this method is called).
*/
MessageReference evictMessage(LinkedList messages) throws IOException;
}

View File

@ -0,0 +1,36 @@
/**
*
* Copyright 2005-2006 The Apache Software Foundation
*
* Licensed 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.broker.region.policy;
import org.apache.activemq.broker.region.MessageReference;
import java.util.LinkedList;
/**
* An eviction strategy which evicts the oldest message first (which is the
* default).
*
* @org.apache.xbean.XBean
*
* @version $Revision$
*/
public class OldestMessageEvictionStrategy implements MessageEvictionStrategy {
public MessageReference evictMessage(LinkedList messages) {
return (MessageReference) messages.removeFirst();
}
}

View File

@ -41,6 +41,7 @@ public class PolicyEntry extends DestinationMapEntry {
private DeadLetterStrategy deadLetterStrategy; private DeadLetterStrategy deadLetterStrategy;
private int messageGroupHashBucketCount = 1024; private int messageGroupHashBucketCount = 1024;
private PendingMessageLimitStrategy pendingMessageLimitStrategy; private PendingMessageLimitStrategy pendingMessageLimitStrategy;
private MessageEvictionStrategy messageEvictionStrategy;
public void configure(Queue queue) { public void configure(Queue queue) {
if (dispatchPolicy != null) { if (dispatchPolicy != null) {
@ -81,6 +82,9 @@ public class PolicyEntry extends DestinationMapEntry {
subscription.setMaximumPendingMessages(value); subscription.setMaximumPendingMessages(value);
} }
} }
if (messageEvictionStrategy != null) {
subscription.setMessageEvictionStrategy(messageEvictionStrategy);
}
} }
// Properties // Properties
@ -156,4 +160,17 @@ public class PolicyEntry extends DestinationMapEntry {
public void setPendingMessageLimitStrategy(PendingMessageLimitStrategy pendingMessageLimitStrategy) { public void setPendingMessageLimitStrategy(PendingMessageLimitStrategy pendingMessageLimitStrategy) {
this.pendingMessageLimitStrategy = pendingMessageLimitStrategy; this.pendingMessageLimitStrategy = pendingMessageLimitStrategy;
} }
public MessageEvictionStrategy getMessageEvictionStrategy() {
return messageEvictionStrategy;
}
/**
* Sets the eviction strategy used to decide which message to evict when the
* slow consumer needs to discard messages
*/
public void setMessageEvictionStrategy(MessageEvictionStrategy messageEvictionStrategy) {
this.messageEvictionStrategy = messageEvictionStrategy;
}
} }