Vert.x Http Proxy
Vert.x Http Proxy is a reverse proxy based on Vert.x, it aims to implement reusable reverse proxy logic to focus on higher concerns.
Warning
|
This module has Tech Preview status, this means the API can change between versions. |
Using Vert.x Http Proxy
To use Vert.x Http Proxy, add the following dependency to the dependencies section of your build descriptor:
-
Maven (in your
pom.xml
):
<dependency>
<groupId>io.vertx</groupId>
<artifactId>vertx-http-proxy</artifactId>
<version>4.3.8</version>
</dependency>
-
Gradle (in your
build.gradle
file):
dependencies {
compile 'io.vertx:vertx-http-proxy:4.3.8'
}
Reverse proxy server
In order to accomplish a reverse proxy with Vert.x Http Proxy you need the following:
-
Proxy Server that handles user-agent requests and forward them to the origin server
-
Origin Server that handles requests from the proxy server and respond accordingly
You can create a proxy server that listens to port 8080
and implement reverse proxy logic
HttpClient proxyClient = vertx.createHttpClient();
HttpProxy proxy = HttpProxy.reverseProxy(proxyClient);
proxy.origin(7070, "origin");
HttpServer proxyServer = vertx.createHttpServer();
proxyServer.requestHandler(proxy).listen(8080);
All user-agent requests are forwarded to the origin server conveniently.
Origin server routing
You can create a proxy that forwards all the traffic to a single server like seen before
You can set an origin selector to route the traffic to a given server
proxy.originSelector(request -> Future.succeededFuture(resolveOriginAddress(request)));
You can set a function to create the client request to the origin server for ultimate flexibility
proxy.originRequestProvider((request, client) -> client.request(resolveOriginOptions(request)));
Headers forwarding
End-to-end headers are forwarded by the proxy, hop-by-hop headers are ignored.
Request authority
As a transparent proxy, the request authority ({@code Host} header for HTTP/1.1, {@code :authority} pseudo header for HTTP/2) is preserved.
You can override the request authority
proxy.addInterceptor(new ProxyInterceptor() {
@Override
public Future<ProxyResponse> handleProxyRequest(ProxyContext context) {
ProxyRequest proxyRequest = context.request();
proxyRequest.setAuthority("example.com:80");
return ProxyInterceptor.super.handleProxyRequest(context);
}
});
When the request authority is overridden, the {@code x-forwarded-host} header is set on the request to the origin server with the original authority value.
Warning
|
changing the request authority can have undesirable side effects and can affect the proxied web server that might rely on the original request authority to handle cookies, URL redirects and such. |
WebSockets
The proxy supports WebSocket by default.
WebSocket handshake requests are forwarded to the origin server (including the connection
header) and the handshake
happens between the user agent and the origin server.
You can configure WebSocket support with setSupportWebSocket
.
Proxy caching
By default the proxy does not cache response and ignores most cache directives, you can enable caching by setting the cache options.
HttpProxy proxy = HttpProxy.reverseProxy(new ProxyOptions().setCacheOptions(new CacheOptions()), proxyClient);
Proxy interception
Interception is a powerful way to extend the proxy withg new features.
You can implement handleProxyRequest
to perform any operation on the proxy request
proxy.addInterceptor(new ProxyInterceptor() {
@Override
public Future<ProxyResponse> handleProxyRequest(ProxyContext context) {
ProxyRequest proxyRequest = context.request();
filter(proxyRequest.headers());
// Continue the interception chain
return context.sendRequest();
}
});
Likewise with the proxy response
proxy.addInterceptor(new ProxyInterceptor() {
@Override
public Future<Void> handleProxyResponse(ProxyContext context) {
ProxyResponse proxyResponse = context.response();
filter(proxyResponse.headers());
// Continue the interception chain
return context.sendResponse();
}
});
Body filtering
You can filter body by simply replacing the original Body
with a new one
proxy.addInterceptor(new ProxyInterceptor() {
@Override
public Future<Void> handleProxyResponse(ProxyContext context) {
ProxyResponse proxyResponse = context.response();
// Create a filtered body
Body filteredBody = filter(proxyResponse.getBody());
// And then let the response use it
proxyResponse.setBody(filteredBody);
// Continue the interception chain
return context.sendResponse();
}
});
Interception control
sendRequest
and sendResponse
continue the
current interception chain and then send the result to the origin server or the user-agent.
You can change the control, e.g you can send a response immediately to the user-agent without even requesting the origin server
proxy.addInterceptor(new ProxyInterceptor() {
@Override
public Future<ProxyResponse> handleProxyRequest(ProxyContext context) {
ProxyRequest proxyRequest = context.request();
// Release the underlying resources
proxyRequest.release();
// Create a response and populate it
ProxyResponse proxyResponse = proxyRequest.response()
.setStatusCode(200)
.putHeader("content-type", "text/plain")
.setBody(Body.body(Buffer.buffer("Hello World")));
return Future.succeededFuture(proxyResponse);
}
});