Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
reactor.netty.http.server.HttpTrafficHandler Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2024 VMware, Inc. or its affiliates, All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package reactor.netty.http.server;
import java.net.SocketAddress;
import java.time.Duration;
import java.time.ZonedDateTime;
import java.util.Optional;
import java.util.Queue;
import java.util.function.BiFunction;
import java.util.function.BiPredicate;
import io.netty.channel.ChannelDuplexHandler;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.DecoderResult;
import io.netty.handler.codec.DecoderResultProvider;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.DefaultHttpContent;
import io.netty.handler.codec.http.DefaultHttpResponse;
import io.netty.handler.codec.http.DefaultLastHttpContent;
import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpObject;
import io.netty.handler.codec.http.HttpRequest;
import io.netty.handler.codec.http.HttpResponse;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpStatusClass;
import io.netty.handler.codec.http.HttpVersion;
import io.netty.handler.codec.http.LastHttpContent;
import io.netty.handler.codec.http.cookie.ServerCookieDecoder;
import io.netty.handler.codec.http.cookie.ServerCookieEncoder;
import io.netty.handler.ssl.SslHandler;
import io.netty.util.ReferenceCountUtil;
import reactor.core.Exceptions;
import reactor.core.publisher.Mono;
import reactor.netty.Connection;
import reactor.netty.ConnectionObserver;
import reactor.netty.ReactorNetty;
import reactor.netty.channel.ChannelOperations;
import reactor.netty.http.logging.HttpMessageArgProviderFactory;
import reactor.netty.http.logging.HttpMessageLogFactory;
import reactor.util.annotation.Nullable;
import reactor.util.concurrent.Queues;
import static io.netty.handler.codec.http.HttpUtil.isContentLengthSet;
import static io.netty.handler.codec.http.HttpUtil.isKeepAlive;
import static io.netty.handler.codec.http.HttpUtil.isTransferEncodingChunked;
import static io.netty.handler.codec.http.HttpUtil.setKeepAlive;
import static io.netty.handler.codec.http.LastHttpContent.EMPTY_LAST_CONTENT;
import static reactor.netty.ReactorNetty.format;
/**
* Replace {@link io.netty.handler.codec.http.HttpServerKeepAliveHandler} with extra
* handler management.
*/
final class HttpTrafficHandler extends ChannelDuplexHandler implements Runnable {
static final String MULTIPART_PREFIX = "multipart";
static final HttpVersion H2 = HttpVersion.valueOf("HTTP/2.0");
static final boolean LAST_FLUSH_WHEN_NO_READ = Boolean.parseBoolean(
System.getProperty("reactor.netty.http.server.lastFlushWhenNoRead", "false"));
final BiPredicate compress;
final ServerCookieDecoder cookieDecoder;
final ServerCookieEncoder cookieEncoder;
final HttpServerFormDecoderProvider formDecoderProvider;
final BiFunction forwardedHeaderHandler;
final HttpMessageLogFactory httpMessageLogFactory;
final Duration idleTimeout;
final ConnectionObserver listener;
final BiFunction super Mono, ? super Connection, ? extends Mono>
mapHandle;
final int maxKeepAliveRequests;
final Duration readTimeout;
final Duration requestTimeout;
ChannelHandlerContext ctx;
boolean nonInformationalResponse;
boolean overflow;
// Track pending responses to support client pipelining: https://tools.ietf.org/html/rfc7230#section-6.3.2
int pendingResponses;
boolean persistentConnection = true;
Queue pipelined;
SocketAddress remoteAddress;
Boolean secure;
boolean read;
boolean needsFlush;
boolean finalizingResponse;
HttpTrafficHandler(
@Nullable BiPredicate compress,
ServerCookieDecoder decoder,
ServerCookieEncoder encoder,
HttpServerFormDecoderProvider formDecoderProvider,
@Nullable BiFunction forwardedHeaderHandler,
HttpMessageLogFactory httpMessageLogFactory,
@Nullable Duration idleTimeout,
ConnectionObserver listener,
@Nullable BiFunction super Mono, ? super Connection, ? extends Mono> mapHandle,
int maxKeepAliveRequests,
@Nullable Duration readTimeout,
@Nullable Duration requestTimeout) {
this.listener = listener;
this.formDecoderProvider = formDecoderProvider;
this.forwardedHeaderHandler = forwardedHeaderHandler;
this.compress = compress;
this.cookieEncoder = encoder;
this.cookieDecoder = decoder;
this.httpMessageLogFactory = httpMessageLogFactory;
this.idleTimeout = idleTimeout;
this.mapHandle = mapHandle;
this.maxKeepAliveRequests = maxKeepAliveRequests;
this.readTimeout = readTimeout;
this.requestTimeout = requestTimeout;
}
@Override
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
super.handlerAdded(ctx);
this.ctx = ctx;
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "New http connection, requesting read"));
}
ctx.read();
}
@Override
public void channelActive(ChannelHandlerContext ctx) {
IdleTimeoutHandler.addIdleTimeoutHandler(ctx.pipeline(), idleTimeout);
ctx.fireChannelActive();
}
@Override
public void channelRead(ChannelHandlerContext ctx, Object msg) {
read = true;
if (secure == null) {
secure = ctx.channel().pipeline().get(SslHandler.class) != null;
}
if (remoteAddress == null) {
remoteAddress =
Optional.ofNullable(HAProxyMessageReader.resolveRemoteAddressFromProxyProtocol(ctx.channel()))
.orElse(ctx.channel().remoteAddress());
}
// read message and track if it was keepAlive
if (msg instanceof HttpRequest) {
finalizingResponse = false;
if (idleTimeout != null) {
IdleTimeoutHandler.removeIdleTimeoutHandler(ctx.pipeline());
}
final HttpRequest request = (HttpRequest) msg;
if (H2.equals(request.protocolVersion())) {
IllegalStateException e = new IllegalStateException(
"Unexpected request [" + request.method() + " " + request.uri() + " HTTP/2.0]");
request.setDecoderResult(DecoderResult.failure(e.getCause() != null ? e.getCause() : e));
sendDecodingFailures(e, msg);
return;
}
if (persistentConnection) {
pendingResponses += 1;
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Increasing pending responses count: {}"),
pendingResponses);
}
persistentConnection = isKeepAlive(request);
}
else {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Dropping pipelined HTTP request, " +
"previous response requested connection close"));
}
ReferenceCountUtil.release(msg);
return;
}
if (pendingResponses > 1) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Buffering pipelined HTTP request, " +
"pending responses count: {}, queue: {}"),
pendingResponses,
pipelined != null ? pipelined.size() : 0);
}
overflow = true;
doPipeline(ctx, new HttpRequestHolder(request));
return;
}
else {
overflow = false;
if (LAST_FLUSH_WHEN_NO_READ) {
ChannelOperations, ?> ops = ChannelOperations.get(ctx.channel());
if (ops instanceof HttpServerOperations) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Last HTTP packet was sent, terminating the channel"));
}
((HttpServerOperations) ops).terminateInternal();
}
}
if (handleDecodingFailures(request.decoderResult(), msg)) {
return;
}
HttpServerOperations ops;
ZonedDateTime timestamp = ZonedDateTime.now(ReactorNetty.ZONE_ID_SYSTEM);
ConnectionInfo connectionInfo = null;
try {
connectionInfo = ConnectionInfo.from(ctx.channel(),
request,
secure,
remoteAddress,
forwardedHeaderHandler);
ops = new HttpServerOperations(Connection.from(ctx.channel()),
listener,
request,
compress,
connectionInfo,
cookieDecoder,
cookieEncoder,
formDecoderProvider,
httpMessageLogFactory,
false,
mapHandle,
readTimeout,
requestTimeout,
secure,
timestamp);
}
catch (RuntimeException e) {
request.setDecoderResult(DecoderResult.failure(e.getCause() != null ? e.getCause() : e));
sendDecodingFailures(e, msg, timestamp, connectionInfo);
return;
}
ops.bind();
listener.onStateChange(ops, ConnectionObserver.State.CONFIGURED);
ctx.fireChannelRead(msg);
return;
}
}
else if (persistentConnection && pendingResponses == 0) {
if (msg == EMPTY_LAST_CONTENT) {
ctx.fireChannelRead(msg);
}
else if (msg.getClass() == DefaultLastHttpContent.class) {
if (handleDecodingFailures(((DefaultLastHttpContent) msg).decoderResult(), msg)) {
return;
}
ctx.fireChannelRead(msg);
}
else if (msg instanceof LastHttpContent) {
if (handleDecodingFailures(((LastHttpContent) msg).decoderResult(), msg)) {
return;
}
ctx.fireChannelRead(msg);
}
else {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(
format(ctx.channel(), "Dropped HTTP content, since response has been sent already: {}"),
msg instanceof HttpObject ?
httpMessageLogFactory.debug(HttpMessageArgProviderFactory.create(msg)) : msg);
}
ReferenceCountUtil.release(msg);
}
ctx.read();
return;
}
else if (overflow) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Buffering pipelined HTTP content, " +
"pending responses count: {}, queue: {}"),
pendingResponses,
pipelined != null ? pipelined.size() : 0);
}
doPipeline(ctx, msg);
return;
}
if (msg instanceof DecoderResultProvider &&
handleDecodingFailures(((DecoderResultProvider) msg).decoderResult(), msg)) {
return;
}
ctx.fireChannelRead(msg);
}
@Override
public void channelReadComplete(ChannelHandlerContext ctx) {
endReadAndFlush();
ctx.fireChannelReadComplete();
}
void endReadAndFlush() {
if (read) {
read = false;
if (LAST_FLUSH_WHEN_NO_READ && needsFlush) {
needsFlush = false;
ctx.flush();
}
}
}
@Override
public void flush(ChannelHandlerContext ctx) {
if (LAST_FLUSH_WHEN_NO_READ && finalizingResponse) {
if (needsFlush || !ctx.channel().isWritable()) {
needsFlush = false;
ctx.flush();
}
else {
needsFlush = true;
}
}
else {
ctx.flush();
}
}
boolean handleDecodingFailures(DecoderResult decoderResult, Object msg) {
if (decoderResult.isFailure()) {
sendDecodingFailures(decoderResult.cause(), msg);
return true;
}
return false;
}
void sendDecodingFailures(Throwable t, Object msg) {
sendDecodingFailures(t, msg, null, null);
}
void sendDecodingFailures(Throwable t, Object msg, @Nullable ZonedDateTime timestamp, @Nullable ConnectionInfo connectionInfo) {
persistentConnection = false;
HttpServerOperations.sendDecodingFailures(ctx, listener, secure, t, msg, httpMessageLogFactory, timestamp, connectionInfo,
remoteAddress);
}
void doPipeline(ChannelHandlerContext ctx, Object msg) {
if (pipelined == null) {
pipelined = Queues.unbounded()
.get();
}
if (!pipelined.offer(msg)) {
ctx.fireExceptionCaught(Exceptions.failWithOverflow());
}
}
@Override
@SuppressWarnings("FutureReturnValueIgnored")
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) {
Class> msgClass = msg.getClass();
// modify message on way out to add headers if needed
if (msgClass == DefaultHttpResponse.class) {
handleDefaultHttpResponse((DefaultHttpResponse) msg, promise);
return;
}
else if (msgClass == DefaultFullHttpResponse.class) {
if (handleDefaultFullHttpResponse((DefaultFullHttpResponse) msg, promise)) {
return;
}
handleLastHttpContent(msg, promise);
return;
}
else if (msg == EMPTY_LAST_CONTENT || msgClass == DefaultLastHttpContent.class) {
handleLastHttpContent(msg, promise);
return;
}
else if (msgClass == DefaultHttpContent.class) {
handleDefaultHttContent((DefaultHttpContent) msg, promise);
return;
}
else if (msg instanceof HttpResponse) {
final HttpResponse response = (HttpResponse) msg;
nonInformationalResponse = !isInformational(response);
// Assume the response writer knows if they can persist or not and sets isKeepAlive on the response
boolean maxKeepAliveRequestsReached = maxKeepAliveRequests != -1 && HttpServerOperations.requestsCounter(ctx.channel()) == maxKeepAliveRequests;
if (maxKeepAliveRequestsReached || !isKeepAlive(response) || !isSelfDefinedMessageLength(response)) {
// No longer keep alive as the client can't tell when the message is done unless we close connection
pendingResponses = 0;
persistentConnection = false;
}
// Server might think it can keep connection alive, but we should fix response header if we know better
if (!shouldKeepAlive()) {
setKeepAlive(response, false);
}
if (response.status().code() == HttpResponseStatus.CONTINUE.code()) {
//"FutureReturnValueIgnored" this is deliberate
ctx.write(msg, promise);
return;
}
}
if (msg instanceof LastHttpContent) {
handleLastHttpContent(msg, promise);
return;
}
if (persistentConnection && pendingResponses == 0) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(
format(ctx.channel(), "Dropped HTTP content, since response has been sent already: {}"),
msg instanceof HttpObject ?
httpMessageLogFactory.debug(HttpMessageArgProviderFactory.create(msg)) : msg);
}
ReferenceCountUtil.release(msg);
promise.setSuccess();
return;
}
//"FutureReturnValueIgnored" this is deliberate
ctx.write(msg, promise);
}
@SuppressWarnings("FutureReturnValueIgnored")
boolean handleDefaultFullHttpResponse(DefaultFullHttpResponse response, ChannelPromise promise) {
nonInformationalResponse = !isInformational(response);
// Assume the response writer knows if they can persist or not and sets isKeepAlive on the response
boolean maxKeepAliveRequestsReached = maxKeepAliveRequests != -1 && HttpServerOperations.requestsCounter(ctx.channel()) == maxKeepAliveRequests;
if (maxKeepAliveRequestsReached || !isKeepAlive(response) || !isSelfDefinedMessageLength(response)) {
// No longer keep alive as the client can't tell when the message is done unless we close connection
pendingResponses = 0;
persistentConnection = false;
}
// Server might think it can keep connection alive, but we should fix response header if we know better
if (!shouldKeepAlive()) {
setKeepAlive(response, false);
}
if (response.status().code() == HttpResponseStatus.CONTINUE.code()) {
//"FutureReturnValueIgnored" this is deliberate
ctx.write(response, promise);
return true;
}
return false;
}
@SuppressWarnings("FutureReturnValueIgnored")
void handleDefaultHttContent(DefaultHttpContent msg, ChannelPromise promise) {
if (persistentConnection && pendingResponses == 0) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(
format(ctx.channel(), "Dropped HTTP content, since response has been sent already: {}"),
httpMessageLogFactory.debug(HttpMessageArgProviderFactory.create(msg)));
}
msg.release();
promise.setSuccess();
return;
}
//"FutureReturnValueIgnored" this is deliberate
ctx.write(msg, promise);
}
@SuppressWarnings("FutureReturnValueIgnored")
void handleDefaultHttpResponse(DefaultHttpResponse response, ChannelPromise promise) {
nonInformationalResponse = !isInformational(response);
// Assume the response writer knows if they can persist or not and sets isKeepAlive on the response
boolean maxKeepAliveRequestsReached = maxKeepAliveRequests != -1 && HttpServerOperations.requestsCounter(ctx.channel()) == maxKeepAliveRequests;
if (maxKeepAliveRequestsReached || !isKeepAlive(response) || !isSelfDefinedMessageLength(response)) {
// No longer keep alive as the client can't tell when the message is done unless we close connection
pendingResponses = 0;
persistentConnection = false;
}
// Server might think it can keep connection alive, but we should fix response header if we know better
if (!shouldKeepAlive()) {
setKeepAlive(response, false);
}
//"FutureReturnValueIgnored" this is deliberate
ctx.write(response, promise);
}
@SuppressWarnings("FutureReturnValueIgnored")
void handleLastHttpContent(Object msg, ChannelPromise promise) {
finalizingResponse = true;
if (LAST_FLUSH_WHEN_NO_READ) {
needsFlush = !read;
}
if (!shouldKeepAlive()) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Detected non persistent http " +
"connection, preparing to close. Pending responses count: {}"),
pendingResponses);
}
//"FutureReturnValueIgnored" this is deliberate
ctx.write(msg, promise.unvoid()).addListener(ChannelFutureListener.CLOSE);
return;
}
//"FutureReturnValueIgnored" this is deliberate
ctx.write(msg, promise);
if (!persistentConnection) {
return;
}
if (nonInformationalResponse) {
nonInformationalResponse = false;
pendingResponses -= 1;
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Decreasing pending responses count: {}"),
pendingResponses);
}
}
if (pipelined != null && !pipelined.isEmpty()) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Draining next pipelined " +
"HTTP request, pending responses count: {}, queued: {}"),
pendingResponses, pipelined.size());
}
ctx.executor().execute(this);
}
else {
IdleTimeoutHandler.addIdleTimeoutHandler(ctx.pipeline(), idleTimeout);
ctx.read();
}
}
@Override
public void run() {
Object next;
HttpRequest nextRequest = null;
while ((next = pipelined.peek()) != null) {
if (next instanceof HttpRequestHolder) {
if (nextRequest != null) {
return;
}
if (!persistentConnection) {
discard();
return;
}
HttpRequestHolder holder = (HttpRequestHolder) next;
nextRequest = holder.request;
finalizingResponse = false;
if (LAST_FLUSH_WHEN_NO_READ) {
ChannelOperations, ?> ops = ChannelOperations.get(ctx.channel());
if (ops instanceof HttpServerOperations) {
if (HttpServerOperations.log.isDebugEnabled()) {
HttpServerOperations.log.debug(format(ctx.channel(), "Last HTTP packet was sent, terminating the channel"));
}
((HttpServerOperations) ops).terminateInternal();
}
}
DecoderResult decoderResult = nextRequest.decoderResult();
if (decoderResult.isFailure()) {
sendDecodingFailures(decoderResult.cause(), nextRequest, holder.timestamp, null);
discard();
return;
}
HttpServerOperations ops;
ConnectionInfo connectionInfo = null;
try {
connectionInfo = ConnectionInfo.from(ctx.channel(),
nextRequest,
secure,
remoteAddress,
forwardedHeaderHandler);
ops = new HttpServerOperations(Connection.from(ctx.channel()),
listener,
nextRequest,
compress,
connectionInfo,
cookieDecoder,
cookieEncoder,
formDecoderProvider,
httpMessageLogFactory,
false,
mapHandle,
readTimeout,
requestTimeout,
secure,
holder.timestamp);
}
catch (RuntimeException e) {
holder.request.setDecoderResult(DecoderResult.failure(e.getCause() != null ? e.getCause() : e));
sendDecodingFailures(e, holder.request, holder.timestamp, connectionInfo);
return;
}
ops.bind();
listener.onStateChange(ops, ConnectionObserver.State.CONFIGURED);
pipelined.poll();
ctx.fireChannelRead(holder.request);
}
else {
ctx.fireChannelRead(pipelined.poll());
}
}
overflow = false;
}
@Override
public void handlerRemoved(ChannelHandlerContext ctx) {
discard();
}
final void discard() {
if (pipelined != null && !pipelined.isEmpty()) {
Object o;
while ((o = pipelined.poll()) != null) {
ReferenceCountUtil.release(o);
}
}
}
boolean shouldKeepAlive() {
return pendingResponses != 0 && persistentConnection;
}
/**
* Keep-alive only works if the client can detect when the message has ended without
* relying on the connection being closed.
*
*
*
* @param response The HttpResponse to check
*
* @return true if the response has a self defined message length.
*/
static boolean isSelfDefinedMessageLength(HttpResponse response) {
return isContentLengthSet(response) || isTransferEncodingChunked(response) || isMultipart(
response) || isInformational(response) || isNotModified(response) || isNoContent(response);
}
static boolean isInformational(HttpResponse response) {
return response.status()
.codeClass() == HttpStatusClass.INFORMATIONAL;
}
static boolean isNoContent(HttpResponse response) {
return HttpResponseStatus.NO_CONTENT.code() == response.status().code();
}
static boolean isNotModified(HttpResponse response) {
return HttpResponseStatus.NOT_MODIFIED.code() == response.status().code();
}
static boolean isMultipart(HttpResponse response) {
String contentType = response.headers()
.get(HttpHeaderNames.CONTENT_TYPE);
return contentType != null && contentType.regionMatches(true,
0,
MULTIPART_PREFIX,
0,
MULTIPART_PREFIX.length());
}
static final class HttpRequestHolder {
final HttpRequest request;
final ZonedDateTime timestamp;
HttpRequestHolder(HttpRequest request) {
this.request = request;
this.timestamp = ZonedDateTime.now(ReactorNetty.ZONE_ID_SYSTEM);
}
}
}