org.mockserver.netty.proxy.relay.RelayConnectHandler Maven / Gradle / Ivy
package org.mockserver.netty.proxy.relay;
import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufUtil;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.http.HttpClientCodec;
import io.netty.handler.codec.http.HttpContentDecompressor;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpServerCodec;
import io.netty.handler.ssl.SslHandler;
import org.mockserver.configuration.ConfigurationProperties;
import org.mockserver.lifecycle.LifeCycle;
import org.mockserver.log.model.LogEntry;
import org.mockserver.logging.LoggingHandler;
import org.mockserver.logging.MockServerLogger;
import org.slf4j.event.Level;
import java.net.InetSocketAddress;
import java.nio.charset.StandardCharsets;
import static org.mockserver.exception.ExceptionHandling.connectionClosedException;
import static org.mockserver.mock.action.http.HttpActionHandler.getRemoteAddress;
import static org.mockserver.netty.unification.PortUnificationHandler.*;
import static org.slf4j.event.Level.DEBUG;
@ChannelHandler.Sharable
public abstract class RelayConnectHandler extends SimpleChannelInboundHandler {
public static final String PROXIED = "PROXIED_";
public static final String PROXIED_SECURE = PROXIED + "SECURE_";
public static final String PROXIED_RESPONSE = "PROXIED_RESPONSE_";
private final LifeCycle server;
private final MockServerLogger mockServerLogger;
protected final String host;
protected final int port;
public RelayConnectHandler(LifeCycle server, MockServerLogger mockServerLogger, String host, int port) {
this.server = server;
this.mockServerLogger = mockServerLogger;
this.host = host;
this.port = port;
}
@Override
public void channelRead0(final ChannelHandlerContext proxyClientCtx, final T request) {
Bootstrap bootstrap = new Bootstrap()
.group(proxyClientCtx.channel().eventLoop())
.channel(NioSocketChannel.class)
.handler(new ChannelInboundHandlerAdapter() {
@Override
public void channelActive(final ChannelHandlerContext mockServerCtx) {
if (isSslEnabledUpstream(proxyClientCtx.channel())) {
mockServerCtx.writeAndFlush(Unpooled.copiedBuffer((PROXIED_SECURE + host + ":" + port).getBytes(StandardCharsets.UTF_8))).awaitUninterruptibly();
} else {
mockServerCtx.writeAndFlush(Unpooled.copiedBuffer((PROXIED + host + ":" + port).getBytes(StandardCharsets.UTF_8))).awaitUninterruptibly();
}
}
@Override
public void channelRead(ChannelHandlerContext mockServerCtx, Object msg) {
if (msg instanceof ByteBuf) {
byte[] bytes = ByteBufUtil.getBytes((ByteBuf) msg);
if (new String(bytes, StandardCharsets.UTF_8).startsWith(PROXIED_RESPONSE)) {
proxyClientCtx
.writeAndFlush(successResponse(request))
.addListener((ChannelFutureListener) channelFuture -> {
removeCodecSupport(proxyClientCtx);
// downstream (to MockServer)
ChannelPipeline pipelineToMockServer = mockServerCtx.channel().pipeline();
if (isSslEnabledDownstream(proxyClientCtx.channel())) {
pipelineToMockServer.addLast(nettySslContextFactory(proxyClientCtx.channel()).createClientSslContext(true).newHandler(mockServerCtx.alloc(), host, port));
}
if (MockServerLogger.isEnabled(Level.TRACE)) {
pipelineToMockServer.addLast(new LoggingHandler("downstream -->"));
}
pipelineToMockServer.addLast(new HttpClientCodec(ConfigurationProperties.maxInitialLineLength(), ConfigurationProperties.maxHeaderSize(), ConfigurationProperties.maxChunkSize()));
pipelineToMockServer.addLast(new HttpContentDecompressor());
pipelineToMockServer.addLast(new HttpObjectAggregator(Integer.MAX_VALUE));
pipelineToMockServer.addLast(new DownstreamProxyRelayHandler(mockServerLogger, proxyClientCtx.channel()));
// upstream (to proxy client)
ChannelPipeline pipelineToProxyClient = proxyClientCtx.channel().pipeline();
if (isSslEnabledUpstream(proxyClientCtx.channel()) && pipelineToProxyClient.get(SslHandler.class) == null) {
pipelineToProxyClient.addLast(nettySslContextFactory(proxyClientCtx.channel()).createServerSslContext().newHandler(proxyClientCtx.alloc()));
}
if (MockServerLogger.isEnabled(Level.TRACE)) {
pipelineToProxyClient.addLast(new LoggingHandler("upstream <-- "));
}
pipelineToProxyClient.addLast(new HttpServerCodec(ConfigurationProperties.maxInitialLineLength(), ConfigurationProperties.maxHeaderSize(), ConfigurationProperties.maxChunkSize()));
pipelineToProxyClient.addLast(new HttpContentDecompressor());
pipelineToProxyClient.addLast(new HttpObjectAggregator(Integer.MAX_VALUE));
pipelineToProxyClient.addLast(new UpstreamProxyRelayHandler(mockServerLogger, proxyClientCtx.channel(), mockServerCtx.channel()));
});
} else {
mockServerCtx.fireChannelRead(msg);
}
}
}
});
final InetSocketAddress remoteSocket = getDownstreamSocket(proxyClientCtx);
bootstrap.connect(remoteSocket).addListener((ChannelFutureListener) future -> {
if (future.isSuccess()) {
if (MockServerLogger.isEnabled(DEBUG)) {
mockServerLogger.logEvent(
new LogEntry()
.setLogLevel(DEBUG)
.setMessageFormat("connected to{}")
.setArguments(remoteSocket)
);
}
} else {
failure("Connection failed to " + remoteSocket, future.cause(), proxyClientCtx, failureResponse(request));
}
});
}
private InetSocketAddress getDownstreamSocket(ChannelHandlerContext ctx) {
InetSocketAddress remoteAddress = getRemoteAddress(ctx);
if (remoteAddress != null) {
return remoteAddress;
} else {
return new InetSocketAddress(server.getLocalPort());
}
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
failure("Exception caught by CONNECT proxy handler -> closing pipeline ", cause, ctx, failureResponse(null));
}
private void failure(String message, Throwable cause, ChannelHandlerContext ctx, Object response) {
if (connectionClosedException(cause)) {
mockServerLogger.logEvent(
new LogEntry()
.setLogLevel(Level.ERROR)
.setMessageFormat(message)
.setThrowable(cause)
);
}
Channel channel = ctx.channel();
channel.writeAndFlush(response);
if (channel.isActive()) {
channel.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
}
}
protected abstract void removeCodecSupport(ChannelHandlerContext ctx);
protected abstract Object successResponse(Object request);
protected abstract Object failureResponse(Object request);
protected void removeHandler(ChannelPipeline pipeline, Class extends ChannelHandler> handlerType) {
if (pipeline.get(handlerType) != null) {
pipeline.remove(handlerType);
}
}
protected void removeHandler(ChannelPipeline pipeline, ChannelHandler channelHandler) {
if (pipeline.toMap().containsValue(channelHandler)) {
pipeline.remove(channelHandler);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy