added a hook to eagerly evict expired messages on non-durable topics first before we apply other eviction policies such as old messages etc

git-svn-id: https://svn.apache.org/repos/asf/incubator/activemq/trunk@392349 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
James Strachan 2006-04-07 17:58:17 +00:00
parent 46d62dff70
commit cc6fec68f4
7 changed files with 99 additions and 6 deletions

View File

@ -59,6 +59,8 @@ public class IndirectMessageReference implements MessageReference {
private int referenceCount; private int referenceCount;
/** the size of the message **/ /** the size of the message **/
private int cachedSize = 0; private int cachedSize = 0;
/** the expiration time of the message */
private long expiration;
/** /**
* Only used by the END_OF_BROWSE_MARKER singleton * Only used by the END_OF_BROWSE_MARKER singleton
@ -71,6 +73,7 @@ public class IndirectMessageReference implements MessageReference {
this.groupID = null; this.groupID = null;
this.groupSequence = 0; this.groupSequence = 0;
this.targetConsumerId=null; this.targetConsumerId=null;
this.expiration = message.getExpiration();
this.cachedSize = message != null ? message.getSize() : 0; this.cachedSize = message != null ? message.getSize() : 0;
} }
@ -82,6 +85,7 @@ public class IndirectMessageReference implements MessageReference {
this.groupID = message.getGroupID(); this.groupID = message.getGroupID();
this.groupSequence = message.getGroupSequence(); this.groupSequence = message.getGroupSequence();
this.targetConsumerId=message.getTargetConsumerId(); this.targetConsumerId=message.getTargetConsumerId();
this.expiration = message.getExpiration();
this.referenceCount=1; this.referenceCount=1;
message.incrementReferenceCount(); message.incrementReferenceCount();
@ -207,6 +211,18 @@ public class IndirectMessageReference implements MessageReference {
return targetConsumerId; return targetConsumerId;
} }
public long getExpiration() {
return expiration;
}
public boolean isExpired() {
long expireTime = getExpiration();
if (expireTime > 0 && System.currentTimeMillis() > expireTime) {
return true;
}
return false;
}
public int getSize(){ public int getSize(){
Message msg = message; Message msg = message;
if (msg != null){ if (msg != null){

View File

@ -47,5 +47,11 @@ public interface MessageReference {
public int decrementReferenceCount(); public int decrementReferenceCount();
public ConsumerId getTargetConsumerId(); public ConsumerId getTargetConsumerId();
public int getSize(); public int getSize();
public long getExpiration();
/**
* Returns true if this message is expired
*/
public boolean isExpired();
} }

View File

@ -73,6 +73,16 @@ public class TopicSubscription extends AbstractSubscription{
matched.addLast(node); matched.addLast(node);
// NOTE - be careful about the slaveBroker! // NOTE - be careful about the slaveBroker!
if (maximumPendingMessages > 0) { if (maximumPendingMessages > 0) {
// calculate the high water mark from which point we will eagerly evict expired messages
int max = messageEvictionStrategy.getEvictExpiredMessagesHighWatermark();
if (maximumPendingMessages > 0 && maximumPendingMessages < max) {
max = maximumPendingMessages;
}
if (!matched.isEmpty() && matched.size() > max) {
removeExpiredMessages(matched);
}
// 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 = messageEvictionStrategy.evictMessage(matched); MessageReference oldMessage = messageEvictionStrategy.evictMessage(matched);
@ -88,6 +98,22 @@ public class TopicSubscription extends AbstractSubscription{
} }
} }
/**
* Discard any expired messages from the matched list. Called from a synchronized block.
* @throws IOException
*/
protected void removeExpiredMessages(LinkedList messages) throws IOException {
for(Iterator i=matched.iterator();i.hasNext();){
MessageReference node=(MessageReference) i.next();
if (node.isExpired()) {
i.remove();
dispatched.incrementAndGet();
node.decrementReferenceCount();
break;
}
}
}
public void processMessageDispatchNotification(MessageDispatchNotification mdn){ public void processMessageDispatchNotification(MessageDispatchNotification mdn){
synchronized(matchedListMutex){ synchronized(matchedListMutex){
for(Iterator i=matched.iterator();i.hasNext();){ for(Iterator i=matched.iterator();i.hasNext();){

View File

@ -37,4 +37,9 @@ public interface MessageEvictionStrategy {
*/ */
MessageReference evictMessage(LinkedList messages) throws IOException; MessageReference evictMessage(LinkedList messages) throws IOException;
/**
* REturns the high water mark on which we will eagerly evict expired messages from RAM
*/
int getEvictExpiredMessagesHighWatermark();
} }

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;
/**
* A useful base class for implementation inheritence.
*
* @version $Revision$
*/
public abstract class MessageEvictionStrategySupport implements MessageEvictionStrategy {
private int evictExpiredMessagesHighWatermark = 1000;
public int getEvictExpiredMessagesHighWatermark() {
return evictExpiredMessagesHighWatermark;
}
/**
* Sets the high water mark on which we will eagerly evict expired messages from RAM
*/
public void setEvictExpiredMessagesHighWatermark(int evictExpiredMessagesHighWaterMark) {
this.evictExpiredMessagesHighWatermark = evictExpiredMessagesHighWaterMark;
}
}

View File

@ -28,7 +28,7 @@ import java.util.LinkedList;
* *
* @version $Revision$ * @version $Revision$
*/ */
public class OldestMessageEvictionStrategy implements MessageEvictionStrategy { public class OldestMessageEvictionStrategy extends MessageEvictionStrategySupport {
public MessageReference evictMessage(LinkedList messages) { public MessageReference evictMessage(LinkedList messages) {
return (MessageReference) messages.removeFirst(); return (MessageReference) messages.removeFirst();

View File

@ -29,7 +29,7 @@ import java.util.LinkedList;
* *
* @version $Revision$messageEvictionStrategy * @version $Revision$messageEvictionStrategy
*/ */
public class OldestMessageWithLowestPriorityEvictionStrategy implements MessageEvictionStrategy { public class OldestMessageWithLowestPriorityEvictionStrategy extends MessageEvictionStrategySupport {
public MessageReference evictMessage(LinkedList messages) throws IOException { public MessageReference evictMessage(LinkedList messages) throws IOException {
byte lowestPriority = Byte.MAX_VALUE; byte lowestPriority = Byte.MAX_VALUE;