From 5cae769c2db6d25fe4cbd9bd3b87ad6f463ec3e0 Mon Sep 17 00:00:00 2001 From: Sam Brannen <104798+sbrannen@users.noreply.github.com> Date: Tue, 1 Jul 2025 19:33:35 +0200 Subject: [PATCH] Remove maxDuration/maxElapsedTime support from RetryPolicy MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit When the core retry functionality was introduced, it had a built-in MaxRetryDurationPolicy. In #35058, that was migrated to a withMaxDuration() factory method, and in #35110 that was renamed to withMaxElapsedTime() (with a corresponding maxElapsedTime() method on the builder) in order to align with the maxElapsedTime feature of ExponentialBackOff. The latter also changed the semantics of the feature in the context of the RetryPolicy. However, @⁠Retryable does not provide maxElapsedTime support. In addition, the maxElapsedTime feature is a bit misleading, since it does not actually track CPU time or wall-clock time but rather only the sum of individual, accumulated back-off intervals/delays, which is likely not very useful. Furthermore, the maxElapsedTime will never apply to a zero-valued delay/interval. In light of the above, this commit removes the maxElapsedTime support from the built-in RetryPolicy. Users can still implement a custom BackOff strategy if they find they need some form of "max elapsed time" or "max duration". See gh-34716 See gh-35058 See gh-34529 See gh-35110 Closes gh-35144 --- .../core/retry/RetryPolicy.java | 43 ++--------- .../core/retry/RetryPolicyTests.java | 76 ++++--------------- 2 files changed, 20 insertions(+), 99 deletions(-) diff --git a/spring-core/src/main/java/org/springframework/core/retry/RetryPolicy.java b/spring-core/src/main/java/org/springframework/core/retry/RetryPolicy.java index c99c11e6778..83f2bab2560 100644 --- a/spring-core/src/main/java/org/springframework/core/retry/RetryPolicy.java +++ b/spring-core/src/main/java/org/springframework/core/retry/RetryPolicy.java @@ -35,8 +35,8 @@ import org.springframework.util.backoff.FixedBackOff; * *

Also provides factory methods and a fluent builder API for creating retry * policies with common configurations. See {@link #withDefaults()}, - * {@link #withMaxAttempts(long)}, {@link #withMaxElapsedTime(Duration)}, - * {@link #builder()}, and the configuration options in {@link Builder} for details. + * {@link #withMaxAttempts(long)}, {@link #builder()}, and the configuration + * options in {@link Builder} for details. * * @author Sam Brannen * @author Mahmoud Ben Hassine @@ -91,18 +91,6 @@ public interface RetryPolicy { return builder().backOff(new FixedBackOff(Builder.DEFAULT_DELAY, maxAttempts)).build(); } - /** - * Create a {@link RetryPolicy} configured with a maximum elapsed time. - *

The returned policy uses a fixed backoff of {@value Builder#DEFAULT_DELAY} - * milliseconds. - * @param maxElapsedTime the maximum elapsed time; must be positive - * @see Builder#maxElapsedTime(Duration) - * @see FixedBackOff - */ - static RetryPolicy withMaxElapsedTime(Duration maxElapsedTime) { - return builder().maxElapsedTime(maxElapsedTime).build(); - } - /** * Create a {@link Builder} to configure a {@link RetryPolicy} with common * configuration options. @@ -152,8 +140,6 @@ public interface RetryPolicy { private @Nullable Duration maxDelay; - private @Nullable Duration maxElapsedTime; - private final Set> includes = new LinkedHashSet<>(); private final Set> excludes = new LinkedHashSet<>(); @@ -173,8 +159,7 @@ public interface RetryPolicy { * strategy, you should not configure any of the following: * {@link #maxAttempts(long) maxAttempts}, {@link #delay(Duration) delay}, * {@link #jitter(Duration) jitter}, {@link #multiplier(double) multiplier}, - * {@link #maxDelay(Duration) maxDelay}, or {@link #maxElapsedTime(Duration) - * maxElapsedTime}. + * or {@link #maxDelay(Duration) maxDelay}. * @param backOff the {@code BackOff} strategy * @return this {@code Builder} instance for chained method invocations */ @@ -291,21 +276,6 @@ public interface RetryPolicy { return this; } - /** - * Specify the maximum elapsed time. - *

The default is unlimited. - *

The supplied value will override any previously configured value. - *

You should not specify this configuration option if you have - * configured a custom {@link #backOff(BackOff) BackOff} strategy. - * @param maxElapsedTime the maximum elapsed time; must be positive - * @return this {@code Builder} instance for chained method invocations - */ - public Builder maxElapsedTime(Duration maxElapsedTime) { - assertIsPositive("maxElapsedTime", maxElapsedTime); - this.maxElapsedTime = maxElapsedTime; - return this; - } - /** * Specify the types of exceptions for which the {@link RetryPolicy} * should retry a failed operation. @@ -415,10 +385,10 @@ public interface RetryPolicy { BackOff backOff = this.backOff; if (backOff != null) { boolean misconfigured = (this.maxAttempts != 0) || (this.delay != null) || (this.jitter != null) || - (this.multiplier != 0) || (this.maxDelay != null) || (this.maxElapsedTime != null); + (this.multiplier != 0) || (this.maxDelay != null); Assert.state(!misconfigured, """ The following configuration options are not supported with a custom BackOff strategy: \ - maxAttempts, delay, jitter, multiplier, maxDelay, or maxElapsedTime."""); + maxAttempts, delay, jitter, multiplier, or maxDelay."""); } else { ExponentialBackOff exponentialBackOff = new ExponentialBackOff(); @@ -429,9 +399,6 @@ public interface RetryPolicy { if (this.jitter != null) { exponentialBackOff.setJitter(this.jitter.toMillis()); } - if (this.maxElapsedTime != null) { - exponentialBackOff.setMaxElapsedTime(this.maxElapsedTime.toMillis()); - } backOff = exponentialBackOff; } return new DefaultRetryPolicy(this.includes, this.excludes, this.predicate, backOff); diff --git a/spring-core/src/test/java/org/springframework/core/retry/RetryPolicyTests.java b/spring-core/src/test/java/org/springframework/core/retry/RetryPolicyTests.java index 60f2da526d3..5ba4469c5c2 100644 --- a/spring-core/src/test/java/org/springframework/core/retry/RetryPolicyTests.java +++ b/spring-core/src/test/java/org/springframework/core/retry/RetryPolicyTests.java @@ -87,29 +87,6 @@ class RetryPolicyTests { assertThat(backOff.getInterval()).isEqualTo(1000); }); } - - @Test - void withMaxElapsedTimePreconditions() { - assertThatIllegalArgumentException() - .isThrownBy(() -> RetryPolicy.withMaxElapsedTime(Duration.ofMillis(0))) - .withMessage("Invalid duration (0ms): maxElapsedTime must be positive."); - assertThatIllegalArgumentException() - .isThrownBy(() -> RetryPolicy.withMaxElapsedTime(Duration.ofMillis(-1))) - .withMessage("Invalid duration (-1ms): maxElapsedTime must be positive."); - } - - @Test - void withMaxElapsedTime() { - var policy = RetryPolicy.withMaxElapsedTime(Duration.ofMillis(42)); - - assertThat(policy.shouldRetry(new AssertionError())).isTrue(); - assertThat(policy.shouldRetry(new IOException())).isTrue(); - - assertThat(policy.getBackOff()) - .asInstanceOf(type(ExponentialBackOff.class)) - .satisfies(hasDefaultMaxAttemptsAndDelay()) - .extracting(ExponentialBackOff::getMaxElapsedTime).isEqualTo(42L); - } } @@ -122,7 +99,7 @@ class RetryPolicyTests { .isThrownBy(() -> RetryPolicy.builder().backOff(mock()).delay(Duration.ofMillis(10)).build()) .withMessage(""" The following configuration options are not supported with a custom BackOff strategy: \ - maxAttempts, delay, jitter, multiplier, maxDelay, or maxElapsedTime."""); + maxAttempts, delay, jitter, multiplier, or maxDelay."""); } @Test @@ -157,7 +134,7 @@ class RetryPolicyTests { assertThat(backOff.getInitialInterval()).isEqualTo(1000); }); - assertToString(policy, 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 5); + assertToString(policy, 1000, 0, 1.0, Long.MAX_VALUE, 5); } @Test @@ -178,7 +155,7 @@ class RetryPolicyTests { assertThat(backOff.getMaxAttempts()).isEqualTo(3); }); - assertToString(policy, 42, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, 42, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -197,7 +174,7 @@ class RetryPolicyTests { .satisfies(hasDefaultMaxAttemptsAndDelay()) .extracting(ExponentialBackOff::getJitter).isEqualTo(42L); - assertToString(policy, 1000, 42, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, 1000, 42, 1.0, Long.MAX_VALUE, 3); } @Test @@ -226,7 +203,7 @@ class RetryPolicyTests { .satisfies(hasDefaultMaxAttemptsAndDelay()) .extracting(ExponentialBackOff::getMultiplier).isEqualTo(1.5); - assertToString(policy, 1000, 0, 1.5, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, 1000, 0, 1.5, Long.MAX_VALUE, 3); } @Test @@ -248,29 +225,7 @@ class RetryPolicyTests { .satisfies(hasDefaultMaxAttemptsAndDelay()) .extracting(ExponentialBackOff::getMaxInterval).isEqualTo(42L); - assertToString(policy, 1000, 0, 1, 42, Long.MAX_VALUE, 3); - } - - @Test - void maxElapsedTimePreconditions() { - assertThatIllegalArgumentException() - .isThrownBy(() -> RetryPolicy.builder().maxElapsedTime(Duration.ofMillis(0))) - .withMessage("Invalid duration (0ms): maxElapsedTime must be positive."); - assertThatIllegalArgumentException() - .isThrownBy(() -> RetryPolicy.builder().maxElapsedTime(Duration.ofMillis(-1))) - .withMessage("Invalid duration (-1ms): maxElapsedTime must be positive."); - } - - @Test - void maxElapsedTime() { - var policy = RetryPolicy.builder().maxElapsedTime(Duration.ofMillis(42)).build(); - - assertThat(policy.getBackOff()) - .asInstanceOf(type(ExponentialBackOff.class)) - .satisfies(hasDefaultMaxAttemptsAndDelay()) - .extracting(ExponentialBackOff::getMaxElapsedTime).isEqualTo(42L); - - assertToString(policy, 1000, 0, 1, Long.MAX_VALUE, 42, 3); + assertToString(policy, 1000, 0, 1.0, 42, 3); } @Test @@ -294,7 +249,7 @@ class RetryPolicyTests { String filters = "includes=" + names(FileNotFoundException.class, IllegalArgumentException.class, NumberFormatException.class, AssertionError.class) + ", "; - assertToString(policy, filters, 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, filters, 1000, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -311,7 +266,7 @@ class RetryPolicyTests { .asInstanceOf(type(ExponentialBackOff.class)) .satisfies(hasDefaultMaxAttemptsAndDelay()); - assertToString(policy, "includes=[java.io.IOException], ", 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, "includes=[java.io.IOException], ", 1000, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -335,7 +290,7 @@ class RetryPolicyTests { String filters = "excludes=" + names(FileNotFoundException.class, IllegalArgumentException.class, NumberFormatException.class, AssertionError.class) + ", "; - assertToString(policy, filters, 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, filters, 1000, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -349,7 +304,7 @@ class RetryPolicyTests { assertThat(policy.shouldRetry(new Throwable())).isTrue(); assertThat(policy.shouldRetry(new AssertionError())).isTrue(); - assertToString(policy, "excludes=[java.io.IOException], ", 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, "excludes=[java.io.IOException], ", 1000, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -368,8 +323,7 @@ class RetryPolicyTests { .asInstanceOf(type(ExponentialBackOff.class)) .satisfies(hasDefaultMaxAttemptsAndDelay()); - assertToString(policy, "predicate=NumberFormatExceptionMatcher, ", - 1000, 0, 1, Long.MAX_VALUE, Long.MAX_VALUE, 3); + assertToString(policy, "predicate=NumberFormatExceptionMatcher, ", 1000, 0, 1.0, Long.MAX_VALUE, 3); } @Test @@ -398,13 +352,13 @@ class RetryPolicyTests { private static void assertToString(RetryPolicy policy, long initialInterval, long jitter, - double multiplier, long maxInterval, long maxElapsedTime, int maxAttempts) { + double multiplier, long maxInterval, int maxAttempts) { - assertToString(policy, "", initialInterval, jitter, multiplier, maxInterval, maxElapsedTime, maxAttempts); + assertToString(policy, "", initialInterval, jitter, multiplier, maxInterval, maxAttempts); } private static void assertToString(RetryPolicy policy, String filters, long initialInterval, long jitter, - double multiplier, long maxInterval, long maxElapsedTime, int maxAttempts) { + double multiplier, long maxInterval, int maxAttempts) { assertThat(policy).asString() .isEqualTo(""" @@ -416,7 +370,7 @@ class RetryPolicyTests { maxElapsedTime=%d, \ maxAttempts=%d\ ]]""", - filters, initialInterval, jitter, multiplier, maxInterval, maxElapsedTime, maxAttempts); + filters, initialInterval, jitter, multiplier, maxInterval, Long.MAX_VALUE, maxAttempts); } @SafeVarargs