mirror of https://github.com/apache/activemq.git
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:
parent
114706a7b4
commit
7c3bb40100
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue