diff --git a/src/java.net.http/share/classes/java/net/http/HttpRequest.java b/src/java.net.http/share/classes/java/net/http/HttpRequest.java
index 6472bd4a77c..5dc0486826a 100644
--- a/src/java.net.http/share/classes/java/net/http/HttpRequest.java
+++ b/src/java.net.http/share/classes/java/net/http/HttpRequest.java
@@ -39,6 +39,7 @@ import java.util.Iterator;
import java.util.Objects;
import java.util.Optional;
import java.util.concurrent.Flow;
+import java.util.function.BiPredicate;
import java.util.function.Supplier;
import jdk.internal.net.http.HttpRequestBuilderImpl;
@@ -90,8 +91,9 @@ public abstract class HttpRequest {
/**
* A builder of {@linkplain HttpRequest HTTP requests}.
*
- *
Instances of {@code HttpRequest.Builder} are created by calling {@link
- * HttpRequest#newBuilder(URI)} or {@link HttpRequest#newBuilder()}.
+ *
Instances of {@code HttpRequest.Builder} are created by calling
+ * {@link HttpRequest#newBuilder()}, {@link HttpRequest#newBuilder(URI)},
+ * or {@link HttpRequest#newBuilder(HttpRequest, BiPredicate)}.
*
*
The builder can be used to configure per-request state, such as: the
* request URI, the request method (default is GET unless explicitly set),
@@ -303,6 +305,74 @@ public abstract class HttpRequest {
return new HttpRequestBuilderImpl(uri);
}
+ /**
+ * Creates a {@code Builder} whose initial state is copied from an existing
+ * {@code HttpRequest}.
+ *
+ *
This builder can be used to build an {@code HttpRequest}, equivalent
+ * to the original, while allowing amendment of the request state prior to
+ * construction - for example, adding additional headers.
+ *
+ *
The {@code filter} is applied to each header name value pair as they
+ * are copied from the given request. When completed, only headers that
+ * satisfy the condition as laid out by the {@code filter} will be present
+ * in the {@code Builder} returned from this method.
+ *
+ * @apiNote
+ * The following scenarios demonstrate typical use-cases of the filter.
+ * Given an {@code HttpRequest} request:
+ *
+ *
+ * - Retain all headers:
+ *
{@code HttpRequest.newBuilder(request, (n, v) -> true)}
+ *
+ * - Remove all headers:
+ *
{@code HttpRequest.newBuilder(request, (n, v) -> false)}
+ *
+ * - Remove a particular header (e.g. Foo-Bar):
+ *
{@code HttpRequest.newBuilder(request, (name, value) -> !name.equalsIgnoreCase("Foo-Bar"))}
+ *
+ *
+ * @param request the original request
+ * @param filter a header filter
+ * @return a new request builder
+ * @throws IllegalArgumentException if a new builder cannot be seeded from
+ * the given request (for instance, if the request contains illegal
+ * parameters)
+ * @since 16
+ */
+ public static Builder newBuilder(HttpRequest request, BiPredicate filter) {
+ Objects.requireNonNull(request);
+ Objects.requireNonNull(filter);
+
+ final HttpRequest.Builder builder = HttpRequest.newBuilder();
+ builder.uri(request.uri());
+ builder.expectContinue(request.expectContinue());
+
+ // Filter unwanted headers
+ HttpHeaders headers = HttpHeaders.of(request.headers().map(), filter);
+ headers.map().forEach((name, values) ->
+ values.forEach(value -> builder.header(name, value)));
+
+ request.version().ifPresent(builder::version);
+ request.timeout().ifPresent(builder::timeout);
+ var method = request.method();
+ request.bodyPublisher().ifPresentOrElse(
+ // if body is present, set it
+ bodyPublisher -> builder.method(method, bodyPublisher),
+ // otherwise, the body is absent, special case for GET/DELETE,
+ // or else use empty body
+ () -> {
+ switch (method) {
+ case "GET" -> builder.GET();
+ case "DELETE" -> builder.DELETE();
+ default -> builder.method(method, HttpRequest.BodyPublishers.noBody());
+ }
+ }
+ );
+ return builder;
+ }
+
/**
* Creates an {@code HttpRequest} builder.
*
diff --git a/test/jdk/java/net/httpclient/HttpRequestNewBuilderTest.java b/test/jdk/java/net/httpclient/HttpRequestNewBuilderTest.java
new file mode 100644
index 00000000000..c1c7e3432b6
--- /dev/null
+++ b/test/jdk/java/net/httpclient/HttpRequestNewBuilderTest.java
@@ -0,0 +1,434 @@
+/*
+* 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.
+*/
+
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.net.http.HttpClient.Version;
+import java.net.http.HttpHeaders;
+import java.net.http.HttpRequest;
+import java.net.http.HttpResponse.BodySubscriber;
+import java.net.http.HttpResponse.BodySubscribers;
+import java.nio.ByteBuffer;
+import java.time.Duration;
+import java.util.Arrays;
+import java.util.List;
+import java.util.Map;
+import java.util.Optional;
+import java.util.concurrent.Flow;
+import java.util.function.BiConsumer;
+import java.util.function.BiPredicate;
+import static java.net.http.HttpClient.Version.HTTP_2;
+import static java.net.http.HttpClient.Version.HTTP_1_1;
+import static java.nio.charset.StandardCharsets.UTF_8;
+
+import org.testng.annotations.Test;
+import org.testng.annotations.DataProvider;
+import static org.testng.Assert.assertEquals;
+import static org.testng.Assert.assertFalse;
+import static org.testng.Assert.assertThrows;
+import static org.testng.Assert.assertTrue;
+import static org.testng.Assert.fail;
+
+/**
+* @test
+* @bug 8252304
+* @summary HttpRequest.newBuilder(HttpRequest) API and behaviour checks
+* @run testng/othervm HttpRequestNewBuilderTest
+*/
+public class HttpRequestNewBuilderTest {
+ static final Class NPE = NullPointerException.class;
+ static final Class IAE = IllegalArgumentException.class;
+
+ record NamedAssertion(String name, BiConsumer test) { }
+
+ List REQUEST_ASSERTIONS = List.of(
+ new NamedAssertion("uri", (r1, r2) -> assertEquals(r1.uri(), r2.uri())),
+ new NamedAssertion("timeout", (r1, r2) -> assertEquals(r1.timeout(), r2.timeout())),
+ new NamedAssertion("version", (r1, r2) -> assertEquals(r1.version(), r2.version())),
+ new NamedAssertion("headers", (r1, r2) -> assertEquals(r1.headers(), r2.headers())),
+ new NamedAssertion("expectContinue", (r1, r2) -> assertEquals(r1.expectContinue(), r2.expectContinue())),
+ new NamedAssertion("method", (r1, r2) -> {
+ assertEquals(r1.method(), r2.method());
+ assertBodyPublisherEqual(r1, r2);
+ })
+ );
+
+ @DataProvider(name = "testRequests")
+ public Object[][] variants() {
+ return new Object[][]{
+ { HttpRequest.newBuilder(URI.create("https://uri-1/")).build() },
+ { HttpRequest.newBuilder(URI.create("https://version-1/")).version(HTTP_1_1).build() },
+ { HttpRequest.newBuilder(URI.create("https://version-2/")).version(HTTP_2).build() },
+ { HttpRequest.newBuilder(URI.create("https://timeout-1/")).timeout(Duration.ofSeconds(30)).build() },
+ { HttpRequest.newBuilder(URI.create("https://header-1/")).header("testName1", "testValue1").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-2/"))
+ .headers("testName1", "testValue1", "a", "1", "b", "2", "c", "3", "d", "4").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-3/"))
+ .headers("testName1", "testValue1", "a", "1", "b", "2", "c", "3", "d", "4", "testName2", "testValue2").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-4/"))
+ .headers("a", "1", "b", "2", "c", "3", "d", "4", "testName1", "testValue1").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-5/"))
+ .headers( "a", "1", "b", "2", "testName1", "testValue1", "testName2", "testValue2", "c", "3", "d", "4").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-6/"))
+ .headers("testName1", "testValue1")
+ .headers("testName1", "v")
+ .headers("testName1", "w")
+ .headers("testName1", "x")
+ .headers("testName1", "y")
+ .headers("testName1", "z").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-7/"))
+ .headers("testName1", "testValue1")
+ .headers("testName1", "v")
+ .headers("testName1", "w")
+ .headers("testName1", "x")
+ .headers("testName1", "y")
+ .headers("testName1", "z")
+ .headers("testName1", "testValue2").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-8/"))
+ .headers("testName1", "v")
+ .headers("testName1", "w")
+ .headers("testName1", "x")
+ .headers("testName1", "y")
+ .headers("testName1", "z")
+ .headers("testName1", "testValue1").build() },
+ { HttpRequest.newBuilder(URI.create("https://header-9/"))
+ .headers("testName1", "v")
+ .headers("testName1", "w")
+ .headers("testName1", "testValue1")
+ .headers("testName1", "testValue2")
+ .headers("testName1", "x")
+ .headers("testName1", "y")
+ .headers("testName1", "z").build() },
+ // dedicated method
+ { HttpRequest.newBuilder(URI.create("https://method-1/")).GET().build() },
+ { HttpRequest.newBuilder(URI.create("https://method-2/")).DELETE().build() },
+ { HttpRequest.newBuilder(URI.create("https://method-3/")).POST(HttpRequest.BodyPublishers.ofString("testData")).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-4/")).PUT(HttpRequest.BodyPublishers.ofString("testData")).build() },
+ // method w/body
+ { HttpRequest.newBuilder(URI.create("https://method-5/")).method("GET", HttpRequest.BodyPublishers.ofString("testData")).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-6/")).method("DELETE", HttpRequest.BodyPublishers.ofString("testData")).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-7/")).method("POST", HttpRequest.BodyPublishers.ofString("testData")).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-8/")).method("PUT", HttpRequest.BodyPublishers.ofString("testData")).build() },
+ // method w/o body
+ { HttpRequest.newBuilder(URI.create("https://method-9/")).method("GET", HttpRequest.BodyPublishers.noBody()).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-10/")).method("DELETE", HttpRequest.BodyPublishers.noBody()).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-11/")).method("POST", HttpRequest.BodyPublishers.noBody()).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-12/")).method("PUT", HttpRequest.BodyPublishers.noBody()).build() },
+ // user defined methods w/ & w/o body
+ { HttpRequest.newBuilder(URI.create("https://method-13/")).method("TEST", HttpRequest.BodyPublishers.noBody()).build() },
+ { HttpRequest.newBuilder(URI.create("https://method-14/")).method("TEST", HttpRequest.BodyPublishers.ofString("testData")).build() },
+
+ { HttpRequest.newBuilder(URI.create("https://all-fields-1/")).GET().expectContinue(true).version(HTTP_2)
+ .timeout(Duration.ofSeconds(1)).header("testName1", "testValue1").build() },
+ };
+ }
+
+ // test methods
+ void assertBodyPublisherEqual(HttpRequest r1, HttpRequest r2) {
+ if (r1.bodyPublisher().isPresent()) {
+ assertTrue(r2.bodyPublisher().isPresent());
+ var bp1 = r1.bodyPublisher().get();
+ var bp2 = r2.bodyPublisher().get();
+
+ assertEquals(bp1.getClass(), bp2.getClass());
+ assertEquals(bp1.contentLength(), bp2.contentLength());
+
+ final class TestSubscriber implements Flow.Subscriber {
+ final BodySubscriber s;
+ TestSubscriber(BodySubscriber s) { this.s = s; }
+ @Override
+ public void onSubscribe(Flow.Subscription subscription) { s.onSubscribe(subscription); }
+ @Override
+ public void onNext(ByteBuffer item) { s.onNext(List.of(item)); }
+ @Override
+ public void onError(Throwable throwable) { fail("TestSubscriber failed"); }
+ @Override
+ public void onComplete() { s.onComplete(); }
+ }
+ var bs1 = BodySubscribers.ofString(UTF_8);
+ bp1.subscribe(new TestSubscriber(bs1));
+ var b1 = bs1.getBody().toCompletableFuture().join().getBytes();
+
+ var bs2 = BodySubscribers.ofString(UTF_8);
+ bp2.subscribe(new TestSubscriber(bs2));
+ var b2 = bs2.getBody().toCompletableFuture().join().getBytes();
+
+ assertEquals(b1, b2);
+ } else {
+ assertFalse(r2.bodyPublisher().isPresent());
+ }
+ }
+
+ void assertAllOtherElementsEqual(HttpRequest r1, HttpRequest r2, String... except) {
+ var ignoreList = Arrays.asList(except);
+ REQUEST_ASSERTIONS.stream()
+ .filter(a -> !ignoreList.contains(a.name()))
+ .forEach(testCaseAssertion -> testCaseAssertion.test().accept(r1, r2));
+ }
+
+ void testBodyPublisher(String methodName, HttpRequest request) {
+ // method w/body
+ var r = HttpRequest.newBuilder(request, (n, v) -> true)
+ .method(methodName, HttpRequest.BodyPublishers.ofString("testData"))
+ .build();
+ assertEquals(r.method(), methodName);
+ assertTrue(r.bodyPublisher().isPresent());
+ assertEquals(r.bodyPublisher().get().contentLength(), 8);
+ assertAllOtherElementsEqual(r, request, "method");
+
+ // method w/o body
+ var noBodyPublisher = HttpRequest.BodyPublishers.noBody();
+ var r1 = HttpRequest.newBuilder(request, (n, v) -> true)
+ .method(methodName, noBodyPublisher)
+ .build();
+ assertEquals(r1.method(), methodName);
+ assertTrue(r1.bodyPublisher().isPresent());
+ assertEquals(r1.bodyPublisher().get(), noBodyPublisher);
+ assertAllOtherElementsEqual(r1, request, "method");
+ }
+
+ @Test
+ public void testNull() {
+ var r = HttpRequest.newBuilder(URI.create("https://foobar/")).build();
+ assertThrows(NPE, () -> HttpRequest.newBuilder(r, null));
+ assertThrows(NPE, () -> HttpRequest.newBuilder(null, (n, v) -> true));
+ assertThrows(NPE, () -> HttpRequest.newBuilder(null, null));
+ }
+
+ @Test(dataProvider = "testRequests")
+ void testBuilder(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true).build();
+ assertEquals(r, request);
+ assertAllOtherElementsEqual(r, request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testURI(HttpRequest request) {
+ URI newURI = URI.create("http://www.newURI.com/");
+ var r = HttpRequest.newBuilder(request, (n, v) -> true).uri(newURI).build();
+
+ assertEquals(r.uri(), newURI);
+ assertAllOtherElementsEqual(r, request, "uri");
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testTimeout(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true).timeout(Duration.ofSeconds(2)).build();
+
+ assertEquals(r.timeout().get().getSeconds(), 2);
+ assertAllOtherElementsEqual(r, request, "timeout");
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testVersion(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true).version(HTTP_1_1).build();
+
+ assertEquals(r.version().get(), HTTP_1_1);
+ assertAllOtherElementsEqual(r, request, "version");
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testGET(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true)
+ .GET()
+ .build();
+ assertEquals(r.method(), "GET");
+ assertTrue(r.bodyPublisher().isEmpty());
+ assertAllOtherElementsEqual(r, request, "method");
+
+ testBodyPublisher("GET", request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testDELETE(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true)
+ .DELETE()
+ .build();
+ assertEquals(r.method(), "DELETE");
+ assertTrue(r.bodyPublisher().isEmpty());
+ assertAllOtherElementsEqual(r, request, "method");
+
+ testBodyPublisher("DELETE", request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testPOST(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true)
+ .POST(HttpRequest.BodyPublishers.ofString("testData"))
+ .build();
+ assertEquals(r.method(), "POST");
+ assertTrue(r.bodyPublisher().isPresent());
+ assertEquals(r.bodyPublisher().get().contentLength(), 8);
+ assertAllOtherElementsEqual(r, request, "method");
+
+ testBodyPublisher("POST", request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testPUT(HttpRequest request) {
+ var r = HttpRequest.newBuilder(request, (n, v) -> true)
+ .PUT(HttpRequest.BodyPublishers.ofString("testData"))
+ .build();
+ assertEquals(r.method(), "PUT");
+ assertTrue(r.bodyPublisher().isPresent());
+ assertEquals(r.bodyPublisher().get().contentLength(), 8);
+ assertAllOtherElementsEqual(r, request, "method");
+
+ testBodyPublisher("PUT", request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testUserDefinedMethod(HttpRequest request) {
+ testBodyPublisher("TEST", request);
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testAddHeader(HttpRequest request) {
+ BiPredicate filter = (n, v) -> true;
+
+ var r = HttpRequest.newBuilder(request, filter).headers("newName", "newValue").build();
+ assertEquals(r.headers().firstValue("newName").get(), "newValue");
+ assertEquals(r.headers().allValues("newName").size(), 1);
+ assertAllOtherElementsEqual(r, request, "headers");
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testRemoveHeader(HttpRequest request) {
+ if(!request.headers().map().isEmpty()) {
+ assertTrue(request.headers().map().containsKey("testName1"));
+ }
+ BiPredicate filter = (n, v) -> !n.equalsIgnoreCase("testName1");
+
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertFalse(r.headers().map().containsKey("testName1"));
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testRemoveSingleHeaderValue(HttpRequest request) {
+ if(!request.headers().map().isEmpty()) {
+ assertTrue(request.headers().allValues("testName1").contains("testValue1"));
+ }
+ BiPredicate filter = (n, v) ->
+ !(n.equalsIgnoreCase("testName1") && v.equals("testValue1"));
+
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertFalse(r.headers().map().containsValue("testValue1"));
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testRemoveMultipleHeaders(HttpRequest request) {
+ BiPredicate isTestName1Value1 = (n ,v) ->
+ n.equalsIgnoreCase("testName1") && v.equals("testValue1");
+ BiPredicate isTestName2Value2 = (n ,v) ->
+ n.equalsIgnoreCase("testName2") && v.equals("testValue2");
+ var filter = (isTestName1Value1.or(isTestName2Value2)).negate();
+
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+
+ BiPredicate filter1 = (n, v) ->
+ !(n.equalsIgnoreCase("testName1") && (v.equals("testValue1") || v.equals("testValue2")));
+
+ var r1 = HttpRequest.newBuilder(request, filter1).build();
+ assertEquals(r1.headers().map(), HttpHeaders.of(request.headers().map(), filter1).map());
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testRemoveAllHeaders(HttpRequest request) {
+ if (!request.headers().map().isEmpty()) {
+ BiPredicate filter = (n, v) -> false;
+
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertTrue(r.headers().map().isEmpty());
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+ }
+ }
+
+ @Test(dataProvider = "testRequests")
+ public void testRetainAllHeaders(HttpRequest request) {
+ if (!request.headers().map().isEmpty()) {
+ BiPredicate filter = (n, v) -> true;
+
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertFalse(r.headers().map().isEmpty());
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+ }
+ }
+
+ @Test
+ public void testHeaderExample() {
+ var request = HttpRequest.newBuilder(URI.create("https://example/"))
+ .header("Foo-Bar", "baz").build();
+
+ BiPredicate filter = (n, v) -> !n.equalsIgnoreCase("Foo-Bar");
+ var r = HttpRequest.newBuilder(request, filter).build();
+ assertFalse(r.headers().map().containsKey("Foo-Bar"));
+ assertEquals(r.headers().map(), HttpHeaders.of(request.headers().map(), filter).map());
+ }
+
+ @Test
+ public void testInvalidMethod() throws URISyntaxException {
+ URI testURI = new URI("http://www.foo.com/");
+ var r = new HttpRequest() {
+ @Override
+ public Optional bodyPublisher() { return Optional.empty(); }
+ @Override
+ public String method() { return "CONNECT"; }
+ @Override
+ public Optional timeout() { return Optional.empty(); }
+ @Override
+ public boolean expectContinue() { return false; }
+ @Override
+ public URI uri() { return testURI; }
+ @Override
+ public Optional version() { return Optional.empty(); }
+ @Override
+ public HttpHeaders headers() { return HttpHeaders.of(Map.of(), (n, v) -> true); }
+ };
+ assertThrows(IAE, () -> HttpRequest.newBuilder(r, (n, v) -> true).build());
+ }
+
+ @Test
+ public void testInvalidURIScheme() throws URISyntaxException {
+ URI badURI = new URI("ftp://foo.com/somefile");
+ var r = new HttpRequest() {
+ @Override
+ public Optional bodyPublisher() { return Optional.empty(); }
+ @Override
+ public String method() { return "GET"; }
+ @Override
+ public Optional timeout() { return Optional.empty(); }
+ @Override
+ public boolean expectContinue() { return false; }
+ @Override
+ public URI uri() { return badURI; }
+ @Override
+ public Optional version() { return Optional.empty(); }
+ @Override
+ public HttpHeaders headers() { return HttpHeaders.of(Map.of(), (n, v) -> true); }
+ };
+ assertThrows(IAE, () -> HttpRequest.newBuilder(r, (n, v) -> true).build());
+ }
+}
diff --git a/test/jdk/java/net/httpclient/examples/JavadocExamples.java b/test/jdk/java/net/httpclient/examples/JavadocExamples.java
index 483107ed771..8d96c05fea4 100644
--- a/test/jdk/java/net/httpclient/examples/JavadocExamples.java
+++ b/test/jdk/java/net/httpclient/examples/JavadocExamples.java
@@ -119,6 +119,18 @@ public class JavadocExamples {
.uri(URI.create("https://foo.com/"))
.POST(BodyPublishers.ofByteArray(new byte[] { /*...*/ }))
.build();
+
+ // HttpRequest.Builder
+ // API note - newBuilder(HttpRequest, BiPredicate)
+ // Retain all headers:
+ HttpRequest.newBuilder(request, (n, v) -> true);
+
+ //Remove all headers:
+ HttpRequest.newBuilder(request, (n, v) -> false);
+
+ // Remove a particular header (e.g. Foo-Bar):
+ HttpRequest.newBuilder(request, (name, value) ->
+ !name.equalsIgnoreCase("Foo-Bar"));
}
void fromHttpResponse() throws Exception {