io.temporal.internal.BackoffThrottler Maven / Gradle / Ivy
/*
* Copyright (C) 2022 Temporal Technologies, Inc. All Rights Reserved.
*
* Copyright (C) 2012-2016 Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Modifications copyright (C) 2017 Uber Technologies, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this material 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 io.temporal.internal;
import io.grpc.Status;
import io.grpc.Status.Code;
import java.time.Duration;
import java.util.Objects;
import javax.annotation.Nullable;
import javax.annotation.concurrent.NotThreadSafe;
/**
* Used to throttle code execution in presence of failures using exponential backoff logic.
*
* The formula used to calculate the next sleep interval is:
*
*
*
*
* jitter = random number in the range [-maxJitterCoefficient, +maxJitterCoefficient];
* sleepTime = min(pow(backoffCoefficient, failureCount - 1) * initialSleep * (1 + jitter), maxSleep);
*
*
* where initialSleep
is either set to regularInitialSleep
or
* congestionInitialSleep
based on the most recent failure. Note that it means that
* attempt X can possibly get a shorter throttle than attempt X-1, if a non-congestion failure
* occurs after a congestion failure. This is the expected behaviour for all SDK.
*
* Example usage:
*
*
* BackoffThrottler throttler = new BackoffThrottler(50, 1000, 60000, 2, 0.1);
* while(!stopped) {
* try {
* long throttleMs = throttler.getSleepTime();
* if (throttleMs > 0) {
* Thread.sleep(throttleMs);
* }
* // some code that can fail and should be throttled
* ...
* throttler.success();
* }
* catch (Exception e) {
* throttler.failure(
* (e instanceof StatusRuntimeException)
* ? ((StatusRuntimeException) e).getStatus().getCode()
* : Status.Code.UNKNOWN);
* }
* }
*
*
* @author fateev
*/
@NotThreadSafe
public final class BackoffThrottler {
private final Duration regularInitialSleep;
private final Duration congestionInitialSleep;
private final Duration maxSleep;
private final double backoffCoefficient;
private final double maxJitterCoefficient;
private int failureCount = 0;
private Status.Code lastFailureCode = Code.OK;
/**
* Construct an instance of the throttler.
*
* @param regularInitialSleep time to sleep on the first failure (assuming regular failures)
* @param congestionInitialSleep time to sleep on the first failure (for congestion failures)
* @param maxSleep maximum time to sleep independently of number of failures
* @param backoffCoefficient coefficient used to calculate the next time to sleep
* @param maxJitterCoefficient maximum jitter coefficient (in the range [0.0, 1.0[) to randomly
* add or subtract to sleep time
*/
public BackoffThrottler(
Duration regularInitialSleep,
Duration congestionInitialSleep,
@Nullable Duration maxSleep,
double backoffCoefficient,
double maxJitterCoefficient) {
Objects.requireNonNull(regularInitialSleep, "regularInitialSleep");
Objects.requireNonNull(congestionInitialSleep, "congestionInitialSleep");
if (backoffCoefficient < 1.0) {
throw new IllegalArgumentException(
"backoff coefficient less than 1.0: " + backoffCoefficient);
}
if (maxJitterCoefficient < 0 || maxJitterCoefficient >= 1.0) {
throw new IllegalArgumentException(
"maxJitterCoefficient has to be >= 0 and < 1.0: " + maxJitterCoefficient);
}
this.regularInitialSleep = regularInitialSleep;
this.congestionInitialSleep = congestionInitialSleep;
this.maxSleep = maxSleep;
this.backoffCoefficient = backoffCoefficient;
this.maxJitterCoefficient = maxJitterCoefficient;
}
public long getSleepTime() {
if (failureCount == 0) return 0;
Duration initial =
(lastFailureCode == Code.RESOURCE_EXHAUSTED) ? congestionInitialSleep : regularInitialSleep;
// Choose a random number in the range [-maxJitterCoefficient, +maxJitterCoefficient];
double jitter = Math.random() * maxJitterCoefficient * 2 - maxJitterCoefficient;
double sleepMillis =
Math.pow(backoffCoefficient, failureCount - 1) * initial.toMillis() * (1 + jitter);
if (maxSleep != null) {
return Math.min((long) sleepMillis, maxSleep.toMillis());
}
return (long) sleepMillis;
}
public int getAttemptCount() {
return failureCount;
}
/** Reset failure count to 0 and clear last failure code. */
public void success() {
failureCount = 0;
lastFailureCode = Code.OK;
}
/** Increment failure count and set last failure code. */
public void failure(Status.Code failureCode) {
failureCount++;
lastFailureCode = failureCode;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy