Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,12 @@
CHANGELOG
=========

5.0.3 (unreleased)
------------------

* Added `WebServiceClient.Builder.maxRetries(int)` to bound transport-failure
retries (default 1; set 0 to disable). See the README for retry semantics.

5.0.2 (2025-12-08)
------------------

Expand Down
44 changes: 44 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,50 @@ are not created for each request.
See the [API documentation](https://maxmind.github.io/GeoIP2-java/) for
more details.

### Connection pooling and transport retries ###

`WebServiceClient` is thread-safe and reuses a pooled `HttpClient` across
requests. Idle connections in the pool can be silently closed by load
balancers or other intermediaries. When the next request reuses one of these
half-closed connections, the JDK reports the failure as a `Connection reset`,
`Broken pipe`, or related transport `IOException`.

To smooth over these intermittent transport failures, the SDK retries once
by default. Any transport-level `IOException` raised by the underlying HTTP
send is retried, with the following exclusions:

* `HttpTimeoutException` — a request-phase timeout. Connect-phase timeouts
(`HttpConnectTimeoutException`) are also excluded because they extend
`HttpTimeoutException`. The SDK honors the timeouts you configure.
* `InterruptedIOException` — the calling thread was interrupted; the SDK
honors the cancellation rather than override it.
* Typically deterministic failures: `UnknownHostException`,
`ConnectException`, `SSLHandshakeException`, `SSLPeerUnverifiedException`.
Retrying these would just delay surfacing a config bug.
* If the calling thread is already interrupted when the predicate runs, the
retry is short-circuited regardless of the exception type.

HTTP 4xx and 5xx responses are not retried — they are returned as
`HttpResponse` objects (not `IOException`s) and surfaced through the existing
exception hierarchy. Web service requests are idempotent GETs, so retried
requests are byte-identical to the original.

You can change the retry budget via the builder:

```java
WebServiceClient client = new WebServiceClient.Builder(42, "license_key")
.maxRetries(2) // up to two retries (three total attempts)
.build();
```

Set `.maxRetries(0)` to disable the retry entirely. Negative values throw
`IllegalArgumentException`.

If you frequently see `Connection reset` errors, you can also reduce the
JDK's keep-alive timeout via the system property
`jdk.httpclient.keepalive.timeout` (in seconds) to evict pooled connections
before any intermediary does so.

## Web Service Example ##

### Country Service ###
Expand Down
30 changes: 30 additions & 0 deletions mise.lock
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
# @generated - this file is auto-generated by `mise lock` https://mise.jdx.dev/dev-tools/mise-lock.html

[[tools.java]]
version = "26.0.0"
backend = "core:java"

[[tools.maven]]
version = "3.9.15"
backend = "aqua:apache/maven"

[tools.maven."platforms.linux-arm64"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.linux-arm64-musl"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.linux-x64"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.linux-x64-musl"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.macos-arm64"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.macos-x64"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"

[tools.maven."platforms.windows-x64"]
url = "https://archive.apache.org/dist/maven/maven-3/3.9.15/binaries/apache-maven-3.9.15-bin.tar.gz"
18 changes: 18 additions & 0 deletions mise.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
[settings]
experimental = true
lockfile = true
disable_backends = [
"asdf",
"vfox",
]

[tools]
java = "latest"
maven = "latest"

[hooks]
enter = "mise install --quiet --locked"

[[watch_files]]
patterns = ["mise.toml", "mise.lock"]
run = "mise install --quiet --locked"
100 changes: 98 additions & 2 deletions src/main/java/com/maxmind/geoip2/WebServiceClient.java
Original file line number Diff line number Diff line change
Expand Up @@ -20,20 +20,26 @@
import com.maxmind.geoip2.model.InsightsResponse;
import java.io.IOException;
import java.io.InputStream;
import java.io.InterruptedIOException;
import java.net.ConnectException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.ProxySelector;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.UnknownHostException;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.net.http.HttpTimeoutException;
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.Base64;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.net.ssl.SSLHandshakeException;
import javax.net.ssl.SSLPeerUnverifiedException;

/**
* <p>
Expand Down Expand Up @@ -112,6 +118,7 @@ public class WebServiceClient implements WebServiceProvider {
private final boolean useHttps;
private final int port;
private final Duration requestTimeout;
private final int maxRetries;
private final String userAgent = "GeoIP2/"
+ getClass().getPackage().getImplementationVersion()
+ " (Java/" + System.getProperty("java.version") + ")";
Expand All @@ -125,6 +132,7 @@ private WebServiceClient(Builder builder) {
this.port = builder.port;
this.useHttps = builder.useHttps;
this.locales = builder.locales;
this.maxRetries = builder.maxRetries;

// HttpClient supports basic auth, but it will only send it after the
// server responds with an unauthorized. As such, we just make the
Expand Down Expand Up @@ -182,6 +190,7 @@ public static final class Builder {
List<String> locales = List.of("en");
private ProxySelector proxy = null;
private HttpClient httpClient = null;
private int maxRetries = 1;

/**
* @param accountId Your MaxMind account ID.
Expand All @@ -197,6 +206,7 @@ public Builder(int accountId, String licenseKey) {
* @param val Timeout duration to establish a connection to the
* web service. The default is 3 seconds.
* @return Builder object
* @apiNote See {@link #maxRetries(int)} for how this timeout interacts with retries.
*/
public Builder connectTimeout(Duration val) {
this.connectTimeout = val;
Expand Down Expand Up @@ -251,6 +261,7 @@ public Builder locales(List<String> val) {
/**
* @param val Request timeout duration. The default is 20 seconds.
* @return Builder object
* @apiNote See {@link #maxRetries(int)} for how this timeout interacts with retries.
*/
public Builder requestTimeout(Duration val) {
this.requestTimeout = val;
Expand All @@ -271,13 +282,42 @@ public Builder proxy(ProxySelector val) {
* @param val the custom HttpClient to use for requests. When providing a
* custom HttpClient, you cannot also set connectTimeout or proxy
* parameters as these should be configured on the provided client.
* <p>
* The SDK applies its own transport-failure retry on top of any
* supplied client; customers can disable it via
* {@link #maxRetries(int)} with {@code .maxRetries(0)}.
* @return Builder object
*/
public Builder httpClient(HttpClient val) {
this.httpClient = val;
return this;
}

/**
* @param val Maximum number of retries on transport-level failures
* (connection reset, broken pipe, EOF, ...).
* Applies uniformly to all endpoints. Defaults to 1.
* Set to 0 to disable.
* @return Builder.
* @throws IllegalArgumentException if {@code val} is negative.
* @apiNote Timeouts are not retried ({@link java.net.http.HttpTimeoutException},
* including the connect-phase subclass
* {@link java.net.http.HttpConnectTimeoutException}). When
* {@code maxRetries > 0}, retries are triggered only by fast transport
* failures, so each attempt is independently bounded by
* {@link #connectTimeout(Duration)} and {@link #requestTimeout(Duration)}.
* The multiplied worst-case wall clock a naive reading suggests is
* unreachable in practice, since hitting the timeout aborts the call
* rather than triggering a retry.
*/
public Builder maxRetries(int val) {
if (val < 0) {
throw new IllegalArgumentException("maxRetries must not be negative");
}
maxRetries = val;
return this;
}

/**
* @return an instance of {@code WebServiceClient} created from the
* fields set on this builder.
Expand Down Expand Up @@ -371,18 +411,74 @@ private <T> T responseFor(String path, InetAddress ipAddress, Class<T> cls)
.GET()
.build();
try {
var response = this.httpClient
.send(request, HttpResponse.BodyHandlers.ofInputStream());
var response = sendWithRetry(request);
try {
return handleResponse(response, cls);
} finally {
response.body().close();
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new GeoIp2Exception("Interrupted sending request", e);
}
}

private HttpResponse<InputStream> sendWithRetry(HttpRequest request)
throws IOException, InterruptedException {
int attempts = 0;
IOException prior = null;
while (true) {
try {
return httpClient.send(request, HttpResponse.BodyHandlers.ofInputStream());
} catch (IOException e) {
if (prior != null) {
e.addSuppressed(prior);
}
if (!isRetriableTransportFailure(e) || attempts >= maxRetries) {
throw e;
}
prior = e;
attempts++;
}
}
}
Comment thread
oschwald marked this conversation as resolved.

private static boolean isRetriableTransportFailure(IOException e) {
if (Thread.currentThread().isInterrupted()) {
return false;
}
// Both connect-phase and request-phase timeouts are customer-set
// budgets that retrying would silently extend.
// HttpConnectTimeoutException extends HttpTimeoutException, so this
// single check covers both.
if (e instanceof HttpTimeoutException) {
return false;
}
// The thread was interrupted during I/O; honor the cancellation.
if (e instanceof InterruptedIOException) {
return false;
}
// Typically deterministic failures: retrying just delays surfacing the
// config bug without recovering the request.
if (e instanceof UnknownHostException) {
return false;
}
if (e instanceof ConnectException) {
return false;
}
if (e instanceof SSLHandshakeException) {
return false;
}
if (e instanceof SSLPeerUnverifiedException) {
return false;
}
// Everything else from httpClient.send() is a transport failure
// (connection reset, broken pipe, EOF, closed channel, ...).
// HTTP 4xx and 5xx responses do not reach this predicate -- they come
// back as HttpResponse objects rather than IOExceptions.
return true;
}

private <T> T handleResponse(HttpResponse<InputStream> response, Class<T> cls)
throws GeoIp2Exception, IOException {
var status = response.statusCode();
Expand Down
Loading
Loading