mirror of
https://github.com/openjdk/jdk.git
synced 2026-01-28 03:58:21 +00:00
8376118: java/net/httpclient/StreamingBody.java fails intermittently on Windows
Reviewed-by: vyazici, jpai
This commit is contained in:
parent
de5c7a9e86
commit
8a9127fc2d
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2018, 2025, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2018, 2026, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -25,11 +25,12 @@
|
|||||||
* @test
|
* @test
|
||||||
* @summary Exercise a streaming subscriber ( InputStream ) without holding a
|
* @summary Exercise a streaming subscriber ( InputStream ) without holding a
|
||||||
* strong (or any ) reference to the client.
|
* strong (or any ) reference to the client.
|
||||||
|
* @key randomness
|
||||||
* @library /test/lib /test/jdk/java/net/httpclient/lib
|
* @library /test/lib /test/jdk/java/net/httpclient/lib
|
||||||
* @build jdk.httpclient.test.lib.common.HttpServerAdapters jdk.test.lib.net.SimpleSSLContext
|
* @build jdk.httpclient.test.lib.common.HttpServerAdapters jdk.test.lib.net.SimpleSSLContext
|
||||||
* @run testng/othervm
|
* @run junit/othervm
|
||||||
* -Djdk.httpclient.HttpClient.log=trace,headers,requests
|
* -Djdk.httpclient.HttpClient.log=trace,headers,requests
|
||||||
* StreamingBody
|
* ${test.main.class}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
@ -40,13 +41,15 @@ import java.net.http.HttpClient;
|
|||||||
import java.net.http.HttpRequest;
|
import java.net.http.HttpRequest;
|
||||||
import java.net.http.HttpResponse;
|
import java.net.http.HttpResponse;
|
||||||
import java.net.http.HttpResponse.BodyHandlers;
|
import java.net.http.HttpResponse.BodyHandlers;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
import javax.net.ssl.SSLContext;
|
import javax.net.ssl.SSLContext;
|
||||||
import jdk.httpclient.test.lib.common.HttpServerAdapters;
|
import jdk.httpclient.test.lib.common.HttpServerAdapters;
|
||||||
|
import jdk.test.lib.RandomFactory;
|
||||||
import jdk.test.lib.net.SimpleSSLContext;
|
import jdk.test.lib.net.SimpleSSLContext;
|
||||||
import org.testng.annotations.AfterTest;
|
|
||||||
import org.testng.annotations.BeforeTest;
|
|
||||||
import org.testng.annotations.DataProvider;
|
|
||||||
import org.testng.annotations.Test;
|
|
||||||
import static java.lang.System.out;
|
import static java.lang.System.out;
|
||||||
import static java.net.http.HttpClient.Version.HTTP_1_1;
|
import static java.net.http.HttpClient.Version.HTTP_1_1;
|
||||||
import static java.net.http.HttpClient.Version.HTTP_2;
|
import static java.net.http.HttpClient.Version.HTTP_2;
|
||||||
@ -55,8 +58,21 @@ import static java.net.http.HttpOption.Http3DiscoveryMode.HTTP_3_URI_ONLY;
|
|||||||
import static java.net.http.HttpOption.H3_DISCOVERY;
|
import static java.net.http.HttpOption.H3_DISCOVERY;
|
||||||
import static java.nio.charset.StandardCharsets.UTF_8;
|
import static java.nio.charset.StandardCharsets.UTF_8;
|
||||||
import static java.net.http.HttpClient.Builder.NO_PROXY;
|
import static java.net.http.HttpClient.Builder.NO_PROXY;
|
||||||
import static org.testng.Assert.assertEquals;
|
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.AfterAll;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Assumptions;
|
||||||
|
import org.junit.jupiter.api.BeforeAll;
|
||||||
|
import org.junit.jupiter.api.TestInstance;
|
||||||
|
import org.junit.jupiter.api.extension.BeforeEachCallback;
|
||||||
|
import org.junit.jupiter.api.extension.ExtensionContext;
|
||||||
|
import org.junit.jupiter.api.extension.RegisterExtension;
|
||||||
|
import org.junit.jupiter.api.extension.TestWatcher;
|
||||||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
import org.junit.jupiter.params.provider.MethodSource;
|
||||||
|
|
||||||
|
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
|
||||||
public class StreamingBody implements HttpServerAdapters {
|
public class StreamingBody implements HttpServerAdapters {
|
||||||
|
|
||||||
private static final SSLContext sslContext = SimpleSSLContext.findSSLContext();
|
private static final SSLContext sslContext = SimpleSSLContext.findSSLContext();
|
||||||
@ -71,10 +87,105 @@ public class StreamingBody implements HttpServerAdapters {
|
|||||||
String https2URI;
|
String https2URI;
|
||||||
String http3URI;
|
String http3URI;
|
||||||
|
|
||||||
|
static final AtomicLong clientCount = new AtomicLong();
|
||||||
|
static final AtomicLong serverCount = new AtomicLong();
|
||||||
|
static final ConcurrentMap<String, Throwable> FAILURES = new ConcurrentHashMap<>();
|
||||||
|
private static boolean stopAfterFirstFailure() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static final long start = System.nanoTime();
|
||||||
|
public static String now() {
|
||||||
|
long now = System.nanoTime() - start;
|
||||||
|
long secs = now / 1000_000_000;
|
||||||
|
long mill = (now % 1000_000_000) / 1000_000;
|
||||||
|
long nan = now % 1000_000;
|
||||||
|
return String.format("[%d s, %d ms, %d ns] ", secs, mill, nan);
|
||||||
|
}
|
||||||
|
|
||||||
|
final static class TestStopper implements TestWatcher, BeforeEachCallback {
|
||||||
|
final AtomicReference<String> failed = new AtomicReference<>();
|
||||||
|
TestStopper() { }
|
||||||
|
@Override
|
||||||
|
public void testFailed(ExtensionContext context, Throwable cause) {
|
||||||
|
if (stopAfterFirstFailure()) {
|
||||||
|
String msg = "Aborting due to: " + cause;
|
||||||
|
failed.compareAndSet(null, msg);
|
||||||
|
FAILURES.putIfAbsent(context.getDisplayName(), cause);
|
||||||
|
System.out.printf("%nTEST FAILED: %s%s%n\tAborting due to %s%n%n",
|
||||||
|
now(), context.getDisplayName(), cause);
|
||||||
|
System.err.printf("%nTEST FAILED: %s%s%n\tAborting due to %s%n%n",
|
||||||
|
now(), context.getDisplayName(), cause);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void beforeEach(ExtensionContext context) {
|
||||||
|
String msg = failed.get();
|
||||||
|
Assumptions.assumeTrue(msg == null, msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@RegisterExtension
|
||||||
|
static final TestStopper stopper = new TestStopper();
|
||||||
|
|
||||||
|
/// The GCTrigger triggers GC at random intervals to
|
||||||
|
/// help garbage collecting HttpClient intances. This test
|
||||||
|
/// wants to verify that HttpClient instances which are no
|
||||||
|
/// longer strongly referenced are not garbage collected
|
||||||
|
/// before pending HTTP requests are finished. The test
|
||||||
|
/// creates many client instances (up to 500) and relies
|
||||||
|
/// on the GC to collect them, since it does not want to
|
||||||
|
/// keep a strong reference, and therefore cannot not call
|
||||||
|
/// close(). This can put extra load on the machine since
|
||||||
|
/// we can't (and don't want to) control when the GC will
|
||||||
|
/// intervene. The purpose of this class is to trigger the
|
||||||
|
/// GC at random intervals to 1. help garbage collect client
|
||||||
|
/// instances earlier, thus reducing the load on the machine,
|
||||||
|
/// and 2. potentially trigger bugs if the client gets
|
||||||
|
/// inadvertently GC'ed before the request is finished
|
||||||
|
/// (which is the bug we're testing for here).
|
||||||
|
static class GCTrigger {
|
||||||
|
private final long gcinterval;
|
||||||
|
private final Thread runner;
|
||||||
|
private volatile boolean stop;
|
||||||
|
private final static Random RANDOM = RandomFactory.getRandom();
|
||||||
|
|
||||||
|
GCTrigger(long gcinterval) {
|
||||||
|
this.gcinterval = Math.clamp(gcinterval, 100, Long.MAX_VALUE/2);
|
||||||
|
runner = Thread.ofPlatform().daemon().unstarted(this::loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loop() {
|
||||||
|
long min = gcinterval / 2;
|
||||||
|
long max = gcinterval + min;
|
||||||
|
while (!stop) {
|
||||||
|
try {
|
||||||
|
Thread.sleep(RANDOM.nextLong(min, max));
|
||||||
|
} catch (InterruptedException x) {
|
||||||
|
stop = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
out.println(now() + "triggering gc");
|
||||||
|
System.gc();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void start() {
|
||||||
|
runner.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void stop() throws InterruptedException {
|
||||||
|
stop = true;
|
||||||
|
runner.interrupt();
|
||||||
|
runner.join();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static GCTrigger gcTrigger;
|
||||||
static final String MESSAGE = "StreamingBody message body";
|
static final String MESSAGE = "StreamingBody message body";
|
||||||
static final int ITERATIONS = 100;
|
static final int ITERATIONS = 100;
|
||||||
|
|
||||||
@DataProvider(name = "positive")
|
|
||||||
public Object[][] positive() {
|
public Object[][] positive() {
|
||||||
return new Object[][] {
|
return new Object[][] {
|
||||||
{ http3URI, },
|
{ http3URI, },
|
||||||
@ -94,17 +205,28 @@ public class StreamingBody implements HttpServerAdapters {
|
|||||||
return builder;
|
return builder;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(dataProvider = "positive")
|
@ParameterizedTest
|
||||||
|
@MethodSource("positive")
|
||||||
void test(String uriString) throws Exception {
|
void test(String uriString) throws Exception {
|
||||||
out.printf("%n---- starting (%s) ----%n", uriString);
|
out.printf("%n%s---- starting (%s) ----%n", now(), uriString);
|
||||||
URI uri = URI.create(uriString);
|
URI uri = URI.create(uriString);
|
||||||
HttpRequest request = newRequestBuilder(uri).build();
|
HttpRequest request = newRequestBuilder(uri).build();
|
||||||
|
|
||||||
for (int i=0; i< ITERATIONS; i++) {
|
for (int i=0; i< ITERATIONS; i++) {
|
||||||
out.println("iteration: " + i);
|
try {
|
||||||
|
out.println(now() + "iteration: " + i);
|
||||||
var builder = uriString.contains("/http3/")
|
var builder = uriString.contains("/http3/")
|
||||||
? newClientBuilderForH3()
|
? newClientBuilderForH3()
|
||||||
: HttpClient.newBuilder();
|
: HttpClient.newBuilder();
|
||||||
|
clientCount.incrementAndGet();
|
||||||
|
|
||||||
|
// we want to relinquish the reference to the HttpClient facade
|
||||||
|
// as soon as possible. We're using `ofInputStream()` because
|
||||||
|
// the HttpResponse will be returned almost immediately, before
|
||||||
|
// the response is read. Similarly we use sendAsync() because
|
||||||
|
// this will return a CompletableFuture and not wait for the
|
||||||
|
// request to complete within a method called on the client
|
||||||
|
// facade.
|
||||||
HttpResponse<InputStream> response = builder
|
HttpResponse<InputStream> response = builder
|
||||||
.sslContext(sslContext)
|
.sslContext(sslContext)
|
||||||
.proxy(NO_PROXY)
|
.proxy(NO_PROXY)
|
||||||
@ -116,50 +238,85 @@ public class StreamingBody implements HttpServerAdapters {
|
|||||||
out.println("Got response: " + response);
|
out.println("Got response: " + response);
|
||||||
out.println("Got body Path: " + body);
|
out.println("Got body Path: " + body);
|
||||||
|
|
||||||
assertEquals(response.statusCode(), 200);
|
assertEquals(200, response.statusCode());
|
||||||
assertEquals(body, MESSAGE);
|
assertEquals(MESSAGE, body);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
String msg = "%stest(%s)[%s] failed: %s"
|
||||||
|
.formatted(now(), uriString, i, t);
|
||||||
|
out.println(msg);
|
||||||
|
throw new AssertionError(msg, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// -- Infrastructure
|
// -- Infrastructure
|
||||||
|
|
||||||
@BeforeTest
|
@BeforeAll
|
||||||
public void setup() throws Exception {
|
public void setup() throws Exception {
|
||||||
httpTestServer = HttpTestServer.create(HTTP_1_1);
|
httpTestServer = HttpTestServer.create(HTTP_1_1);
|
||||||
httpTestServer.addHandler(new MessageHandler(), "/http1/streamingbody/");
|
httpTestServer.addHandler(new MessageHandler(), "/http1/streamingbody/");
|
||||||
httpURI = "http://" + httpTestServer.serverAuthority() + "/http1/streamingbody/w";
|
httpURI = "http://" + httpTestServer.serverAuthority() + "/http1/streamingbody/w";
|
||||||
|
serverCount.incrementAndGet();
|
||||||
|
|
||||||
httpsTestServer = HttpTestServer.create(HTTP_1_1, sslContext);
|
httpsTestServer = HttpTestServer.create(HTTP_1_1, sslContext);
|
||||||
httpsTestServer.addHandler(new MessageHandler(),"/https1/streamingbody/");
|
httpsTestServer.addHandler(new MessageHandler(),"/https1/streamingbody/");
|
||||||
httpsURI = "https://" + httpsTestServer.serverAuthority() + "/https1/streamingbody/x";
|
httpsURI = "https://" + httpsTestServer.serverAuthority() + "/https1/streamingbody/x";
|
||||||
|
serverCount.incrementAndGet();
|
||||||
|
|
||||||
http2TestServer = HttpTestServer.create(HTTP_2);
|
http2TestServer = HttpTestServer.create(HTTP_2);
|
||||||
http2TestServer.addHandler(new MessageHandler(), "/http2/streamingbody/");
|
http2TestServer.addHandler(new MessageHandler(), "/http2/streamingbody/");
|
||||||
http2URI = "http://" + http2TestServer.serverAuthority() + "/http2/streamingbody/y";
|
http2URI = "http://" + http2TestServer.serverAuthority() + "/http2/streamingbody/y";
|
||||||
|
serverCount.incrementAndGet();
|
||||||
|
|
||||||
https2TestServer = HttpTestServer.create(HTTP_2, sslContext);
|
https2TestServer = HttpTestServer.create(HTTP_2, sslContext);
|
||||||
https2TestServer.addHandler(new MessageHandler(), "/https2/streamingbody/");
|
https2TestServer.addHandler(new MessageHandler(), "/https2/streamingbody/");
|
||||||
https2URI = "https://" + https2TestServer.serverAuthority() + "/https2/streamingbody/z";
|
https2URI = "https://" + https2TestServer.serverAuthority() + "/https2/streamingbody/z";
|
||||||
|
serverCount.incrementAndGet();
|
||||||
|
|
||||||
http3TestServer = HttpTestServer.create(HTTP_3_URI_ONLY, sslContext);
|
http3TestServer = HttpTestServer.create(HTTP_3_URI_ONLY, sslContext);
|
||||||
http3TestServer.addHandler(new MessageHandler(), "/http3/streamingbody/");
|
http3TestServer.addHandler(new MessageHandler(), "/http3/streamingbody/");
|
||||||
http3URI = "https://" + http3TestServer.serverAuthority() + "/http3/streamingbody/z";
|
http3URI = "https://" + http3TestServer.serverAuthority() + "/http3/streamingbody/z";
|
||||||
|
serverCount.incrementAndGet();
|
||||||
|
|
||||||
|
gcTrigger = new GCTrigger(500);
|
||||||
|
|
||||||
httpTestServer.start();
|
httpTestServer.start();
|
||||||
httpsTestServer.start();
|
httpsTestServer.start();
|
||||||
http2TestServer.start();
|
http2TestServer.start();
|
||||||
https2TestServer.start();
|
https2TestServer.start();
|
||||||
http3TestServer.start();
|
http3TestServer.start();
|
||||||
|
gcTrigger.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
@AfterTest
|
@AfterAll
|
||||||
public void teardown() throws Exception {
|
public void teardown() throws Exception {
|
||||||
|
try {
|
||||||
httpTestServer.stop();
|
httpTestServer.stop();
|
||||||
httpsTestServer.stop();
|
httpsTestServer.stop();
|
||||||
http2TestServer.stop();
|
http2TestServer.stop();
|
||||||
https2TestServer.stop();
|
https2TestServer.stop();
|
||||||
http3TestServer.stop();
|
http3TestServer.stop();
|
||||||
|
gcTrigger.stop();
|
||||||
|
} finally {
|
||||||
|
printFailedTests();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static final void printFailedTests() {
|
||||||
|
out.println("\n=========================");
|
||||||
|
try {
|
||||||
|
out.printf("%n%sCreated %s servers and %s clients%n",
|
||||||
|
now(), serverCount.get(), clientCount.get());
|
||||||
|
if (FAILURES.isEmpty()) return;
|
||||||
|
out.println("Failed tests: ");
|
||||||
|
FAILURES.entrySet().forEach((e) -> {
|
||||||
|
out.printf("\t%s: %s%n", e.getKey(), e.getValue());
|
||||||
|
e.getValue().printStackTrace(out);
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
out.println("\n=========================\n");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static class MessageHandler implements HttpTestHandler {
|
static class MessageHandler implements HttpTestHandler {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user