ReduceMemoryFootprint now applies to non-persistent messages if they
have been marshalled and topics now clear memory after the recovery
policy check
This commit is contained in:
Christopher L. Shannon (cshannon) 2016-10-25 10:17:04 -04:00
parent 114706a7b4
commit 7c3bb40100
8 changed files with 236 additions and 15 deletions

View File

@ -839,9 +839,6 @@ public class Queue extends BaseDestination implements Task, UsageListener, Index
} else { } else {
store.addMessage(context, message); store.addMessage(context, message);
} }
if (isReduceMemoryFootprint()) {
message.clearMarshalledState();
}
} catch (Exception e) { } catch (Exception e) {
// we may have a store in inconsistent state, so reset the cursor // we may have a store in inconsistent state, so reset the cursor
// before restarting normal broker operations // before restarting normal broker operations
@ -849,6 +846,13 @@ public class Queue extends BaseDestination implements Task, UsageListener, Index
throw e; throw e;
} }
} }
//Clear the unmarshalled state if the message is marshalled
//Persistent messages will always be marshalled but non-persistent may not be
//Specially non-persistent messages over the VM transport won't be
if (isReduceMemoryFootprint() && message.isMarshalled()) {
message.clearUnMarshalledState();
}
if(tryOrderedCursorAdd(message, context)) { if(tryOrderedCursorAdd(message, context)) {
break; break;
} }

View File

@ -516,9 +516,7 @@ public class Topic extends BaseDestination implements Task {
} }
result = topicStore.asyncAddTopicMessage(context, message,isOptimizeStorage()); result = topicStore.asyncAddTopicMessage(context, message,isOptimizeStorage());
if (isReduceMemoryFootprint()) { //Moved the reduceMemoryfootprint clearing to the dispatch method
message.clearMarshalledState();
}
} }
message.incrementReferenceCount(); message.incrementReferenceCount();
@ -758,6 +756,13 @@ public class Topic extends BaseDestination implements Task {
return; return;
} }
} }
// Clear memory before dispatch - need to clear here because the call to
//subscriptionRecoveryPolicy.add() will unmarshall the state
if (isReduceMemoryFootprint() && message.isMarshalled()) {
message.clearUnMarshalledState();
}
msgContext = context.getMessageEvaluationContext(); msgContext = context.getMessageEvaluationContext();
msgContext.setDestination(destination); msgContext.setDestination(destination);
msgContext.setMessageReference(message); msgContext.setMessageReference(message);

View File

@ -130,8 +130,8 @@ public class ActiveMQMapMessage extends ActiveMQMessage implements MapMessage {
} }
@Override @Override
public void clearMarshalledState() throws JMSException { public void clearUnMarshalledState() throws JMSException {
super.clearMarshalledState(); super.clearUnMarshalledState();
map.clear(); map.clear();
} }

View File

@ -224,8 +224,8 @@ public class ActiveMQObjectMessage extends ActiveMQMessage implements ObjectMess
} }
@Override @Override
public void clearMarshalledState() throws JMSException { public void clearUnMarshalledState() throws JMSException {
super.clearMarshalledState(); super.clearUnMarshalledState();
this.object = null; this.object = null;
} }

View File

@ -153,8 +153,8 @@ public class ActiveMQTextMessage extends ActiveMQMessage implements TextMessage
// see https://issues.apache.org/activemq/browse/AMQ-2103 // see https://issues.apache.org/activemq/browse/AMQ-2103
// and https://issues.apache.org/activemq/browse/AMQ-2966 // and https://issues.apache.org/activemq/browse/AMQ-2966
@Override @Override
public void clearMarshalledState() throws JMSException { public void clearUnMarshalledState() throws JMSException {
super.clearMarshalledState(); super.clearUnMarshalledState();
this.text = null; this.text = null;
} }

View File

@ -110,11 +110,24 @@ public abstract class Message extends BaseCommand implements MarshallAware, Mess
public abstract void storeContent(); public abstract void storeContent();
public abstract void storeContentAndClear(); public abstract void storeContentAndClear();
// useful to reduce the memory footprint of a persisted message /**
* @deprecated - This method name is misnamed
* @throws JMSException
*/
public void clearMarshalledState() throws JMSException { public void clearMarshalledState() throws JMSException {
clearUnMarshalledState();
}
// useful to reduce the memory footprint of a persisted message
public void clearUnMarshalledState() throws JMSException {
properties = null; properties = null;
} }
public boolean isMarshalled() {
return content != null && (marshalledProperties != null ||
(marshalledProperties == null && properties == null));
}
protected void copy(Message copy) { protected void copy(Message copy) {
super.copy(copy); super.copy(copy);
copy.producerId = producerId; copy.producerId = producerId;

View File

@ -21,7 +21,7 @@ import javax.jms.Message;
import javax.jms.MessageConsumer; import javax.jms.MessageConsumer;
import javax.jms.MessageProducer; import javax.jms.MessageProducer;
import javax.jms.Session; import javax.jms.Session;
import junit.framework.Test;
import org.apache.activemq.ActiveMQConnectionFactory; import org.apache.activemq.ActiveMQConnectionFactory;
import org.apache.activemq.ActiveMQSession; import org.apache.activemq.ActiveMQSession;
import org.apache.activemq.broker.BrokerTestSupport; import org.apache.activemq.broker.BrokerTestSupport;
@ -32,7 +32,16 @@ import org.apache.activemq.command.ActiveMQObjectMessage;
import org.apache.activemq.command.ActiveMQQueue; import org.apache.activemq.command.ActiveMQQueue;
import org.apache.activemq.command.ActiveMQTextMessage; import org.apache.activemq.command.ActiveMQTextMessage;
import org.apache.activemq.usecases.MyObject; import org.apache.activemq.usecases.MyObject;
import org.junit.Ignore;
import junit.framework.Test;
/**
* AMQ-6477 changes the behavior to only clear memory if the marshalled state exists
* so this test no longer works
*/
@Ignore
public class AMQ2103Test extends BrokerTestSupport { public class AMQ2103Test extends BrokerTestSupport {
static PolicyEntry reduceMemoryFootprint = new PolicyEntry(); static PolicyEntry reduceMemoryFootprint = new PolicyEntry();
static { static {
@ -47,7 +56,7 @@ public class AMQ2103Test extends BrokerTestSupport {
} }
public void initCombosForTestVerifyMarshalledStateIsCleared() throws Exception { public void initCombosForTestVerifyMarshalledStateIsCleared() throws Exception {
addCombinationValues("defaultPolicy", new Object[]{defaultPolicy, null}); addCombinationValues("defaultPolicy", new Object[]{defaultPolicy, null});
} }
public static Test suite() { public static Test suite() {
@ -60,6 +69,8 @@ public class AMQ2103Test extends BrokerTestSupport {
* With vm transport and deferred serialisation and no persistence (mem persistence), * With vm transport and deferred serialisation and no persistence (mem persistence),
* we see the message as sent by the client so we can validate the contents against * we see the message as sent by the client so we can validate the contents against
* the policy * the policy
*
*
* @throws Exception * @throws Exception
*/ */
public void testVerifyMarshalledStateIsCleared() throws Exception { public void testVerifyMarshalledStateIsCleared() throws Exception {

View File

@ -0,0 +1,188 @@
/*
* 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.usecases;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertNull;
import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import javax.jms.Connection;
import javax.jms.MessageConsumer;
import javax.jms.MessageProducer;
import javax.jms.Session;
import javax.jms.TextMessage;
import org.apache.activemq.ActiveMQConnectionFactory;
import org.apache.activemq.broker.BrokerService;
import org.apache.activemq.broker.TransportConnector;
import org.apache.activemq.broker.region.MessageReference;
import org.apache.activemq.broker.region.PrefetchSubscription;
import org.apache.activemq.broker.region.Subscription;
import org.apache.activemq.broker.region.TopicSubscription;
import org.apache.activemq.broker.region.policy.PolicyEntry;
import org.apache.activemq.broker.region.policy.PolicyMap;
import org.apache.activemq.command.ActiveMQDestination;
import org.apache.activemq.command.ActiveMQQueue;
import org.apache.activemq.command.ActiveMQTextMessage;
import org.apache.activemq.command.ActiveMQTopic;
import org.apache.activemq.command.Message;
import org.junit.After;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Test that clearUnMarshalled data gets called properly to reduce memory usage
*/
@RunWith(Parameterized.class)
public class AMQ6477Test {
static final Logger LOG = LoggerFactory.getLogger(AMQ6477Test.class);
@Rule
public TemporaryFolder dataDir = new TemporaryFolder();
private BrokerService brokerService;
private String connectionUri;
private final ActiveMQQueue queue = new ActiveMQQueue("queue");
private final ActiveMQTopic topic = new ActiveMQTopic("topic");
private final int numMessages = 10;
private Connection connection;
private Session session;
private SubType subType;
private boolean persistent;
protected enum SubType {QUEUE, TOPIC, DURABLE};
@Parameters(name="subType={0},isPersistent={1}")
public static Collection<Object[]> data() {
return Arrays.asList(new Object[][] {
{SubType.QUEUE, false},
{SubType.TOPIC, false},
{SubType.DURABLE, false},
{SubType.QUEUE, true},
{SubType.TOPIC, true},
{SubType.DURABLE, true}
});
}
/**
*/
public AMQ6477Test(SubType subType, boolean persistent) {
super();
this.subType = subType;
this.persistent = persistent;
}
@Before
public void before() throws Exception {
brokerService = new BrokerService();
TransportConnector connector = brokerService.addConnector("tcp://localhost:0");
connectionUri = connector.getPublishableConnectString();
brokerService.setPersistent(persistent);
brokerService.setDataDirectory(dataDir.getRoot().getAbsolutePath());
brokerService.setUseJmx(false);
PolicyMap policyMap = new PolicyMap();
PolicyEntry entry = new PolicyEntry();
entry.setReduceMemoryFootprint(true);
policyMap.setDefaultEntry(entry);
brokerService.setDestinationPolicy(policyMap);
brokerService.start();
brokerService.waitUntilStarted();
final ActiveMQConnectionFactory factory = new ActiveMQConnectionFactory(connectionUri);
connection = factory.createConnection();
connection.setClientID("clientId");
connection.start();
session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
}
@After
public void after() throws Exception {
if (connection != null) {
connection.stop();
}
if (brokerService != null) {
brokerService.stop();
brokerService.waitUntilStopped();
}
}
@Test(timeout=30000)
public void testReduceMemoryFootprint() throws Exception {
ActiveMQDestination destination = subType.equals(SubType.QUEUE) ? queue : topic;
MessageConsumer consumer = subType.equals(SubType.DURABLE) ?
session.createDurableSubscriber(topic, "sub1") : session.createConsumer(destination);
MessageProducer producer = session.createProducer(destination);
for (int i = 0; i < numMessages; i++) {
TextMessage m = session.createTextMessage("test");
m.setStringProperty("test", "test");
producer.send(m);
}
Subscription sub = brokerService.getDestination(destination).getConsumers().get(0);
List<MessageReference> messages = getSubscriptionMessages(sub);
//Go through each message and make sure the unmarshalled fields are null
//then call the getters which will unmarshall the data again to show the marshalled
//data exists
for (MessageReference ref : messages) {
ActiveMQTextMessage message = (ActiveMQTextMessage) ref.getMessage();
Field propertiesField = Message.class.getDeclaredField("properties");
propertiesField.setAccessible(true);
Field textField = ActiveMQTextMessage.class.getDeclaredField("text");
textField.setAccessible(true);
assertNull(textField.get(message));
assertNull(propertiesField.get(message));
assertNotNull(message.getProperties());
assertNotNull(message.getText());
}
consumer.close();
}
@SuppressWarnings("unchecked")
protected List<MessageReference> getSubscriptionMessages(Subscription sub) throws Exception {
Field f = null;
if (sub instanceof TopicSubscription) {
f = TopicSubscription.class.getDeclaredField("dispatched");
} else {
f = PrefetchSubscription.class.getDeclaredField("dispatched");
}
f.setAccessible(true);
return (List<MessageReference>) f.get(sub);
}
}