committed by
GitHub
4 changed files with 205 additions and 1 deletions
@ -0,0 +1,21 @@ |
|||
/** |
|||
* Copyright © 2016-2025 The Thingsboard Authors |
|||
* |
|||
* 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.thingsboard.mqtt; |
|||
|
|||
@FunctionalInterface |
|||
public interface ReconnectStrategy { |
|||
long getNextReconnectDelay(); |
|||
} |
|||
@ -0,0 +1,82 @@ |
|||
/** |
|||
* Copyright © 2016-2025 The Thingsboard Authors |
|||
* |
|||
* 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.thingsboard.mqtt; |
|||
|
|||
import lombok.Getter; |
|||
import lombok.extern.slf4j.Slf4j; |
|||
|
|||
import java.util.concurrent.ThreadLocalRandom; |
|||
import java.util.concurrent.TimeUnit; |
|||
|
|||
@Getter |
|||
@Slf4j |
|||
public class ReconnectStrategyExponential implements ReconnectStrategy { |
|||
|
|||
public static final int DEFAULT_RECONNECT_INTERVAL_SEC = 10; |
|||
public static final int MAX_RECONNECT_INTERVAL_SEC = 60; |
|||
public static final int EXP_MAX = 8; |
|||
public static final long JITTER_MAX = 1; |
|||
private final long reconnectIntervalMinSeconds; |
|||
private final long reconnectIntervalMaxSeconds; |
|||
private long lastDisconnectNanoTime = 0; //isotonic time
|
|||
private long retryCount = 0; |
|||
|
|||
public ReconnectStrategyExponential(long reconnectIntervalMinSeconds) { |
|||
this.reconnectIntervalMaxSeconds = calculateIntervalMax(reconnectIntervalMinSeconds); |
|||
this.reconnectIntervalMinSeconds = calculateIntervalMin(reconnectIntervalMinSeconds); |
|||
} |
|||
|
|||
long calculateIntervalMax(long reconnectIntervalMinSeconds) { |
|||
return reconnectIntervalMinSeconds > MAX_RECONNECT_INTERVAL_SEC ? reconnectIntervalMinSeconds : MAX_RECONNECT_INTERVAL_SEC; |
|||
} |
|||
|
|||
long calculateIntervalMin(long reconnectIntervalMinSeconds) { |
|||
return Math.min((reconnectIntervalMinSeconds > 0 ? reconnectIntervalMinSeconds : DEFAULT_RECONNECT_INTERVAL_SEC), this.reconnectIntervalMaxSeconds); |
|||
} |
|||
|
|||
@Override |
|||
synchronized public long getNextReconnectDelay() { |
|||
final long currentNanoTime = getNanoTime(); |
|||
final long coolDownSpentNanos = currentNanoTime - lastDisconnectNanoTime; |
|||
lastDisconnectNanoTime = currentNanoTime; |
|||
if (isCooledDown(coolDownSpentNanos)) { |
|||
retryCount = 0; |
|||
return reconnectIntervalMinSeconds; |
|||
} |
|||
return calculateNextReconnectDelay() + calculateJitter(); |
|||
} |
|||
|
|||
long calculateJitter() { |
|||
return ThreadLocalRandom.current().nextInt() >= 0 ? JITTER_MAX : 0; |
|||
} |
|||
|
|||
long calculateNextReconnectDelay() { |
|||
return Math.min(reconnectIntervalMaxSeconds, reconnectIntervalMinSeconds + calculateExp(retryCount++)); |
|||
} |
|||
|
|||
long calculateExp(long e) { |
|||
return 1L << Math.min(e, EXP_MAX); |
|||
} |
|||
|
|||
boolean isCooledDown(long coolDownSpentNanos) { |
|||
return TimeUnit.NANOSECONDS.toSeconds(coolDownSpentNanos) > reconnectIntervalMaxSeconds + reconnectIntervalMinSeconds; |
|||
} |
|||
|
|||
long getNanoTime() { |
|||
return System.nanoTime(); |
|||
} |
|||
|
|||
} |
|||
@ -0,0 +1,95 @@ |
|||
/** |
|||
* Copyright © 2016-2025 The Thingsboard Authors |
|||
* |
|||
* 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.thingsboard.mqtt; |
|||
|
|||
import lombok.extern.slf4j.Slf4j; |
|||
import org.junit.jupiter.api.parallel.Execution; |
|||
import org.junit.jupiter.api.parallel.ExecutionMode; |
|||
import org.junit.jupiter.params.ParameterizedTest; |
|||
import org.junit.jupiter.params.provider.ValueSource; |
|||
import org.mockito.Mockito; |
|||
import org.mockito.stubbing.Answer; |
|||
|
|||
import java.util.ArrayList; |
|||
import java.util.Collection; |
|||
import java.util.concurrent.BlockingQueue; |
|||
import java.util.concurrent.LinkedBlockingDeque; |
|||
import java.util.concurrent.TimeUnit; |
|||
import java.util.concurrent.atomic.AtomicLong; |
|||
|
|||
import static org.assertj.core.api.Assertions.assertThat; |
|||
import static org.assertj.core.data.Offset.offset; |
|||
import static org.mockito.ArgumentMatchers.anyLong; |
|||
import static org.mockito.BDDMockito.willAnswer; |
|||
import static org.thingsboard.mqtt.ReconnectStrategyExponential.EXP_MAX; |
|||
import static org.thingsboard.mqtt.ReconnectStrategyExponential.JITTER_MAX; |
|||
|
|||
@Slf4j |
|||
class ReconnectStrategyExponentialTest { |
|||
|
|||
@Execution(ExecutionMode.SAME_THREAD) // just for convenient log reading
|
|||
@ParameterizedTest |
|||
@ValueSource(ints = {1, 0, 60}) |
|||
public void exponentialReconnectDelayTest(final int reconnectIntervalMinSeconds) { |
|||
final ReconnectStrategyExponential strategy = Mockito.spy(new ReconnectStrategyExponential(reconnectIntervalMinSeconds)); |
|||
log.info("=== Reconnect delay test for ReconnectStrategyExponential({}) : calculated min [{}] max [{}] ===", reconnectIntervalMinSeconds, strategy.getReconnectIntervalMinSeconds(), strategy.getReconnectIntervalMaxSeconds()); |
|||
final AtomicLong nanoTime = new AtomicLong(System.nanoTime()); |
|||
willAnswer((x) -> nanoTime.get()).given(strategy).getNanoTime(); |
|||
final LinkedBlockingDeque<Long> jittersCaptured = new LinkedBlockingDeque<>(); |
|||
final LinkedBlockingDeque<Long> expCaptured = new LinkedBlockingDeque<>(); |
|||
|
|||
willAnswer(captureResult(jittersCaptured)).given(strategy).calculateJitter(); |
|||
willAnswer(captureResult(expCaptured)).given(strategy).calculateExp(anyLong()); |
|||
|
|||
for (int phase = 0; phase < 3; phase++) { |
|||
log.info("== Phase {} ==", phase); |
|||
long previousDelay = 0; |
|||
for (int i = 0; i < EXP_MAX + 4; i++) { |
|||
final long nextReconnectDelay = strategy.getNextReconnectDelay(); |
|||
nanoTime.addAndGet(TimeUnit.SECONDS.toNanos(nextReconnectDelay)); |
|||
log.info("Retry [{}] Delay [{}] : min [{}] exp [{}] jitter [{}]", strategy.getRetryCount(), nextReconnectDelay, strategy.getReconnectIntervalMinSeconds(), expCaptured.peekLast(), jittersCaptured.peekLast()); |
|||
assertThat(previousDelay).satisfiesAnyOf( |
|||
v -> assertThat(v).isLessThanOrEqualTo(nextReconnectDelay), |
|||
v -> assertThat(v).isCloseTo(nextReconnectDelay, offset(JITTER_MAX)) // Adjust tolerance as needed
|
|||
); |
|||
previousDelay = nextReconnectDelay; |
|||
} |
|||
log.info("Jitters captured: {}", drainAll(jittersCaptured)); |
|||
log.info("Exponents captured: {}", drainAll(expCaptured)); |
|||
assertThat(previousDelay).isCloseTo(strategy.getReconnectIntervalMaxSeconds(), offset(JITTER_MAX)); |
|||
|
|||
final long coolDownPeriodSec = strategy.getReconnectIntervalMinSeconds() + strategy.getReconnectIntervalMaxSeconds() + 1; |
|||
log.info("Cooling down for [{}] seconds ...", coolDownPeriodSec); |
|||
nanoTime.addAndGet(TimeUnit.SECONDS.toNanos(coolDownPeriodSec)); |
|||
assertThat(strategy.isCooledDown(TimeUnit.SECONDS.toNanos(coolDownPeriodSec))).as("cooled down").isTrue(); |
|||
} |
|||
} |
|||
|
|||
private Answer<Long> captureResult(Collection<Long> collection) { |
|||
return invocation -> { |
|||
long result = (long) invocation.callRealMethod(); |
|||
collection.add(result); |
|||
return result; |
|||
}; |
|||
} |
|||
|
|||
private Collection<Long> drainAll(BlockingQueue<Long> jittersCaptured) { |
|||
Collection<Long> elements = new ArrayList<>(); |
|||
jittersCaptured.drainTo(elements); |
|||
return elements; |
|||
} |
|||
|
|||
} |
|||
Loading…
Reference in new issue