test/jdk/java/net/httpclient/HeadersTest1.java
author dfuchs
Thu, 29 Mar 2018 09:55:50 +0100
branchhttp-client-branch
changeset 56365 7b2e4c363335
parent 56265 ec34ae013fbe
child 56451 9585061fdb04
permissions -rw-r--r--
http-client-branch: fix header field-value validation

/*
 * Copyright (c) 2015, 2018, 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 8153142 8195138
 * @modules java.net.http
 *          jdk.httpserver
 * @run testng/othervm HeadersTest1
 */

import java.io.IOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpHeaders;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import com.sun.net.httpserver.Headers;
import com.sun.net.httpserver.HttpExchange;
import com.sun.net.httpserver.HttpHandler;
import com.sun.net.httpserver.HttpServer;
import org.testng.annotations.Test;
import static java.net.http.HttpResponse.BodyHandlers.ofString;
import static java.nio.charset.StandardCharsets.US_ASCII;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertNotNull;
import static org.testng.Assert.assertTrue;


public class HeadersTest1 {

    private static final String RESPONSE = "Hello world";
    private static final String QUOTED = "a=\"quote\", b=\"\\\"quote\\\"  \\\"quote\\\"  codec\"";

    @Test
    public void test() throws Exception {
        InetSocketAddress addr = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0);
        HttpServer server = HttpServer.create(addr, 10);
        Handler h = new Handler();
        server.createContext("/test", h);
        int port = server.getAddress().getPort();
        System.out.println("Server port = " + port);

        ExecutorService e = Executors.newCachedThreadPool();
        server.setExecutor(e);
        server.start();
        HttpClient client = HttpClient.newBuilder()
                                      .executor(e)
                                      .build();

        try {
            URI uri = new URI("http://localhost:" + port + "/test/foo");
            HttpRequest req = HttpRequest.newBuilder(uri)
                                         .headers("X-Bar", "foo1")
                                         .headers("X-Bar", "foo2")
                                         .headers("X-Quote",QUOTED)
                                         .GET()
                                         .build();

            HttpResponse<?> resp = client.send(req, ofString());
            if (resp.statusCode() != 200) {
                throw new RuntimeException("Expected 200, got: " + resp.statusCode());
            }
            HttpHeaders hd = resp.headers();

            assertTrue(!hd.firstValue("Non-Existent-Header").isPresent());

            List<String> v1 = hd.allValues("Non-Existent-Header");
            assertNotNull(v1);
            assertTrue(v1.isEmpty(), String.valueOf(v1));
            TestKit.assertUnmodifiableList(v1);

            // case insensitive
            List<String> headernames = List.of("X-Foo-Response",
                                               "x-foo-Response",
                                               "x-fOo-REspoNse");
            for (String headerName : headernames) {
                List<String> v2 = hd.allValues(headerName);
                assertNotNull(v2);
                assertEquals(new HashSet<>(v2), Set.of("resp1", "resp2"));
                TestKit.assertUnmodifiableList(v2);
            }

            Map<String, List<String>> map = hd.map();
            TestKit.assertUnmodifiableMap(map);
            for (List<String> values : map.values()) {
                TestKit.assertUnmodifiableList(values);
            }

            // toString
            assertTrue(hd.toString().toLowerCase().contains("content-length"));
            assertTrue(hd.toString().toLowerCase().contains("x-foo-response"));
            assertTrue(hd.toString().toLowerCase().contains("x-multi-line-response"));

            // multi-line
            List<String> multiline = hd.allValues("x-multi-line-response");
            assertTrue(multiline.get(0).startsWith("Custom "));
            assertTrue(multiline.get(0).contains(" foo=\"bar\""));
            assertTrue(multiline.get(0).contains(" bar=\"foo\""));
            assertTrue(multiline.get(0).contains(" foobar=\"barfoo\""));

            // quote
            List<String> quote = hd.allValues("X-Quote-Response");
            assertEquals(quote, List.of(QUOTED));
        } finally {
            server.stop(0);
            e.shutdownNow();
        }
        System.out.println("OK");
    }

    private static final class Handler implements HttpHandler {

        @Override
        public void handle(HttpExchange he) throws IOException {
            List<String> l = he.getRequestHeaders().get("X-Bar");
            if (!l.contains("foo1") || !l.contains("foo2")) {
                for (String s : l) {
                    System.out.println("HH: " + s);
                }
                he.sendResponseHeaders(500, -1);
                he.close();
                return;
            }
            l = he.getRequestHeaders().get("X-Quote");
            if (l.isEmpty() || l.size() != 1 || !QUOTED.equals(l.get(0))) {
                System.out.println("Bad X-Quote: " + l);
                he.sendResponseHeaders(500, -1);
                he.close();
                return;
            }
            Headers h = he.getResponseHeaders();
            h.add("X-Foo-Response", "resp1");
            h.add("X-Foo-Response", "resp2");
            h.add("X-multi-line-response", "Custom foo=\"bar\","
                    + "\r\n    bar=\"foo\","
                    + "\r\n    foobar=\"barfoo\"");
            h.add("X-Quote-Response", he.getRequestHeaders().getFirst("X-Quote"));
            he.sendResponseHeaders(200, RESPONSE.length());
            OutputStream os = he.getResponseBody();
            os.write(RESPONSE.getBytes(US_ASCII));
            os.close();
        }
    }
}