Skip to content

Commit 1acf8a3

Browse files
map AWS_IO_SOCKET_TIMEOUT HttpException to java.net.ConnectException when acquiring connection (#5940)
Co-authored-by: Zoe Wang <[email protected]>
1 parent a516e29 commit 1acf8a3

File tree

4 files changed

+71
-1
lines changed

4 files changed

+71
-1
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"category": "AWS CRT HTTP Client",
3+
"contributor": "thomasjinlo",
4+
"type": "feature",
5+
"description": "Map AWS_IO_SOCKET_TIMEOUT to ConnectException when acquiring a connection to improve error handling"
6+
}

http-clients/aws-crt-client/src/main/java/software/amazon/awssdk/http/crt/internal/CrtUtils.java

+6-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
import static software.amazon.awssdk.utils.NumericUtils.saturatedCast;
2424

2525
import java.io.IOException;
26+
import java.net.ConnectException;
2627
import java.time.Duration;
2728
import javax.net.ssl.SSLHandshakeException;
2829
import software.amazon.awssdk.annotations.SdkInternalApi;
@@ -35,6 +36,7 @@
3536
@SdkInternalApi
3637
public final class CrtUtils {
3738
public static final int CRT_TLS_NEGOTIATION_ERROR_CODE = 1029;
39+
public static final int CRT_SOCKET_TIMEOUT = 1048;
3840

3941
private CrtUtils() {
4042
}
@@ -54,9 +56,12 @@ public static Throwable wrapConnectionFailureException(Throwable throwable) {
5456
Throwable toThrow = new IOException("An exception occurred when acquiring a connection", throwable);
5557
if (throwable instanceof HttpException) {
5658
HttpException httpException = (HttpException) throwable;
59+
int httpErrorCode = httpException.getErrorCode();
5760

58-
if (httpException.getErrorCode() == CRT_TLS_NEGOTIATION_ERROR_CODE) {
61+
if (httpErrorCode == CRT_TLS_NEGOTIATION_ERROR_CODE) {
5962
toThrow = new SSLHandshakeException(httpException.getMessage());
63+
} else if (httpErrorCode == CRT_SOCKET_TIMEOUT) {
64+
toThrow = new ConnectException(httpException.getMessage());
6065
}
6166
}
6267

http-clients/aws-crt-client/src/test/java/software/amazon/awssdk/http/crt/internal/CrtAsyncRequestExecutorTest.java

+31
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,19 @@
2121
import static software.amazon.awssdk.http.crt.CrtHttpClientTestUtils.createRequest;
2222

2323
import java.io.IOException;
24+
import java.net.ConnectException;
2425
import java.net.URI;
26+
import javax.net.ssl.SSLHandshakeException;
2527
import java.util.concurrent.CompletableFuture;
28+
import java.util.stream.Stream;
29+
import java.util.AbstractMap.SimpleEntry;
30+
import java.util.Map.Entry;
2631
import org.junit.jupiter.api.AfterEach;
2732
import org.junit.jupiter.api.BeforeEach;
2833
import org.junit.jupiter.api.Test;
2934
import org.junit.jupiter.api.extension.ExtendWith;
35+
import org.junit.jupiter.params.ParameterizedTest;
36+
import org.junit.jupiter.params.provider.MethodSource;
3037
import org.mockito.ArgumentCaptor;
3138
import org.mockito.Mock;
3239
import org.mockito.Mockito;
@@ -56,6 +63,13 @@ public class CrtAsyncRequestExecutorTest {
5663
@Mock
5764
private HttpClientConnection httpClientConnection;
5865

66+
public static Stream<Entry<Integer, Class<? extends Throwable>>> mappedExceptions() {
67+
return Stream.of(
68+
new SimpleEntry<>(0x0405, SSLHandshakeException.class), // For AWS_IO_TLS_ERROR_NEGOTIATION_FAILURE (1029)
69+
new SimpleEntry<>(0x0418, ConnectException.class) // For AWS_IO_SOCKET_TIMEOUT (1048)
70+
);
71+
}
72+
5973
@BeforeEach
6074
public void setup() {
6175
requestExecutor = new CrtAsyncRequestExecutor();
@@ -149,6 +163,23 @@ public void execute_AcquireConnectionFailure_shouldAlwaysWrapIOException() {
149163
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(IOException.class).hasRootCause(exception);
150164
}
151165

166+
@ParameterizedTest
167+
@MethodSource("mappedExceptions")
168+
public void execute_AcquireConnectionFailure_shouldAlwaysBeInstanceOfIOException(Entry<Integer, Class<? extends Throwable>> entry) {
169+
int errorCode = entry.getKey();
170+
Class<? extends Throwable> ioExceptionSubclass = entry.getValue();
171+
172+
CrtAsyncRequestContext context = crtAsyncRequestContext();
173+
HttpException exception = new HttpException(errorCode);
174+
CompletableFuture<HttpClientConnection> completableFuture = CompletableFutureUtils.failedFuture(exception);
175+
176+
Mockito.when(connectionManager.acquireConnection()).thenReturn(completableFuture);
177+
178+
CompletableFuture<Void> executeFuture = requestExecutor.execute(context);
179+
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(IOException.class).hasMessageContaining(exception.getMessage());
180+
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(ioExceptionSubclass);
181+
}
182+
152183
@Test
153184
public void executeRequest_failedOfIllegalStateException_shouldWrapIOException() {
154185
IllegalStateException exception = new IllegalStateException("connection closed");

http-clients/aws-crt-client/src/test/java/software/amazon/awssdk/http/crt/internal/CrtRequestExecutorTest.java

+28
Original file line numberDiff line numberDiff line change
@@ -20,9 +20,13 @@
2020
import static software.amazon.awssdk.http.crt.CrtHttpClientTestUtils.createRequest;
2121

2222
import java.io.IOException;
23+
import java.net.ConnectException;
2324
import java.net.URI;
25+
import javax.net.ssl.SSLHandshakeException;
2426
import java.util.concurrent.CompletableFuture;
2527
import java.util.stream.Stream;
28+
import java.util.AbstractMap.SimpleEntry;
29+
import java.util.Map.Entry;
2630
import org.junit.jupiter.api.AfterEach;
2731
import org.junit.jupiter.api.BeforeEach;
2832
import org.junit.jupiter.api.Test;
@@ -59,6 +63,13 @@ public static Stream<Throwable> retryableExceptions() {
5963
"connection closed"));
6064
}
6165

66+
public static Stream<Entry<Integer, Class<? extends Throwable>>> mappedExceptions() {
67+
return Stream.of(
68+
new SimpleEntry<>(0x0405, SSLHandshakeException.class), // For AWS_IO_TLS_ERROR_NEGOTIATION_FAILURE (1029)
69+
new SimpleEntry<>(0x0418, ConnectException.class) // For AWS_IO_SOCKET_TIMEOUT (1048)
70+
);
71+
}
72+
6273
@BeforeEach
6374
public void setup() {
6475
requestExecutor = new CrtRequestExecutor();
@@ -111,6 +122,23 @@ public void execute_AcquireConnectionFailure_shouldAlwaysWrapIOException() {
111122
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(IOException.class).hasRootCause(exception);
112123
}
113124

125+
@ParameterizedTest
126+
@MethodSource("mappedExceptions")
127+
public void execute_AcquireConnectionFailure_shouldAlwaysBeInstanceOfIOException(Entry<Integer, Class<? extends Throwable>> entry) {
128+
int errorCode = entry.getKey();
129+
Class<? extends Throwable> ioExceptionSubclass = entry.getValue();
130+
131+
CrtRequestContext context = crtRequestContext();
132+
HttpException exception = new HttpException(errorCode);
133+
CompletableFuture<HttpClientConnection> completableFuture = CompletableFutureUtils.failedFuture(exception);
134+
135+
Mockito.when(connectionManager.acquireConnection()).thenReturn(completableFuture);
136+
137+
CompletableFuture<SdkHttpFullResponse> executeFuture = requestExecutor.execute(context);
138+
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(IOException.class).hasMessageContaining(exception.getMessage());
139+
assertThatThrownBy(executeFuture::join).hasCauseInstanceOf(ioExceptionSubclass);
140+
}
141+
114142
@Test
115143
public void executeRequest_failedOfNonRetryableHttpException_shouldNotWrapIOException() {
116144
HttpException exception = new HttpException(0x0801); // AWS_ERROR_HTTP_HEADER_NOT_FOUND

0 commit comments

Comments
 (0)