Skip to content

Commit

Permalink
Merge #3280 into 2.0.0-M4
Browse files Browse the repository at this point in the history
  • Loading branch information
violetagg committed Jun 4, 2024
2 parents f0120d2 + 8abd279 commit 48ff9db
Show file tree
Hide file tree
Showing 4 changed files with 62 additions and 14 deletions.
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2020-2023 VMware, Inc. or its affiliates, All Rights Reserved.
* Copyright (c) 2020-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.
Expand All @@ -19,11 +19,13 @@
import io.netty5.handler.codec.http.headers.HttpCookiePair;
import reactor.netty5.ReactorNetty;
import reactor.netty5.http.server.ConnectionInformation;
import reactor.netty5.internal.util.MapUtils;
import reactor.util.annotation.Nullable;

import java.net.SocketAddress;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.function.Supplier;
Expand Down Expand Up @@ -56,6 +58,20 @@ abstract class AbstractAccessLogArgProvider<SELF extends AbstractAccessLogArgPro
this.remoteAddress = remoteAddress;
}

AbstractAccessLogArgProvider(AbstractAccessLogArgProvider<?> copy) {
this.remoteAddress = copy.remoteAddress;
this.connectionInfo = copy.connectionInfo;
this.accessDateTime = copy.accessDateTime;
this.method = copy.method;
this.uri = copy.uri;
this.protocol = copy.protocol;
this.chunked = copy.chunked;
this.contentLength = copy.contentLength;
this.startTime = copy.startTime;
this.cookies = new HashMap<>(MapUtils.calculateInitialCapacity(copy.cookies.size()));
this.cookies.putAll(copy.cookies);
}

@Override
@Nullable
public ZonedDateTime accessDateTime() {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2020-2023 VMware, Inc. or its affiliates, All Rights Reserved.
* Copyright (c) 2020-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.
Expand Down Expand Up @@ -36,6 +36,12 @@ final class AccessLogArgProviderH1 extends AbstractAccessLogArgProvider<AccessLo
super(remoteAddress);
}

AccessLogArgProviderH1(AccessLogArgProviderH1 copy) {
super(copy);
this.request = copy.request;
this.response = copy.response;
}

AccessLogArgProviderH1 request(HttpServerRequest request) {
this.request = Objects.requireNonNull(request, "request");
onRequest();
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2018-2023 VMware, Inc. or its affiliates, All Rights Reserved.
* Copyright (c) 2018-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.
Expand Down Expand Up @@ -38,6 +38,9 @@
*/
final class AccessLogHandlerH1 extends BaseAccessLogHandler {

static final boolean LAST_FLUSH_WHEN_NO_READ = Boolean.parseBoolean(
System.getProperty("reactor.netty.http.server.lastFlushWhenNoRead", "false"));

AccessLogArgProviderH1 accessLogArgProvider;

AccessLogHandlerH1(@Nullable Function<AccessLogArgProvider, AccessLog> accessLog) {
Expand All @@ -56,6 +59,9 @@ public Future<Void> write(ChannelHandlerContext ctx, Object msg) {
if (accessLogArgProvider == null) {
accessLogArgProvider = new AccessLogArgProviderH1(ctx.channel().remoteAddress());
}
else {
accessLogArgProvider.clear();
}

ChannelOperations<?, ?> ops = ChannelOperations.get(ctx.channel());
if (ops instanceof HttpServerRequest) {
Expand All @@ -71,14 +77,14 @@ public Future<Void> write(ChannelHandlerContext ctx, Object msg) {
}
if (msg instanceof LastHttpContent<?> lastHttpContent) {
accessLogArgProvider.increaseContentLength(lastHttpContent.payload().readableBytes());
AccessLogArgProviderH1 copy = LAST_FLUSH_WHEN_NO_READ ? new AccessLogArgProviderH1(accessLogArgProvider) : null;
return ctx.write(msg)
.addListener(future -> {
if (future.isSuccess()) {
AccessLog log = accessLog.apply(accessLogArgProvider);
AccessLog log = copy != null ? accessLog.apply(copy) : accessLog.apply(accessLogArgProvider);
if (log != null) {
log.log();
}
accessLogArgProvider.clear();
}
});
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,16 +27,17 @@
import io.netty5.handler.codec.http.HttpUtil;
import io.netty5.handler.codec.http.LastHttpContent;
import io.netty5.util.concurrent.Future;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.EnumSource;
import org.junit.jupiter.params.provider.ValueSource;
import org.reactivestreams.Publisher;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.Signal;
import reactor.netty5.BaseHttpTest;
import reactor.netty5.Connection;
import reactor.netty5.DisposableServer;
import reactor.netty5.LogTracker;
import reactor.netty5.http.HttpProtocol;
import reactor.netty5.http.client.HttpClient;
import reactor.netty5.tcp.TcpClient;
Expand Down Expand Up @@ -232,16 +233,18 @@ void httpGetRespondsSendObject(HttpProtocol protocol) throws Exception {
assertThat(recorder.onTerminateIsReceived.get()).isEqualTo(1);
}

@Test
void httpPipeliningGetRespondsSendMono() throws Exception {
@ParameterizedTest
@ValueSource(booleans = {false, true})
void httpPipeliningGetRespondsSendMono(boolean enableAccessLog) throws Exception {
String oldValue = System.getProperty("reactor.netty5.http.server.lastFlushWhenNoRead", "false");
System.setProperty("reactor.netty5.http.server.lastFlushWhenNoRead", "true");
try {
String message = "\"GET /1 HTTP/1.1\" 200 1024";
try (LogTracker logTracker = new LogTracker("reactor.netty5.http.server.AccessLog", 16, message)) {
CountDownLatch latch = new CountDownLatch(64);
EventsRecorder recorder = new EventsRecorder(latch);
disposableServer = createServer(recorder, HttpProtocol.HTTP11,
r -> r.get("/1", (req, res) -> res.sendString(Mono.just(REPEAT).delayElement(Duration.ofMillis(10))
.doOnEach(recorder).doOnCancel(recorder))));
.doOnEach(recorder).doOnCancel(recorder))), enableAccessLog);

Connection client =
TcpClient.create()
Expand Down Expand Up @@ -277,22 +280,29 @@ void httpPipeliningGetRespondsSendMono() throws Exception {
assertThat(recorder.fullResponseIsSent.get()).isEqualTo(16);
assertThat(recorder.onCompleteIsReceived.get()).isEqualTo(16);
assertThat(recorder.onTerminateIsReceived.get()).isEqualTo(16);

if (enableAccessLog) {
assertThat(logTracker.latch.await(5, TimeUnit.SECONDS)).isTrue();
assertThat(logTracker.actualMessages).hasSize(16);
}
}
finally {
System.setProperty("reactor.netty5.http.server.lastFlushWhenNoRead", oldValue);
}
}

@Test
void httpPipeliningGetRespondsSendObject() throws Exception {
@ParameterizedTest
@ValueSource(booleans = {false, true})
void httpPipeliningGetRespondsSendObject(boolean enableAccessLog) throws Exception {
String oldValue = System.getProperty("reactor.netty5.http.server.lastFlushWhenNoRead", "false");
System.setProperty("reactor.netty5.http.server.lastFlushWhenNoRead", "true");
try {
String message = "\"GET /1 HTTP/1.1\" 200 1024";
try (LogTracker logTracker = new LogTracker("reactor.netty5.http.server.AccessLog", 16, message)) {
CountDownLatch latch = new CountDownLatch(64);
EventsRecorder recorder = new EventsRecorder(latch);
disposableServer = createServer(recorder, HttpProtocol.HTTP11,
r -> r.get("/1", (req, res) -> res.sendObject(res.alloc().copyOf(REPEAT, Charset.defaultCharset()))
.then().doOnEach(recorder).doOnCancel(recorder)));
.then().doOnEach(recorder).doOnCancel(recorder)), enableAccessLog);

Connection client =
TcpClient.create()
Expand Down Expand Up @@ -328,6 +338,11 @@ void httpPipeliningGetRespondsSendObject() throws Exception {
assertThat(recorder.fullResponseIsSent.get()).isEqualTo(16);
assertThat(recorder.onCompleteIsReceived.get()).isEqualTo(16);
assertThat(recorder.onTerminateIsReceived.get()).isEqualTo(16);

if (enableAccessLog) {
assertThat(logTracker.latch.await(5, TimeUnit.SECONDS)).isTrue();
assertThat(logTracker.actualMessages).hasSize(16);
}
}
finally {
System.setProperty("reactor.netty5.http.server.lastFlushWhenNoRead", oldValue);
Expand Down Expand Up @@ -469,6 +484,10 @@ void httpPostRespondsSendObject(HttpProtocol protocol) throws Exception {
}

static DisposableServer createServer(EventsRecorder recorder, HttpProtocol protocol, Consumer<? super HttpServerRoutes> routes) {
return createServer(recorder, protocol, routes, false);
}

static DisposableServer createServer(EventsRecorder recorder, HttpProtocol protocol, Consumer<? super HttpServerRoutes> routes, boolean enableAccessLog) {
return createServer()
.protocol(protocol)
.doOnChannelInit((obs, ch, addr) -> {
Expand All @@ -482,6 +501,7 @@ static DisposableServer createServer(EventsRecorder recorder, HttpProtocol proto
conn.channel().pipeline().addBefore(HttpTrafficHandler, "eventsRecorderHandler", new EventsRecorderHandler(recorder));
}
})
.accessLog(enableAccessLog)
.route(routes)
.bindNow();
}
Expand Down

0 comments on commit 48ff9db

Please sign in to comment.