Skip to content

Commit c42b50d

Browse files
authored
Merge pull request #47 from reactiverse/feat/configurable-proxy
feat/configurable-proxy
2 parents 263e075 + 906501c commit c42b50d

File tree

6 files changed

+152
-16
lines changed

6 files changed

+152
-16
lines changed

.editorconfig

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
root = true
2+
3+
[*]
4+
charset = utf-8
5+
indent_style = space
6+
indent_size = 2
7+
trim_trailing_whitespace = true
8+
end_of_line = lf
9+
insert_final_newline = true
10+
11+
[Makefile]
12+
indent_style = tab
13+
14+
[**/examples/**.java]
15+
# 84 looks like a odd number, however
16+
# it accounts for 4 spaces (class and example method indentation)
17+
max_line_length = 84

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ Artifacts are published [here](https://search.maven.org/search?q=a:vertx-aws-sdk
1212

1313
| Project | Vert.x | AWS sdk |
1414
| ------- | ------ | ------- |
15+
| 0.6.0 | 3.9.2 | 2.14.7 |
1516
| 0.5.1 | 3.9.2 | 2.13.6 |
1617
| 0.5.0 | 3.9.0 | 2.12.0 |
1718
| 0.4.0 | 3.8.3 | 2.10.16 |

build.gradle.kts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
11
val vertxVersion = "3.9.2"
2-
val awsSdkVersion = "2.13.6"
2+
val awsSdkVersion = "2.14.7"
33
val junit5Version = "5.4.0"
44
val logbackVersion = "1.2.3"
55
val integrationOption = "tests.integration"
66

77
group = "io.reactiverse"
8-
version = "0.5.1"
8+
version = "0.6.0"
99

1010
plugins {
1111
`java-library`

src/main/java/io/reactiverse/awssdk/VertxNioAsyncHttpClient.java

+20-7
Original file line numberDiff line numberDiff line change
@@ -14,23 +14,36 @@
1414
import software.amazon.awssdk.http.async.SdkAsyncHttpResponseHandler;
1515
import software.amazon.awssdk.http.async.SdkHttpContentPublisher;
1616

17+
import java.util.Optional;
1718
import java.util.concurrent.CompletableFuture;
1819

20+
import static java.util.Objects.requireNonNull;
21+
1922
public class VertxNioAsyncHttpClient implements SdkAsyncHttpClient {
2023

2124
private final Context context;
2225
private final HttpClient client;
26+
private final HttpClientOptions clientOptions;
27+
28+
private static final HttpClientOptions DEFAULT_CLIENT_OPTIONS = new HttpClientOptions()
29+
.setSsl(true)
30+
.setKeepAlive(true);
2331

2432
public VertxNioAsyncHttpClient(Context context) {
25-
this.context = context;
26-
this.client = createVertxHttpClient(context.owner());
33+
this.context = context;
34+
this.clientOptions = DEFAULT_CLIENT_OPTIONS;
35+
this.client = createVertxHttpClient(context.owner());
36+
}
37+
38+
public VertxNioAsyncHttpClient(Context context, HttpClientOptions clientOptions) {
39+
requireNonNull(clientOptions);
40+
this.context = context;
41+
this.clientOptions = clientOptions;
42+
this.client = createVertxHttpClient(context.owner());
2743
}
2844

29-
private static HttpClient createVertxHttpClient(Vertx vertx) {
30-
HttpClientOptions options = new HttpClientOptions()
31-
.setSsl(true)
32-
.setKeepAlive(true);
33-
return vertx.createHttpClient(options);
45+
private HttpClient createVertxHttpClient(Vertx vertx) {
46+
return vertx.createHttpClient(clientOptions);
3447
}
3548

3649
@Override
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,27 @@
11
package io.reactiverse.awssdk;
22

33
import io.vertx.core.Context;
4+
import io.vertx.core.http.HttpClientOptions;
45
import software.amazon.awssdk.awscore.client.builder.AwsAsyncClientBuilder;
56
import software.amazon.awssdk.core.SdkClient;
67
import software.amazon.awssdk.core.client.config.SdkAdvancedAsyncClientOption;
78

89
public interface VertxSdkClient {
910

10-
static<C extends SdkClient, B extends AwsAsyncClientBuilder<B, C>> B withVertx(B builder, Context context) {
11-
return builder
12-
.httpClient(new VertxNioAsyncHttpClient(context))
13-
.asyncConfiguration(conf ->
14-
conf.advancedOption(SdkAdvancedAsyncClientOption.FUTURE_COMPLETION_EXECUTOR, new VertxExecutor(context))
15-
);
16-
}
11+
static<C extends SdkClient, B extends AwsAsyncClientBuilder<B, C>> B withVertx(B builder, Context context) {
12+
return builder
13+
.httpClient(new VertxNioAsyncHttpClient(context))
14+
.asyncConfiguration(conf ->
15+
conf.advancedOption(SdkAdvancedAsyncClientOption.FUTURE_COMPLETION_EXECUTOR, new VertxExecutor(context))
16+
);
17+
}
18+
19+
static<C extends SdkClient, B extends AwsAsyncClientBuilder<B, C>> B withVertx(B builder, HttpClientOptions clientOptions, Context context) {
20+
return builder
21+
.httpClient(new VertxNioAsyncHttpClient(context, clientOptions))
22+
.asyncConfiguration(conf ->
23+
conf.advancedOption(SdkAdvancedAsyncClientOption.FUTURE_COMPLETION_EXECUTOR, new VertxExecutor(context))
24+
);
25+
}
1726

1827
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
package io.reactiverse.awssdk;
2+
3+
import io.vertx.core.Vertx;
4+
import io.vertx.core.http.HttpClientOptions;
5+
import io.vertx.core.http.HttpServer;
6+
import io.vertx.core.net.ProxyOptions;
7+
import io.vertx.junit5.Timeout;
8+
import io.vertx.junit5.VertxExtension;
9+
import io.vertx.junit5.VertxTestContext;
10+
import org.junit.jupiter.api.AfterEach;
11+
import org.junit.jupiter.api.BeforeEach;
12+
import org.junit.jupiter.api.Test;
13+
import org.junit.jupiter.api.extension.ExtendWith;
14+
import software.amazon.awssdk.auth.credentials.AwsCredentials;
15+
import software.amazon.awssdk.auth.credentials.AwsCredentialsProvider;
16+
import software.amazon.awssdk.regions.Region;
17+
import software.amazon.awssdk.services.kinesis.KinesisAsyncClient;
18+
import software.amazon.awssdk.services.kinesis.KinesisAsyncClientBuilder;
19+
import software.amazon.awssdk.services.kinesis.model.ListStreamsResponse;
20+
21+
import java.net.URI;
22+
import java.util.concurrent.CompletableFuture;
23+
import java.util.concurrent.TimeUnit;
24+
import java.util.concurrent.atomic.AtomicInteger;
25+
26+
import static org.junit.jupiter.api.Assertions.assertEquals;
27+
import static org.junit.jupiter.api.Assertions.assertTrue;
28+
29+
@ExtendWith(VertxExtension.class)
30+
public class ProxyTest {
31+
32+
private Vertx vertx;
33+
private HttpServer proxyServer;
34+
35+
private static final int PROXY_PORT = 8000;
36+
private static final String PROXY_HOST = "localhost";
37+
38+
private final AtomicInteger proxyAccess = new AtomicInteger(0);
39+
private static final AwsCredentialsProvider credentialsProvider = () -> new AwsCredentials() {
40+
@Override
41+
public String accessKeyId() {
42+
return "a";
43+
}
44+
45+
@Override
46+
public String secretAccessKey() {
47+
return "a";
48+
}
49+
};
50+
51+
@BeforeEach
52+
public void setUp(VertxTestContext ctx) {
53+
vertx = Vertx.vertx();
54+
proxyServer = vertx.createHttpServer();
55+
proxyServer.requestHandler(req -> {
56+
proxyAccess.incrementAndGet();
57+
req.response().end();
58+
});
59+
proxyServer.listen(PROXY_PORT, PROXY_HOST, res -> {
60+
assertTrue(res.succeeded());
61+
ctx.completeNow();
62+
});
63+
}
64+
65+
@AfterEach
66+
public void tearDown(VertxTestContext ctx) {
67+
if (proxyServer == null) {
68+
return;
69+
}
70+
proxyAccess.set(0);
71+
proxyServer.close(res -> {
72+
assertTrue(res.succeeded());
73+
ctx.completeNow();
74+
});
75+
}
76+
77+
@Test
78+
@Timeout(value = 15, timeUnit = TimeUnit.SECONDS)
79+
public void testGetThroughProxy(VertxTestContext ctx) throws Exception {
80+
final KinesisAsyncClientBuilder builder = KinesisAsyncClient.builder()
81+
.region(Region.EU_WEST_1)
82+
.endpointOverride(new URI("http://localhost:1111")) // something that just doesn't exist, the only thing that matters is that every request has traveled through proxy
83+
.credentialsProvider(credentialsProvider);
84+
HttpClientOptions throughProxyOptions = new HttpClientOptions().setProxyOptions(new ProxyOptions().setHost(PROXY_HOST).setPort(PROXY_PORT));
85+
KinesisAsyncClient kinesis = VertxSdkClient.withVertx(builder, throughProxyOptions, vertx.getOrCreateContext()).build();
86+
assertEquals(proxyAccess.get(), 0, "Proxy access count should have been reset");
87+
kinesis
88+
.listStreams()
89+
.handle((res, err) -> {
90+
assertTrue(proxyAccess.get() > 0, "Requests should have been transferred through proxy");
91+
ctx.completeNow();
92+
return null;
93+
});
94+
}
95+
96+
}

0 commit comments

Comments
 (0)