Skip to content

Commit

Permalink
GH-395: Expose maximum number of attempts property
Browse files Browse the repository at this point in the history
Fixes #395

It is sometimes useful to be aware, within a retry context, whether the current attempt is the last attempt when using a 'maximum number of attempts before failure' type of retry policy.
In order to achieve the desired result, for all policies, a new attribute is going to be added to the context named "context.max-attempts" that will contain the number of max attempts set in the policy, if any, or -1 otherwise.

* GH-395: make the retry context aware of the policies that have a maximum number of attempts set
* GH-395: add author details to all modified classes that already had an author list
  • Loading branch information
e-ivaldi authored Nov 17, 2023
1 parent 3a82012 commit 7516219
Show file tree
Hide file tree
Showing 9 changed files with 98 additions and 1 deletion.
8 changes: 8 additions & 0 deletions src/main/java/org/springframework/retry/RetryContext.java
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
* used to alter the course of the retry, e.g. force an early termination.
*
* @author Dave Syer
* @author Emanuele Ivaldi
*
*/
public interface RetryContext extends AttributeAccessor {
Expand Down Expand Up @@ -61,6 +62,13 @@ public interface RetryContext extends AttributeAccessor {
*/
String NO_RECOVERY = "context.no-recovery";

/**
* Retry context attribute that represent the maximum number of attempts for policies
* that provide a maximum number of attempts before failure. For other policies the
* value returned is {@link RetryPolicy#NO_MAXIMUM_ATTEMPTS_SET}
*/
String MAX_ATTEMPTS = "context.max-attempts";

/**
* Signal to the framework that no more attempts should be made to try or retry the
* current {@link RetryCallback}.
Expand Down
17 changes: 17 additions & 0 deletions src/main/java/org/springframework/retry/RetryPolicy.java
Original file line number Diff line number Diff line change
Expand Up @@ -26,10 +26,17 @@
* API for a range of different platforms for the external context.
*
* @author Dave Syer
* @author Emanuele Ivaldi
*
*/
public interface RetryPolicy extends Serializable {

/**
* The value returned by {@link RetryPolicy#getMaxAttempts()} when the policy doesn't
* provide a maximum number of attempts before failure
*/
int NO_MAXIMUM_ATTEMPTS_SET = -1;

/**
* @param context the current retry status
* @return true if the operation can proceed
Expand Down Expand Up @@ -59,4 +66,14 @@ public interface RetryPolicy extends Serializable {
*/
void registerThrowable(RetryContext context, Throwable throwable);

/**
* Called to understand if the policy has a fixed number of maximum attempts before
* failure
* @return -1 if the policy doesn't provide a fixed number of maximum attempts before
* failure, the number of maximum attempts before failure otherwise
*/
default int getMaxAttempts() {
return NO_MAXIMUM_ATTEMPTS_SET;
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@
*
* @author Dave Syer
* @author Michael Minella
* @author Emanuele Ivaldi
*
*/
@SuppressWarnings("serial")
Expand Down Expand Up @@ -145,6 +146,22 @@ public void registerThrowable(RetryContext context, Throwable throwable) {
((RetryContextSupport) context).registerThrowable(throwable);
}

/**
* @return the lower 'maximum number of attempts before failure' between all policies
* that have a 'maximum number of attempts before failure' set, if at least one is
* present among the policies, return {@link RetryPolicy#NO_MAXIMUM_ATTEMPTS_SET}
* otherwise
*/
@Override
public int getMaxAttempts() {
return Arrays.stream(policies)
.map(RetryPolicy::getMaxAttempts)
.filter(maxAttempts -> maxAttempts != NO_MAXIMUM_ATTEMPTS_SET)
.sorted()
.findFirst()
.orElse(NO_MAXIMUM_ATTEMPTS_SET);
}

private static class CompositeRetryContext extends RetryContextSupport {

RetryContext[] contexts;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ public void setMaxAttempts(int maxAttempts) {
* The maximum number of attempts before failure.
* @return the maximum number of attempts
*/
@Override
public int getMaxAttempts() {
return this.maxAttempts;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@
* @author Gary Russell
* @author Aleksandr Shamukov
* @author Artem Bilan
* @author Emanuele Ivaldi
*/
@SuppressWarnings("serial")
public class SimpleRetryPolicy implements RetryPolicy {
Expand Down Expand Up @@ -194,6 +195,7 @@ public void maxAttemptsSupplier(Supplier<Integer> maxAttemptsSupplier) {
* The maximum number of attempts before failure.
* @return the maximum number of attempts
*/
@Override
public int getMaxAttempts() {
if (this.maxAttemptsSupplier != null) {
return this.maxAttemptsSupplier.get();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@
* @author Artem Bilan
* @author Josh Long
* @author Aleksandr Shamukov
* @author Emanuele Ivaldi
*/
public class RetryTemplate implements RetryOperations {

Expand Down Expand Up @@ -296,6 +297,10 @@ protected <T, E extends Throwable> T doExecute(RetryCallback<T, E> retryCallback
throw new TerminatedRetryException("Retry terminated abnormally by interceptor before first attempt");
}

if (!context.hasAttribute(RetryContext.MAX_ATTEMPTS)) {
context.setAttribute(RetryContext.MAX_ATTEMPTS, retryPolicy.getMaxAttempts());
}

// Get or Start the backoff context...
BackOffContext backOffContext = null;
Object resource = context.getAttribute("backOffContext");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@
* @author Henning Pöttker
* @author Yanming Zhou
* @author Anton Aharkau
* @author Emanuele Ivaldi
* @since 1.1
*/
public class EnableRetryTests {
Expand Down Expand Up @@ -295,7 +296,7 @@ void runtimeExpressions() throws Exception {
ExpressionService service = context.getBean(ExpressionService.class);
service.service6();
RuntimeConfigs runtime = context.getBean(RuntimeConfigs.class);
verify(runtime, times(5)).getMaxAttempts();
verify(runtime, times(6)).getMaxAttempts();
verify(runtime, times(1)).getInitial();
verify(runtime, times(2)).getMax();
verify(runtime, times(2)).getMult();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -160,4 +160,23 @@ public boolean canRetry(RetryContext context) {
assertThat(policy.canRetry(context)).isTrue();
}

@Test
public void testMaximumAttemptsForNonSuitablePolicies() {
CompositeRetryPolicy policy = new CompositeRetryPolicy();
policy.setOptimistic(true);
policy.setPolicies(new RetryPolicy[] { new NeverRetryPolicy(), new NeverRetryPolicy() });

assertThat(policy.getMaxAttempts()).isEqualTo(RetryPolicy.NO_MAXIMUM_ATTEMPTS_SET);
}

@Test
public void testMaximumAttemptsForSuitablePolicies() {
CompositeRetryPolicy policy = new CompositeRetryPolicy();
policy.setOptimistic(true);
policy.setPolicies(
new RetryPolicy[] { new SimpleRetryPolicy(6), new SimpleRetryPolicy(3), new SimpleRetryPolicy(4) });

assertThat(policy.getMaxAttempts()).isEqualTo(3);
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -26,11 +26,13 @@
import org.springframework.retry.RetryCallback;
import org.springframework.retry.RetryContext;
import org.springframework.retry.RetryListener;
import org.springframework.retry.RetryPolicy;
import org.springframework.retry.TerminatedRetryException;
import org.springframework.retry.backoff.BackOffContext;
import org.springframework.retry.backoff.BackOffInterruptedException;
import org.springframework.retry.backoff.BackOffPolicy;
import org.springframework.retry.backoff.StatelessBackOffPolicy;
import org.springframework.retry.policy.AlwaysRetryPolicy;
import org.springframework.retry.policy.NeverRetryPolicy;
import org.springframework.retry.policy.SimpleRetryPolicy;

Expand All @@ -49,6 +51,7 @@
* @author Dave Syer
* @author Gary Russell
* @author Henning Pöttker
* @author Emanuele Ivaldi
*/
public class RetryTemplateTests {

Expand Down Expand Up @@ -333,6 +336,30 @@ public <T, E extends Throwable> void onSuccess(RetryContext context, RetryCallba
assertThat(callCount.get()).isEqualTo(2);
}

@Test
public void testContextForPolicyWithMaximumNumberOfAttempts() throws Throwable {
RetryTemplate retryTemplate = new RetryTemplate();
RetryPolicy retryPolicy = new SimpleRetryPolicy(2);
retryTemplate.setRetryPolicy(retryPolicy);

Integer result = retryTemplate.execute((RetryCallback<Integer, Throwable>) context -> (Integer) context
.getAttribute(RetryContext.MAX_ATTEMPTS), context -> RetryPolicy.NO_MAXIMUM_ATTEMPTS_SET);

assertThat(result).isEqualTo(2);
}

@Test
public void testContextForPolicyWithNoMaximumNumberOfAttempts() throws Throwable {
RetryTemplate retryTemplate = new RetryTemplate();
RetryPolicy retryPolicy = new AlwaysRetryPolicy();
retryTemplate.setRetryPolicy(retryPolicy);

Integer result = retryTemplate.execute((RetryCallback<Integer, Throwable>) context -> (Integer) context
.getAttribute(RetryContext.MAX_ATTEMPTS), context -> RetryPolicy.NO_MAXIMUM_ATTEMPTS_SET);

assertThat(result).isEqualTo(RetryPolicy.NO_MAXIMUM_ATTEMPTS_SET);
}

private static class MockRetryCallback implements RetryCallback<Object, Exception> {

private int attempts;
Expand Down

0 comments on commit 7516219

Please sign in to comment.