How to Use Spring Retry
Need to automatically re-invoke a failed operation in Spring? Check out this tutorial on Spring Retry to learn more.
Join the DZone community and get the full member experience.
Join For FreeA few days ago, I noticed that there is a group of people asking how to use Spring Retry. Before I go into the sample code, let me quickly explain the purpose behind Spring Retry. Spring Retry provides the ability to automatically re-invoke a failed operation. This is helpful when errors may be transient in nature (like a momentary network glitch). Spring Retry provides a declarative control of the process and policy-based behavior that is easy to extend and customize.
You can find the complete source code in here.
Maven Dependencies
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.retry</groupId>
<artifactId>spring-retry</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
Enable Retry
package com.chrisshayan.example.springretry;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.retry.annotation.EnableRetry;
public class SpringRetryApplication {
public static void main(String[] args) {
SpringApplication.run(SpringRetryApplication.class, args);
}
}
Using Retry With Annotations
package com.chrisshayan.example.springretry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.retry.annotation.Backoff;
import org.springframework.retry.annotation.Recover;
import org.springframework.retry.annotation.Retryable;
import org.springframework.stereotype.Service;
public class SampleRetryService {
private static final Logger LOGGER = LoggerFactory.getLogger(SampleRetryService.class);
private static int COUNTER = 0;
(
value = {TypeOneException.class, TypeTwoException.class},
maxAttempts = 4, backoff = (2000))
public String retryWhenException() throws TypeOneException, TypeTwoException {
COUNTER++;
LOGGER.info("COUNTER = " + COUNTER);
if(COUNTER == 1)
throw new TypeOneException();
else if(COUNTER == 2)
throw new TypeTwoException();
else
throw new RuntimeException();
}
public String recover(Throwable t) {
LOGGER.info("SampleRetryService.recover");
return "Error Class :: " + t.getClass().getName();
}
}
In order for your test class to work, the retry needs to be in the proper context. This is because we need to have another service that wraps around the retry. This is called:
package com.chrisshayan.example.springretry;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
public class SampleRetryClientService {
private SampleRetryService sampleRetryService;
public String callRetryService() throws TypeOneException, TypeTwoException {
return sampleRetryService.retryWhenException();
}
}
Test Class
331package com.chrisshayan.example.springretry;
2
3import org.junit.Test;
4import org.junit.runner.RunWith;
5import org.slf4j.Logger;
6import org.slf4j.LoggerFactory;
7import org.springframework.beans.factory.annotation.Autowired;
8import org.springframework.boot.test.context.SpringBootTest;
9import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
10
11SpringJUnit4ClassRunner.class) (
12
13public class SpringRetryApplicationTests {
14
15 private static final Logger LOGGER = LoggerFactory.getLogger(SpringRetryApplicationTests.class);
16
17 private SampleRetryClientService client;
18
19
20public void contextLoads() {
21}
22
23
24 public void sampleRetryService() {
25 try {
26 final String message = client.callRetryService();
27 LOGGER.info("message = " + message);
28 } catch (TypeOneException | TypeTwoException e) {
29 e.printStackTrace();
30 }
31 }
32
33}
Console
package com.chrisshayan.example.springretry;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
SpringJUnit4ClassRunner.class) (
public class SpringRetryApplicationTests {
private static final Logger LOGGER = LoggerFactory.getLogger(SpringRetryApplicationTests.class);
private SampleRetryClientService client;
public void contextLoads() {
}
public void sampleRetryService() {
try {
final String message = client.callRetryService();
LOGGER.info("message = " + message);
} catch (TypeOneException | TypeTwoException e) {
e.printStackTrace();
}
}
}
2018-07-10 23:42:45.528 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 1
2018-07-10 23:42:47.534 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 2
2018-07-10 23:42:49.538 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 3
2018-07-10 23:42:49.539 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : SampleRetryService.recover
2018-07-10 23:42:49.539 INFO 14583 --- [ main] c.c.e.s.SpringRetryApplicationTests : message = Error Class :: java.lang.RuntimeException
There are more capabilities in Spring Retry, such as Stateless Retry, Stateful Retry, and different retry policies and listeners. You can read more here.
Published at DZone with permission of Chris Shayan. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments