/* * Copyright (c) 2001, 2013, 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 Unit test for java.net.URL (Based on the URI tests that is authored by Mark Reinhold) * @bug 4496251 */ import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.PrintStream; import java.net.URL; import java.net.MalformedURLException; public class Test { static PrintStream out = System.out; static int testCount = 0; // Properties that we check static final int PARSEFAIL = 1 << 0; static final int PROTOCOL = 1 << 1; static final int USERINFO = 1 << 2; static final int HOST = 1 << 3; static final int PORT = 1 << 4; static final int PATH = 1 << 5; static final int QUERY = 1 << 6; static final int REF = 1 << 7; String input; URL url = null; URL originalURL; URL base = null; // Base for resolution/relativization String op = null; // Op performed if url != originalURL int checked = 0; // Mask for checked properties int failed = 0; // Mask for failed properties Exception exc = null; private Test(String s) { testCount++; input = s; try { url = new URL(s); } catch (MalformedURLException x) { exc = x; } originalURL = url; } static Test test(String s) { return new Test(s); } private Test(String s, boolean xxx) { testCount++; try { url = new URL(s); } catch (Exception x) { exc = x; } if (url != null) input = url.toString(); originalURL = url; } static Test test(URL base, String spec) { return new Test(base, spec); } private Test(URL base, String spec) { testCount++; try { url = new URL(base, spec); } catch (Exception x) { exc = x; } if (url != null) input = url.toString(); originalURL = url; } static Test test(String protocol, String host, int port, String file) { return new Test(protocol, host, port, file); } private Test(String protocol, String host, int port, String file) { testCount++; try { url = new URL(protocol, host, port, file); } catch (Exception x) { exc = x; } if (url != null) input = url.toString(); originalURL = url; } boolean parsed() { return url != null; } boolean resolved() { return base != null; } URL url() { return url; } // Operations on Test instances // // These are short so as to make test cases compact. // // s Scheme // u User info // h Host // n port Number // p Path // q Query // f Fragment // // rslv Resolve against given base // rtvz Relativize // psa Parse server Authority // norm Normalize // // x Check that parse failed as expected // z End -- ensure that unchecked components are null private boolean check1(int prop) { checked |= prop; if (!parsed()) { failed |= prop; return false; } return true; } private void check2(String s, String ans, int prop) { if (s == null && ans == null) return; if ((s == null) || !s.equals(ans)) failed |= prop; } Test s(String s) { if (check1(PROTOCOL)) check2(url.getProtocol(), s, PROTOCOL); return this; } Test u(String s) { if (check1(USERINFO)) check2(url.getUserInfo(), s, USERINFO); return this; } Test h(String s) { if (check1(HOST)) check2(url.getHost(), s, HOST); return this; } Test n(int n) { checked |= PORT; if (!parsed() || (url.getPort() != n)) failed |= PORT; return this; } Test p(String s) { if (check1(PATH)) check2(url.getPath(), s, PATH); return this; } Test q(String s) { if (check1(QUERY)) check2(url.getQuery(), s, QUERY); return this; } Test f(String s) { if (check1(REF)) check2(url.getRef(), s, REF); return this; } Test x() { checked |= PARSEFAIL; if (parsed()) failed |= PARSEFAIL; return this; } private void checkEmpty(String s, int prop) { if (((checked & prop) == 0) && (s != null)) failed |= prop; } // Check that unchecked component properties are not defined, // and report any failures Test z() { if (!parsed()) { report(); return this; } checkEmpty(url.getProtocol(), PROTOCOL); checkEmpty(url.getUserInfo(), USERINFO); checkEmpty(url.getHost(), HOST); if (((checked & PORT) == 0) && (url.getPort() != -1)) failed |= PORT; checkEmpty(url.getPath(), PATH); checkEmpty(url.getQuery(), QUERY); checkEmpty(url.getRef(), REF); report(); return this; } // Summarization and reporting static void header(String s) { out.println(); out.println(); out.println("-- " + s + " --"); } static void show(String prefix, MalformedURLException x) { out.println(prefix + ": " + x.getMessage()); } private void summarize() { out.println(); StringBuffer sb = new StringBuffer(); if (input.length() == 0) sb.append("\"\""); else sb.append(input); if (base != null) { sb.append(" "); sb.append(base); } if (!parsed()) { String s = (((checked & PARSEFAIL) != 0) ? "Correct exception" : "UNEXPECTED EXCEPTION"); if (exc instanceof MalformedURLException) show(s, (MalformedURLException)exc); else { out.println(sb.toString()); out.print(s + ": "); exc.printStackTrace(out); } } else { if (url != originalURL) { sb.append(" "); sb.append(op); sb.append(" --> "); sb.append(url); } out.println(sb.toString()); } } static void show(String n, String v) { out.println(" " + n + " = ".substring(n.length()) + v); } public static void show(URL u) { show("scheme", u.getProtocol()); show("authority", u.getAuthority()); show("userInfo", u.getUserInfo()); show("host", u.getHost()); show("port", "" + u.getPort()); show("path", u.getPath()); show("query", u.getQuery()); show("ref", u.getRef()); } private void report() { summarize(); if (failed == 0) return; StringBuffer sb = new StringBuffer(); sb.append("FAIL:"); if ((failed & PARSEFAIL) != 0) sb.append(" parsefail"); if ((failed & PROTOCOL) != 0) sb.append(" scheme"); if ((failed & USERINFO) != 0) sb.append(" userinfo"); if ((failed & HOST) != 0) sb.append(" host"); if ((failed & PORT) != 0) sb.append(" port"); if ((failed & PATH) != 0) sb.append(" path"); if ((failed & QUERY) != 0) sb.append(" query"); if ((failed & REF) != 0) sb.append(" fragment"); out.println(sb.toString()); if (url != null) show(url); throw new RuntimeException("Test failed"); } private static boolean hasFtp() { try { return new java.net.URL("ftp://") != null; } catch (java.net.MalformedURLException x) { System.out.println("FTP not supported by this runtime."); return false; } } // -- Tests -- static void rfc2396() { header("RFC2396: Basic examples"); if (hasFtp()) test("ftp://ftp.is.co.za/rfc/rfc1808.txt") .s("ftp").h("ftp.is.co.za").p("/rfc/rfc1808.txt").z(); test("http://www.math.uio.no/faq/compression-faq/part1.html") .s("http").h("www.math.uio.no").p("/faq/compression-faq/part1.html").z(); test("http://www.w3.org/Addressing/") .s("http").h("www.w3.org").p("/Addressing/").z(); if (hasFtp()) test("ftp://ds.internic.net/rfc/") .s("ftp").h("ds.internic.net").p("/rfc/").z(); test("http://www.ics.uci.edu/pub/ietf/url/historical.html#WARNING") .s("http").h("www.ics.uci.edu").p("/pub/ietf/url/historical.html") .f("WARNING").z(); test("http://www.ics.uci.edu/pub/ietf/url/#Related") .s("http").h("www.ics.uci.edu").p("/pub/ietf/url/") .f("Related").z(); test("file:/home/someone/dir1/dir2/file").s("file").h("").p("/home/someone/dir1/dir2/file").z(); header("RFC2396: Normal relative-URL examples (appendix C)"); URL base = (test("http://a/b/c/d;p?q") .s("http").h("a").p("/b/c/d;p").q("q").z().url()); // g:h g:h // test(base, "http:h").s("g").p("h").z(); // g http://a/b/c/g test(base, "g").s("http").h("a").p("/b/c/g").z(); // ./g http://a/b/c/g test(base, "./g").s("http").h("a").p("/b/c/g").z(); // g/ http://a/b/c/g/ test(base, "g/").s("http").h("a").p("/b/c/g/").z(); // /g http://a/g test(base, "/g").s("http").h("a").p("/g").z(); // //g http://g test(base,"//g").s("http").h("g").p("").z(); // ?y http://a/b/c/?y test(base, "?y").s("http").h("a").p("/b/c/").q("y").z(); // g?y http://a/b/c/g?y test(base, "g?y").s("http").h("a").p("/b/c/g").q("y").z(); // #s (current document)#s // DEVIATION: Lone fragment parses as relative URL with empty path, // and resolves without removing the last segment of the base path. // test(base,"#s").s("http").h("a").p("/b/c/d;p").f("s").z(); test(base,"#s").s("http").h("a").p("/b/c/d;p").q("q").f("s").z(); // g#s http://a/b/c/g#s test(base, "g#s").s("http").h("a").p("/b/c/g").f("s").z(); // g?y#s http://a/b/c/g?y#s test(base,"g?y#s").s("http").h("a").p("/b/c/g").q("y").f("s").z(); // ;x http://a/b/c/;x test(base,";x").s("http").h("a").p("/b/c/;x").z(); // g;x http://a/b/c/g;x test(base,"g;x").s("http").h("a").p("/b/c/g;x").z(); // g;x?y#s http://a/b/c/g;x?y#s test(base,"g;x?y#s").s("http").h("a").p("/b/c/g;x").q("y").f("s").z(); // . http://a/b/c/ test(base,".").s("http").h("a").p("/b/c/").z(); // ./ http://a/b/c/ test(base,"./").s("http").h("a").p("/b/c/").z(); // .. http://a/b/ test(base,"..").s("http").h("a").p("/b/").z(); // ../ http://a/b/ test(base,"../").s("http").h("a").p("/b/").z(); // ../g http://a/b/g test(base,"../g").s("http").h("a").p("/b/g").z(); // ../.. http://a/ test(base,"../..").s("http").h("a").p("/").z(); // ../../ http://a/ test(base,"../../").s("http").h("a").p("/").z(); // ../../g http://a/g test(base,"../../g").s("http").h("a").p("/g").z(); // http://u@s1/p1 http://s2/p2 test(test("http://u:p@s1/p1").url(),"http://s2/p2") .s("http").h("s2").u(null).p("/p2").z(); header("RFC2396: Abnormal relative-URL examples (appendix C)"); // ../../../g = http://a/../g test(base,"../../../g").s("http").h("a").p("/../g").z(); // ../../../../g = http://a/../../g test(base, "../../../../g").s("http").h("a").p("/../../g").z(); // /./g = http://a/./g test(base,"/./g").s("http").h("a").p("/./g").z(); // /../g = http://a/../g test(base,"/../g").s("http").h("a").p("/../g").z(); // g. = http://a/b/c/g. test(base,"g.").s("http").h("a").p("/b/c/g.").z(); // .g = http://a/b/c/.g test(base,".g").s("http").h("a").p("/b/c/.g").z(); // g.. = http://a/b/c/g.. test(base,"g..").s("http").h("a").p("/b/c/g..").z(); // ..g = http://a/b/c/..g test(base,"..g").s("http").h("a").p("/b/c/..g").z(); // ./../g = http://a/b/g test(base,"./../g").s("http").h("a").p("/b/g").z(); // ./g/. = http://a/b/c/g/ test(base,"./g/.").s("http").h("a").p("/b/c/g/").z(); // g/./h = http://a/b/c/g/h test(base,"g/./h").s("http").h("a").p("/b/c/g/h").z(); // g/../h = http://a/b/c/h test(base,"g/../h").s("http").h("a").p("/b/c/h").z(); // g;x=1/./y = http://a/b/c/g;x=1/y test(base,"g;x=1/./y").s("http").h("a").p("/b/c/g;x=1/y").z(); // g;x=1/../y = http://a/b/c/y test(base,"g;x=1/../y").s("http").h("a").p("/b/c/y").z(); // g?y/./x = http://a/b/c/g?y/./x test(base,"g?y/./x").s("http").h("a").p("/b/c/g").q("y/./x").z(); // g?y/../x = http://a/b/c/g?y/../x test(base,"g?y/../x").s("http").h("a").p("/b/c/g").q("y/../x").z(); // g#s/./x = http://a/b/c/g#s/./x test(base,"g#s/./x").s("http").h("a").p("/b/c/g").f("s/./x").z(); // g#s/../x = http://a/b/c/g#s/../x test(base,"g#s/../x").s("http").h("a").p("/b/c/g").f("s/../x").z(); // http:g = http:g // test(base,"http:g").s("http").p("g").z(); } static void ip() { header("IP addresses"); test("http://1.2.3.4:5") .s("http").h("1.2.3.4").n(5).p("").z(); // From RFC2732 test("http://[FEDC:BA98:7654:3210:FEDC:BA98:7654:3210]:80/index.html") .s("http").h("[FEDC:BA98:7654:3210:FEDC:BA98:7654:3210]") .n(80).p("/index.html").z(); test("http://[FEDC:BA98:7654:3210:FEDC:BA98:7654:3210%12]:80/index.html") .s("http").h("[FEDC:BA98:7654:3210:FEDC:BA98:7654:3210%12]") .n(80).p("/index.html").z(); test("http://[1080:0:0:0:8:800:200C:417A]/index.html") .s("http").h("[1080:0:0:0:8:800:200C:417A]").p("/index.html").z(); test("http://[1080:0:0:0:8:800:200C:417A%1]/index.html") .s("http").h("[1080:0:0:0:8:800:200C:417A%1]").p("/index.html").z(); test("http://[3ffe:2a00:100:7031::1]") .s("http").h("[3ffe:2a00:100:7031::1]").p("").z(); test("http://[1080::8:800:200C:417A]/foo") .s("http").h("[1080::8:800:200C:417A]").p("/foo").z(); test("http://[::192.9.5.5]/ipng") .s("http").h("[::192.9.5.5]").p("/ipng").z(); test("http://[::192.9.5.5%interface]/ipng") .s("http").h("[::192.9.5.5%interface]").p("/ipng").z(); test("http://[::FFFF:129.144.52.38]:80/index.html") .s("http").h("[::FFFF:129.144.52.38]").n(80).p("/index.html").z(); test("http://[2010:836B:4179::836B:4179]") .s("http").h("[2010:836B:4179::836B:4179]").p("").z(); // From RFC2373 test("http://[FF01::101]") .s("http").h("[FF01::101]").p("").z(); test("http://[::1]") .s("http").h("[::1]").p("").z(); test("http://[::]") .s("http").h("[::]").p("").z(); test("http://[::%hme0]") .s("http").h("[::%hme0]").p("").z(); test("http://[0:0:0:0:0:0:13.1.68.3]") .s("http").h("[0:0:0:0:0:0:13.1.68.3]").p("").z(); test("http://[0:0:0:0:0:FFFF:129.144.52.38]") .s("http").h("[0:0:0:0:0:FFFF:129.144.52.38]").p("").z(); test("http://[0:0:0:0:0:FFFF:129.144.52.38%33]") .s("http").h("[0:0:0:0:0:FFFF:129.144.52.38%33]").p("").z(); test("http://[::13.1.68.3]") .s("http").h("[::13.1.68.3]").p("").z(); test("http://[::13.1.68.3]:") .s("http").h("[::13.1.68.3]").p("").z(); // Error cases test("http://[ff01:234/foo").x().z(); test("http://[ff01:234:zzz]/foo").x().z(); test("http://[foo]").x().z(); test("http://[]").x().z(); test("http://[129.33.44.55]").x().z(); test("http://[ff:ee:dd::cc:bb::aa:9:8]").x().z(); test("http://[1:2:3:4:5:6:7:8%]").x().z(); test("http://[1:2:3:4:5:6:7:8%!/]").x().z(); test("http://[1:2:3:4:5:6:7:8:9]").x().z(); test("http://[::1.2.3.300]").x().z(); test("http://[1.2.3.4:5]").x().z(); // Optional IPv6 brackets in constructors test("http", "1:2:3:4:5:6:7:8", -1, "") .s("http").h("[1:2:3:4:5:6:7:8]").p("").z(); test("http", "1:2:3:4:5:6:7:8%hme0", -1, "") .s("http").h("[1:2:3:4:5:6:7:8%hme0]").p("").z(); test("http", "[1:2:3:4:5:6:7:8]", -1, "") .s("http").h("[1:2:3:4:5:6:7:8]").p("").z(); } static void serial() throws IOException, MalformedURLException { header("Serialization"); ByteArrayOutputStream bo = new ByteArrayOutputStream(); ObjectOutputStream oo = new ObjectOutputStream(bo); URL u = new URL("http://java.sun.com/jdk/1.4?release#beta"); oo.writeObject(u); oo.close(); ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray()); ObjectInputStream oi = new ObjectInputStream(bi); try { Object o = oi.readObject(); u.equals(o); } catch (ClassNotFoundException x) { x.printStackTrace(); throw new RuntimeException(x.toString()); } } static void tests() throws IOException, MalformedURLException { rfc2396(); ip(); serial(); } // -- Command-line invocation -- static void usage() { out.println("Usage:"); out.println(" java Test -- Runs all tests in this file"); out.println(" java Test <url> -- Parses url, shows components"); out.println(" java Test <base> <url> -- Parses url and base, then resolves"); out.println(" url against base"); } public static void main(String[] args) throws Exception { switch (args.length) { case 0: tests(); out.println(); out.println("Test cases: " + testCount); break; case 1: if (args[0].equals("-help")) { usage(); break; } // clargs(null, args[0]); break; case 2: // clargs(args[0], args[1]); break; default: usage(); break; } } }