blob: cff57e410cf6cd9a523dab696b30a1cc41f4a210 [file] [log] [blame]
/*
* 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;
}
}
}