How to Have a Fallback on Errors Calling Downstream Services in Spring Boot Webflux
The source code for this post is available on Github.
Things break. When you start adding more and more microservices, things will break a lot more. This post is about how to provide a degraded experience to your users when things break.
I’m going to build off of some of the boilerplate code written in previous blog posts. If you’ll recall, we had a WebClient configured like so:
@Configuration
public class Config {
@Bean("service-a-web-client")
public WebClient serviceAWebClient() {
HttpClient httpClient = HttpClient.create().tcpConfiguration(tcpClient ->
tcpClient.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 1000)
.doOnConnected(connection -> connection.addHandlerLast(new ReadTimeoutHandler(1000, TimeUnit.MILLISECONDS)))
);
return WebClient.builder()
.baseUrl("http://your-base-url.com")
.clientConnector(new ReactorClientHttpConnector(httpClient))
.build();
}
}
[You will obviously want to replace http://your-base-url.com with your actual url]
And with that in place, let’s create the boilerplate for a java service that will wrap a call to an external client, one that sometimes behaves badly and fails:
@Service
public class FallbackService {
private final WebClient serviceAWebClient;
public FallbackService(@Qualifier("service-a-web-client")
WebClient serviceAWebClient) {
this.serviceAWebClient = serviceAWebClient;
}
public Mono<WelcomeMessage> getWelcomeMessageByLocale(String locale) {
return Mono.empty();
}
}
This method will allow us to get a WelcomeMessage that is locale specific, so that somebody who wants to receive a welcome message in french [because, say, that’s the only language they speak] can do so. That DTO looks like this:
public class WelcomeMessage {
private String message;
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
}
Following TDD, we will now create the test class:
@ExtendWith(MockServerExtension.class)
public class FallbackServiceIT {
private WebClient webClient;
private ClientAndServer clientAndServer;
private FallbackService fallbackService;
public FallbackServiceIT(ClientAndServer clientAndServer) {
this.clientAndServer = clientAndServer;
this.webClient = WebClient.builder()
.baseUrl("http://localhost:" + clientAndServer.getPort())
.build();
}
@BeforeEach
public void setup() {
fallbackService = new FallbackService(webClient);
}
@AfterEach
public void reset() {
clientAndServer.reset();
}
@Test
public void welcomeMessage_worksWhenNoErrors() {
this.clientAndServer.when(
request()
.withPath("/locale/en_US/message")
.withMethod(HttpMethod.GET.name())
).respond(
HttpResponse
.response()
.withBody("{\"message\": \"hello\"}")
.withContentType(MediaType.APPLICATION_JSON)
);
StepVerifier.create(fallbackService.getWelcomeMessageByLocale("en_US"))
.expectNextMatches(welcomeMessage -> "hello".equals(welcomeMessage.getMessage()))
.verifyComplete();
}
}
Similar to previous posts, we’re leveraging MockServer here to simulate a response to a predefined endpoint at "/locale/en_US/message". The response is a json response that matches our DTO. If you run this test, it will predictably fail.
Now let’s change the service code to make it pass:
public Mono<WelcomeMessage> getWelcomeMessageByLocale(String locale) {
return this.serviceAWebClient.get()
.uri(uriBuilder -> uriBuilder.path("/locale/{locale}/message").build(locale))
.retrieve()
.bodyToMono(WelcomeMessage.class);
}
As advertised, our test now passes.
Okay, so this is fine if the downstream service is behaving normally, but what if the service is misbehaving and barfing up 500s? In that case, our WebClient will just propagate up the error to our service. Now, let’s say that 90% of your user base speaks English, it would seem pretty dumb to bring down this entire portion of the app just because you couldn’t get a specific welcome message, even though that welcome message is almost always going to be in English.
To simulate this failure, we can similarly use MockServer:
@Test
public void welcomeMessage_fallsBackToEnglishWhenError() {
this.clientAndServer.when(
request()
.withPath("/locale/fr/message")
.withMethod(HttpMethod.GET.name())
).respond(
HttpResponse.response()
.withStatusCode(503)
);
StepVerifier.create(fallbackService.getWelcomeMessageByLocale("fr"))
.expectNextMatches(welcomeMessage -> "hello fallback!".equals(welcomeMessage.getMessage()))
.verifyComplete();
}
With this now in place, we can start examining the different ways we can accomplish our goals here. One option is to just use onErrorReturn:
public Mono<WelcomeMessage> getWelcomeMessageByLocale(String locale) {
return this.serviceAWebClient.get()
.uri(uriBuilder -> uriBuilder.path("/locale/{locale}/message").build(locale))
.retrieve()
.bodyToMono(WelcomeMessage.class)
.onErrorReturn(new WelcomeMessage("hello fallback!"));
}
This is obviously very simple, but pretty crude. I generally prefer to use a different overloaded method for that, which is to use a Predicate to first check that the error type is one we are okay with falling back on:
public Mono<WelcomeMessage> getWelcomeMessageByLocale(String locale) {
return this.serviceAWebClient.get()
.uri(uriBuilder -> uriBuilder.path("/locale/{locale}/message").build(locale))
.retrieve()
.bodyToMono(WelcomeMessage.class)
.onErrorReturn(
throwable -> throwable instanceof WebClientResponseException
&& ((WebClientResponseException)throwable).getStatusCode().is5xxServerError(),
new WelcomeMessage("hello fallback!")
);
}
With either of those changes, our test now passes, and we’ve added a small but meaningful win to our app! Remember to check out the source code on Github to see this in action.