jdk-24/test/jdk/java/net/httpclient/AggregateRequestBodyTest.java

840 lines
34 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8252374
* @library /test/lib http2/server
* @build jdk.test.lib.net.SimpleSSLContext HttpServerAdapters
* ReferenceTracker AggregateRequestBodyTest
* @modules java.base/sun.net.www.http
* java.net.http/jdk.internal.net.http.common
* java.net.http/jdk.internal.net.http.frame
* java.net.http/jdk.internal.net.http.hpack
* @run testng/othervm -Djdk.internal.httpclient.debug=true
* -Djdk.httpclient.HttpClient.log=requests,responses,errors
* AggregateRequestBodyTest
* @summary Tests HttpRequest.BodyPublishers::concat
*/
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpRequest.BodyPublisher;
import java.net.http.HttpRequest.BodyPublishers;
import java.net.http.HttpResponse;
import java.net.http.HttpResponse.BodyHandlers;
import java.nio.ByteBuffer;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import java.util.concurrent.Flow;
import java.util.concurrent.Flow.Subscriber;
import java.util.concurrent.Flow.Subscription;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicLong;
import java.util.function.Consumer;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.LongStream;
import java.util.stream.Stream;
import javax.net.ssl.SSLContext;
import com.sun.net.httpserver.HttpServer;
import com.sun.net.httpserver.HttpsConfigurator;
import com.sun.net.httpserver.HttpsServer;
import jdk.test.lib.net.SimpleSSLContext;
import org.testng.Assert;
import org.testng.ITestContext;
import org.testng.annotations.AfterClass;
import org.testng.annotations.AfterTest;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
import static java.lang.System.out;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertFalse;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.expectThrows;
public class AggregateRequestBodyTest implements HttpServerAdapters {
SSLContext sslContext;
HttpTestServer http1TestServer; // HTTP/1.1 ( http )
HttpTestServer https1TestServer; // HTTPS/1.1 ( https )
HttpTestServer http2TestServer; // HTTP/2 ( h2c )
HttpTestServer https2TestServer; // HTTP/2 ( h2 )
String http1URI;
String https1URI;
String http2URI;
String https2URI;
static final int RESPONSE_CODE = 200;
static final int ITERATION_COUNT = 4;
static final Class<IllegalArgumentException> IAE = IllegalArgumentException.class;
static final Class<CompletionException> CE = CompletionException.class;
// a shared executor helps reduce the amount of threads created by the test
static final Executor executor = new TestExecutor(Executors.newCachedThreadPool());
static final ConcurrentMap<String, Throwable> FAILURES = new ConcurrentHashMap<>();
static volatile boolean tasksFailed;
static final AtomicLong serverCount = new AtomicLong();
static final AtomicLong clientCount = new AtomicLong();
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 ReferenceTracker TRACKER = ReferenceTracker.INSTANCE;
private volatile HttpClient sharedClient;
static class TestExecutor implements Executor {
final AtomicLong tasks = new AtomicLong();
Executor executor;
TestExecutor(Executor executor) {
this.executor = executor;
}
@Override
public void execute(Runnable command) {
long id = tasks.incrementAndGet();
executor.execute(() -> {
try {
command.run();
} catch (Throwable t) {
tasksFailed = true;
System.out.printf(now() + "Task %s failed: %s%n", id, t);
System.err.printf(now() + "Task %s failed: %s%n", id, t);
FAILURES.putIfAbsent("Task " + id, t);
throw t;
}
});
}
}
protected boolean stopAfterFirstFailure() {
return Boolean.getBoolean("jdk.internal.httpclient.debug");
}
@BeforeMethod
void beforeMethod(ITestContext context) {
if (stopAfterFirstFailure() && context.getFailedTests().size() > 0) {
throw new RuntimeException("some tests failed");
}
}
@AfterClass
static final void printFailedTests() {
out.println("\n=========================");
try {
out.printf("%n%sCreated %d servers and %d 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);
e.getValue().printStackTrace();
});
if (tasksFailed) {
System.out.println("WARNING: Some tasks failed");
}
} finally {
out.println("\n=========================\n");
}
}
private String[] uris() {
return new String[] {
http1URI,
https1URI,
http2URI,
https2URI,
};
}
static AtomicLong URICOUNT = new AtomicLong();
@DataProvider(name = "variants")
public Object[][] variants(ITestContext context) {
if (stopAfterFirstFailure() && context.getFailedTests().size() > 0) {
return new Object[0][];
}
String[] uris = uris();
Object[][] result = new Object[uris.length * 2][];
int i = 0;
for (boolean sameClient : List.of(false, true)) {
for (String uri : uris()) {
result[i++] = new Object[]{uri, sameClient};
}
}
assert i == uris.length * 2;
return result;
}
private HttpClient makeNewClient() {
clientCount.incrementAndGet();
HttpClient client = HttpClient.newBuilder()
.proxy(HttpClient.Builder.NO_PROXY)
.executor(executor)
.sslContext(sslContext)
.build();
return TRACKER.track(client);
}
HttpClient newHttpClient(boolean share) {
if (!share) return makeNewClient();
HttpClient shared = sharedClient;
if (shared != null) return shared;
synchronized (this) {
shared = sharedClient;
if (shared == null) {
shared = sharedClient = makeNewClient();
}
return shared;
}
}
static final List<String> BODIES = List.of(
"Lorem ipsum",
"dolor sit amet",
"consectetur adipiscing elit, sed do eiusmod tempor",
"quis nostrud exercitation ullamco",
"laboris nisi",
"ut",
"aliquip ex ea commodo consequat." +
"Duis aute irure dolor in reprehenderit in voluptate velit esse" +
"cillum dolore eu fugiat nulla pariatur.",
"Excepteur sint occaecat cupidatat non proident."
);
static BodyPublisher[] publishers(String... content) {
if (content == null) return null;
BodyPublisher[] result = new BodyPublisher[content.length];
for (int i=0; i < content.length ; i++) {
result[i] = content[i] == null ? null : BodyPublishers.ofString(content[i]);
}
return result;
}
static String[] strings(String... s) {
return s;
}
@DataProvider(name = "sparseContent")
Object[][] nulls() {
return new Object[][] {
{"null array", null},
{"null element", strings((String)null)},
{"null first element", strings(null, "one")},
{"null second element", strings( "one", null)},
{"null third element", strings( "one", "two", null)},
{"null fourth element", strings( "one", "two", "three", null)},
{"null random element", strings( "one", "two", "three", null, "five")},
};
}
static List<Long> lengths(long... lengths) {
return LongStream.of(lengths)
.mapToObj(Long::valueOf)
.collect(Collectors.toList());
}
@DataProvider(name = "contentLengths")
Object[][] contentLengths() {
return new Object[][] {
{-1, lengths(-1)},
{-42, lengths(-42)},
{42, lengths(42)},
{42, lengths(10, 0, 20, 0, 12)},
{-1, lengths(10, 0, 20, -1, 12)},
{-1, lengths(-1, 0, 20, 10, 12)},
{-1, lengths(10, 0, 20, 12, -1)},
{-1, lengths(10, 0, 20, -10, 12)},
{-1, lengths(-10, 0, 20, 10, 12)},
{-1, lengths(10, 0, 20, 12, -10)},
{-1, lengths(10, 0, Long.MIN_VALUE, -1, 12)},
{-1, lengths(-1, 0, Long.MIN_VALUE, 10, 12)},
{-1, lengths(10, Long.MIN_VALUE, 20, 12, -1)},
{Long.MAX_VALUE, lengths(10, Long.MAX_VALUE - 42L, 20, 0, 12)},
{-1, lengths(10, Long.MAX_VALUE - 40L, 20, 0, 12)},
{-1, lengths(10, Long.MAX_VALUE - 12L, 20, 0, 12)},
{-1, lengths(10, Long.MAX_VALUE/2L, Long.MAX_VALUE/2L + 1L, 0, 12)},
{-1, lengths(10, Long.MAX_VALUE/2L, -1, Long.MAX_VALUE/2L + 1L, 12)},
{-1, lengths(10, Long.MAX_VALUE, 12, Long.MAX_VALUE, 20)},
{-1, lengths(10, Long.MAX_VALUE, Long.MAX_VALUE, 12, 20)},
{-1, lengths(0, Long.MAX_VALUE, Long.MAX_VALUE, 12, 20)},
{-1, lengths(Long.MAX_VALUE, Long.MAX_VALUE, 12, 0, 20)}
};
}
@DataProvider(name="negativeRequests")
Object[][] negativeRequests() {
return new Object[][] {
{0L}, {-1L}, {-2L}, {Long.MIN_VALUE + 1L}, {Long.MIN_VALUE}
};
}
static class ContentLengthPublisher implements BodyPublisher {
final long length;
ContentLengthPublisher(long length) {
this.length = length;
}
@Override
public long contentLength() {
return length;
}
@Override
public void subscribe(Subscriber<? super ByteBuffer> subscriber) {
}
static ContentLengthPublisher[] of(List<Long> lengths) {
return lengths.stream()
.map(ContentLengthPublisher::new)
.toArray(ContentLengthPublisher[]::new);
}
}
/**
* A dummy publisher that allows to call onError on its subscriber (or not...).
*/
static class PublishWithError implements BodyPublisher {
final ConcurrentHashMap<Subscriber<?>, ErrorSubscription> subscribers = new ConcurrentHashMap<>();
final long length;
final List<String> content;
final int errorAt;
final Supplier<? extends Throwable> errorSupplier;
PublishWithError(List<String> content, int errorAt, Supplier<? extends Throwable> supplier) {
this.content = content;
this.errorAt = errorAt;
this.errorSupplier = supplier;
length = content.stream().mapToInt(String::length).sum();
}
boolean hasErrors() {
return errorAt < content.size();
}
@Override
public long contentLength() {
return length;
}
@Override
public void subscribe(Subscriber<? super ByteBuffer> subscriber) {
ErrorSubscription subscription = new ErrorSubscription(subscriber);
subscribers.put(subscriber, subscription);
subscriber.onSubscribe(subscription);
}
class ErrorSubscription implements Flow.Subscription {
volatile boolean cancelled;
volatile int at;
final Subscriber<? super ByteBuffer> subscriber;
ErrorSubscription(Subscriber<? super ByteBuffer> subscriber) {
this.subscriber = subscriber;
}
@Override
public void request(long n) {
while (!cancelled && --n >= 0 && at < Math.min(errorAt+1, content.size())) {
if (at++ == errorAt) {
subscriber.onError(errorSupplier.get());
return;
} else if (at <= content.size()){
subscriber.onNext(ByteBuffer.wrap(
content.get(at-1).getBytes()));
if (at == content.size()) {
subscriber.onComplete();
return;
}
}
}
}
@Override
public void cancel() {
cancelled = true;
}
}
}
static class RequestSubscriber implements Flow.Subscriber<ByteBuffer> {
CompletableFuture<Subscription> subscriptionCF = new CompletableFuture<>();
ConcurrentLinkedDeque<ByteBuffer> items = new ConcurrentLinkedDeque<>();
CompletableFuture<List<ByteBuffer>> resultCF = new CompletableFuture<>();
@Override
public void onSubscribe(Subscription subscription) {
this.subscriptionCF.complete(subscription);
}
@Override
public void onNext(ByteBuffer item) {
items.addLast(item);
}
@Override
public void onError(Throwable throwable) {
resultCF.completeExceptionally(throwable);
}
@Override
public void onComplete() {
resultCF.complete(items.stream().collect(Collectors.toUnmodifiableList()));
}
CompletableFuture<List<ByteBuffer>> resultCF() { return resultCF; }
}
static String stringFromBuffer(ByteBuffer buffer) {
byte[] bytes = new byte[buffer.remaining()];
buffer.get(bytes);
return new String(bytes);
}
String stringFromBytes(Stream<ByteBuffer> buffers) {
return buffers.map(AggregateRequestBodyTest::stringFromBuffer)
.collect(Collectors.joining());
}
static PublishWithError withNoError(String content) {
return new PublishWithError(List.of(content), 1,
() -> new AssertionError("Should not happen!"));
}
static PublishWithError withNoError(List<String> content) {
return new PublishWithError(content, content.size(),
() -> new AssertionError("Should not happen!"));
}
@Test(dataProvider = "sparseContent") // checks that NPE is thrown
public void testNullPointerException(String description, String[] content) {
BodyPublisher[] publishers = publishers(content);
Assert.assertThrows(NullPointerException.class, () -> BodyPublishers.concat(publishers));
}
// Verifies that an empty array creates a "noBody" publisher
@Test
public void testEmpty() {
BodyPublisher publisher = BodyPublishers.concat();
RequestSubscriber subscriber = new RequestSubscriber();
assertEquals(publisher.contentLength(), 0);
publisher.subscribe(subscriber);
subscriber.subscriptionCF.thenAccept(s -> s.request(1));
List<ByteBuffer> result = subscriber.resultCF.join();
assertEquals(result, List.of());
assertTrue(subscriber.items.isEmpty());;
}
// verifies that error emitted by upstream publishers are propagated downstream.
@Test(dataProvider = "sparseContent") // nulls are replaced with error publisher
public void testOnError(String description, String[] content) {
final RequestSubscriber subscriber = new RequestSubscriber();
final PublishWithError errorPublisher;
final BodyPublisher[] publishers;
String result = BODIES.stream().collect(Collectors.joining());
if (content == null) {
content = List.of(result).toArray(String[]::new);
errorPublisher = new PublishWithError(BODIES, BODIES.size(),
() -> new AssertionError("Unexpected!!"));
publishers = List.of(errorPublisher).toArray(new BodyPublisher[0]);
description = "No error";
} else {
publishers = publishers(content);
description = description.replace("null", "error at");
errorPublisher = new PublishWithError(BODIES, 2, () -> new Exception("expected"));
}
result = "";
boolean hasErrors = false;
for (int i=0; i < content.length; i++) {
if (content[i] == null) {
publishers[i] = errorPublisher;
if (hasErrors) continue;
if (!errorPublisher.hasErrors()) {
result = result + errorPublisher
.content.stream().collect(Collectors.joining());
} else {
result = result + errorPublisher.content
.stream().limit(errorPublisher.errorAt)
.collect(Collectors.joining());
result = result + "<error>";
hasErrors = true;
}
} else if (!hasErrors) {
result = result + content[i];
}
}
BodyPublisher publisher = BodyPublishers.concat(publishers);
publisher.subscribe(subscriber);
subscriber.subscriptionCF.thenAccept(s -> s.request(Long.MAX_VALUE));
if (errorPublisher.hasErrors()) {
CompletionException ce = expectThrows(CompletionException.class,
() -> subscriber.resultCF.join());
out.println(description + ": got expected " + ce);
assertEquals(ce.getCause().getClass(), Exception.class);
assertEquals(stringFromBytes(subscriber.items.stream()) + "<error>", result);
} else {
assertEquals(stringFromBytes(subscriber.resultCF.join().stream()), result);
out.println(description + ": got expected result: " + result);
}
}
// Verifies that if an upstream publisher has an unknown length, the
// aggregate publisher will have an unknown length as well. Otherwise
// the length should be known.
@Test(dataProvider = "sparseContent") // nulls are replaced with unknown length
public void testUnknownContentLength(String description, String[] content) {
if (content == null) {
content = BODIES.toArray(String[]::new);
description = "BODIES (known length)";
} else {
description = description.replace("null", "length(-1)");
}
BodyPublisher[] publishers = publishers(content);
BodyPublisher nolength = new BodyPublisher() {
final BodyPublisher missing = BodyPublishers.ofString("missing");
@Override
public long contentLength() { return -1; }
@Override
public void subscribe(Subscriber<? super ByteBuffer> subscriber) {
missing.subscribe(subscriber);
}
};
long length = 0;
for (int i=0; i < content.length; i++) {
if (content[i] == null) {
publishers[i] = nolength;
length = -1;
} else if (length >= 0) {
length += content[i].length();
}
}
out.printf("testUnknownContentLength(%s): %d%n", description, length);
BodyPublisher publisher = BodyPublishers.concat(publishers);
assertEquals(publisher.contentLength(), length,
description.replace("null", "length(-1)"));
}
private static final Throwable completionCause(CompletionException x) {
while (x.getCause() instanceof CompletionException) {
x = (CompletionException)x.getCause();
}
return x.getCause();
}
@Test(dataProvider = "negativeRequests")
public void testNegativeRequest(long n) {
assert n <= 0 : "test for negative request called with n > 0 : " + n;
BodyPublisher[] publishers = ContentLengthPublisher.of(List.of(1L, 2L, 3L));
BodyPublisher publisher = BodyPublishers.concat(publishers);
RequestSubscriber subscriber = new RequestSubscriber();
publisher.subscribe(subscriber);
Subscription subscription = subscriber.subscriptionCF.join();
subscription.request(n);
CompletionException expected = expectThrows(CE, () -> subscriber.resultCF.join());
Throwable cause = completionCause(expected);
if (cause instanceof IllegalArgumentException) {
System.out.printf("Got expected IAE for %d: %s%n", n, cause);
} else {
throw new AssertionError("Unexpected exception: " + cause,
(cause == null) ? expected : cause);
}
}
static BodyPublisher[] ofStrings(String... strings) {
return Stream.of(strings).map(BodyPublishers::ofString).toArray(BodyPublisher[]::new);
}
@Test
public void testPositiveRequests() {
// A composite array of publishers
BodyPublisher[] publishers = Stream.of(
Stream.of(ofStrings("Lorem", " ", "ipsum", " ")),
Stream.of(BodyPublishers.concat(ofStrings("dolor", " ", "sit", " ", "amet", ", "))),
Stream.<BodyPublisher>of(withNoError(List.of("consectetur", " ", "adipiscing"))),
Stream.of(ofStrings(" ")),
Stream.of(BodyPublishers.concat(ofStrings("elit", ".")))
).flatMap((s) -> s).toArray(BodyPublisher[]::new);
BodyPublisher publisher = BodyPublishers.concat(publishers);
// Test that we can request all 13 items in a single request call.
RequestSubscriber requestSubscriber1 = new RequestSubscriber();
publisher.subscribe(requestSubscriber1);
Subscription subscription1 = requestSubscriber1.subscriptionCF.join();
subscription1.request(16);
assertTrue(requestSubscriber1.resultCF().isDone());
List<ByteBuffer> list1 = requestSubscriber1.resultCF().join();
String result1 = stringFromBytes(list1.stream());
assertEquals(result1, "Lorem ipsum dolor sit amet, consectetur adipiscing elit.");
System.out.println("Got expected sentence with one request: \"%s\"".formatted(result1));
// Test that we can split our requests call any which way we want
// (whether in the 'middle of a publisher' or at the boundaries.
RequestSubscriber requestSubscriber2 = new RequestSubscriber();
publisher.subscribe(requestSubscriber2);
Subscription subscription2 = requestSubscriber2.subscriptionCF.join();
subscription2.request(1);
assertFalse(requestSubscriber2.resultCF().isDone());
subscription2.request(10);
assertFalse(requestSubscriber2.resultCF().isDone());
subscription2.request(4);
assertFalse(requestSubscriber2.resultCF().isDone());
subscription2.request(1);
assertTrue(requestSubscriber2.resultCF().isDone());
List<ByteBuffer> list2 = requestSubscriber2.resultCF().join();
String result2 = stringFromBytes(list2.stream());
assertEquals(result2, "Lorem ipsum dolor sit amet, consectetur adipiscing elit.");
System.out.println("Got expected sentence with 4 requests: \"%s\"".formatted(result1));
}
@Test(dataProvider = "contentLengths")
public void testContentLength(long expected, List<Long> lengths) {
BodyPublisher[] publishers = ContentLengthPublisher.of(lengths);
BodyPublisher aggregate = BodyPublishers.concat(publishers);
assertEquals(aggregate.contentLength(), expected,
"Unexpected result for %s".formatted(lengths));
}
// Verifies that cancelling the subscription ensure that downstream
// publishers are no longer subscribed etc...
@Test
public void testCancel() {
BodyPublisher[] publishers = BODIES.stream()
.map(BodyPublishers::ofString)
.toArray(BodyPublisher[]::new);
BodyPublisher publisher = BodyPublishers.concat(publishers);
assertEquals(publisher.contentLength(),
BODIES.stream().mapToInt(String::length).sum());
Map<RequestSubscriber, String> subscribers = new LinkedHashMap<>();
for (int n=0; n < BODIES.size(); n++) {
String description = String.format(
"cancel after %d/%d onNext() invocations",
n, BODIES.size());
RequestSubscriber subscriber = new RequestSubscriber();
publisher.subscribe(subscriber);
Subscription subscription = subscriber.subscriptionCF.join();
subscribers.put(subscriber, description);
// receive half the data
for (int i = 0; i < n; i++) {
subscription.request(1);
ByteBuffer buffer = subscriber.items.pop();
}
// cancel subscription
subscription.cancel();
// request the rest...
subscription.request(Long.MAX_VALUE);
}
CompletableFuture[] results = subscribers.keySet()
.stream().map(RequestSubscriber::resultCF)
.toArray(CompletableFuture[]::new);
CompletableFuture<?> any = CompletableFuture.anyOf(results);
// subscription was cancelled, so nothing should be received...
try {
TimeoutException x = Assert.expectThrows(TimeoutException.class,
() -> any.get(5, TimeUnit.SECONDS));
out.println("Got expected " + x);
} finally {
subscribers.keySet().stream()
.filter(rs -> rs.resultCF.isDone())
.forEach(rs -> System.err.printf(
"Failed: %s completed with %s",
subscribers.get(rs), rs.resultCF));
}
Consumer<RequestSubscriber> check = (rs) -> {
Assert.assertTrue(rs.items.isEmpty(), subscribers.get(rs) + " has items");
Assert.assertFalse(rs.resultCF.isDone(), subscribers.get(rs) + " was not cancelled");
out.println(subscribers.get(rs) + ": PASSED");
};
subscribers.keySet().stream().forEach(check);
}
// Verifies that cancelling the subscription is propagated downstream
@Test
public void testCancelSubscription() {
PublishWithError upstream = new PublishWithError(BODIES, BODIES.size(),
() -> new AssertionError("should not come here"));
BodyPublisher publisher = BodyPublishers.concat(upstream);
assertEquals(publisher.contentLength(),
BODIES.stream().mapToInt(String::length).sum());
Map<RequestSubscriber, String> subscribers = new LinkedHashMap<>();
for (int n=0; n < BODIES.size(); n++) {
String description = String.format(
"cancel after %d/%d onNext() invocations",
n, BODIES.size());
RequestSubscriber subscriber = new RequestSubscriber();
publisher.subscribe(subscriber);
Subscription subscription = subscriber.subscriptionCF.join();
subscribers.put(subscriber, description);
// receive half the data
for (int i = 0; i < n; i++) {
subscription.request(1);
ByteBuffer buffer = subscriber.items.pop();
}
// cancel subscription
subscription.cancel();
// request the rest...
subscription.request(Long.MAX_VALUE);
assertTrue(upstream.subscribers.get(subscriber).cancelled,
description + " upstream subscription not cancelled");
out.println(description + " upstream subscription was properly cancelled");
}
CompletableFuture[] results = subscribers.keySet()
.stream().map(RequestSubscriber::resultCF)
.toArray(CompletableFuture[]::new);
CompletableFuture<?> any = CompletableFuture.anyOf(results);
// subscription was cancelled, so nothing should be received...
try {
TimeoutException x = Assert.expectThrows(TimeoutException.class,
() -> any.get(5, TimeUnit.SECONDS));
out.println("Got expected " + x);
} finally {
subscribers.keySet().stream()
.filter(rs -> rs.resultCF.isDone())
.forEach(rs -> System.err.printf(
"Failed: %s completed with %s",
subscribers.get(rs), rs.resultCF));
}
Consumer<RequestSubscriber> check = (rs) -> {
Assert.assertTrue(rs.items.isEmpty(), subscribers.get(rs) + " has items");
Assert.assertFalse(rs.resultCF.isDone(), subscribers.get(rs) + " was not cancelled");
out.println(subscribers.get(rs) + ": PASSED");
};
subscribers.keySet().stream().forEach(check);
}
@Test(dataProvider = "variants")
public void test(String uri, boolean sameClient) throws Exception {
System.out.println("Request to " + uri);
HttpClient client = newHttpClient(sameClient);
BodyPublisher publisher = BodyPublishers.concat(
BODIES.stream()
.map(BodyPublishers::ofString)
.toArray(HttpRequest.BodyPublisher[]::new)
);
HttpRequest request = HttpRequest.newBuilder(URI.create(uri))
.POST(publisher)
.build();
for (int i = 0; i < ITERATION_COUNT; i++) {
System.out.println("Iteration: " + i);
HttpResponse<String> response = client.send(request, BodyHandlers.ofString());
int expectedResponse = RESPONSE_CODE;
if (response.statusCode() != expectedResponse)
throw new RuntimeException("wrong response code " + Integer.toString(response.statusCode()));
assertEquals(response.body(), BODIES.stream().collect(Collectors.joining()));
}
System.out.println("test: DONE");
}
@BeforeTest
public void setup() throws Exception {
sslContext = new SimpleSSLContext().get();
if (sslContext == null)
throw new AssertionError("Unexpected null sslContext");
HttpTestHandler handler = new HttpTestEchoHandler();
InetSocketAddress loopback = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0);
HttpServer http1 = HttpServer.create(loopback, 0);
http1TestServer = HttpTestServer.of(http1);
http1TestServer.addHandler(handler, "/http1/echo/");
http1URI = "http://" + http1TestServer.serverAuthority() + "/http1/echo/x";
HttpsServer https1 = HttpsServer.create(loopback, 0);
https1.setHttpsConfigurator(new HttpsConfigurator(sslContext));
https1TestServer = HttpTestServer.of(https1);
https1TestServer.addHandler(handler, "/https1/echo/");
https1URI = "https://" + https1TestServer.serverAuthority() + "/https1/echo/x";
// HTTP/2
http2TestServer = HttpTestServer.of(new Http2TestServer("localhost", false, 0));
http2TestServer.addHandler(handler, "/http2/echo/");
http2URI = "http://" + http2TestServer.serverAuthority() + "/http2/echo/x";
https2TestServer = HttpTestServer.of(new Http2TestServer("localhost", true, sslContext));
https2TestServer.addHandler(handler, "/https2/echo/");
https2URI = "https://" + https2TestServer.serverAuthority() + "/https2/echo/x";
serverCount.addAndGet(4);
http1TestServer.start();
https1TestServer.start();
http2TestServer.start();
https2TestServer.start();
}
@AfterTest
public void teardown() throws Exception {
String sharedClientName =
sharedClient == null ? null : sharedClient.toString();
sharedClient = null;
Thread.sleep(100);
AssertionError fail = TRACKER.check(500);
try {
http1TestServer.stop();
https1TestServer.stop();
http2TestServer.stop();
https2TestServer.stop();
} finally {
if (fail != null) {
if (sharedClientName != null) {
System.err.println("Shared client name is: " + sharedClientName);
}
throw fail;
}
}
}
}