/* * Copyright (c) 2022, 2023, 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 * @summary Tests that a Content-length header is not sent for GET requests * that do not have a body. Also checks that the header is sent when * a body is present on the GET request. * @library /test/lib /test/jdk/java/net/httpclient/lib * @build jdk.test.lib.net.SimpleSSLContext * jdk.httpclient.test.lib.common.HttpServerAdapters * @bug 8283544 * @run testng/othervm ContentLengthHeaderTest */ import jdk.httpclient.test.lib.common.HttpServerAdapters; 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 java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.PrintStream; import java.net.URI; import java.net.URISyntaxException; import java.net.http.HttpClient; import java.net.http.HttpClient.Version; import java.net.http.HttpRequest; import java.net.http.HttpResponse; import java.util.Optional; import javax.net.ssl.SSLContext; import jdk.test.lib.net.URIBuilder; import static java.net.http.HttpClient.Version.HTTP_1_1; import static java.net.http.HttpClient.Version.HTTP_2; import static java.nio.charset.StandardCharsets.UTF_8; import static org.testng.Assert.assertEquals; public class ContentLengthHeaderTest implements HttpServerAdapters { final String NO_BODY_PATH = "/no_body"; final String BODY_PATH = "/body"; static HttpTestServer testContentLengthServerH1; static HttpTestServer testContentLengthServerH2; static PrintStream testLog = System.err; static SSLContext sslContext; HttpClient hc; URI testContentLengthURIH1; URI testContentLengthURIH2; @BeforeTest public void setup() throws IOException, URISyntaxException { sslContext = new SimpleSSLContext().get(); testContentLengthServerH1 = HttpTestServer.create(HTTP_1_1); testContentLengthServerH2 = HttpTestServer.create(HTTP_2, sslContext); // Create handlers for tests that check for the presence of a Content-length header testContentLengthServerH1.addHandler(new NoContentLengthHandler(), NO_BODY_PATH); testContentLengthServerH2.addHandler(new NoContentLengthHandler(), NO_BODY_PATH); testContentLengthServerH1.addHandler(new ContentLengthHandler(), BODY_PATH); testContentLengthServerH2.addHandler(new OptionalContentLengthHandler(), BODY_PATH); testContentLengthURIH1 = URIBuilder.newBuilder() .scheme("http") .loopback() .port(testContentLengthServerH1.getAddress().getPort()) .build(); testContentLengthURIH2 = URIBuilder.newBuilder() .scheme("https") .loopback() .port(testContentLengthServerH2.getAddress().getPort()) .build(); testContentLengthServerH1.start(); testLog.println("HTTP/1.1 Server up at address: " + testContentLengthServerH1.getAddress()); testLog.println("Request URI for Client: " + testContentLengthURIH1); testContentLengthServerH2.start(); testLog.println("HTTP/2 Server up at address: " + testContentLengthServerH2.getAddress()); testLog.println("Request URI for Client: " + testContentLengthURIH2); hc = HttpClient.newBuilder() .proxy(HttpClient.Builder.NO_PROXY) .sslContext(sslContext) .build(); } @AfterTest public void teardown() { if (testContentLengthServerH1 != null) testContentLengthServerH1.stop(); if (testContentLengthServerH2 != null) testContentLengthServerH2.stop(); } @DataProvider(name = "bodies") Object[][] bodies() { return new Object[][]{ {HTTP_1_1, URI.create(testContentLengthURIH1 + BODY_PATH)}, {HTTP_2, URI.create(testContentLengthURIH2 + BODY_PATH)} }; } @DataProvider(name = "nobodies") Object[][] nobodies() { return new Object[][]{ {HTTP_1_1, URI.create(testContentLengthURIH1 + NO_BODY_PATH)}, {HTTP_2, URI.create(testContentLengthURIH2 + NO_BODY_PATH)} }; } @Test(dataProvider = "nobodies") // A GET request with no request body should have no Content-length header public void getWithNoBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking GET with no request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .GET() .uri(uri) .build(); HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } @Test(dataProvider = "bodies") // A GET request with a request body should have a Content-length header // in HTTP/1.1 public void getWithBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking GET with request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .method("GET", HttpRequest.BodyPublishers.ofString("GET Body")) .uri(uri) .build(); HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } @Test(dataProvider = "nobodies") // A DELETE request with no request body should have no Content-length header public void deleteWithNoBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking DELETE with no request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .DELETE() .uri(uri) .build(); HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } @Test(dataProvider = "bodies") // A DELETE request with a request body should have a Content-length header // in HTTP/1.1 public void deleteWithBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking DELETE with request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .method("DELETE", HttpRequest.BodyPublishers.ofString("DELETE Body")) .uri(uri) .build(); HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } @Test(dataProvider = "nobodies") // A HEAD request with no request body should have no Content-length header public void headWithNoBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking HEAD with no request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .HEAD() .uri(uri) .build(); HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } @Test(dataProvider = "bodies") // A HEAD request with a request body should have a Content-length header // in HTTP/1.1 public void headWithBody(Version version, URI uri) throws IOException, InterruptedException { testLog.println(version + " Checking HEAD with request body"); HttpRequest req = HttpRequest.newBuilder() .version(version) .method("HEAD", HttpRequest.BodyPublishers.ofString("HEAD Body")) .uri(uri) .build(); // Sending this request invokes sendResponseHeaders which emits a warning about including // a Content-length header with a HEAD request HttpResponse resp = hc.send(req, HttpResponse.BodyHandlers.ofString(UTF_8)); assertEquals(resp.statusCode(), 200, resp.body()); assertEquals(resp.version(), version); } public static void handleResponse(long expected, HttpTestExchange ex, String body, int rCode) throws IOException { try (InputStream is = ex.getRequestBody()) { byte[] reqBody = is.readAllBytes(); if (expected == -1 || expected == reqBody.length || rCode != 200) { sendResponse(ex, body, rCode); } else { body = body + ", but Content-Length:%s doesn't match body size %s" .formatted(expected, reqBody.length); testLog.println("Unexpected content length value: " + body); sendResponse(ex, body, 400); } } } public static void sendResponse(HttpTestExchange ex, String body, int rCode) throws IOException { try (OutputStream os = ex.getResponseBody()) { byte[] bytes = body.getBytes(UTF_8); ex.sendResponseHeaders(rCode, bytes.length); os.write(bytes); } } static class NoContentLengthHandler implements HttpTestHandler { @Override public void handle(HttpTestExchange exchange) throws IOException { testLog.println("NoContentLengthHandler: Received Headers " + exchange.getRequestHeaders().entrySet() + " from " + exchange.getRequestMethod() + " request."); Optional contentLength = exchange.getRequestHeaders() .firstValue("Content-length"); // Check Content-length header was not set if (contentLength.isPresent()) { String responseBody = exchange.getRequestMethod() + " request contained an unexpected " + "Content-length header of value: " + exchange.getRequestHeaders().firstValue("Content-length").get(); handleResponse(-1, exchange, responseBody, 400); } else { handleResponse(0, exchange, "Request completed",200); } } } static class ContentLengthHandler implements HttpTestHandler { @Override public void handle(HttpTestExchange exchange) throws IOException { testLog.println("ContentLengthHandler: Received Headers " + exchange.getRequestHeaders().entrySet() + " from " + exchange.getRequestMethod() + " request."); Optional contentLength = exchange.getRequestHeaders() .firstValue("Content-length"); // Check Content-length header was set if (contentLength.isPresent()) { handleResponse(Long.parseLong(contentLength.get()), exchange, "Request completed", 200); } else { String responseBody = "Expected a Content-length header in " + exchange.getRequestMethod() + " request but was not present."; handleResponse(-1, exchange, responseBody, 400); } } } /** * A handler used for cases where the presence of a Content-Length * header is optional. If present, its value must match the number of * bytes sent in the request body. */ static class OptionalContentLengthHandler implements HttpTestHandler { @Override public void handle(HttpTestExchange exchange) throws IOException { testLog.println("OptionalContentLengthHandler: Received Headers " + exchange.getRequestHeaders().entrySet() + " from " + exchange.getRequestMethod() + " request."); Optional contentLength = exchange.getRequestHeaders().firstValue("Content-Length"); // Check Content-length header was set if (contentLength.isPresent()) { handleResponse(Long.parseLong(contentLength.get()), exchange, "Request completed", 200); } else { handleResponse(-1, exchange, "Request completed, no content length", 200); } } } }