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
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ CHANGELOG
* Added `FAT_ZEBRA` to the `Payment.Processor` enum.
* Added `CLEAR` to the `TransactionReport.Tag` enum for use with the Report
Transaction API.
* Added `WebServiceClient.Builder.maxRetries(int)` to bound transport-failure
retries (default 1; set 0 to disable). See the README for retry semantics.

4.2.0 (2026-02-26)
------------------
Expand Down
44 changes: 44 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,50 @@ exception will be thrown.

See the API documentation 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. POST bodies are replayable, so retried requests are
byte-identical to the original.

You can change the retry budget via the builder:

```java
WebServiceClient client = new WebServiceClient.Builder(6, "ABCD567890")
.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.

### Exceptions ###

Runtime exceptions:
Expand Down
105 changes: 102 additions & 3 deletions src/main/java/com/maxmind/minfraud/WebServiceClient.java
Original file line number Diff line number Diff line change
Expand Up @@ -16,19 +16,25 @@
import com.maxmind.minfraud.response.ScoreResponse;
import java.io.IOException;
import java.io.InputStream;
import java.io.InterruptedIOException;
import java.net.ConnectException;
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.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.net.ssl.SSLHandshakeException;
import javax.net.ssl.SSLPeerUnverifiedException;

/**
* Client for MaxMind minFraud Score, Insights, and Factors
Expand All @@ -45,6 +51,7 @@ public final class WebServiceClient {
private final boolean useHttps;
private final List<String> locales;
private final Duration requestTimeout;
private final int maxRetries;

private final HttpClient httpClient;

Expand All @@ -63,6 +70,7 @@ private WebServiceClient(WebServiceClient.Builder builder) {
.getBytes(StandardCharsets.UTF_8));

requestTimeout = builder.requestTimeout;
maxRetries = builder.maxRetries;
if (builder.httpClient != null) {
httpClient = builder.httpClient;
} else {
Expand Down Expand Up @@ -106,6 +114,7 @@ public static final class Builder {
List<String> locales = List.of("en");
private ProxySelector proxy;
private HttpClient httpClient;
private int maxRetries = 1;

/**
* @param accountId Your MaxMind account ID.
Expand All @@ -120,6 +129,7 @@ public Builder(int accountId, String licenseKey) {
* @param val Timeout duration to establish a connection to the web service. There is no
* timeout by default.
* @return Builder object
* @apiNote See {@link #maxRetries(int)} for how this timeout interacts with retries.
*/
public WebServiceClient.Builder connectTimeout(Duration val) {
connectTimeout = val;
Expand Down Expand Up @@ -173,8 +183,9 @@ public WebServiceClient.Builder locales(List<String> val) {


/**
* @param val Request timeout duration. here is no timeout by default.
* @param val Request timeout duration. There is no timeout by default.
* @return Builder object
* @apiNote See {@link #maxRetries(int)} for how this timeout interacts with retries.
*/
public Builder requestTimeout(Duration val) {
requestTimeout = val;
Expand All @@ -195,13 +206,42 @@ public Builder proxy(ProxySelector val) {
* @param val the HttpClient to use when making requests. When provided,
* connectTimeout and proxy settings will be ignored as the
* custom client should handle these configurations.
* <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) {
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 @@ -311,10 +351,11 @@ public void reportTransaction(TransactionReport transaction) throws IOException,

HttpResponse<InputStream> response = null;
try {
response = httpClient.send(request, HttpResponse.BodyHandlers.ofInputStream());
response = sendWithRetry(request);
maybeThrowException(response, uri);
exhaustBody(response);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new MinFraudException("Interrupted sending request", e);
} finally {
if (response != null) {
Expand All @@ -333,9 +374,10 @@ private <T> T responseFor(String service, AbstractModel transaction, Class<T> cl

HttpResponse<InputStream> response = null;
try {
response = httpClient.send(request, HttpResponse.BodyHandlers.ofInputStream());
response = sendWithRetry(request);
return handleResponse(response, uri, cls);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new MinFraudException("Interrupted sending request", e);
} finally {
if (response != null) {
Expand All @@ -344,6 +386,62 @@ private <T> T responseFor(String service, AbstractModel transaction, Class<T> cl
}
}

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++;
}
}
}

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 HttpRequest requestFor(AbstractModel transaction, URI uri)
throws MinFraudException, IOException {
var builder = HttpRequest.newBuilder()
Expand All @@ -354,6 +452,7 @@ private HttpRequest requestFor(AbstractModel transaction, URI uri)
.header("User-Agent", userAgent)
// XXX - creating this JSON string is somewhat wasteful. We
// could use an input stream instead.
// BodyPublishers.ofString() is replayable; safe for retry attempts
.POST(HttpRequest.BodyPublishers.ofString(transaction.toJson()));

if (requestTimeout != null) {
Expand Down
Loading
Loading