org.mockserver.netty.proxy.relay.UpstreamProxyRelayHandler Maven / Gradle / Ivy
package org.mockserver.netty.proxy.relay;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.ssl.SslHandler;
import org.mockserver.log.model.LogEntry;
import org.mockserver.logging.MockServerLogger;
import org.slf4j.event.Level;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.ClosedSelectorException;
import static org.mockserver.exception.ExceptionHandling.closeOnFlush;
import static org.mockserver.exception.ExceptionHandling.connectionClosedException;
import static org.mockserver.netty.unification.PortUnificationHandler.isSslEnabledDownstream;
import static org.mockserver.netty.unification.PortUnificationHandler.nettySslContextFactory;
public class UpstreamProxyRelayHandler extends SimpleChannelInboundHandler {
private final MockServerLogger mockServerLogger;
private final Channel upstreamChannel;
private final Channel downstreamChannel;
public UpstreamProxyRelayHandler(MockServerLogger mockServerLogger, Channel upstreamChannel, Channel downstreamChannel) {
super(false);
this.upstreamChannel = upstreamChannel;
this.downstreamChannel = downstreamChannel;
this.mockServerLogger = mockServerLogger;
}
@Override
public void channelActive(ChannelHandlerContext ctx) {
ctx.read();
ctx.write(Unpooled.EMPTY_BUFFER);
}
@Override
public void channelRead0(final ChannelHandlerContext ctx, final FullHttpRequest request) {
if (isSslEnabledDownstream(upstreamChannel) && downstreamChannel.pipeline().get(SslHandler.class) == null) {
downstreamChannel.pipeline().addFirst(nettySslContextFactory(ctx.channel()).createClientSslContext(true).newHandler(ctx.alloc()));
}
downstreamChannel.writeAndFlush(request).addListener((ChannelFutureListener) future -> {
if (future.isSuccess()) {
ctx.channel().read();
} else {
if (isNotSocketClosedException(future.cause())) {
mockServerLogger.logEvent(
new LogEntry()
.setType(LogEntry.LogMessageType.EXCEPTION)
.setLogLevel(Level.ERROR)
.setMessageFormat("exception while returning response for request \"" + request.method() + " " + request.uri() + "\"")
.setThrowable(future.cause())
);
}
future.channel().close();
}
});
}
private boolean isNotSocketClosedException(Throwable cause) {
return !(cause instanceof ClosedChannelException || cause instanceof ClosedSelectorException);
}
@Override
public void channelInactive(ChannelHandlerContext ctx) {
closeOnFlush(downstreamChannel);
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
if (connectionClosedException(cause)) {
mockServerLogger.logEvent(
new LogEntry()
.setType(LogEntry.LogMessageType.EXCEPTION)
.setLogLevel(Level.ERROR)
.setMessageFormat("exception caught by upstream relay handler -> closing pipeline " + ctx.channel())
.setThrowable(cause)
);
}
closeOnFlush(ctx.channel());
}
}