com.palantir.atlasdb.transaction.impl.TransactionRetryStrategy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of atlasdb-client Show documentation
Show all versions of atlasdb-client Show documentation
Palantir open source project
/*
* (c) Copyright 2019 Palantir Technologies Inc. All rights reserved.
*
* 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 com.palantir.atlasdb.transaction.impl;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.github.rholder.retry.Attempt;
import com.github.rholder.retry.BlockStrategies;
import com.github.rholder.retry.BlockStrategy;
import com.github.rholder.retry.RetryException;
import com.github.rholder.retry.RetryListener;
import com.github.rholder.retry.Retryer;
import com.github.rholder.retry.RetryerBuilder;
import com.github.rholder.retry.WaitStrategies;
import com.github.rholder.retry.WaitStrategy;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.primitives.Ints;
import com.palantir.atlasdb.transaction.api.TransactionFailedException;
import com.palantir.exception.NotInitializedException;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tritium.ids.UniqueIds;
import java.util.Random;
import java.util.UUID;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.function.IntPredicate;
public final class TransactionRetryStrategy {
@SuppressWarnings("ImmutableEnumChecker") // TransactionRetryStrategy is immutable, despite what error-prone thinks
public enum Strategies {
LEGACY(createLegacy(BlockStrategies.threadSleepStrategy())),
EXPONENTIAL(createExponential(BlockStrategies.threadSleepStrategy(), new Random()));
private final TransactionRetryStrategy strategy;
Strategies(TransactionRetryStrategy strategy) {
this.strategy = strategy;
}
public TransactionRetryStrategy get() {
return strategy;
}
}
private static final SafeLogger log = SafeLoggerFactory.get(TransactionRetryStrategy.class);
private final WaitStrategy waitStrategy;
private final BlockStrategy blockStrategy;
private TransactionRetryStrategy(WaitStrategy waitStrategy, BlockStrategy blockStrategy) {
this.waitStrategy = waitStrategy;
this.blockStrategy = blockStrategy;
}
@SuppressWarnings("rawtypes") // StopStrategy uses raw Attempt
public T runWithRetry(IntPredicate shouldStopRetrying, Retryable task) throws E {
UUID runId = UniqueIds.pseudoRandomUuidV4(); // runId is just diagnostic so do not need CSPRNG
Retryer retryer = RetryerBuilder.newBuilder()
.retryIfException(TransactionRetryStrategy::shouldRetry)
.withBlockStrategy(blockStrategy)
.withWaitStrategy(waitStrategy)
.withStopStrategy(
failedAttempt -> shouldStopRetrying.test(Ints.checkedCast(failedAttempt.getAttemptNumber())))
.withRetryListener(new RetryListener() {
@Override
public void onRetry(Attempt attempt) {
logAttempt(runId, attempt, shouldStopRetrying);
}
})
.build();
try {
return retryer.call(task::run);
} catch (RetryException e) {
throw throwTaskException(e.getLastFailedAttempt().getExceptionCause());
} catch (ExecutionException e) {
throw throwTaskException(e.getCause());
}
}
private void logAttempt(UUID runId, Attempt attempt, IntPredicate shouldStopRetrying) {
int failureCount = Ints.checkedCast(attempt.getAttemptNumber()) - 1;
if (attempt.hasResult()) {
if (failureCount > 0) {
log.info(
"[{}] Successfully completed transaction after {} retries.",
SafeArg.of("runId", runId),
SafeArg.of("failureCount", failureCount));
}
} else {
Throwable thrown = attempt.getExceptionCause();
if (thrown instanceof TransactionFailedException) {
TransactionFailedException exception = (TransactionFailedException) thrown;
if (!exception.canTransactionBeRetried()) {
log.warn(
"[{}] Non-retriable exception while processing transaction.",
SafeArg.of("runId", runId),
SafeArg.of("failureCount", failureCount));
} else if (shouldStopRetrying.test(failureCount)) {
log.warn(
"[{}] Failing after {} tries.",
SafeArg.of("runId", runId),
SafeArg.of("failureCount", failureCount),
exception);
} else if (failureCount > 2) {
log.info(
"[{}] Retrying transaction after {} failure(s).",
SafeArg.of("runId", runId),
SafeArg.of("failureCount", failureCount),
thrown);
}
} else if (thrown instanceof NotInitializedException) {
log.info("TransactionManager is not initialized. Aborting transaction with runTaskWithRetry", thrown);
} else if (thrown instanceof RuntimeException) {
log.debug("[{}] RuntimeException while processing transaction.", SafeArg.of("runId", runId), thrown);
}
}
}
/**
* This is overly unchecked in order to handle the transaction tasks's potential for checked exceptions.
*/
private static RuntimeException throwTaskException(Throwable thrown) throws E {
if (thrown instanceof Error) {
throw (Error) thrown;
} else if (thrown instanceof Exception) {
throw (E) thrown;
} else {
throw new RuntimeException(thrown);
}
}
@FunctionalInterface
public interface Retryable {
T run() throws E;
}
@SuppressWarnings("rawtypes") // WaitStrategy uses raw Attempt
private static WaitStrategy randomize(Random random, WaitStrategy strategy) {
return attempt -> random.nextInt(Ints.checkedCast(strategy.computeSleepTime(attempt)));
}
private static boolean shouldRetry(Throwable e) {
return e instanceof TransactionFailedException && ((TransactionFailedException) e).canTransactionBeRetried();
}
@JsonCreator
static TransactionRetryStrategy of(Strategies strategy) {
return strategy.strategy;
}
@VisibleForTesting
static TransactionRetryStrategy createLegacy(BlockStrategy blockStrategy) {
return new TransactionRetryStrategy(WaitStrategies.noWait(), blockStrategy);
}
@VisibleForTesting
static TransactionRetryStrategy createExponential(BlockStrategy blockStrategy, Random random) {
return new TransactionRetryStrategy(
randomize(random, WaitStrategies.exponentialWait(100, 1, TimeUnit.MINUTES)), blockStrategy);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy