This closes #777
This commit is contained in:
commit
4e444d53f9
artemis-commons/src
main/java/org/apache/activemq/artemis/core/server
test/java/org/apache/activemq/artemis/utils
artemis-jdbc-store/src/main/java/org/apache/activemq/artemis/jdbc/store/journal
artemis-server/src/main/java/org/apache/activemq/artemis/core
paging/impl
postoffice/impl
server
tests/integration-tests/src/test/java/org/apache/activemq/artemis/tests/integration/cluster/distribution
|
@ -28,14 +28,18 @@ import org.jboss.logging.Logger;
|
|||
/** This is for components with a scheduled at a fixed rate. */
|
||||
public abstract class ActiveMQScheduledComponent implements ActiveMQComponent, Runnable {
|
||||
|
||||
|
||||
private static final Logger logger = Logger.getLogger(ActiveMQScheduledComponent.class);
|
||||
private final ScheduledExecutorService scheduledExecutorService;
|
||||
private long period;
|
||||
private long millisecondsPeriod;
|
||||
private TimeUnit timeUnit;
|
||||
private final Executor executor;
|
||||
private ScheduledFuture future;
|
||||
private final boolean onDemand;
|
||||
|
||||
long lastTime = 0;
|
||||
|
||||
private final AtomicInteger delayed = new AtomicInteger(0);
|
||||
|
||||
public ActiveMQScheduledComponent(ScheduledExecutorService scheduledExecutorService,
|
||||
|
@ -58,6 +62,8 @@ public abstract class ActiveMQScheduledComponent implements ActiveMQComponent, R
|
|||
if (future != null) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.millisecondsPeriod = timeUnit.convert(period, TimeUnit.MILLISECONDS);
|
||||
if (onDemand) {
|
||||
return;
|
||||
}
|
||||
|
@ -113,11 +119,6 @@ public abstract class ActiveMQScheduledComponent implements ActiveMQComponent, R
|
|||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
delayed.decrementAndGet();
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized boolean isStarted() {
|
||||
return future != null;
|
||||
|
@ -132,10 +133,30 @@ public abstract class ActiveMQScheduledComponent implements ActiveMQComponent, R
|
|||
}
|
||||
}
|
||||
|
||||
final Runnable runForExecutor = new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
if (onDemand && delayed.get() > 0) {
|
||||
delayed.decrementAndGet();
|
||||
}
|
||||
|
||||
if (!onDemand && lastTime > 0) {
|
||||
if (System.currentTimeMillis() - lastTime < millisecondsPeriod) {
|
||||
logger.trace("Execution ignored due to too many simultaneous executions, probably a previous delayed execution");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
lastTime = System.currentTimeMillis();
|
||||
|
||||
ActiveMQScheduledComponent.this.run();
|
||||
}
|
||||
};
|
||||
|
||||
final Runnable runForScheduler = new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
executor.execute(ActiveMQScheduledComponent.this);
|
||||
executor.execute(runForExecutor);
|
||||
}
|
||||
};
|
||||
|
||||
|
|
|
@ -0,0 +1,80 @@
|
|||
/**
|
||||
* 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.artemis.utils;
|
||||
|
||||
import java.util.concurrent.ExecutorService;
|
||||
import java.util.concurrent.Executors;
|
||||
import java.util.concurrent.ScheduledExecutorService;
|
||||
import java.util.concurrent.ScheduledThreadPoolExecutor;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.atomic.AtomicInteger;
|
||||
|
||||
import org.apache.activemq.artemis.core.server.ActiveMQScheduledComponent;
|
||||
import org.junit.After;
|
||||
import org.junit.Assert;
|
||||
import org.junit.Before;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
|
||||
public class ActiveMQScheduledComponentTest {
|
||||
|
||||
@Rule
|
||||
public ThreadLeakCheckRule rule = new ThreadLeakCheckRule();
|
||||
|
||||
ScheduledExecutorService scheduledExecutorService;
|
||||
ExecutorService executorService;
|
||||
@Before
|
||||
public void before() {
|
||||
scheduledExecutorService = new ScheduledThreadPoolExecutor(5);
|
||||
executorService = Executors.newSingleThreadExecutor();
|
||||
}
|
||||
|
||||
@After
|
||||
public void after() {
|
||||
executorService.shutdown();
|
||||
scheduledExecutorService.shutdown();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testAccumulation() throws Exception {
|
||||
final AtomicInteger count = new AtomicInteger(0);
|
||||
|
||||
|
||||
final ActiveMQScheduledComponent local = new ActiveMQScheduledComponent(scheduledExecutorService, executorService, 100, TimeUnit.MILLISECONDS, false) {
|
||||
public void run() {
|
||||
if (count.get() == 0) {
|
||||
try {
|
||||
Thread.sleep(800);
|
||||
}
|
||||
catch (Exception e) {
|
||||
}
|
||||
}
|
||||
count.incrementAndGet();
|
||||
}
|
||||
};
|
||||
|
||||
local.start();
|
||||
|
||||
Thread.sleep(1000);
|
||||
|
||||
local.stop();
|
||||
|
||||
Assert.assertTrue("just because one took a lot of time, it doesn't mean we can accumulate many, we got " + count + " executions", count.get() < 5);
|
||||
}
|
||||
|
||||
}
|
|
@ -38,7 +38,6 @@ public class JDBCJournalSync extends ActiveMQScheduledComponent {
|
|||
|
||||
@Override
|
||||
public void run() {
|
||||
super.run();
|
||||
if (journal.isStarted()) {
|
||||
journal.sync();
|
||||
}
|
||||
|
|
|
@ -78,7 +78,6 @@ final class PageSyncTimer extends ActiveMQScheduledComponent {
|
|||
|
||||
@Override
|
||||
public void run() {
|
||||
super.run();
|
||||
tick();
|
||||
}
|
||||
|
||||
|
|
|
@ -27,13 +27,13 @@ import java.util.Map;
|
|||
import java.util.Set;
|
||||
import java.util.concurrent.ConcurrentHashMap;
|
||||
import java.util.concurrent.ConcurrentMap;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.Executor;
|
||||
import java.util.concurrent.ScheduledExecutorService;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
|
||||
import org.apache.activemq.artemis.api.core.ActiveMQAddressFullException;
|
||||
import org.apache.activemq.artemis.api.core.ActiveMQDuplicateIdException;
|
||||
import org.apache.activemq.artemis.api.core.ActiveMQInterruptedException;
|
||||
import org.apache.activemq.artemis.api.core.ActiveMQNonExistentQueueException;
|
||||
import org.apache.activemq.artemis.api.core.Message;
|
||||
import org.apache.activemq.artemis.api.core.Pair;
|
||||
|
@ -57,6 +57,7 @@ import org.apache.activemq.artemis.core.postoffice.PostOffice;
|
|||
import org.apache.activemq.artemis.core.postoffice.QueueInfo;
|
||||
import org.apache.activemq.artemis.core.postoffice.RoutingStatus;
|
||||
import org.apache.activemq.artemis.core.server.ActiveMQMessageBundle;
|
||||
import org.apache.activemq.artemis.core.server.ActiveMQScheduledComponent;
|
||||
import org.apache.activemq.artemis.core.server.ActiveMQServer;
|
||||
import org.apache.activemq.artemis.core.server.ActiveMQServerLogger;
|
||||
import org.apache.activemq.artemis.core.server.LargeServerMessage;
|
||||
|
@ -112,8 +113,6 @@ public class PostOfficeImpl implements PostOffice, NotificationListener, Binding
|
|||
|
||||
private Reaper reaperRunnable;
|
||||
|
||||
private volatile Thread reaperThread;
|
||||
|
||||
private final long reaperPeriod;
|
||||
|
||||
private final int reaperPriority;
|
||||
|
@ -198,12 +197,6 @@ public class PostOfficeImpl implements PostOffice, NotificationListener, Binding
|
|||
if (reaperRunnable != null)
|
||||
reaperRunnable.stop();
|
||||
|
||||
if (reaperThread != null) {
|
||||
reaperThread.join();
|
||||
|
||||
reaperThread = null;
|
||||
}
|
||||
|
||||
addressManager.clear();
|
||||
|
||||
queueInfos.clear();
|
||||
|
@ -1244,12 +1237,9 @@ public class PostOfficeImpl implements PostOffice, NotificationListener, Binding
|
|||
if (reaperPeriod > 0) {
|
||||
if (reaperRunnable != null)
|
||||
reaperRunnable.stop();
|
||||
reaperRunnable = new Reaper();
|
||||
reaperThread = new Thread(reaperRunnable, "activemq-expiry-reaper-thread");
|
||||
reaperRunnable = new Reaper(server.getScheduledPool(), server.getExecutorFactory().getExecutor(), reaperPeriod, TimeUnit.MILLISECONDS, false);
|
||||
|
||||
reaperThread.setPriority(reaperPriority);
|
||||
|
||||
reaperThread.start();
|
||||
reaperRunnable.start();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1268,48 +1258,38 @@ public class PostOfficeImpl implements PostOffice, NotificationListener, Binding
|
|||
return message;
|
||||
}
|
||||
|
||||
private final class Reaper implements Runnable {
|
||||
private final class Reaper extends ActiveMQScheduledComponent {
|
||||
|
||||
private final CountDownLatch latch = new CountDownLatch(1);
|
||||
|
||||
public void stop() {
|
||||
latch.countDown();
|
||||
Reaper(ScheduledExecutorService scheduledExecutorService,
|
||||
Executor executor,
|
||||
long checkPeriod,
|
||||
TimeUnit timeUnit,
|
||||
boolean onDemand) {
|
||||
super(scheduledExecutorService, executor, checkPeriod, timeUnit, onDemand);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
// The reaper thread should be finished case the PostOffice is gone
|
||||
// This is to avoid leaks on PostOffice between stops and starts
|
||||
while (isStarted()) {
|
||||
Map<SimpleString, Binding> nameMap = addressManager.getBindings();
|
||||
|
||||
List<Queue> queues = new ArrayList<>();
|
||||
|
||||
for (Binding binding : nameMap.values()) {
|
||||
if (binding.getType() == BindingType.LOCAL_QUEUE) {
|
||||
Queue queue = (Queue) binding.getBindable();
|
||||
|
||||
queues.add(queue);
|
||||
}
|
||||
}
|
||||
|
||||
for (Queue queue : queues) {
|
||||
try {
|
||||
if (latch.await(reaperPeriod, TimeUnit.MILLISECONDS))
|
||||
return;
|
||||
queue.expireReferences();
|
||||
}
|
||||
catch (InterruptedException e1) {
|
||||
throw new ActiveMQInterruptedException(e1);
|
||||
}
|
||||
if (!isStarted())
|
||||
return;
|
||||
|
||||
Map<SimpleString, Binding> nameMap = addressManager.getBindings();
|
||||
|
||||
List<Queue> queues = new ArrayList<>();
|
||||
|
||||
for (Binding binding : nameMap.values()) {
|
||||
if (binding.getType() == BindingType.LOCAL_QUEUE) {
|
||||
Queue queue = (Queue) binding.getBindable();
|
||||
|
||||
queues.add(queue);
|
||||
}
|
||||
}
|
||||
|
||||
for (Queue queue : queues) {
|
||||
try {
|
||||
queue.expireReferences();
|
||||
}
|
||||
catch (Exception e) {
|
||||
ActiveMQServerLogger.LOGGER.errorExpiringMessages(e);
|
||||
}
|
||||
catch (Exception e) {
|
||||
ActiveMQServerLogger.LOGGER.errorExpiringMessages(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -74,7 +74,6 @@ public class FileStoreMonitor extends ActiveMQScheduledComponent {
|
|||
|
||||
@Override
|
||||
public void run() {
|
||||
super.run();
|
||||
tick();
|
||||
}
|
||||
|
||||
|
|
|
@ -1100,13 +1100,17 @@ public class QueueImpl implements Queue {
|
|||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void expire(final MessageReference ref) throws Exception {
|
||||
if (expiryAddress != null) {
|
||||
SimpleString messageExpiryAddress = expiryAddressFromMessageAddress(ref);
|
||||
if (messageExpiryAddress == null) {
|
||||
messageExpiryAddress = expiryAddressFromAddressSettings(ref);
|
||||
}
|
||||
|
||||
if (messageExpiryAddress != null) {
|
||||
if (logger.isTraceEnabled()) {
|
||||
logger.trace("moving expired reference " + ref + " to address = " + expiryAddress + " from queue=" + this.getName());
|
||||
logger.trace("moving expired reference " + ref + " to address = " + messageExpiryAddress + " from queue=" + this.getName());
|
||||
}
|
||||
move(null, expiryAddress, ref, false, AckReason.EXPIRED);
|
||||
move(null, messageExpiryAddress, ref, false, AckReason.EXPIRED);
|
||||
}
|
||||
else {
|
||||
if (logger.isTraceEnabled()) {
|
||||
|
@ -1116,6 +1120,40 @@ public class QueueImpl implements Queue {
|
|||
}
|
||||
}
|
||||
|
||||
private SimpleString expiryAddressFromMessageAddress(MessageReference ref) {
|
||||
SimpleString messageAddress = extractAddress(ref.getMessage());
|
||||
SimpleString expiryAddress = null;
|
||||
|
||||
if (messageAddress == null || messageAddress.equals(getAddress())) {
|
||||
expiryAddress = getExpiryAddress();
|
||||
}
|
||||
|
||||
return expiryAddress;
|
||||
}
|
||||
|
||||
private SimpleString expiryAddressFromAddressSettings(MessageReference ref) {
|
||||
SimpleString messageAddress = extractAddress(ref.getMessage());
|
||||
SimpleString expiryAddress = null;
|
||||
|
||||
if (messageAddress != null) {
|
||||
AddressSettings addressSettings = addressSettingsRepository.getMatch(messageAddress.toString());
|
||||
|
||||
expiryAddress = addressSettings.getExpiryAddress();
|
||||
}
|
||||
|
||||
return expiryAddress;
|
||||
}
|
||||
|
||||
private SimpleString extractAddress(ServerMessage message) {
|
||||
if (message.containsProperty(Message.HDR_ORIG_MESSAGE_ID)) {
|
||||
return message.getSimpleStringProperty(Message.HDR_ORIGINAL_ADDRESS);
|
||||
}
|
||||
else {
|
||||
return message.getAddress();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public SimpleString getExpiryAddress() {
|
||||
return this.expiryAddress;
|
||||
|
|
|
@ -44,7 +44,6 @@ public class ReloadManagerImpl extends ActiveMQScheduledComponent implements Rel
|
|||
|
||||
@Override
|
||||
public void run() {
|
||||
super.run();
|
||||
tick();
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,115 @@
|
|||
/**
|
||||
* 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.artemis.tests.integration.cluster.distribution;
|
||||
|
||||
import org.apache.activemq.artemis.api.core.SimpleString;
|
||||
import org.apache.activemq.artemis.api.core.client.ClientConsumer;
|
||||
import org.apache.activemq.artemis.api.core.client.ClientMessage;
|
||||
import org.apache.activemq.artemis.api.core.client.ClientProducer;
|
||||
import org.apache.activemq.artemis.api.core.client.ClientSession;
|
||||
import org.apache.activemq.artemis.api.core.client.ClientSessionFactory;
|
||||
import org.apache.activemq.artemis.core.server.cluster.ClusterConnection;
|
||||
import org.apache.activemq.artemis.core.server.cluster.MessageFlowRecord;
|
||||
import org.apache.activemq.artemis.core.server.cluster.impl.ClusterConnectionImpl;
|
||||
import org.apache.activemq.artemis.core.server.cluster.impl.MessageLoadBalancingType;
|
||||
import org.apache.activemq.artemis.core.settings.impl.AddressSettings;
|
||||
import org.junit.Assert;
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
|
||||
public class ExpireWhileLoadBalanceTest extends ClusterTestBase {
|
||||
|
||||
@Before
|
||||
public void setUp() throws Exception {
|
||||
super.setUp();
|
||||
|
||||
setupServer(0, isFileStorage(), true);
|
||||
setupServer(1, isFileStorage(), true);
|
||||
setupServer(2, isFileStorage(), true);
|
||||
|
||||
for (int i = 0; i < 3; i++) {
|
||||
servers[i].getConfiguration().setMessageExpiryScanPeriod(100);
|
||||
}
|
||||
|
||||
setupClusterConnection("cluster0", "queues", MessageLoadBalancingType.STRICT, 1, true, 0, 1, 2);
|
||||
|
||||
setupClusterConnection("cluster1", "queues", MessageLoadBalancingType.STRICT, 1, true, 1, 0, 2);
|
||||
|
||||
setupClusterConnection("cluster2", "queues", MessageLoadBalancingType.STRICT, 1, true, 2, 0, 1);
|
||||
|
||||
startServers(0, 1, 2);
|
||||
|
||||
setupSessionFactory(0, true);
|
||||
setupSessionFactory(1, true);
|
||||
setupSessionFactory(2, true);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testSend() throws Exception {
|
||||
waitForTopology(getServer(0), 3);
|
||||
waitForTopology(getServer(1), 3);
|
||||
waitForTopology(getServer(2), 3);
|
||||
|
||||
SimpleString expiryQueue = SimpleString.toSimpleString("expiryQueue");
|
||||
|
||||
AddressSettings as = new AddressSettings();
|
||||
as.setDeadLetterAddress(expiryQueue);
|
||||
as.setExpiryAddress(expiryQueue);
|
||||
|
||||
for (int i = 0; i <= 2; i++) {
|
||||
createQueue(i, "queues.testaddress", "queue0", null, true);
|
||||
getServer(i).createQueue(expiryQueue, expiryQueue, null, true, false);
|
||||
getServer(i).getAddressSettingsRepository().addMatch("queues.*", as);
|
||||
|
||||
}
|
||||
|
||||
// this will pause all the cluster bridges
|
||||
for (ClusterConnection clusterConnection : getServer(0).getClusterManager().getClusterConnections()) {
|
||||
for (MessageFlowRecord record : ((ClusterConnectionImpl) clusterConnection).getRecords().values()) {
|
||||
record.getBridge().pause();
|
||||
}
|
||||
}
|
||||
|
||||
ClientSessionFactory sf = sfs[0];
|
||||
|
||||
ClientSession session = sf.createSession(false, false);
|
||||
ClientProducer producer = session.createProducer("queues.testaddress");
|
||||
|
||||
for (int i = 0; i < 1000; i++) {
|
||||
ClientMessage message = session.createMessage(true);
|
||||
message.setExpiration(500);
|
||||
producer.send(message);
|
||||
}
|
||||
|
||||
session.commit();
|
||||
|
||||
session.start();
|
||||
|
||||
ClientConsumer consumer = session.createConsumer("expiryQueue");
|
||||
for (int i = 0; i < 1000; i++) {
|
||||
ClientMessage message = consumer.receive(2000);
|
||||
Assert.assertNotNull(message);
|
||||
message.acknowledge();
|
||||
}
|
||||
|
||||
session.commit();
|
||||
|
||||
session.close();
|
||||
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue