From 08e4d4409e380dcff2ad401b242144d3a1c017b4 Mon Sep 17 00:00:00 2001 From: Martin Siegemund Date: Sat, 28 Oct 2023 14:19:05 +0200 Subject: [PATCH] introduce disable-time-limiter property --- README.adoc | 18 ++++ ...ReactiveResilience4JAutoConfiguration.java | 5 +- .../ReactiveResilience4JCircuitBreaker.java | 65 ++++++++---- ...tiveResilience4JCircuitBreakerFactory.java | 23 ++++- .../Resilience4JCircuitBreaker.java | 49 +++++++--- .../Resilience4JCircuitBreakerFactory.java | 3 +- .../Resilience4JConfigurationProperties.java | 10 ++ .../Resilience4jBulkheadProvider.java | 6 +- ...4JAutoConfigurationWithoutMetricsTest.java | 2 +- ...eactiveResilience4JCircuitBreakerTest.java | 98 ++++++++++++++++++- ...ce4JAutoConfigurationMetricsConfigRun.java | 2 +- .../Resilience4JCircuitBreakerTest.java | 72 ++++++++++++++ 12 files changed, 310 insertions(+), 43 deletions(-) diff --git a/README.adoc b/README.adoc index 5d0742fe..60c21f32 100644 --- a/README.adoc +++ b/README.adoc @@ -253,6 +253,24 @@ resilience4j.timelimiter: For more information on Resilience4j property configuration, see https://resilience4j.readme.io/docs/getting-started-3#configuration[Resilience4J Spring Boot 2 Configuration]. +===== Disabling the TimeLimiter + +By default, the `TimeLimiter` is enabled and every execution is backed by a time limit. This time limit is either defined explicitly or the default time limit (provided by `io.github.resilience4j.timelimiter.TimeLimiterConfig#ofDefaults`) is used. + +The `TimeLimiter` can be globally disabled by setting the property `spring.cloud.circuitbreaker.resilience4j.disable-time-limiter` to `true`. + +[source,yaml] +---- +spring: + cloud: + circuitbreaker: + resilience4j: + disable-time-limiter: true +---- + +This type of option is only provided on a global scope within the `spring-cloud-circuitbreaker` and applies to the +basic and to the reactive circuitbreaker implementation. + ==== Bulkhead pattern supporting If `resilience4j-bulkhead` is on the classpath, Spring Cloud CircuitBreaker will wrap all methods with a Resilience4j Bulkhead. You can disable the Resilience4j Bulkhead by setting `spring.cloud.circuitbreaker.bulkhead.resilience4j.enabled` to `false`. diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfiguration.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfiguration.java index 6589933a..e2775545 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfiguration.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfiguration.java @@ -54,9 +54,10 @@ public class ReactiveResilience4JAutoConfiguration { @Bean @ConditionalOnMissingBean(ReactiveCircuitBreakerFactory.class) public ReactiveResilience4JCircuitBreakerFactory reactiveResilience4JCircuitBreakerFactory( - CircuitBreakerRegistry circuitBreakerRegistry, TimeLimiterRegistry timeLimiterRegistry) { + CircuitBreakerRegistry circuitBreakerRegistry, TimeLimiterRegistry timeLimiterRegistry, + Resilience4JConfigurationProperties resilience4JConfigurationProperties) { ReactiveResilience4JCircuitBreakerFactory factory = new ReactiveResilience4JCircuitBreakerFactory( - circuitBreakerRegistry, timeLimiterRegistry); + circuitBreakerRegistry, timeLimiterRegistry, resilience4JConfigurationProperties); customizers.forEach(customizer -> customizer.customize(factory)); return factory; } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreaker.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreaker.java index 175e67d6..af340e4c 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreaker.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreaker.java @@ -16,6 +16,7 @@ package org.springframework.cloud.circuitbreaker.resilience4j; +import java.time.Duration; import java.util.Map; import java.util.Optional; import java.util.concurrent.TimeUnit; @@ -59,10 +60,21 @@ public class ReactiveResilience4JCircuitBreaker implements ReactiveCircuitBreake private final Optional> circuitBreakerCustomizer; + private final boolean disableTimeLimiter; + + @Deprecated + public ReactiveResilience4JCircuitBreaker(String id, String groupName, + Resilience4JConfigBuilder.Resilience4JCircuitBreakerConfiguration config, + CircuitBreakerRegistry circuitBreakerRegistry, TimeLimiterRegistry timeLimiterRegistry, + Optional> circuitBreakerCustomizer) { + this(id, groupName, config, circuitBreakerRegistry, timeLimiterRegistry, circuitBreakerCustomizer, false); + } + public ReactiveResilience4JCircuitBreaker(String id, String groupName, Resilience4JConfigBuilder.Resilience4JCircuitBreakerConfiguration config, CircuitBreakerRegistry circuitBreakerRegistry, TimeLimiterRegistry timeLimiterRegistry, - Optional> circuitBreakerCustomizer) { + Optional> circuitBreakerCustomizer, + boolean disableTimeLimiter) { this.id = id; this.groupName = groupName; this.circuitBreakerConfig = config.getCircuitBreakerConfig(); @@ -70,18 +82,24 @@ public ReactiveResilience4JCircuitBreaker(String id, String groupName, this.circuitBreakerCustomizer = circuitBreakerCustomizer; this.timeLimiterConfig = config.getTimeLimiterConfig(); this.timeLimiterRegistry = timeLimiterRegistry; + this.disableTimeLimiter = disableTimeLimiter; } @Override public Mono run(Mono toRun, Function> fallback) { - Tuple2 tuple = buildCircuitBreakerAndTimeLimiter(); - Mono toReturn = toRun.transform(CircuitBreakerOperator.of(tuple.getT1())) - .timeout(tuple.getT2().getTimeLimiterConfig().getTimeoutDuration()) - // Since we are using the Mono timeout we need to tell the circuit breaker - // about the error - .doOnError(TimeoutException.class, - t -> tuple.getT1().onError(tuple.getT2().getTimeLimiterConfig().getTimeoutDuration().toMillis(), - TimeUnit.MILLISECONDS, t)); + Tuple2> tuple = buildCircuitBreakerAndTimeLimiter(); + Mono toReturn = toRun.transform(CircuitBreakerOperator.of(tuple.getT1())); + if (tuple.getT2().isPresent()) { + final Duration timeoutDuration = tuple.getT2().get().getTimeLimiterConfig().getTimeoutDuration(); + toReturn = toReturn + .timeout(timeoutDuration) + // Since we are using the Mono timeout we need to tell the circuit breaker + // about the error + .doOnError(TimeoutException.class, + t -> tuple.getT1() + .onError(timeoutDuration.toMillis(), + TimeUnit.MILLISECONDS, t)); + } if (fallback != null) { toReturn = toReturn.onErrorResume(fallback); } @@ -90,28 +108,37 @@ public Mono run(Mono toRun, Function> fallback) { @Override public Flux run(Flux toRun, Function> fallback) { - Tuple2 tuple = buildCircuitBreakerAndTimeLimiter(); - Flux toReturn = toRun.transform(CircuitBreakerOperator.of(tuple.getT1())) - .timeout(tuple.getT2().getTimeLimiterConfig().getTimeoutDuration()) - // Since we are using the Flux timeout we need to tell the circuit breaker - // about the error - .doOnError(TimeoutException.class, - t -> tuple.getT1().onError(tuple.getT2().getTimeLimiterConfig().getTimeoutDuration().toMillis(), - TimeUnit.MILLISECONDS, t)); + Tuple2> tuple = buildCircuitBreakerAndTimeLimiter(); + Flux toReturn = toRun.transform(CircuitBreakerOperator.of(tuple.getT1())); + if (tuple.getT2().isPresent()) { + final Duration timeoutDuration = tuple.getT2().get().getTimeLimiterConfig().getTimeoutDuration(); + toReturn = toReturn.timeout(timeoutDuration) + // Since we are using the Flux timeout we need to tell the circuit breaker + // about the error + .doOnError(TimeoutException.class, + t -> tuple.getT1() + .onError(timeoutDuration.toMillis(), + TimeUnit.MILLISECONDS, t)); + } if (fallback != null) { toReturn = toReturn.onErrorResume(fallback); } return toReturn; } - private Tuple2 buildCircuitBreakerAndTimeLimiter() { + private Tuple2> buildCircuitBreakerAndTimeLimiter() { final Map tags = Map.of(CIRCUIT_BREAKER_GROUP_TAG, this.groupName); CircuitBreaker circuitBreaker = circuitBreakerRegistry.circuitBreaker(id, circuitBreakerConfig, tags); circuitBreakerCustomizer.ifPresent(customizer -> customizer.customize(circuitBreaker)); + if (disableTimeLimiter) { + /* do not provide/load time-limiter */ + return Tuples.of(circuitBreaker, Optional.empty()); + } + /* provide time-limiter */ TimeLimiter timeLimiter = this.timeLimiterRegistry.find(this.id) .orElseGet(() -> this.timeLimiterRegistry.find(this.groupName) .orElseGet(() -> this.timeLimiterRegistry.timeLimiter(this.id, this.timeLimiterConfig, tags))); - return Tuples.of(circuitBreaker, timeLimiter); + return Tuples.of(circuitBreaker, Optional.of(timeLimiter)); } } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerFactory.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerFactory.java index f14f08d9..5c9eeef5 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerFactory.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerFactory.java @@ -46,15 +46,25 @@ public class ReactiveResilience4JCircuitBreakerFactory extends private TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); - private Map> circuitBreakerCustomizers = new HashMap<>(); + private final Map> circuitBreakerCustomizers = new HashMap<>(); + + private final Resilience4JConfigurationProperties resilience4JConfigurationProperties; + + @Deprecated + public ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry circuitBreakerRegistry, + TimeLimiterRegistry timeLimiterRegistry) { + this(circuitBreakerRegistry, timeLimiterRegistry, null); + } public ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry circuitBreakerRegistry, - TimeLimiterRegistry timeLimiterRegistry) { + TimeLimiterRegistry timeLimiterRegistry, + Resilience4JConfigurationProperties resilience4JConfigurationProperties) { this.circuitBreakerRegistry = circuitBreakerRegistry; this.timeLimiterRegistry = timeLimiterRegistry; this.defaultConfiguration = id -> new Resilience4JConfigBuilder(id) .circuitBreakerConfig(this.circuitBreakerRegistry.getDefaultConfig()) .timeLimiterConfig(this.timeLimiterRegistry.getDefaultConfig()).build(); + this.resilience4JConfigurationProperties = resilience4JConfigurationProperties; } @Override @@ -91,7 +101,14 @@ public ReactiveCircuitBreaker create(String id, String groupName) { Resilience4JConfigBuilder.Resilience4JCircuitBreakerConfiguration config = new Resilience4JConfigBuilder(id) .circuitBreakerConfig(circuitBreakerConfig).timeLimiterConfig(timeLimiterConfig).build(); return new ReactiveResilience4JCircuitBreaker(id, groupName, config, circuitBreakerRegistry, - timeLimiterRegistry, Optional.ofNullable(circuitBreakerCustomizers.get(id))); + timeLimiterRegistry, Optional.ofNullable(circuitBreakerCustomizers.get(id)), isDisableTimeLimiter()); + } + + private boolean isDisableTimeLimiter() { + if (resilience4JConfigurationProperties != null) { + return resilience4JConfigurationProperties.isDisableTimeLimiter(); + } + return false; } @Override diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreaker.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreaker.java index f2c9c185..fe9e3d16 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreaker.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreaker.java @@ -45,6 +45,7 @@ public class Resilience4JCircuitBreaker implements CircuitBreaker { private final String id; private final String groupName; + private final Map tags; private Resilience4jBulkheadProvider bulkheadProvider; @@ -60,12 +61,26 @@ public class Resilience4JCircuitBreaker implements CircuitBreaker { private final Optional> circuitBreakerCustomizer; + private final boolean disableTimeLimiter; + + @Deprecated + public Resilience4JCircuitBreaker(String id, String groupName, + io.github.resilience4j.circuitbreaker.CircuitBreakerConfig circuitBreakerConfig, + TimeLimiterConfig timeLimiterConfig, CircuitBreakerRegistry circuitBreakerRegistry, + TimeLimiterRegistry timeLimiterRegistry, ExecutorService executorService, + Optional> circuitBreakerCustomizer, + Resilience4jBulkheadProvider bulkheadProvider) { + this(id, groupName, circuitBreakerConfig, timeLimiterConfig, circuitBreakerRegistry, timeLimiterRegistry, + executorService, circuitBreakerCustomizer, bulkheadProvider, false); + } + public Resilience4JCircuitBreaker(String id, String groupName, io.github.resilience4j.circuitbreaker.CircuitBreakerConfig circuitBreakerConfig, TimeLimiterConfig timeLimiterConfig, CircuitBreakerRegistry circuitBreakerRegistry, TimeLimiterRegistry timeLimiterRegistry, ExecutorService executorService, Optional> circuitBreakerCustomizer, - Resilience4jBulkheadProvider bulkheadProvider) { + Resilience4jBulkheadProvider bulkheadProvider, + boolean disableTimeLimiter) { this.id = id; this.groupName = groupName; this.circuitBreakerConfig = circuitBreakerConfig; @@ -75,6 +90,8 @@ public Resilience4JCircuitBreaker(String id, String groupName, this.executorService = executorService; this.circuitBreakerCustomizer = circuitBreakerCustomizer; this.bulkheadProvider = bulkheadProvider; + this.disableTimeLimiter = disableTimeLimiter; + this.tags = Map.of(CIRCUIT_BREAKER_GROUP_TAG, this.groupName); } public Resilience4JCircuitBreaker(String id, String groupName, @@ -90,40 +107,42 @@ public Resilience4JCircuitBreaker(String id, String groupName, @Override public T run(Supplier toRun, Function fallback) { final Map tags = Map.of(CIRCUIT_BREAKER_GROUP_TAG, this.groupName); - TimeLimiter timeLimiter = this.timeLimiterRegistry.find(this.id) - .orElseGet(() -> this.timeLimiterRegistry.find(this.groupName) - .orElseGet(() -> this.timeLimiterRegistry.timeLimiter(this.id, this.timeLimiterConfig, tags))); + Optional timeLimiter = loadTimeLimiter(); io.github.resilience4j.circuitbreaker.CircuitBreaker defaultCircuitBreaker = registry.circuitBreaker(this.id, - this.circuitBreakerConfig, tags); + this.circuitBreakerConfig, tags); circuitBreakerCustomizer.ifPresent(customizer -> customizer.customize(defaultCircuitBreaker)); if (bulkheadProvider != null) { if (executorService != null) { Supplier> futureSupplier = () -> executorService.submit(toRun::get); - Callable timeLimitedCall = TimeLimiter.decorateFutureSupplier(timeLimiter, futureSupplier); + /* conditionally wrap in time-limiter */ + Callable timeLimitedCall = timeLimiter.map(tl -> TimeLimiter.decorateFutureSupplier(tl, futureSupplier)) + .orElse(() -> futureSupplier.get().get()); Callable bulkheadCall = bulkheadProvider.decorateCallable(this.groupName, tags, timeLimitedCall); Callable circuitBreakerCall = io.github.resilience4j.circuitbreaker.CircuitBreaker - .decorateCallable(defaultCircuitBreaker, bulkheadCall); + .decorateCallable(defaultCircuitBreaker, bulkheadCall); return getAndApplyFallback(circuitBreakerCall, fallback); } else { Callable bulkheadCall = bulkheadProvider.decorateCallable(this.groupName, tags, toRun::get); Callable circuitBreakerCall = io.github.resilience4j.circuitbreaker.CircuitBreaker - .decorateCallable(defaultCircuitBreaker, bulkheadCall); + .decorateCallable(defaultCircuitBreaker, bulkheadCall); return getAndApplyFallback(circuitBreakerCall, fallback); } } else { if (executorService != null) { Supplier> futureSupplier = () -> executorService.submit(toRun::get); - Callable restrictedCall = TimeLimiter.decorateFutureSupplier(timeLimiter, futureSupplier); + /* conditionally wrap in time-limiter */ + Callable restrictedCall = timeLimiter.map(tl -> TimeLimiter.decorateFutureSupplier(tl, futureSupplier)) + .orElse(() -> futureSupplier.get().get()); Callable callable = io.github.resilience4j.circuitbreaker.CircuitBreaker - .decorateCallable(defaultCircuitBreaker, restrictedCall); + .decorateCallable(defaultCircuitBreaker, restrictedCall); return getAndApplyFallback(callable, fallback); } else { Supplier decorator = io.github.resilience4j.circuitbreaker.CircuitBreaker - .decorateSupplier(defaultCircuitBreaker, toRun); + .decorateSupplier(defaultCircuitBreaker, toRun); return getAndApplyFallback(decorator, fallback); } } @@ -147,4 +166,12 @@ private static T getAndApplyFallback(Callable callable, Function loadTimeLimiter() { + if (disableTimeLimiter) { + return Optional.empty(); + } + return Optional.of(this.timeLimiterRegistry.find(this.id) + .orElseGet(() -> this.timeLimiterRegistry.find(this.groupName) + .orElseGet(() -> this.timeLimiterRegistry.timeLimiter(this.id, this.timeLimiterConfig, this.tags)))); + } } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerFactory.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerFactory.java index 717b7827..9a6ca07b 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerFactory.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerFactory.java @@ -172,7 +172,8 @@ private Resilience4JCircuitBreaker create(String id, String groupName, else { return new Resilience4JCircuitBreaker(id, groupName, circuitBreakerConfig, timeLimiterConfig, circuitBreakerRegistry, timeLimiterRegistry, circuitBreakerExecutorService, - Optional.ofNullable(circuitBreakerCustomizers.get(id)), bulkheadProvider); + Optional.ofNullable(circuitBreakerCustomizers.get(id)), bulkheadProvider, + this.resilience4JConfigurationProperties.isDisableTimeLimiter()); } } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JConfigurationProperties.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JConfigurationProperties.java index fcfc2ea7..d04701d7 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JConfigurationProperties.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JConfigurationProperties.java @@ -32,6 +32,8 @@ public class Resilience4JConfigurationProperties { private boolean disableThreadPool = false; + private boolean disableTimeLimiter = false; + public boolean isEnableGroupMeterFilter() { return enableGroupMeterFilter; } @@ -64,4 +66,12 @@ public void setDisableThreadPool(boolean disableThreadPool) { this.disableThreadPool = disableThreadPool; } + + boolean isDisableTimeLimiter() { + return disableTimeLimiter; + } + + void setDisableTimeLimiter(boolean disableTimeLimiter) { + this.disableTimeLimiter = disableTimeLimiter; + } } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4jBulkheadProvider.java b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4jBulkheadProvider.java index 70ab52f9..ea426469 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4jBulkheadProvider.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/main/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4jBulkheadProvider.java @@ -154,8 +154,12 @@ private boolean useSemaphoreBulkhead(String id) { || (bulkheadRegistry.find(id).isPresent() && threadPoolBulkheadRegistry.find(id).isEmpty()); } - private Callable decorateTimeLimiter(final Supplier> supplier, TimeLimiter timeLimiter) { + private static Callable decorateTimeLimiter(final Supplier> supplier, TimeLimiter timeLimiter) { final Supplier> futureSupplier = () -> supplier.get().toCompletableFuture(); + if (timeLimiter == null) { + /* execute without time-limiter */ + return () -> futureSupplier.get().get(); + } return timeLimiter.decorateFutureSupplier(futureSupplier); } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfigurationWithoutMetricsTest.java b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfigurationWithoutMetricsTest.java index f038247e..c47775eb 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfigurationWithoutMetricsTest.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JAutoConfigurationWithoutMetricsTest.java @@ -44,7 +44,7 @@ public class ReactiveResilience4JAutoConfigurationWithoutMetricsTest { static ReactiveResilience4JCircuitBreakerFactory circuitBreakerFactory = spy( new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults())); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties())); @Test public void testWithoutMetrics() { diff --git a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerTest.java b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerTest.java index 352ee51b..0a10dee9 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerTest.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/ReactiveResilience4JCircuitBreakerTest.java @@ -18,13 +18,17 @@ import java.util.Arrays; import java.util.Collections; +import java.util.concurrent.TimeUnit; import io.github.resilience4j.circuitbreaker.CircuitBreakerRegistry; import io.github.resilience4j.timelimiter.TimeLimiterRegistry; +import org.junit.Assert; import org.junit.Test; import reactor.core.publisher.Flux; import reactor.core.publisher.Mono; +import reactor.core.scheduler.Schedulers; +import org.springframework.cloud.client.circuitbreaker.NoFallbackAvailableException; import org.springframework.cloud.client.circuitbreaker.ReactiveCircuitBreaker; import static org.assertj.core.api.Assertions.assertThat; @@ -38,14 +42,14 @@ public class ReactiveResilience4JCircuitBreakerTest { @Test public void runMono() { ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults()).create("foo"); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties()).create("foo"); assertThat(Mono.just("foobar").transform(cb::run).block()).isEqualTo("foobar"); } @Test public void runMonoWithFallback() { ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults()).create("foo"); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties()).create("foo"); assertThat(Mono.error(new RuntimeException("boom")).transform(it -> cb.run(it, t -> Mono.just("fallback"))) .block()).isEqualTo("fallback"); } @@ -53,7 +57,7 @@ public void runMonoWithFallback() { @Test public void runFlux() { ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults()).create("foo"); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties()).create("foo"); assertThat(Flux.just("foobar", "hello world").transform(cb::run).collectList().block()) .isEqualTo(Arrays.asList("foobar", "hello world")); } @@ -61,9 +65,95 @@ public void runFlux() { @Test public void runFluxWithFallback() { ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults()).create("foo"); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties()).create("foo"); assertThat(Flux.error(new RuntimeException("boom")).transform(it -> cb.run(it, t -> Flux.just("fallback"))) .collectList().block()).isEqualTo(Collections.singletonList("fallback")); } + /** + * Run circuit breaker with default time limiter and expects everything to run without errors. + */ + @Test + public void runWithDefaultTimeLimiter() { + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, new Resilience4JConfigurationProperties()).create("foo"); + + assertThat(Mono.fromCallable(() -> { + try { + /* sleep less than time limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.min(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration() + .toMillis() / 2L, 0L)); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException("thread got interrupted", e); + } + return "foobar"; + }) + .subscribeOn(Schedulers.single()) + .transform(cb::run) + .block() + ).isEqualTo("foobar"); + } + + /** + * Run circuit breaker with default time limiter and expects the time limit to get exceeded. + */ + @Test(expected = NoFallbackAvailableException.class) + public void runWithDefaultTimeLimiterTooSlow() { + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, new Resilience4JConfigurationProperties()).create("foo"); + + Mono.fromCallable(() -> { + try { + /* sleep longer than time limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.max(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration().toMillis(), 100L) * 2); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException("thread got interrupted", e); + } + return "foobar"; + }) + .subscribeOn(Schedulers.single()) + .transform(cb::run) + .doOnSuccess(s -> { + throw new AssertionError("timeout did not occur"); + }) + .block(); + + Assert.fail("execution did not cause exception"); + } + + /** + * Run circuit breaker with default time limiter and exceed time limit. Due to the disabled time limiter execution, + * everything should finish without errors. + */ + @Test + public void runWithDisabledTimeLimiter() { + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + final Resilience4JConfigurationProperties resilience4JConfigurationProperties = new Resilience4JConfigurationProperties(); + resilience4JConfigurationProperties.setDisableTimeLimiter(true); + ReactiveCircuitBreaker cb = new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, resilience4JConfigurationProperties).create("foo"); + + assertThat(Mono.fromCallable(() -> { + try { + /* sleep longer than timit limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.max(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration() + .toMillis(), 100L) * 2); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException("thread got interrupted", e); + } + return "foobar"; + }) + .subscribeOn(Schedulers.single()) + .transform(cb::run) + .block() + ).isEqualTo("foobar"); + } } diff --git a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JAutoConfigurationMetricsConfigRun.java b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JAutoConfigurationMetricsConfigRun.java index fcdbf968..9a95c5fa 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JAutoConfigurationMetricsConfigRun.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JAutoConfigurationMetricsConfigRun.java @@ -42,7 +42,7 @@ public class Resilience4JAutoConfigurationMetricsConfigRun { static ReactiveResilience4JCircuitBreakerFactory reactiveCircuitBreakerFactory = spy( new ReactiveResilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), - TimeLimiterRegistry.ofDefaults())); + TimeLimiterRegistry.ofDefaults(), new Resilience4JConfigurationProperties())); static Resilience4JCircuitBreakerFactory circuitBreakerFactory = spy( new Resilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), TimeLimiterRegistry.ofDefaults(), diff --git a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerTest.java b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerTest.java index 388b8572..7d6ed1f0 100644 --- a/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerTest.java +++ b/spring-cloud-circuitbreaker-resilience4j/src/test/java/org/springframework/cloud/circuitbreaker/resilience4j/Resilience4JCircuitBreakerTest.java @@ -16,14 +16,18 @@ package org.springframework.cloud.circuitbreaker.resilience4j; +import java.util.concurrent.TimeUnit; + import io.github.resilience4j.bulkhead.BulkheadRegistry; import io.github.resilience4j.bulkhead.ThreadPoolBulkheadRegistry; import io.github.resilience4j.circuitbreaker.CircuitBreakerRegistry; import io.github.resilience4j.timelimiter.TimeLimiterRegistry; +import org.assertj.core.api.Assertions; import org.junit.Before; import org.junit.Test; import org.springframework.cloud.client.circuitbreaker.CircuitBreaker; +import org.springframework.cloud.client.circuitbreaker.NoFallbackAvailableException; import static org.assertj.core.api.Assertions.assertThat; @@ -155,4 +159,72 @@ public void runWithFallbackBulkheadProviderAndGroupName() { }, t -> "fallback")).isEqualTo("fallback"); } + /** + * Run circuit breaker with default time limiter and expects everything to run without errors. + */ + @Test + public void runWithDefaultTimeLimiter() { + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + CircuitBreaker cb = new Resilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, null, properties).create("foo"); + assertThat(cb.run(() -> { + try { + /* sleep less than time limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.min(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration() + .toMillis() / 2L, 0L)); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException("thread got interrupted", e); + } + return "foobar"; + })).isEqualTo("foobar"); + } + + /** + * Run circuit breaker with default time limiter and expects the time limit to get exceeded. + */ + @Test(expected = NoFallbackAvailableException.class) + public void runWithDefaultTimeLimiterTooSlow() { + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + CircuitBreaker cb = new Resilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, null, properties).create("foo"); + cb.run(() -> { + try { + /* sleep longer than time limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.max(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration() + .toMillis(), 100L) * 2); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new AssertionError("thread got interrupted -> sleep failed", e); + } + return null; + }); + Assertions.fail("timeout did not happen as expected"); + } + + /** + * Run circuit breaker with default time limiter and exceed time limit. Due to the disabled time limiter execution, + * everything should finish without errors. + */ + @Test + public void runWithDisabledTimeLimiter() { + properties.setDisableTimeLimiter(true); + final TimeLimiterRegistry timeLimiterRegistry = TimeLimiterRegistry.ofDefaults(); + CircuitBreaker cb = new Resilience4JCircuitBreakerFactory(CircuitBreakerRegistry.ofDefaults(), + timeLimiterRegistry, null, properties).create("foo"); + assertThat(cb.run(() -> { + try { + /* sleep longer than limit limit allows us to */ + TimeUnit.MILLISECONDS.sleep(Math.max(timeLimiterRegistry.getDefaultConfig().getTimeoutDuration() + .toMillis(), 100L) * 2); + } + catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException("thread got interrupted", e); + } + return "foobar"; + })).isEqualTo("foobar"); + } }