DnsNameResolverTimeoutException in Spring Boot WebFlux
Posted on June 17, 2024
A few weeks ago, I started noticing errors during the
Polidict
authentication flow. The application was throwing a
DnsNameResolverTimeoutException
when making network requests. The error messages appeared as follows:
io.netty.resolver.dns.DnsNameResolverTimeoutException: [10812: /[fdaa:0:0:0:0:0:0:3]:53] DefaultDnsQuestion( www.googleapis.com . IN AAAA) query '10812' via UDP timed out after 5000 milliseconds (no stack trace available)
Caused by: java.net.UnknownHostException: Failed to resolve ' www.googleapis.com ' [A(1), AAAA(28)] after 2 queries
Problem
Initially, I suspected a network issue and criticized fly.io for it on Twitter.
To resolve the issue, I attempted several approaches:
However, none of these solutions remedied the problem.
Further investigation revealed that the issue was tied to the DNS resolver used by the application. I found related issues on GitHub:
The problem seemed to originate from Netty's
DnsAddressResolverGroup
.
Potential resolutions include:
My Solution
I implemented a global
WebClientCustomizer
that switches the DNS resolver to
DefaultAddressResolverGroup
.
@Component
public class AddressResolverWebClientCustomizer implements WebClientCustomizer {
@Override
public void customize(WebClient.Builder webClientBuilder) {
webClientBuilder.clientConnector(
new ReactorClientHttpConnector(HttpClient.create()
.resolver(DefaultAddressResolverGroup.INSTANCE)));
However, this wasn't sufficient. The issue persisted during the OAuth2 login flow. The WebClientReactiveAuthorizationCodeTokenResponseClient
uses the default WebClient.
To address this, I created a custom WebClientReactiveAuthorizationCodeTokenResponseClient
using a customized WebClient.
* Client with customized WebClient. By default, it uses WebClient.builder().build() which is not customized.
* Refer to {@link AddressResolverWebClientCustomizer} for customization details.
@Bean
public WebClientReactiveAuthorizationCodeTokenResponseClient webClientReactiveAuthorizationCodeTokenResponseClient(
WebClient.Builder webClientBuilder
var webClientReactiveAuthorizationCodeTokenResponseClient = new WebClientReactiveAuthorizationCodeTokenResponseClient();
webClientReactiveAuthorizationCodeTokenResponseClient.setWebClient(webClientBuilder.build());
return webClientReactiveAuthorizationCodeTokenResponseClient;