From 755b987ab132408b187ed12c676a4a87601f27a3 Mon Sep 17 00:00:00 2001 From: Justin Bertram Date: Fri, 26 Aug 2022 10:28:09 -0500 Subject: [PATCH] ARTEMIS-3958 sending LWT may recurse infinitely if disk full Due to the changes in 682f505e32f9b6472665212acd6f58c32c7bf98d we now send "Last Will & Testament" MQTT messages via ServerSession. This means sending will fail if the disk is full. For MQTT this triggers a connection failure which in turns triggers sending an LWT message. This process will recurse infinitely until it results in a java.lang.StackOverflowError. This commit fixes that by tracking whether or not sending a LWT message is already in progress. --- .../protocol/mqtt/MQTTProtocolManager.java | 4 +- .../core/protocol/mqtt/MQTTSession.java | 46 +++++++++---------- .../core/protocol/mqtt/MQTTSessionState.java | 42 ++++++++++++++--- .../core/paging/impl/PagingManagerImpl.java | 7 +++ .../impl/PagingManagerImplAccessor.java | 23 ++++++++++ .../tests/integration/mqtt5/MQTT5Test.java | 21 +++++++++ 6 files changed, 111 insertions(+), 32 deletions(-) create mode 100644 tests/integration-tests/src/test/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImplAccessor.java diff --git a/artemis-protocols/artemis-mqtt-protocol/src/main/java/org/apache/activemq/artemis/core/protocol/mqtt/MQTTProtocolManager.java b/artemis-protocols/artemis-mqtt-protocol/src/main/java/org/apache/activemq/artemis/core/protocol/mqtt/MQTTProtocolManager.java index c48a9802eb..86bbf18c6b 100644 --- a/artemis-protocols/artemis-mqtt-protocol/src/main/java/org/apache/activemq/artemis/core/protocol/mqtt/MQTTProtocolManager.java +++ b/artemis-protocols/artemis-mqtt-protocol/src/main/java/org/apache/activemq/artemis/core/protocol/mqtt/MQTTProtocolManager.java @@ -195,7 +195,7 @@ public class MQTTProtocolManager extends AbstractProtocolManager 0 && state.getDisconnectedTime() + (sessionExpiryInterval * 1000) < System.currentTimeMillis()) { toRemove.add(entry.getKey()); } - if (state.isWill() && !state.isAttached() && state.isFailed() && !state.isWillSent() && state.getWillDelayInterval() > 0 && state.getDisconnectedTime() + (state.getWillDelayInterval() * 1000) < System.currentTimeMillis()) { + if (state.isWill() && !state.isAttached() && state.isFailed() && state.getWillDelayInterval() > 0 && state.getDisconnectedTime() + (state.getWillDelayInterval() * 1000) < System.currentTimeMillis()) { state.getSession().sendWillMessage(); } } @@ -203,7 +203,7 @@ public class MQTTProtocolManager extends AbstractProtocolManager willUserProperties; - private boolean willSent = false; + private WillStatus willStatus = WillStatus.NOT_SENT; private boolean failed = false; @@ -113,7 +113,7 @@ public class MQTTSessionState { willMessage.clear(); willMessage = null; } - willSent = false; + willStatus = WillStatus.NOT_SENT; failed = false; willDelayInterval = 0; willRetain = false; @@ -282,12 +282,12 @@ public class MQTTSessionState { return willUserProperties; } - public boolean isWillSent() { - return willSent; + public WillStatus getWillStatus() { + return willStatus; } - public void setWillSent(boolean willSent) { - this.willSent = willSent; + public void setWillStatus(WillStatus willStatus) { + this.willStatus = willStatus; } public boolean isFailed() { @@ -448,4 +448,34 @@ public class MQTTSessionState { public String toString() { return "MQTTSessionState[" + "session=" + session + ", clientId='" + clientId + "', subscriptions=" + subscriptions + ", messageRefStore=" + messageRefStore + ", addressMessageMap=" + addressMessageMap + ", pubRec=" + pubRec + ", attached=" + attached + ", outboundStore=" + outboundStore + ", disconnectedTime=" + disconnectedTime + ", sessionExpiryInterval=" + clientSessionExpiryInterval + ", isWill=" + isWill + ", willMessage=" + willMessage + ", willTopic='" + willTopic + "', willQoSLevel=" + willQoSLevel + ", willRetain=" + willRetain + ", willDelayInterval=" + willDelayInterval + ", failed=" + failed + ", maxPacketSize=" + clientMaxPacketSize + ']'; } + + public enum WillStatus { + NOT_SENT, SENT, SENDING; + + public byte getStatus() { + switch (this) { + case NOT_SENT: + return 0; + case SENT: + return 1; + case SENDING: + return 2; + default: + return -1; + } + } + + public static WillStatus getStatus(byte status) { + switch (status) { + case 0: + return NOT_SENT; + case 1: + return SENT; + case 2: + return SENDING; + default: + return null; + } + } + } } diff --git a/artemis-server/src/main/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImpl.java b/artemis-server/src/main/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImpl.java index b04bbd7c2f..f3257b33a1 100644 --- a/artemis-server/src/main/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImpl.java +++ b/artemis-server/src/main/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImpl.java @@ -251,6 +251,13 @@ public final class PagingManagerImpl implements PagingManager { } } + /* + * For tests only! + */ + protected void setDiskFull(boolean diskFull) { + this.diskFull = diskFull; + } + @Override public boolean isDiskFull() { return diskFull; diff --git a/tests/integration-tests/src/test/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImplAccessor.java b/tests/integration-tests/src/test/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImplAccessor.java new file mode 100644 index 0000000000..5aa5a5117e --- /dev/null +++ b/tests/integration-tests/src/test/java/org/apache/activemq/artemis/core/paging/impl/PagingManagerImplAccessor.java @@ -0,0 +1,23 @@ +/** + * 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.core.paging.impl; + +public class PagingManagerImplAccessor { + public static void setDiskFull(PagingManagerImpl pagingManager, boolean diskFull) { + pagingManager.setDiskFull(diskFull); + } +} diff --git a/tests/integration-tests/src/test/java/org/apache/activemq/artemis/tests/integration/mqtt5/MQTT5Test.java b/tests/integration-tests/src/test/java/org/apache/activemq/artemis/tests/integration/mqtt5/MQTT5Test.java index c2af69ede9..2485f19d2a 100644 --- a/tests/integration-tests/src/test/java/org/apache/activemq/artemis/tests/integration/mqtt5/MQTT5Test.java +++ b/tests/integration-tests/src/test/java/org/apache/activemq/artemis/tests/integration/mqtt5/MQTT5Test.java @@ -27,10 +27,14 @@ import java.util.concurrent.CountDownLatch; import java.util.concurrent.TimeUnit; import org.apache.activemq.artemis.api.core.QueueConfiguration; +import org.apache.activemq.artemis.api.core.RoutingType; import org.apache.activemq.artemis.api.core.SimpleString; +import org.apache.activemq.artemis.core.paging.impl.PagingManagerImpl; +import org.apache.activemq.artemis.core.paging.impl.PagingManagerImplAccessor; import org.apache.activemq.artemis.core.protocol.mqtt.MQTTReasonCodes; import org.apache.activemq.artemis.core.protocol.mqtt.MQTTUtil; import org.apache.activemq.artemis.core.settings.impl.AddressSettings; +import org.apache.activemq.artemis.logs.AssertionLoggerHandler; import org.apache.activemq.artemis.tests.util.RandomUtil; import org.apache.activemq.artemis.utils.Wait; import org.eclipse.paho.mqttv5.client.MqttAsyncClient; @@ -225,4 +229,21 @@ public class MQTT5Test extends MQTT5TestSupport { server.start(); org.apache.activemq.artemis.tests.util.Wait.assertTrue(() -> getSubscriptionQueue(topic, clientId) != null, 3000, 10); } + + @Test(timeout = DEFAULT_TIMEOUT) + public void testRecursiveWill() throws Exception { + AssertionLoggerHandler.startCapture(true); + try { + final String WILL_QUEUE = "will"; + server.createQueue(new QueueConfiguration(WILL_QUEUE).setRoutingType(RoutingType.ANYCAST)); + PagingManagerImplAccessor.setDiskFull((PagingManagerImpl) server.getPagingManager(), true); + MqttClient client = createPahoClient("willGenerator"); + MqttConnectionOptions options = new MqttConnectionOptionsBuilder().will(WILL_QUEUE, new MqttMessage(RandomUtil.randomBytes())).build(); + client.connect(options); + client.disconnectForcibly(0, 0, false); + Wait.assertTrue(() -> AssertionLoggerHandler.findText("AMQ229119"), 2000, 100); + } finally { + AssertionLoggerHandler.stopCapture(); + } + } }