Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

4.x: Resource limits #7302

Merged
merged 6 commits into from
Aug 4, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions bom/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -1185,6 +1185,11 @@
<artifactId>helidon-nima-testing-junit5-webserver</artifactId>
<version>${helidon.version}</version>
</dependency>
<dependency>
<groupId>io.helidon.nima.testing.junit5</groupId>
<artifactId>helidon-nima-testing-junit5-http2</artifactId>
<version>${helidon.version}</version>
</dependency>
<dependency>
<groupId>io.helidon.nima.testing.junit5</groupId>
<artifactId>helidon-nima-testing-junit5-websocket</artifactId>
Expand Down
63 changes: 33 additions & 30 deletions common/http/src/main/java/io/helidon/common/http/Http.java
Original file line number Diff line number Diff line change
Expand Up @@ -24,13 +24,12 @@
import java.time.format.DateTimeParseException;
import java.time.format.SignStyle;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.Timer;
import java.util.TimerTask;
import java.util.function.Predicate;

import io.helidon.common.buffers.Ascii;
Expand Down Expand Up @@ -1767,24 +1766,23 @@ public static final class DateTime {
*/
private static final Map<Long, String> MONTH_NAME_3D;

private static volatile ZonedDateTime time;
private static volatile String rfc1123String;
private static volatile byte[] http1valueBytes;

static {
Map<Long, String> map = new HashMap<>();
map.put(1L, "Jan");
map.put(2L, "Feb");
map.put(3L, "Mar");
map.put(4L, "Apr");
map.put(5L, "May");
map.put(6L, "Jun");
map.put(7L, "Jul");
map.put(8L, "Aug");
map.put(9L, "Sep");
map.put(10L, "Oct");
map.put(11L, "Nov");
map.put(12L, "Dec");
MONTH_NAME_3D = Collections.unmodifiableMap(map);
MONTH_NAME_3D = Map.ofEntries(Map.entry(1L, "Jan"),
Map.entry(2L, "Feb"),
Map.entry(3L, "Mar"),
Map.entry(4L, "Apr"),
Map.entry(5L, "May"),
Map.entry(6L, "Jun"),
Map.entry(7L, "Jul"),
Map.entry(8L, "Aug"),
Map.entry(9L, "Sep"),
Map.entry(10L, "Oct"),
Map.entry(11L, "Nov"),
Map.entry(12L, "Dec"));

// manually code maps to ensure correct data always used
// (locale data can be changed by application code)
Expand Down Expand Up @@ -1851,18 +1849,13 @@ public static final class DateTime {

update();

Thread thread = new Thread(() -> {
while (true) {
update();
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
return;
}
}
}, "helidon-http-timer");
thread.setDaemon(true);
thread.start();
// start a timer, scheduled every second to update server time (we do not need better precision)
new Timer("helidon-http-timer", true)
.schedule(new TimerTask() {
public void run() {
update();
}
}, 1000, 1000);
}

private DateTime() {
Expand Down Expand Up @@ -1890,6 +1883,15 @@ public static ZonedDateTime parse(String text) {
}
}

/**
* Last recorded timestamp.
*
* @return timestamp
*/
public static ZonedDateTime timestamp() {
return time;
}
tomas-langer marked this conversation as resolved.
Show resolved Hide resolved

/**
* Get current time as RFC-1123 string.
*
Expand All @@ -1910,7 +1912,8 @@ public static byte[] http1Bytes() {
}

static void update() {
rfc1123String = ZonedDateTime.now().format(RFC_1123_DATE_TIME);
time = ZonedDateTime.now();
rfc1123String = time.format(RFC_1123_DATE_TIME);
http1valueBytes = (rfc1123String + "\r\n").getBytes(StandardCharsets.US_ASCII);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -233,34 +233,34 @@ public void assertConnectionIsClosed() {
* @return the exact string returned by webserver (including {@code HTTP/1.1 200 OK} line for instance)
*/
public String sendAndReceive(Http.Method method, String payload) {
return sendAndReceive("/", method, payload);
return sendAndReceive(method, "/", payload);
}

/**
* A helper method that sends the given payload at the given path with the provided method and headers to the server.
*
* @param path the path to access
* @param method the http method
* @param path the path to access
* @param payload the payload to send (must be without the newlines;
* otherwise it's not a valid payload)
* @return the exact string returned by webserver (including {@code HTTP/1.1 200 OK} line for instance)
*/
public String sendAndReceive(String path, Http.Method method, String payload) {
return sendAndReceive(path, method, payload, Collections.emptyList());
public String sendAndReceive(Http.Method method, String path, String payload) {
return sendAndReceive(method, path, payload, Collections.emptyList());
}

/**
* A helper method that sends the given payload at the given path with the provided method to the server.
*
* @param path the path to access
* @param method the http method
* @param path the path to access
* @param payload the payload to send (must be without the newlines;
* otherwise it's not a valid payload)
* @param headers HTTP request headers
* @return the exact string returned by webserver (including {@code HTTP/1.1 200 OK} line for instance)
*/
public String sendAndReceive(String path,
Http.Method method,
public String sendAndReceive(Http.Method method,
String path,
String payload,
Iterable<String> headers) {

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,14 @@

import java.lang.System.Logger.Level;
import java.nio.ByteBuffer;
import java.time.Duration;
import java.time.ZonedDateTime;
import java.util.Optional;
import java.util.concurrent.Semaphore;

import io.helidon.common.buffers.BufferData;
import io.helidon.common.buffers.DataReader;
import io.helidon.common.http.Http;
import io.helidon.nima.webserver.ConnectionContext;
import io.helidon.nima.webserver.spi.ServerConnection;
import io.helidon.nima.websocket.WsCloseCodes;
Expand Down Expand Up @@ -49,25 +53,45 @@ class TyrusConnection implements ServerConnection, WsSession {
private final WebSocketEngine.UpgradeInfo upgradeInfo;
private final TyrusListener listener;

private volatile Thread myThread;
private volatile boolean canRun = true;
private volatile boolean readingNetwork;
private volatile ZonedDateTime lastRequestTimestamp;

TyrusConnection(ConnectionContext ctx, WebSocketEngine.UpgradeInfo upgradeInfo) {
this.ctx = ctx;
this.upgradeInfo = upgradeInfo;
this.listener = new TyrusListener();
this.lastRequestTimestamp = Http.DateTime.timestamp();
}

@Override
public void handle() {
public void handle(Semaphore requestSemaphore) {
myThread = Thread.currentThread();
DataReader dataReader = ctx.dataReader();
listener.onOpen(this);
while (true) {
if (requestSemaphore.tryAcquire()) {
try {
BufferData buffer = dataReader.readBuffer();
listener.onMessage(this, buffer, true);
} catch (Exception e) {
listener.onError(this, e);
listener.onClose(this, WsCloseCodes.UNEXPECTED_CONDITION, e.getMessage());
return;
while (canRun) {
try {
readingNetwork = true;
BufferData buffer = dataReader.readBuffer();
readingNetwork = false;
lastRequestTimestamp = Http.DateTime.timestamp();
listener.onMessage(this, buffer, true);
lastRequestTimestamp = Http.DateTime.timestamp();
} catch (Exception e) {
listener.onError(this, e);
listener.onClose(this, WsCloseCodes.UNEXPECTED_CONDITION, e.getMessage());
return;
}
}
listener.onClose(this, WsCloseCodes.NORMAL_CLOSE, "Idle timeout");
} finally {
requestSemaphore.release();
}
} else {
listener.onClose(this, WsCloseCodes.TRY_AGAIN_LATER, "Too Many Concurrent Requests");
}
}

Expand Down Expand Up @@ -106,6 +130,28 @@ public Optional<String> subProtocol() {
return Optional.empty();
}

@Override
public Duration idleTime() {
return Duration.between(lastRequestTimestamp, Http.DateTime.timestamp());
}

@Override
public void close(boolean interrupt) {
// either way, finish
this.canRun = false;

if (interrupt) {
// interrupt regardless of current state
if (myThread != null) {
myThread.interrupt();
}
} else if (readingNetwork) {
// only interrupt when not processing a request (there is a chance of a race condition, this edge case
// is ignored
myThread.interrupt();
}
}

class TyrusListener implements WsListener {
private static final int MAX_RETRIES = 5;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@
import io.helidon.nima.webserver.http1.Http1Route;

import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;
Expand Down Expand Up @@ -80,8 +81,8 @@ class Http2WebClientTest {
.build());

Http2WebClientTest(WebServer server) {
this.plainPort = server.port();
this.tlsPort = server.port("https");
plainPort = server.port();
tlsPort = server.port("https");
}

@SetUpServer
Expand Down Expand Up @@ -233,6 +234,7 @@ void clientPost(String clientType, LazyValue<Http2Client> client) {
}
}

@Disabled("Failing intermittently, to be investigated")
@ParameterizedTest(name = "{0}")
@MethodSource("clientTypes")
void multiplexParallelStreamsGet(String clientType, LazyValue<Http2Client> client)
Expand Down
Loading