Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.smallrye.faulttolerance.FaultToleranceInterceptor Maven / Gradle / Ivy
/*
* Copyright 2017 Red Hat, Inc, and individual contributors.
*
* 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 io.smallrye.faulttolerance;
import static io.smallrye.faulttolerance.core.Invocation.invocation;
import static io.smallrye.faulttolerance.core.util.Durations.timeInMillis;
import static io.smallrye.faulttolerance.core.util.SneakyThrow.sneakyThrow;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.UUID;
import java.util.concurrent.Callable;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
import jakarta.annotation.Priority;
import jakarta.enterprise.context.control.RequestContextController;
import jakarta.enterprise.inject.Any;
import jakarta.enterprise.inject.Instance;
import jakarta.enterprise.inject.Intercepted;
import jakarta.enterprise.inject.spi.Bean;
import jakarta.inject.Inject;
import jakarta.interceptor.AroundInvoke;
import jakarta.interceptor.Interceptor;
import jakarta.interceptor.InvocationContext;
import org.eclipse.microprofile.faulttolerance.ExecutionContext;
import org.eclipse.microprofile.faulttolerance.FallbackHandler;
import org.eclipse.microprofile.faulttolerance.exceptions.FaultToleranceException;
import io.smallrye.common.annotation.Identifier;
import io.smallrye.faulttolerance.api.AlwaysOnException;
import io.smallrye.faulttolerance.api.BeforeRetryHandler;
import io.smallrye.faulttolerance.api.CustomBackoffStrategy;
import io.smallrye.faulttolerance.api.FaultTolerance;
import io.smallrye.faulttolerance.api.NeverOnResult;
import io.smallrye.faulttolerance.config.FaultToleranceOperation;
import io.smallrye.faulttolerance.core.FailureContext;
import io.smallrye.faulttolerance.core.FaultToleranceStrategy;
import io.smallrye.faulttolerance.core.apiimpl.LazyFaultTolerance;
import io.smallrye.faulttolerance.core.async.CompletionStageExecution;
import io.smallrye.faulttolerance.core.async.FutureExecution;
import io.smallrye.faulttolerance.core.async.RememberEventLoop;
import io.smallrye.faulttolerance.core.bulkhead.CompletionStageThreadPoolBulkhead;
import io.smallrye.faulttolerance.core.bulkhead.FutureThreadPoolBulkhead;
import io.smallrye.faulttolerance.core.bulkhead.SemaphoreBulkhead;
import io.smallrye.faulttolerance.core.circuit.breaker.CircuitBreaker;
import io.smallrye.faulttolerance.core.circuit.breaker.CircuitBreakerEvents;
import io.smallrye.faulttolerance.core.circuit.breaker.CompletionStageCircuitBreaker;
import io.smallrye.faulttolerance.core.event.loop.EventLoop;
import io.smallrye.faulttolerance.core.fallback.AsyncFallbackFunction;
import io.smallrye.faulttolerance.core.fallback.CompletionStageFallback;
import io.smallrye.faulttolerance.core.fallback.Fallback;
import io.smallrye.faulttolerance.core.invocation.AsyncSupport;
import io.smallrye.faulttolerance.core.invocation.AsyncSupportRegistry;
import io.smallrye.faulttolerance.core.invocation.Invoker;
import io.smallrye.faulttolerance.core.invocation.StrategyInvoker;
import io.smallrye.faulttolerance.core.metrics.CompletionStageMetricsCollector;
import io.smallrye.faulttolerance.core.metrics.MeteredOperation;
import io.smallrye.faulttolerance.core.metrics.MeteredOperationName;
import io.smallrye.faulttolerance.core.metrics.MetricsCollector;
import io.smallrye.faulttolerance.core.metrics.MetricsProvider;
import io.smallrye.faulttolerance.core.rate.limit.CompletionStageRateLimit;
import io.smallrye.faulttolerance.core.rate.limit.RateLimit;
import io.smallrye.faulttolerance.core.retry.BackOff;
import io.smallrye.faulttolerance.core.retry.CompletionStageRetry;
import io.smallrye.faulttolerance.core.retry.ConstantBackOff;
import io.smallrye.faulttolerance.core.retry.CustomBackOff;
import io.smallrye.faulttolerance.core.retry.ExponentialBackOff;
import io.smallrye.faulttolerance.core.retry.FibonacciBackOff;
import io.smallrye.faulttolerance.core.retry.Jitter;
import io.smallrye.faulttolerance.core.retry.RandomJitter;
import io.smallrye.faulttolerance.core.retry.Retry;
import io.smallrye.faulttolerance.core.retry.ThreadSleepDelay;
import io.smallrye.faulttolerance.core.retry.TimerDelay;
import io.smallrye.faulttolerance.core.stopwatch.SystemStopwatch;
import io.smallrye.faulttolerance.core.timeout.AsyncTimeout;
import io.smallrye.faulttolerance.core.timeout.CompletionStageTimeout;
import io.smallrye.faulttolerance.core.timeout.Timeout;
import io.smallrye.faulttolerance.core.timer.Timer;
import io.smallrye.faulttolerance.core.util.DirectExecutor;
import io.smallrye.faulttolerance.core.util.ExceptionDecision;
import io.smallrye.faulttolerance.core.util.PredicateBasedExceptionDecision;
import io.smallrye.faulttolerance.core.util.PredicateBasedResultDecision;
import io.smallrye.faulttolerance.core.util.ResultDecision;
import io.smallrye.faulttolerance.core.util.SetBasedExceptionDecision;
import io.smallrye.faulttolerance.core.util.SetOfThrowables;
import io.smallrye.faulttolerance.internal.BeforeRetryMethod;
import io.smallrye.faulttolerance.internal.FallbackMethod;
import io.smallrye.faulttolerance.internal.FallbackMethodCandidates;
import io.smallrye.faulttolerance.internal.InterceptionInvoker;
import io.smallrye.faulttolerance.internal.InterceptionPoint;
import io.smallrye.faulttolerance.internal.RequestScopeActivator;
import io.smallrye.faulttolerance.internal.StrategyCache;
import io.smallrye.faulttolerance.metrics.CdiMeteredOperationImpl;
/**
* The interceptor for fault tolerance strategies.
*
* @author Antoine Sabot-Durand
* @author Martin Kouba
* @author Michal Szynkiewicz
*/
@Interceptor
@FaultToleranceBinding
@Priority(Interceptor.Priority.PLATFORM_AFTER + 10)
public class FaultToleranceInterceptor {
private final Bean> interceptedBean;
private final FaultToleranceOperationProvider operationProvider;
private final StrategyCache cache;
private final FallbackHandlerProvider fallbackHandlerProvider;
private final BeforeRetryHandlerProvider beforeRetryHandlerProvider;
private final MetricsProvider metricsProvider;
private final ExecutorService asyncExecutor;
private final EventLoop eventLoop;
private final Timer timer;
private final RequestContextController requestContextController;
private final CircuitBreakerMaintenanceImpl cbMaintenance;
private final SpecCompatibility specCompatibility;
private final Instance> configuredFaultTolerance;
@Inject
public FaultToleranceInterceptor(
@Intercepted Bean> interceptedBean,
FaultToleranceOperationProvider operationProvider,
StrategyCache cache,
FallbackHandlerProvider fallbackHandlerProvider,
BeforeRetryHandlerProvider beforeRetryHandlerProvider,
MetricsProvider metricsProvider,
ExecutorHolder executorHolder,
RequestContextIntegration requestContextIntegration,
CircuitBreakerMaintenanceImpl cbMaintenance,
SpecCompatibility specCompatibility,
@Any Instance> configuredFaultTolerance) {
this.interceptedBean = interceptedBean;
this.operationProvider = operationProvider;
this.cache = cache;
this.fallbackHandlerProvider = fallbackHandlerProvider;
this.beforeRetryHandlerProvider = beforeRetryHandlerProvider;
this.metricsProvider = metricsProvider;
asyncExecutor = executorHolder.getAsyncExecutor();
eventLoop = executorHolder.getEventLoop();
timer = executorHolder.getTimer();
requestContextController = requestContextIntegration.get();
this.cbMaintenance = cbMaintenance;
this.specCompatibility = specCompatibility;
this.configuredFaultTolerance = configuredFaultTolerance;
}
@AroundInvoke
public Object intercept(InvocationContext interceptionContext) throws Exception {
Method method = interceptionContext.getMethod();
Class> beanClass = interceptedBean != null ? interceptedBean.getBeanClass() : method.getDeclaringClass();
InterceptionPoint point = new InterceptionPoint(beanClass, interceptionContext.getMethod());
FaultToleranceOperation operation = operationProvider.get(beanClass, method);
if (operation.hasApplyFaultTolerance()) {
return preconfiguredFlow(operation, interceptionContext);
} else if (specCompatibility.isOperationTrulyAsynchronous(operation)) {
return asyncFlow(operation, interceptionContext, point);
} else if (specCompatibility.isOperationPseudoAsynchronous(operation)) {
return futureFlow(operation, interceptionContext, point);
} else {
return syncFlow(operation, interceptionContext, point);
}
}
private Object preconfiguredFlow(FaultToleranceOperation operation, InvocationContext interceptionContext)
throws Exception {
String identifier = operation.getApplyFaultTolerance().value();
Instance> instance = configuredFaultTolerance.select(Identifier.Literal.of(identifier));
if (!instance.isResolvable()) {
throw new FaultToleranceException("Can't resolve a bean of type " + FaultTolerance.class.getName()
+ " with qualifier @" + Identifier.class.getName() + "(\"" + identifier + "\")");
}
FaultTolerance faultTolerance = (FaultTolerance) instance.get();
if (!(faultTolerance instanceof LazyFaultTolerance)) {
throw new FaultToleranceException("Configured fault tolerance '" + identifier
+ "' is not created by the FaultTolerance API, this is not supported");
}
LazyFaultTolerance lazyFaultTolerance = (LazyFaultTolerance) faultTolerance;
Class> asyncType = lazyFaultTolerance.internalGetAsyncType();
MeteredOperationName meteredOperationName = new MeteredOperationName(operation.getName());
AsyncSupport, ?> forOperation = AsyncSupportRegistry.get(operation.getParameterTypes(), operation.getReturnType());
AsyncSupport, ?> fromConfigured = asyncType == null ? null : AsyncSupportRegistry.get(new Class[0], asyncType);
if (forOperation == null && fromConfigured == null) {
return lazyFaultTolerance.call(interceptionContext::proceed, meteredOperationName);
} else if (forOperation == null) {
throw new FaultToleranceException("Configured fault tolerance '" + identifier
+ "' expects the operation to " + fromConfigured.mustDescription()
+ ", but the operation is synchronous: " + operation);
} else if (fromConfigured == null) {
throw new FaultToleranceException("Configured fault tolerance '" + identifier
+ "' expects the operation to be synchronous, but it "
+ forOperation.doesDescription() + ": " + operation);
} else if (!forOperation.getClass().equals(fromConfigured.getClass())) {
throw new FaultToleranceException("Configured fault tolerance '" + identifier
+ "' expects the operation to " + fromConfigured.mustDescription()
+ ", but it " + forOperation.doesDescription() + ": " + operation);
} else {
return lazyFaultTolerance.call(interceptionContext::proceed, meteredOperationName);
}
}
private AT asyncFlow(FaultToleranceOperation operation, InvocationContext interceptionContext,
InterceptionPoint point) {
AsyncSupport asyncSupport = AsyncSupportRegistry.get(operation.getParameterTypes(), operation.getReturnType());
if (asyncSupport == null) {
throw new FaultToleranceException("Unknown async invocation: " + operation);
}
FaultToleranceStrategy> strategy = cache.getStrategy(point,
() -> prepareAsyncStrategy(operation, point));
Invoker invoker = new InterceptionInvoker<>(interceptionContext);
io.smallrye.faulttolerance.core.InvocationContext> ctx = invocationContext(
() -> asyncSupport.toCompletionStage(invoker), interceptionContext, operation);
Invoker> wrapper = new StrategyInvoker<>(interceptionContext.getParameters(),
strategy, ctx);
try {
return asyncSupport.fromCompletionStage(wrapper);
} catch (Exception e) {
throw sneakyThrow(e);
}
}
private T syncFlow(FaultToleranceOperation operation, InvocationContext interceptionContext, InterceptionPoint point)
throws Exception {
FaultToleranceStrategy strategy = cache.getStrategy(point, () -> prepareSyncStrategy(operation, point));
io.smallrye.faulttolerance.core.InvocationContext ctx = invocationContext(
() -> (T) interceptionContext.proceed(), interceptionContext, operation);
return strategy.apply(ctx);
}
private Future futureFlow(FaultToleranceOperation operation, InvocationContext interceptionContext,
InterceptionPoint point) throws Exception {
FaultToleranceStrategy> strategy = cache.getStrategy(point, () -> prepareFutureStrategy(operation, point));
io.smallrye.faulttolerance.core.InvocationContext> ctx = invocationContext(
() -> (Future) interceptionContext.proceed(), interceptionContext, operation);
return strategy.apply(ctx);
}
private io.smallrye.faulttolerance.core.InvocationContext invocationContext(Callable callable,
InvocationContext interceptionContext, FaultToleranceOperation operation) {
io.smallrye.faulttolerance.core.InvocationContext result = new io.smallrye.faulttolerance.core.InvocationContext<>(
callable);
result.set(InvocationContext.class, interceptionContext);
if (operation.hasCircuitBreaker() && operation.hasCircuitBreakerName()) {
result.registerEventHandler(CircuitBreakerEvents.StateTransition.class,
cbMaintenance.stateTransitionEventHandler(operation.getCircuitBreakerName().value()));
}
return result;
}
private FaultToleranceStrategy> prepareAsyncStrategy(FaultToleranceOperation operation,
InterceptionPoint point) {
FaultToleranceStrategy> result = invocation();
result = new RequestScopeActivator<>(result, requestContextController);
Executor executor = operation.isThreadOffloadRequired() ? asyncExecutor : DirectExecutor.INSTANCE;
result = new CompletionStageExecution<>(result, executor);
if (operation.hasBulkhead()) {
int size = operation.getBulkhead().value();
int queueSize = operation.getBulkhead().waitingTaskQueue();
result = new CompletionStageThreadPoolBulkhead<>(result, point.toString(), size, queueSize);
}
if (operation.hasTimeout()) {
long timeoutMs = timeInMillis(operation.getTimeout().value(), operation.getTimeout().unit());
result = new CompletionStageTimeout<>(result, point.toString(), timeoutMs, timer);
}
if (operation.hasRateLimit()) {
result = new CompletionStageRateLimit<>(result, point.toString(),
operation.getRateLimit().value(),
timeInMillis(operation.getRateLimit().window(), operation.getRateLimit().windowUnit()),
timeInMillis(operation.getRateLimit().minSpacing(), operation.getRateLimit().minSpacingUnit()),
operation.getRateLimit().type(),
SystemStopwatch.INSTANCE);
}
if (operation.hasCircuitBreaker()) {
long delayInMillis = timeInMillis(operation.getCircuitBreaker().delay(), operation.getCircuitBreaker().delayUnit());
result = new CompletionStageCircuitBreaker<>(result, point.toString(),
createExceptionDecision(operation.getCircuitBreaker().skipOn(), operation.getCircuitBreaker().failOn()),
delayInMillis,
operation.getCircuitBreaker().requestVolumeThreshold(),
operation.getCircuitBreaker().failureRatio(),
operation.getCircuitBreaker().successThreshold(),
SystemStopwatch.INSTANCE,
timer);
String cbName = operation.hasCircuitBreakerName()
? operation.getCircuitBreakerName().value()
: UUID.randomUUID().toString();
cbMaintenance.register(cbName, (CircuitBreaker>) result);
}
if (operation.hasRetry()) {
long maxDurationMs = timeInMillis(operation.getRetry().maxDuration(), operation.getRetry().durationUnit());
Supplier backoff = prepareRetryBackoff(operation);
result = new CompletionStageRetry<>(result, point.toString(),
createResultDecision(operation.hasRetryWhen() ? operation.getRetryWhen().result() : null),
createExceptionDecision(operation.getRetry().abortOn(), operation.getRetry().retryOn(),
operation.hasRetryWhen() ? operation.getRetryWhen().exception() : null),
operation.getRetry().maxRetries(),
maxDurationMs,
() -> new TimerDelay(backoff.get(), timer),
SystemStopwatch.INSTANCE,
operation.hasBeforeRetry() ? prepareBeforeRetryFunction(point, operation) : null);
}
if (operation.hasFallback()) {
result = new CompletionStageFallback<>(result, point.toString(),
prepareFallbackFunction(point, operation),
createExceptionDecision(operation.getFallback().skipOn(), operation.getFallback().applyOn()));
}
if (metricsProvider.isEnabled()) {
MeteredOperation meteredOperation = new CdiMeteredOperationImpl(operation, point, specCompatibility);
result = new CompletionStageMetricsCollector<>(result, metricsProvider.create(meteredOperation),
meteredOperation);
}
if (!operation.isThreadOffloadRequired()) {
result = new RememberEventLoop<>(result, eventLoop);
}
return result;
}
private FaultToleranceStrategy prepareSyncStrategy(FaultToleranceOperation operation, InterceptionPoint point) {
FaultToleranceStrategy result = invocation();
if (operation.hasBulkhead()) {
result = new SemaphoreBulkhead<>(result, point.toString(), operation.getBulkhead().value());
}
if (operation.hasTimeout()) {
long timeoutMs = timeInMillis(operation.getTimeout().value(), operation.getTimeout().unit());
result = new Timeout<>(result, point.toString(), timeoutMs, timer);
}
if (operation.hasRateLimit()) {
result = new RateLimit<>(result, point.toString(),
operation.getRateLimit().value(),
timeInMillis(operation.getRateLimit().window(), operation.getRateLimit().windowUnit()),
timeInMillis(operation.getRateLimit().minSpacing(), operation.getRateLimit().minSpacingUnit()),
operation.getRateLimit().type(),
SystemStopwatch.INSTANCE);
}
if (operation.hasCircuitBreaker()) {
long delayInMillis = timeInMillis(operation.getCircuitBreaker().delay(), operation.getCircuitBreaker().delayUnit());
result = new CircuitBreaker<>(result, point.toString(),
createExceptionDecision(operation.getCircuitBreaker().skipOn(), operation.getCircuitBreaker().failOn()),
delayInMillis,
operation.getCircuitBreaker().requestVolumeThreshold(),
operation.getCircuitBreaker().failureRatio(),
operation.getCircuitBreaker().successThreshold(),
SystemStopwatch.INSTANCE,
timer);
String cbName = operation.hasCircuitBreakerName()
? operation.getCircuitBreakerName().value()
: UUID.randomUUID().toString();
cbMaintenance.register(cbName, (CircuitBreaker>) result);
}
if (operation.hasRetry()) {
long maxDurationMs = timeInMillis(operation.getRetry().maxDuration(), operation.getRetry().durationUnit());
Supplier backoff = prepareRetryBackoff(operation);
result = new Retry<>(result, point.toString(),
createResultDecision(operation.hasRetryWhen() ? operation.getRetryWhen().result() : null),
createExceptionDecision(operation.getRetry().abortOn(), operation.getRetry().retryOn(),
operation.hasRetryWhen() ? operation.getRetryWhen().exception() : null),
operation.getRetry().maxRetries(),
maxDurationMs,
() -> new ThreadSleepDelay(backoff.get()),
SystemStopwatch.INSTANCE,
operation.hasBeforeRetry() ? prepareBeforeRetryFunction(point, operation) : null);
}
if (operation.hasFallback()) {
result = new Fallback<>(result, point.toString(),
prepareFallbackFunction(point, operation),
createExceptionDecision(operation.getFallback().skipOn(), operation.getFallback().applyOn()));
}
if (metricsProvider.isEnabled()) {
MeteredOperation meteredOperation = new CdiMeteredOperationImpl(operation, point, specCompatibility);
result = new MetricsCollector<>(result, metricsProvider.create(meteredOperation), meteredOperation);
}
return result;
}
private FaultToleranceStrategy> prepareFutureStrategy(FaultToleranceOperation operation,
InterceptionPoint point) {
FaultToleranceStrategy> result = invocation();
result = new RequestScopeActivator<>(result, requestContextController);
if (operation.hasBulkhead()) {
int size = operation.getBulkhead().value();
int queueSize = operation.getBulkhead().waitingTaskQueue();
result = new FutureThreadPoolBulkhead<>(result, point.toString(), size, queueSize);
}
if (operation.hasTimeout()) {
long timeoutMs = timeInMillis(operation.getTimeout().value(), operation.getTimeout().unit());
Timeout> timeout = new Timeout<>(result, point.toString(), timeoutMs, timer);
result = new AsyncTimeout<>(timeout, asyncExecutor);
}
if (operation.hasRateLimit()) {
result = new RateLimit<>(result, point.toString(),
operation.getRateLimit().value(),
timeInMillis(operation.getRateLimit().window(), operation.getRateLimit().windowUnit()),
timeInMillis(operation.getRateLimit().minSpacing(), operation.getRateLimit().minSpacingUnit()),
operation.getRateLimit().type(),
SystemStopwatch.INSTANCE);
}
if (operation.hasCircuitBreaker()) {
long delayInMillis = timeInMillis(operation.getCircuitBreaker().delay(), operation.getCircuitBreaker().delayUnit());
result = new CircuitBreaker<>(result, point.toString(),
createExceptionDecision(operation.getCircuitBreaker().skipOn(), operation.getCircuitBreaker().failOn()),
delayInMillis,
operation.getCircuitBreaker().requestVolumeThreshold(),
operation.getCircuitBreaker().failureRatio(),
operation.getCircuitBreaker().successThreshold(),
SystemStopwatch.INSTANCE,
timer);
String cbName = operation.hasCircuitBreakerName()
? operation.getCircuitBreakerName().value()
: UUID.randomUUID().toString();
cbMaintenance.register(cbName, (CircuitBreaker>) result);
}
if (operation.hasRetry()) {
long maxDurationMs = timeInMillis(operation.getRetry().maxDuration(), operation.getRetry().durationUnit());
Supplier backoff = prepareRetryBackoff(operation);
result = new Retry<>(result, point.toString(),
createResultDecision(operation.hasRetryWhen() ? operation.getRetryWhen().result() : null),
createExceptionDecision(operation.getRetry().abortOn(), operation.getRetry().retryOn(),
operation.hasRetryWhen() ? operation.getRetryWhen().exception() : null),
operation.getRetry().maxRetries(),
maxDurationMs,
() -> new ThreadSleepDelay(backoff.get()),
SystemStopwatch.INSTANCE,
operation.hasBeforeRetry() ? prepareBeforeRetryFunction(point, operation) : null);
}
if (operation.hasFallback()) {
result = new Fallback<>(result, point.toString(),
prepareFallbackFunction(point, operation),
createExceptionDecision(operation.getFallback().skipOn(), operation.getFallback().applyOn()));
}
if (metricsProvider.isEnabled()) {
MeteredOperation meteredOperation = new CdiMeteredOperationImpl(operation, point, specCompatibility);
result = new MetricsCollector<>(result, metricsProvider.create(meteredOperation), meteredOperation);
}
result = new FutureExecution<>(result, asyncExecutor);
return result;
}
private Supplier prepareRetryBackoff(FaultToleranceOperation operation) {
long delayMs = timeInMillis(operation.getRetry().delay(), operation.getRetry().delayUnit());
long jitterMs = timeInMillis(operation.getRetry().jitter(), operation.getRetry().jitterDelayUnit());
Jitter jitter = jitterMs == 0 ? Jitter.ZERO : new RandomJitter(jitterMs);
if (operation.hasExponentialBackoff()) {
int factor = operation.getExponentialBackoff().factor();
long maxDelay = timeInMillis(operation.getExponentialBackoff().maxDelay(),
operation.getExponentialBackoff().maxDelayUnit());
return () -> new ExponentialBackOff(delayMs, factor, jitter, maxDelay);
} else if (operation.hasFibonacciBackoff()) {
long maxDelay = timeInMillis(operation.getFibonacciBackoff().maxDelay(),
operation.getFibonacciBackoff().maxDelayUnit());
return () -> new FibonacciBackOff(delayMs, jitter, maxDelay);
} else if (operation.hasCustomBackoff()) {
Class extends CustomBackoffStrategy> strategy = operation.getCustomBackoff().value();
return () -> {
CustomBackoffStrategy instance;
try {
instance = strategy.getConstructor().newInstance();
} catch (ReflectiveOperationException e) {
throw sneakyThrow(e);
}
instance.init(delayMs);
return new CustomBackOff(instance::nextDelayInMillis);
};
} else {
return () -> new ConstantBackOff(delayMs, jitter);
}
}
private Function prepareFallbackFunction(InterceptionPoint point,
FaultToleranceOperation operation) {
AsyncSupport asyncSupport = AsyncSupportRegistry.get(operation.getParameterTypes(), operation.getReturnType());
String fallbackMethodName = operation.getFallback().fallbackMethod();
FallbackMethodCandidates candidates = !"".equals(fallbackMethodName)
? cache.getFallbackMethodCandidates(point, operation)
: null;
Function fallbackFunction;
if (candidates != null) {
fallbackFunction = ctx -> {
FallbackMethod fallbackMethod = candidates.select(ctx.failure.getClass());
if (fallbackMethod == null) {
throw sneakyThrow(ctx.failure);
}
try {
Invoker> invoker = fallbackMethod.createInvoker(ctx);
return asyncSupport == null ? (V) invoker.proceed() : (V) asyncSupport.toCompletionStage(invoker);
} catch (InvocationTargetException e) {
throw sneakyThrow(e.getCause());
} catch (Throwable e) {
throw new FaultToleranceException("Error during fallback method invocation", e);
}
};
} else {
FallbackHandler fallbackHandler = fallbackHandlerProvider.get(operation);
if (fallbackHandler != null) {
fallbackFunction = ctx -> {
ExecutionContext executionContext = new ExecutionContextImpl(
ctx.invocationContext.get(InvocationContext.class), ctx.failure);
V result = fallbackHandler.handle(executionContext);
if (asyncSupport != null) {
result = (V) asyncSupport.fallbackResultToCompletionStage(result);
}
return result;
};
} else {
throw new FaultToleranceException("Could not obtain fallback handler for " + point);
}
}
if (specCompatibility.isOperationTrulyAsynchronous(operation) && operation.isThreadOffloadRequired()) {
fallbackFunction = new AsyncFallbackFunction(fallbackFunction, asyncExecutor);
}
return fallbackFunction;
}
private Consumer prepareBeforeRetryFunction(InterceptionPoint point, FaultToleranceOperation operation) {
String methodName = operation.getBeforeRetry().methodName();
BeforeRetryMethod method = !"".equals(methodName)
? cache.getBeforeRetryMethod(point, operation)
: null;
Consumer beforeRetryFunction;
if (method != null) {
beforeRetryFunction = ctx -> {
try {
method.createInvoker(ctx).proceed();
} catch (InvocationTargetException e) {
throw sneakyThrow(e.getCause());
} catch (Throwable e) {
throw new FaultToleranceException("Error during before retry method invocation", e);
}
};
} else {
BeforeRetryHandler beforeRetryHandler = beforeRetryHandlerProvider.get(operation);
if (beforeRetryHandler != null) {
beforeRetryFunction = ctx -> {
ExecutionContext executionContext = new ExecutionContextImpl(
ctx.invocationContext.get(InvocationContext.class), ctx.failure);
beforeRetryHandler.handle(executionContext);
};
} else {
throw new FaultToleranceException("Could not obtain before retry handler for " + point);
}
}
return beforeRetryFunction;
}
private ResultDecision createResultDecision(Class extends Predicate> whenResult) {
if (whenResult != null && whenResult != NeverOnResult.class) {
Predicate predicate;
try {
predicate = whenResult.getConstructor().newInstance();
} catch (ReflectiveOperationException e) {
throw sneakyThrow(e);
}
return new PredicateBasedResultDecision(predicate.negate());
}
return ResultDecision.ALWAYS_EXPECTED;
}
private ExceptionDecision createExceptionDecision(Class extends Throwable>[] consideredExpected,
Class extends Throwable>[] consideredFailure) {
return new SetBasedExceptionDecision(createSetOfThrowables(consideredFailure),
createSetOfThrowables(consideredExpected), specCompatibility.inspectExceptionCauseChain());
}
private ExceptionDecision createExceptionDecision(Class extends Throwable>[] consideredExpected,
Class extends Throwable>[] consideredFailure, Class extends Predicate> whenException) {
if (whenException != null && whenException != AlwaysOnException.class) {
Predicate predicate;
try {
predicate = whenException.getConstructor().newInstance();
} catch (ReflectiveOperationException e) {
throw sneakyThrow(e);
}
return new PredicateBasedExceptionDecision(predicate.negate());
}
return new SetBasedExceptionDecision(createSetOfThrowables(consideredFailure),
createSetOfThrowables(consideredExpected), specCompatibility.inspectExceptionCauseChain());
}
private SetOfThrowables createSetOfThrowables(Class extends Throwable>[] throwableClasses) {
if (throwableClasses == null || throwableClasses.length == 0) {
return SetOfThrowables.EMPTY;
}
return SetOfThrowables.create(Arrays.asList(throwableClasses));
}
}