b0e54328c5
Reviewed-by: jpai
1818 lines
58 KiB
Java
1818 lines
58 KiB
Java
/*
|
|
* Copyright (c) 2000, 2022, 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.URI
|
|
* @bug 4464135 4505046 4503239 4438319 4991359 4866303 7023363 7041800
|
|
* 7171415 6339649 6933879 8037396 8272072 8051627 8297687
|
|
* @author Mark Reinhold
|
|
*/
|
|
|
|
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.URI;
|
|
import java.net.URISyntaxException;
|
|
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 SCHEME = 1 << 1;
|
|
static final int SSP = 1 << 2;
|
|
static final int SSP_D = 1 << 3; // Decoded form
|
|
static final int OPAQUEPART = 1 << 4; // SSP, and URI is opaque
|
|
static final int USERINFO = 1 << 5;
|
|
static final int USERINFO_D = 1 << 6; // Decoded form
|
|
static final int HOST = 1 << 7;
|
|
static final int PORT = 1 << 8;
|
|
static final int REGISTRY = 1 << 9;
|
|
static final int REGISTRY_D = 1 << 10; // Decoded form
|
|
static final int PATH = 1 << 11;
|
|
static final int PATH_D = 1 << 12; // Decoded form
|
|
static final int QUERY = 1 << 13;
|
|
static final int QUERY_D = 1 << 14; // Decoded form
|
|
static final int FRAGMENT = 1 << 15;
|
|
static final int FRAGMENT_D = 1 << 16; // Decoded form
|
|
static final int TOASCII = 1 << 17;
|
|
static final int IDENT_STR = 1 << 18; // Identities
|
|
static final int IDENT_URI1 = 1 << 19;
|
|
static final int IDENT_URI3 = 1 << 20;
|
|
static final int IDENT_URI5 = 1 << 21;
|
|
static final int IDENT_URI7 = 1 << 22;
|
|
static final int TOSTRING = 1 << 23;
|
|
|
|
String input;
|
|
URI uri = null;
|
|
URI originalURI;
|
|
URI base = null; // Base for resolution/relativization
|
|
String op = null; // Op performed if uri != originalURI
|
|
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 {
|
|
uri = new URI(s);
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
}
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test test(String s) {
|
|
return new Test(s);
|
|
}
|
|
|
|
private Test(String s, String u, String h, int n,
|
|
String p, String q, String f)
|
|
{
|
|
testCount++;
|
|
try {
|
|
uri = new URI(s, u, h, n, p, q, f);
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
input = x.getInput();
|
|
}
|
|
if (uri != null)
|
|
input = uri.toString();
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test test(String s, String u, String h, int n,
|
|
String p, String q, String f) {
|
|
return new Test(s, u, h, n, p, q, f);
|
|
}
|
|
|
|
private Test(String s, String a,
|
|
String p, String q, String f)
|
|
{
|
|
testCount++;
|
|
try {
|
|
uri = new URI(s, a, p, q, f);
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
input = x.getInput();
|
|
}
|
|
if (uri != null)
|
|
input = uri.toString();
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test test(String s, String a,
|
|
String p, String q, String f) {
|
|
return new Test(s, a, p, q, f);
|
|
}
|
|
|
|
private Test(String s, String h, String p, String f) {
|
|
testCount++;
|
|
try {
|
|
uri = new URI(s, h, p, f);
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
input = x.getInput();
|
|
}
|
|
if (uri != null)
|
|
input = uri.toString();
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test test(String s, String h, String p, String f) {
|
|
return new Test(s, h, p, f);
|
|
}
|
|
|
|
private Test(String s, String ssp, String f) {
|
|
testCount++;
|
|
try {
|
|
uri = new URI(s, ssp, f);
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
input = x.getInput();
|
|
}
|
|
if (uri != null)
|
|
input = uri.toString();
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test test(String s, String ssp, String f) {
|
|
return new Test(s, ssp, f);
|
|
}
|
|
|
|
private Test(String s, boolean xxx) {
|
|
testCount++;
|
|
try {
|
|
uri = URI.create(s);
|
|
} catch (IllegalArgumentException x) {
|
|
exc = x;
|
|
}
|
|
if (uri != null)
|
|
input = uri.toString();
|
|
originalURI = uri;
|
|
}
|
|
|
|
static Test testCreate(String s) {
|
|
return new Test(s, false);
|
|
}
|
|
|
|
boolean parsed() {
|
|
return uri != null;
|
|
}
|
|
|
|
boolean resolved() {
|
|
return base != null;
|
|
}
|
|
|
|
URI uri() {
|
|
return uri;
|
|
}
|
|
|
|
|
|
// Operations on Test instances
|
|
//
|
|
// These are short so as to make test cases compact.
|
|
//
|
|
// s Scheme
|
|
// sp Scheme-specific part
|
|
// spd Scheme-specific part, decoded
|
|
// o Opaque part (isOpaque() && ssp matches)
|
|
// g reGistry (authority matches, and host is not defined)
|
|
// gd reGistry, decoded
|
|
// u User info
|
|
// ud User info, decoded
|
|
// h Host
|
|
// n port Number
|
|
// p Path
|
|
// pd Path, decoded
|
|
// q Query
|
|
// qd Query, decoded
|
|
// f Fragment
|
|
// fd Fragment, decoded
|
|
//
|
|
// rslv Resolve against given base
|
|
// rtvz Relativize
|
|
// psa Parse server Authority
|
|
// norm Normalize
|
|
// ta ASCII form
|
|
//
|
|
// 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) || !s.equals(ans))
|
|
failed |= prop;
|
|
}
|
|
|
|
Test s(String s) {
|
|
if (check1(SCHEME)) check2(uri.getScheme(), s, SCHEME);
|
|
return this;
|
|
}
|
|
|
|
Test u(String s) {
|
|
if (check1(USERINFO)) check2(uri.getRawUserInfo(), s, USERINFO);
|
|
return this;
|
|
}
|
|
|
|
Test ud(String s) {
|
|
if (check1(USERINFO_D)) {
|
|
check2(uri.getUserInfo(), s, USERINFO_D);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
Test h(String s) {
|
|
if (check1(HOST)) check2(uri.getHost(), s, HOST);
|
|
return this;
|
|
}
|
|
|
|
Test g(String s) {
|
|
if (check1(REGISTRY)) {
|
|
if (uri.getHost() != null)
|
|
failed |= REGISTRY;
|
|
else
|
|
check2(uri.getRawAuthority(), s, REGISTRY);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
Test gd(String s) {
|
|
if (check1(REGISTRY_D)) {
|
|
if (uri.getHost() != null)
|
|
failed |= REGISTRY_D;
|
|
else
|
|
check2(uri.getAuthority(), s, REGISTRY_D);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
Test n(int n) {
|
|
checked |= PORT;
|
|
if (!parsed() || (uri.getPort() != n))
|
|
failed |= PORT;
|
|
return this;
|
|
}
|
|
|
|
Test p(String s) {
|
|
if (check1(PATH)) check2(uri.getRawPath(), s, PATH);
|
|
return this;
|
|
}
|
|
|
|
Test pd(String s) {
|
|
if (check1(PATH_D)) check2(uri.getPath(), s, PATH_D);
|
|
return this;
|
|
}
|
|
|
|
Test o(String s) {
|
|
if (check1(OPAQUEPART)) {
|
|
if (!uri.isOpaque())
|
|
failed |= OPAQUEPART;
|
|
else
|
|
check2(uri.getSchemeSpecificPart(), s, OPAQUEPART);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
Test sp(String s) {
|
|
if (check1(SSP)) check2(uri.getRawSchemeSpecificPart(), s, SSP);
|
|
return this;
|
|
}
|
|
|
|
Test spd(String s) {
|
|
if (check1(SSP_D)) check2(uri.getSchemeSpecificPart(), s, SSP_D);
|
|
return this;
|
|
}
|
|
|
|
Test q(String s) {
|
|
if (check1(QUERY)) check2(uri.getRawQuery(), s, QUERY);
|
|
return this;
|
|
}
|
|
|
|
Test qd(String s) {
|
|
if (check1(QUERY_D)) check2(uri.getQuery(), s, QUERY_D);
|
|
return this;
|
|
}
|
|
|
|
Test f(String s) {
|
|
if (check1(FRAGMENT)) check2(uri.getRawFragment(), s, FRAGMENT);
|
|
return this;
|
|
}
|
|
|
|
Test fd(String s) {
|
|
if (check1(FRAGMENT_D)) check2(uri.getFragment(), s, FRAGMENT_D);
|
|
return this;
|
|
}
|
|
|
|
Test ta(String s) {
|
|
if (check1(TOASCII))
|
|
check2(uri.toASCIIString(), s, TOASCII);
|
|
return this;
|
|
}
|
|
|
|
Test ts(String s) {
|
|
if (check1(TOSTRING))
|
|
check2(uri.toString(), s, TOSTRING);
|
|
return this;
|
|
}
|
|
|
|
Test x() {
|
|
checked |= PARSEFAIL;
|
|
if (parsed())
|
|
failed |= PARSEFAIL;
|
|
return this;
|
|
}
|
|
|
|
Test rslv(URI base) {
|
|
if (!parsed())
|
|
return this;
|
|
this.base = base;
|
|
op = "rslv";
|
|
URI u = uri;
|
|
uri = null;
|
|
try {
|
|
this.uri = base.resolve(u);
|
|
} catch (IllegalArgumentException x) {
|
|
exc = x;
|
|
}
|
|
checked = 0;
|
|
failed = 0;
|
|
return this;
|
|
}
|
|
|
|
Test norm() {
|
|
if (!parsed())
|
|
return this;
|
|
op = "norm";
|
|
uri = uri.normalize();
|
|
return this;
|
|
}
|
|
|
|
Test rtvz(URI base) {
|
|
if (!parsed())
|
|
return this;
|
|
this.base = base;
|
|
op = "rtvz";
|
|
uri = base.relativize(uri);
|
|
checked = 0;
|
|
failed = 0;
|
|
return this;
|
|
}
|
|
|
|
Test psa() {
|
|
try {
|
|
uri.parseServerAuthority();
|
|
} catch (URISyntaxException x) {
|
|
exc = x;
|
|
uri = null;
|
|
}
|
|
checked = 0;
|
|
failed = 0;
|
|
return this;
|
|
}
|
|
|
|
private void checkEmpty(String s, int prop) {
|
|
if (((checked & prop) == 0) && (s != null))
|
|
failed |= prop;
|
|
}
|
|
|
|
// Check identity for the seven-argument URI constructor
|
|
//
|
|
void checkURI7() {
|
|
// Only works on hierarchical URIs
|
|
if (uri.isOpaque())
|
|
return;
|
|
// Only works with server-based authorities
|
|
if ((uri.getAuthority() == null)
|
|
!= ((uri.getUserInfo() == null) && (uri.getHost() == null)))
|
|
return;
|
|
// Not true if non-US-ASCII chars are encoded unnecessarily
|
|
if (uri.getPath().indexOf('\u20AC') >= 0)
|
|
return;
|
|
try {
|
|
URI u2 = new URI(uri.getScheme(), uri.getUserInfo(),
|
|
uri.getHost(), uri.getPort(), uri.getPath(),
|
|
uri.getQuery(), uri.getFragment());
|
|
if (!uri.equals(u2))
|
|
failed |= IDENT_URI7;
|
|
} catch (URISyntaxException x) {
|
|
failed |= IDENT_URI7;
|
|
}
|
|
}
|
|
|
|
// Check identity for the five-argument URI constructor
|
|
//
|
|
void checkURI5() {
|
|
// Only works on hierarchical URIs
|
|
if (uri.isOpaque())
|
|
return;
|
|
try {
|
|
URI u2 = new URI(uri.getScheme(), uri.getAuthority(),
|
|
uri.getPath(), uri.getQuery(), uri.getFragment());
|
|
if (!uri.equals(u2))
|
|
failed |= IDENT_URI5;
|
|
} catch (URISyntaxException x) {
|
|
failed |= IDENT_URI5;
|
|
}
|
|
}
|
|
|
|
// Check identity for the three-argument URI constructor
|
|
//
|
|
void checkURI3() {
|
|
try {
|
|
URI u2 = new URI(uri.getScheme(),
|
|
uri.getSchemeSpecificPart(),
|
|
uri.getFragment());
|
|
if (!uri.equals(u2))
|
|
failed |= IDENT_URI3;
|
|
} catch (URISyntaxException x) {
|
|
failed |= IDENT_URI3;
|
|
}
|
|
}
|
|
|
|
// Check all identities mentioned in the URI class specification
|
|
//
|
|
void checkIdentities() {
|
|
if (input != null) {
|
|
if (!uri.toString().equals(input))
|
|
failed |= IDENT_STR;
|
|
}
|
|
try {
|
|
if (!(new URI(uri.toString())).equals(uri))
|
|
failed |= IDENT_URI1;
|
|
} catch (URISyntaxException x) {
|
|
failed |= IDENT_URI1;
|
|
}
|
|
|
|
// Remaining identities fail if "//" given but authority is undefined
|
|
if ((uri.getAuthority() == null)
|
|
&& (uri.getSchemeSpecificPart() != null)
|
|
&& (uri.getSchemeSpecificPart().startsWith("///")
|
|
|| uri.getSchemeSpecificPart().startsWith("//?")
|
|
|| uri.getSchemeSpecificPart().equals("//")))
|
|
return;
|
|
|
|
// Remaining identities fail if ":" given but port is undefined
|
|
if ((uri.getHost() != null)
|
|
&& (uri.getAuthority() != null)
|
|
&& (uri.getAuthority().equals(uri.getHost() + ":")))
|
|
return;
|
|
|
|
// Remaining identities fail if non-US-ASCII chars are encoded
|
|
// unnecessarily
|
|
if ((uri.getPath() != null) && uri.getPath().indexOf('\u20AC') >= 0)
|
|
return;
|
|
|
|
checkURI3();
|
|
checkURI5();
|
|
checkURI7();
|
|
}
|
|
|
|
// Check identities, check that unchecked component properties are not
|
|
// defined, and report any failures
|
|
//
|
|
Test z() {
|
|
if (!parsed()) {
|
|
report();
|
|
return this;
|
|
}
|
|
|
|
if (op == null)
|
|
checkIdentities();
|
|
|
|
// Check that unchecked components are undefined
|
|
checkEmpty(uri.getScheme(), SCHEME);
|
|
checkEmpty(uri.getUserInfo(), USERINFO);
|
|
checkEmpty(uri.getHost(), HOST);
|
|
if (((checked & PORT) == 0) && (uri.getPort() != -1)) failed |= PORT;
|
|
checkEmpty(uri.getPath(), PATH);
|
|
checkEmpty(uri.getQuery(), QUERY);
|
|
checkEmpty(uri.getFragment(), FRAGMENT);
|
|
|
|
// Report failures
|
|
report();
|
|
return this;
|
|
}
|
|
|
|
|
|
// Summarization and reporting
|
|
|
|
static void header(String s) {
|
|
out.println();
|
|
out.println();
|
|
out.println("-- " + s + " --");
|
|
}
|
|
|
|
static void show(String prefix, URISyntaxException x) {
|
|
out.println(uquote(x.getInput()));
|
|
if (x.getIndex() >= 0) {
|
|
for (int i = 0; i < x.getIndex(); i++) {
|
|
if (x.getInput().charAt(i) >= '\u0080')
|
|
out.print(" "); // Skip over \u1234
|
|
else
|
|
out.print(" ");
|
|
}
|
|
out.println("^");
|
|
}
|
|
out.println(prefix + ": " + x.getReason());
|
|
}
|
|
|
|
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 URISyntaxException)
|
|
show(s, (URISyntaxException)exc);
|
|
else {
|
|
out.println(uquote(sb.toString()));
|
|
out.print(s + ": ");
|
|
exc.printStackTrace(out);
|
|
}
|
|
} else {
|
|
if (uri != originalURI) {
|
|
sb.append(" ");
|
|
sb.append(op);
|
|
sb.append(" --> ");
|
|
sb.append(uri);
|
|
}
|
|
out.println(uquote(sb.toString()));
|
|
}
|
|
}
|
|
|
|
public static String uquote(String str) {
|
|
if (str == null)
|
|
return str;
|
|
StringBuffer sb = new StringBuffer();
|
|
int n = str.length();
|
|
for (int i = 0; i < n; i++) {
|
|
char c = str.charAt(i);
|
|
if ((c >= ' ') && (c < 0x7f)) {
|
|
sb.append(c);
|
|
continue;
|
|
}
|
|
sb.append("\\u");
|
|
String s = Integer.toHexString(c).toUpperCase();
|
|
while (s.length() < 4)
|
|
s = "0" + s;
|
|
sb.append(s);
|
|
}
|
|
return sb.toString();
|
|
}
|
|
|
|
static void show(String n, String v) {
|
|
out.println(" " + n
|
|
+ " = ".substring(n.length())
|
|
+ uquote(v));
|
|
}
|
|
|
|
static void show(String n, String v, String vd) {
|
|
if ((v == null) || v.equals(vd))
|
|
show(n, v);
|
|
else {
|
|
out.println(" " + n
|
|
+ " = ".substring(n.length())
|
|
+ uquote(v)
|
|
+ " = " + uquote(vd));
|
|
}
|
|
}
|
|
|
|
public static void show(URI u) {
|
|
show("opaque", "" + u.isOpaque());
|
|
show("scheme", u.getScheme());
|
|
show("ssp", u.getRawSchemeSpecificPart(), u.getSchemeSpecificPart());
|
|
show("authority", u.getRawAuthority(), u.getAuthority());
|
|
show("userinfo", u.getRawUserInfo(), u.getUserInfo());
|
|
show("host", u.getHost());
|
|
show("port", "" + u.getPort());
|
|
show("path", u.getRawPath(), u.getPath());
|
|
show("query", u.getRawQuery(), u.getQuery());
|
|
show("fragment", u.getRawFragment(), u.getFragment());
|
|
if (!u.toString().equals(u.toASCIIString()))
|
|
show("toascii", u.toASCIIString());
|
|
}
|
|
|
|
private void report() {
|
|
summarize();
|
|
if (failed == 0) return;
|
|
StringBuffer sb = new StringBuffer();
|
|
sb.append("FAIL:");
|
|
if ((failed & PARSEFAIL) != 0) sb.append(" parsefail");
|
|
if ((failed & SCHEME) != 0) sb.append(" scheme");
|
|
if ((failed & SSP) != 0) sb.append(" ssp");
|
|
if ((failed & OPAQUEPART) != 0) sb.append(" opaquepart");
|
|
if ((failed & USERINFO) != 0) sb.append(" userinfo");
|
|
if ((failed & USERINFO_D) != 0) sb.append(" userinfod");
|
|
if ((failed & HOST) != 0) sb.append(" host");
|
|
if ((failed & PORT) != 0) sb.append(" port");
|
|
if ((failed & REGISTRY) != 0) sb.append(" registry");
|
|
if ((failed & PATH) != 0) sb.append(" path");
|
|
if ((failed & PATH_D) != 0) sb.append(" pathd");
|
|
if ((failed & QUERY) != 0) sb.append(" query");
|
|
if ((failed & QUERY_D) != 0) sb.append(" queryd");
|
|
if ((failed & FRAGMENT) != 0) sb.append(" fragment");
|
|
if ((failed & FRAGMENT_D) != 0) sb.append(" fragmentd");
|
|
if ((failed & TOASCII) != 0) sb.append(" toascii");
|
|
if ((failed & IDENT_STR) != 0) sb.append(" ident-str");
|
|
if ((failed & IDENT_URI1) != 0) sb.append(" ident-uri1");
|
|
if ((failed & IDENT_URI3) != 0) sb.append(" ident-uri3");
|
|
if ((failed & IDENT_URI5) != 0) sb.append(" ident-uri5");
|
|
if ((failed & IDENT_URI7) != 0) sb.append(" ident-uri7");
|
|
if ((failed & TOSTRING) != 0) sb.append(" tostring");
|
|
out.println(sb.toString());
|
|
if (uri != null) show(uri);
|
|
throw new RuntimeException("Test failed");
|
|
}
|
|
|
|
|
|
|
|
// -- Tests --
|
|
|
|
static void rfc2396() {
|
|
|
|
|
|
header("RFC2396: Basic examples");
|
|
|
|
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("mailto:mduerst@ifi.unizh.ch")
|
|
.s("mailto").o("mduerst@ifi.unizh.ch").z();
|
|
|
|
test("news:comp.infosystems.www.servers.unix")
|
|
.s("news").o("comp.infosystems.www.servers.unix").z();
|
|
|
|
test("telnet://melvyl.ucop.edu/")
|
|
.s("telnet").h("melvyl.ucop.edu").p("/").z();
|
|
|
|
test("http://www.w3.org/Addressing/")
|
|
.s("http").h("www.w3.org").p("/Addressing/").z();
|
|
|
|
test("ftp://ds.internic.net/rfc/")
|
|
.s("ftp").h("ds.internic.net").p("/rfc/").z();
|
|
|
|
test("http://www.ics.uci.edu/pub/ietf/uri/historical.html#WARNING")
|
|
.s("http").h("www.ics.uci.edu").p("/pub/ietf/uri/historical.html")
|
|
.f("WARNING").z();
|
|
|
|
test("http://www.ics.uci.edu/pub/ietf/uri/#Related")
|
|
.s("http").h("www.ics.uci.edu").p("/pub/ietf/uri/")
|
|
.f("Related").z();
|
|
|
|
|
|
header("RFC2396: Normal relative-URI examples (appendix C)");
|
|
|
|
URI base = (test("http://a/b/c/d;p?q")
|
|
.s("http").h("a").p("/b/c/d;p").q("q").z().uri());
|
|
|
|
// g:h g:h
|
|
test("g:h")
|
|
.s("g").o("h").z()
|
|
.rslv(base).s("g").o("h").z();
|
|
|
|
// g http://a/b/c/g
|
|
test("g")
|
|
.p("g").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").z();
|
|
|
|
// ./g http://a/b/c/g
|
|
test("./g")
|
|
.p("./g").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").z();
|
|
|
|
// g/ http://a/b/c/g/
|
|
test("g/")
|
|
.p("g/").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g/").z();
|
|
|
|
// /g http://a/g
|
|
test("/g")
|
|
.p("/g").z()
|
|
.rslv(base).s("http").h("a").p("/g").z();
|
|
|
|
// //g http://g
|
|
test("//g")
|
|
.h("g").p("").z()
|
|
.rslv(base).s("http").h("g").p("").z();
|
|
|
|
// ?y http://a/b/c/?y
|
|
test("?y")
|
|
.p("").q("y").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/").q("y").z();
|
|
|
|
// g?y http://a/b/c/g?y
|
|
test("g?y")
|
|
.p("g").q("y").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").q("y").z();
|
|
|
|
// #s (current document)#s
|
|
// DEVIATION: Lone fragment parses as relative URI with empty path
|
|
test("#s")
|
|
.p("").f("s").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/d;p").f("s").q("q").z();
|
|
|
|
// g#s http://a/b/c/g#s
|
|
test("g#s")
|
|
.p("g").f("s").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").f("s").z();
|
|
|
|
// g?y#s http://a/b/c/g?y#s
|
|
test("g?y#s")
|
|
.p("g").q("y").f("s").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").q("y").f("s").z();
|
|
|
|
// ;x http://a/b/c/;x
|
|
test(";x")
|
|
.p(";x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/;x").z();
|
|
|
|
// g;x http://a/b/c/g;x
|
|
test("g;x")
|
|
.p("g;x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g;x").z();
|
|
|
|
// g;x?y#s http://a/b/c/g;x?y#s
|
|
test("g;x?y#s")
|
|
.p("g;x").q("y").f("s").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g;x").q("y").f("s").z();
|
|
|
|
// . http://a/b/c/
|
|
test(".")
|
|
.p(".").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/").z();
|
|
|
|
// ./ http://a/b/c/
|
|
test("./")
|
|
.p("./").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/").z();
|
|
|
|
// .. http://a/b/
|
|
test("..")
|
|
.p("..").z()
|
|
.rslv(base).s("http").h("a").p("/b/").z();
|
|
|
|
// ../ http://a/b/
|
|
test("../")
|
|
.p("../").z()
|
|
.rslv(base).s("http").h("a").p("/b/").z();
|
|
|
|
// ../g http://a/b/g
|
|
test("../g")
|
|
.p("../g").z()
|
|
.rslv(base).s("http").h("a").p("/b/g").z();
|
|
|
|
// ../.. http://a/
|
|
test("../..")
|
|
.p("../..").z()
|
|
.rslv(base).s("http").h("a").p("/").z();
|
|
|
|
// ../../ http://a/
|
|
test("../../")
|
|
.p("../../").z()
|
|
.rslv(base).s("http").h("a").p("/").z();
|
|
|
|
// ../../g http://a/g
|
|
test("../../g")
|
|
.p("../../g").z()
|
|
.rslv(base).s("http").h("a").p("/g").z();
|
|
|
|
|
|
header("RFC2396: Abnormal relative-URI examples (appendix C)");
|
|
|
|
// ../../../g = http://a/../g
|
|
test("../../../g")
|
|
.p("../../../g").z()
|
|
.rslv(base).s("http").h("a").p("/../g").z();
|
|
|
|
// ../../../../g = http://a/../../g
|
|
test("../../../../g")
|
|
.p("../../../../g").z()
|
|
.rslv(base).s("http").h("a").p("/../../g").z();
|
|
|
|
|
|
// /./g = http://a/./g
|
|
test("/./g")
|
|
.p("/./g").z()
|
|
.rslv(base).s("http").h("a").p("/./g").z();
|
|
|
|
// /../g = http://a/../g
|
|
test("/../g")
|
|
.p("/../g").z()
|
|
.rslv(base).s("http").h("a").p("/../g").z();
|
|
|
|
// g. = http://a/b/c/g.
|
|
test("g.")
|
|
.p("g.").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g.").z();
|
|
|
|
// .g = http://a/b/c/.g
|
|
test(".g")
|
|
.p(".g").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/.g").z();
|
|
|
|
// g.. = http://a/b/c/g..
|
|
test("g..")
|
|
.p("g..").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g..").z();
|
|
|
|
// ..g = http://a/b/c/..g
|
|
test("..g")
|
|
.p("..g").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/..g").z();
|
|
|
|
// ./../g = http://a/b/g
|
|
test("./../g")
|
|
.p("./../g").z()
|
|
.rslv(base).s("http").h("a").p("/b/g").z();
|
|
|
|
// ./g/. = http://a/b/c/g/
|
|
test("./g/.")
|
|
.p("./g/.").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g/").z();
|
|
|
|
// g/./h = http://a/b/c/g/h
|
|
test("g/./h")
|
|
.p("g/./h").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g/h").z();
|
|
|
|
// g/../h = http://a/b/c/h
|
|
test("g/../h")
|
|
.p("g/../h").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/h").z();
|
|
|
|
// g;x=1/./y = http://a/b/c/g;x=1/y
|
|
test("g;x=1/./y")
|
|
.p("g;x=1/./y").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g;x=1/y").z();
|
|
|
|
// g;x=1/../y = http://a/b/c/y
|
|
test("g;x=1/../y")
|
|
.p("g;x=1/../y").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/y").z();
|
|
|
|
// g?y/./x = http://a/b/c/g?y/./x
|
|
test("g?y/./x")
|
|
.p("g").q("y/./x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").q("y/./x").z();
|
|
|
|
// g?y/../x = http://a/b/c/g?y/../x
|
|
test("g?y/../x")
|
|
.p("g").q("y/../x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").q("y/../x").z();
|
|
|
|
// g#s/./x = http://a/b/c/g#s/./x
|
|
test("g#s/./x")
|
|
.p("g").f("s/./x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").f("s/./x").z();
|
|
|
|
// g#s/../x = http://a/b/c/g#s/../x
|
|
test("g#s/../x")
|
|
.p("g").f("s/../x").z()
|
|
.rslv(base).s("http").h("a").p("/b/c/g").f("s/../x").z();
|
|
|
|
// http:g = http:g
|
|
test("http:g")
|
|
.s("http").o("g").z()
|
|
.rslv(base).s("http").o("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:10%12]:80/index.html")
|
|
.s("http").h("[FEDC:BA98:7654:3210:FEDC:BA98:7654:10%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://[0:0:0:0:0:ffff:1.2.3.4]")
|
|
.s("http").h("[0:0:0:0:0:ffff:1.2.3.4]").p("").z();
|
|
|
|
test("http://[::13.1.68.3]")
|
|
.s("http").h("[::13.1.68.3]").p("").z();
|
|
|
|
// Optional IPv6 brackets in constructors
|
|
|
|
test("s", null, "1:2:3:4:5:6:7:8", -1, null, null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", null, "[1:2:3:4:5:6:7:8]", -1, null, null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", null, "[1:2:3:4:5:6:7:8]", -1, null, null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", "1:2:3:4:5:6:7:8", null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", "1:2:3:4:5:6:7:8%hme0", null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8%hme0]").p("").z();
|
|
|
|
test("s", "1:2:3:4:5:6:7:8%1", null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8%1]").p("").z();
|
|
|
|
test("s", "[1:2:3:4:5:6:7:8]", null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", "[1:2:3:4:5:6:7:8]", null, null, null)
|
|
.s("s").h("[1:2:3:4:5:6:7:8]").p("").z();
|
|
|
|
test("s", "1:2:3:4:5:6:7:8", null, null, null)
|
|
.s("s").g("1:2:3:4:5:6:7:8").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://[fffff::1]").x().z();
|
|
test("http://[ff::ee::8]").x().z();
|
|
test("http://[1:2:3:4::5:6:7:8]").x().z();
|
|
test("http://[1:2]").x().z();
|
|
test("http://[1:2:3:4:5:6:7:8:9]").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.300]").x().z();
|
|
test("http://1.2.3").psa().x().z();
|
|
test("http://1.2.3.300").psa().x().z();
|
|
test("http://1.2.3.4.5").psa().x().z();
|
|
test("http://[1.2.3.4:5]").x().z();
|
|
test("http://1:2:3:4:5:6:7:8").psa().x().z();
|
|
test("http://[1.2.3.4]/").x().z();
|
|
test("http://[1.2.3.4/").x().z();
|
|
test("http://[foo]/").x().z();
|
|
test("http://[foo/").x().z();
|
|
test("s", "[foo]", "/", null, null).x().z();
|
|
test("s", "[foo", "/", null, null).x().z();
|
|
test("s", "[::foo", "/", null, null).x().z();
|
|
|
|
// Test hostnames that might initially look like IPv4 addresses
|
|
|
|
test("s://1.2.3.com").psa().s("s").h("1.2.3.com").p("").z();
|
|
test("s://1.2.3.4me.com").psa().s("s").h("1.2.3.4me.com").p("").z();
|
|
|
|
test("s://7up.com").psa().s("s").h("7up.com").p("").z();
|
|
test("s://7up.com/p").psa().s("s").h("7up.com").p("/p").z();
|
|
test("s://7up").psa().s("s").h("7up").p("").z();
|
|
test("s://7up/p").psa().s("s").h("7up").p("/p").z();
|
|
test("s://7up.").psa().s("s").h("7up.").p("").z();
|
|
test("s://7up./p").psa().s("s").h("7up.").p("/p").z();
|
|
}
|
|
|
|
|
|
static void misc() throws URISyntaxException {
|
|
|
|
URI base = new URI("s://h/a/b");
|
|
URI rbase = new URI("a/b/c/d");
|
|
|
|
|
|
header("Corner cases");
|
|
|
|
// The empty URI parses as a relative URI with an empty path
|
|
test("").p("").z()
|
|
.rslv(base).s("s").h("h").p("/a/").z();
|
|
|
|
// Resolving solo queries and fragments
|
|
test("#f").p("").f("f").z()
|
|
.rslv(base).s("s").h("h").p("/a/b").f("f").z();
|
|
test("?q").p("").q("q").z()
|
|
.rslv(base).s("s").h("h").p("/a/").q("q").z();
|
|
|
|
// Fragment is not part of ssp
|
|
test("p#f").p("p").f("f").sp("p").z();
|
|
test("s:p#f").s("s").o("p").f("f").z();
|
|
test("p#f")
|
|
.rslv(base).s("s").h("h").p("/a/p").f("f").sp("//h/a/p").z();
|
|
test("").p("").sp("").z();
|
|
|
|
|
|
|
|
header("Emptiness");
|
|
|
|
// Components that may be empty
|
|
test("///p").p("/p").z(); // Authority (w/ path)
|
|
test("//@h/p").u("").h("h").p("/p").z(); // User info
|
|
test("//h:/p").h("h").p("/p").z(); // Port
|
|
test("//h").h("h").p("").z(); // Path
|
|
test("//h?q").h("h").p("").q("q").z(); // Path (w/query)
|
|
test("//?q").p("").q("q").z(); // Authority (w/query)
|
|
test("//#f").p("").f("f").z(); // Authority (w/fragment)
|
|
test("p?#").p("p").q("").f("").z(); // Query & fragment
|
|
|
|
// Components that may not be empty
|
|
test(":").x().z(); // Scheme
|
|
test("x:").x().z(); // Hier/opaque
|
|
test("//").x().z(); // Authority (w/o path)
|
|
|
|
|
|
header("Resolution, normalization, and relativization");
|
|
|
|
// Resolving relative paths
|
|
test("../e/f").p("../e/f").z()
|
|
.rslv(rbase).p("a/b/e/f").z();
|
|
test("../../../../d").p("../../../../d").z()
|
|
.rslv(rbase).p("../d").z();
|
|
test("../../../d:e").p("../../../d:e").z()
|
|
.rslv(rbase).p("./d:e").z();
|
|
test("../../../d:e/f").p("../../../d:e/f").z()
|
|
.rslv(rbase).p("./d:e/f").z();
|
|
|
|
// Normalization
|
|
test("a/./c/../d/f").p("a/./c/../d/f").z()
|
|
.norm().p("a/d/f").z();
|
|
test("http://a/./b/c/../d?q#f")
|
|
.s("http").h("a").p("/./b/c/../d").q("q").f("f").z()
|
|
.norm().s("http").h("a").p("/b/d").q("q").f("f").z();
|
|
test("a/../b").p("a/../b").z().
|
|
norm().p("b");
|
|
test("a/../b:c").p("a/../b:c").z()
|
|
.norm().p("./b:c").z();
|
|
|
|
// Normalization of already normalized URI should yield the
|
|
// same URI
|
|
URI u1 = URI.create("s://h/../p");
|
|
URI u2 = u1.normalize();
|
|
eq(u1, u2);
|
|
eqeq(u1, u2);
|
|
|
|
// Relativization
|
|
test("/a/b").p("/a/b").z()
|
|
.rtvz(new URI("/a")).p("b").z();
|
|
test("/a/b").p("/a/b").z()
|
|
.rtvz(new URI("/a/")).p("b").z();
|
|
test("a/b").p("a/b").z()
|
|
.rtvz(new URI("a")).p("b").z();
|
|
test("/a/b").p("/a/b").z()
|
|
.rtvz(new URI("/a/b")).p("").z(); // Result is empty path
|
|
test("a/../b:c/d").p("a/../b:c/d").z()
|
|
.rtvz(new URI("./b:c/")).p("d").z();
|
|
|
|
test("http://a/b/d/e?q#f")
|
|
.s("http").h("a").p("/b/d/e").q("q").f("f").z()
|
|
.rtvz(new URI("http://a/b/?r#g"))
|
|
.p("d/e").q("q").f("f").z();
|
|
|
|
// parseServerAuthority
|
|
test("/a/b").psa().p("/a/b").z();
|
|
test("s://u@h:1/p")
|
|
.psa().s("s").u("u").h("h").n(1).p("/p").z();
|
|
test("s://u@h:-foo/p").s("s").g("u@h:-foo").p("/p").z()
|
|
.psa().x().z();
|
|
test("s://h:999999999999999999999999").psa().x().z();
|
|
test("s://:/b").psa().x().z();
|
|
|
|
|
|
header("Constructors and factories");
|
|
|
|
test("s", null, null, -1, "p", null, null).x().z();
|
|
test(null, null, null, -1, null, null, null).p("").z();
|
|
test(null, null, null, -1, "p", null, null).p("p").z();
|
|
test(null, null, "foo%20bar", -1, null, null, null).x().z();
|
|
test(null, null, "foo", -100, null, null, null).x().z();
|
|
test("s", null, null, -1, "", null, null).x().z();
|
|
test("s", null, null, -1, "/p", null, null).s("s").p("/p").z();
|
|
test("s", "u", "h", 10, "/p", "q", "f")
|
|
.s("s").u("u").h("h").n(10).p("/p").q("q").f("f").z();
|
|
test("s", "a:b", "/p", "q", "f")
|
|
.s("s").g("a:b").p("/p").q("q").f("f").z();
|
|
test("s", "h", "/p", "f")
|
|
.s("s").h("h").p("/p").f("f").z();
|
|
test("s", "p", "f").s("s").o("p").f("f").z();
|
|
test("s", "/p", "f").s("s").p("/p").f("f").z();
|
|
testCreate("s://u@h/p?q#f")
|
|
.s("s").u("u").h("h").p("/p").q("q").f("f").z();
|
|
}
|
|
|
|
static void npes() throws URISyntaxException {
|
|
|
|
header("NullPointerException");
|
|
|
|
URI base = URI.create("mailto:root@foobar.com");
|
|
|
|
out.println();
|
|
|
|
try {
|
|
base.resolve((URI)null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
out.println("resolve((URI)null) -->");
|
|
out.println("Correct exception: " + x);
|
|
}
|
|
|
|
out.println();
|
|
|
|
try {
|
|
base.resolve((String)null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
out.println("resolve((String)null) -->");
|
|
out.println("Correct exception: " + x);
|
|
}
|
|
|
|
out.println();
|
|
|
|
try {
|
|
base.relativize((URI)null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
out.println("relativize((String)null) -->");
|
|
out.println("Correct exception: " + x);
|
|
}
|
|
|
|
testCount += 3;
|
|
}
|
|
|
|
|
|
static void chars() throws URISyntaxException {
|
|
|
|
header("Escapes and non-US-ASCII characters");
|
|
|
|
URI uri;
|
|
|
|
// Escape pairs
|
|
test("%0a%0A%0f%0F%01%09zz")
|
|
.p("%0a%0A%0f%0F%01%09zz").z();
|
|
test("foo%1").x().z();
|
|
test("foo%z").x().z();
|
|
test("foo%9z").x().z();
|
|
|
|
// Escapes not permitted in scheme, host
|
|
test("s%20t://a").x().z();
|
|
test("//a%20b").g("a%20b").p("").z(); // Parses as registry
|
|
|
|
// Escapes permitted in opaque part, userInfo, registry, path,
|
|
// query, and fragment
|
|
test("//u%20v@a").u("u%20v").h("a").p("").z();
|
|
test("/p%20q").p("/p%20q").z();
|
|
test("/p?q%20").p("/p").q("q%20").z();
|
|
test("/p#%20f").p("/p").f("%20f").z();
|
|
|
|
// Non-US-ASCII chars
|
|
test("s\u00a7t://a").x().z();
|
|
test("//\u00a7/b").g("\u00a7").p("/b").z(); // Parses as registry
|
|
test("//u\u00a7v@a").u("u\u00a7v").h("a").p("").z();
|
|
test("/p\u00a7q").p("/p\u00a7q").z();
|
|
test("/p?q\u00a7").p("/p").q("q\u00a7").z();
|
|
test("/p#\u00a7f").p("/p").f("\u00a7f").z();
|
|
|
|
// 4648111 - Escapes quoted by toString after resolution
|
|
uri = new URI("http://a/b/c/d;p?q");
|
|
test("/p%20p")
|
|
.rslv(uri).s("http").h("a").p("/p%20p").ts("http://a/p%20p").z();
|
|
|
|
// 4464135: Forbid unwise characters throughout opaque part
|
|
test("foo:x{bar").x().z();
|
|
test("foo:{bar").x().z();
|
|
|
|
// 4438319: Single-argument constructor requires quotation,
|
|
// preserves escapes
|
|
test("//u%01@h/a/b/%02/c?q%03#f%04")
|
|
.u("u%01").ud("u\1")
|
|
.h("h")
|
|
.p("/a/b/%02/c").pd("/a/b/\2/c")
|
|
.q("q%03").qd("q\3")
|
|
.f("f%04").fd("f\4")
|
|
.z();
|
|
test("/a/b c").x().z();
|
|
|
|
// 4438319: Multi-argument constructors quote illegal chars and
|
|
// preserve legal non-ASCII chars
|
|
// \uA001-\uA009 are visible characters, \u2000 is a space character
|
|
test(null, "u\uA001\1", "h", -1,
|
|
"/p% \uA002\2\u2000",
|
|
"q% \uA003\3\u2000",
|
|
"f% \uA004\4\u2000")
|
|
.u("u\uA001%01").h("h")
|
|
.p("/p%25%20\uA002%02%E2%80%80").pd("/p% \uA002\2\u2000")
|
|
.q("q%25%20\uA003%03%E2%80%80").qd("q% \uA003\3\u2000")
|
|
.f("f%25%20\uA004%04%E2%80%80").fd("f% \uA004\4\u2000").z();
|
|
test(null, "g\uA001\1",
|
|
"/p% \uA002\2\u2000",
|
|
"q% \uA003\3\u2000",
|
|
"f% \uA004\4\u2000")
|
|
.g("g\uA001%01")
|
|
.p("/p%25%20\uA002%02%E2%80%80").pd("/p% \uA002\2\u2000")
|
|
.q("q%25%20\uA003%03%E2%80%80").qd("q% \uA003\3\u2000")
|
|
.f("f%25%20\uA004%04%E2%80%80").fd("f% \uA004\4\u2000").z();
|
|
test(null, null, "/p% \uA002\2\u2000", "f% \uA004\4\u2000")
|
|
.p("/p%25%20\uA002%02%E2%80%80").pd("/p% \uA002\2\u2000")
|
|
.f("f%25%20\uA004%04%E2%80%80").fd("f% \uA004\4\u2000").z();
|
|
test(null, "/sp% \uA001\1\u2000", "f% \uA004\4\u2000")
|
|
.sp("/sp%25%20\uA001%01%E2%80%80").spd("/sp% \uA001\1\u2000")
|
|
.p("/sp%25%20\uA001%01%E2%80%80").pd("/sp% \uA001\1\u2000")
|
|
.f("f%25%20\uA004%04%E2%80%80").fd("f% \uA004\4\u2000").z();
|
|
|
|
// 4438319: Non-raw accessors decode all escaped octets
|
|
test("/%25%20%E2%82%AC%E2%80%80")
|
|
.p("/%25%20%E2%82%AC%E2%80%80").pd("/% \u20Ac\u2000").z();
|
|
|
|
// 4438319: toASCIIString
|
|
test("/\uCAFE\uBABE")
|
|
.p("/\uCAFE\uBABE").ta("/%EC%AB%BE%EB%AA%BE").z();
|
|
|
|
// 4991359 and 4866303: bad quoting by defineSchemeSpecificPart()
|
|
URI base = new URI ("http://host/foo%20bar/a/b/c/d");
|
|
test ("resolve")
|
|
.rslv(base).spd("//host/foo bar/a/b/c/resolve")
|
|
.sp("//host/foo%20bar/a/b/c/resolve").s("http")
|
|
.pd("/foo bar/a/b/c/resolve").h("host")
|
|
.p("/foo%20bar/a/b/c/resolve").z();
|
|
|
|
// 6773270: java.net.URI fails to escape u0000
|
|
test("s", "a", "/\u0000", null)
|
|
.s("s").p("/%00").h("a")
|
|
.ta("s://a/%00").z();
|
|
}
|
|
|
|
|
|
static void eq0(URI u, URI v) throws URISyntaxException {
|
|
testCount++;
|
|
if (!u.equals(v))
|
|
throw new RuntimeException("Not equal: " + u + " " + v);
|
|
int uh = u.hashCode();
|
|
int vh = v.hashCode();
|
|
if (uh != vh)
|
|
throw new RuntimeException("Hash codes not equal: "
|
|
+ u + " " + Integer.toHexString(uh) + " "
|
|
+ v + " " + Integer.toHexString(vh));
|
|
out.println();
|
|
out.println(u + " == " + v
|
|
+ " [" + Integer.toHexString(uh) + "]");
|
|
}
|
|
|
|
static void cmp0(URI u, URI v, boolean same)
|
|
throws URISyntaxException
|
|
{
|
|
int c = u.compareTo(v);
|
|
if ((c == 0) != same)
|
|
throw new RuntimeException("Comparison inconsistent: " + u + " " + v
|
|
+ " " + c);
|
|
}
|
|
|
|
static void eq(URI u, URI v) throws URISyntaxException {
|
|
eq0(u, v);
|
|
cmp0(u, v, true);
|
|
}
|
|
|
|
static void eq(String expected, String actual) {
|
|
testCount++;
|
|
if (expected == null && actual == null) {
|
|
return;
|
|
}
|
|
if (expected != null && expected.equals(actual)) {
|
|
return;
|
|
}
|
|
throw new AssertionError(String.format(
|
|
"Strings are not equal: '%s', '%s'", expected, actual));
|
|
}
|
|
|
|
static void eqeq(URI u, URI v) {
|
|
testCount++;
|
|
if (u != v)
|
|
throw new RuntimeException("Not ==: " + u + " " + v);
|
|
}
|
|
|
|
static void ne0(URI u, URI v) throws URISyntaxException {
|
|
testCount++;
|
|
if (u.equals(v))
|
|
throw new RuntimeException("Equal: " + u + " " + v);
|
|
out.println();
|
|
out.println(u + " != " + v
|
|
+ " [" + Integer.toHexString(u.hashCode())
|
|
+ " " + Integer.toHexString(v.hashCode())
|
|
+ "]");
|
|
}
|
|
|
|
static void ne(URI u, URI v) throws URISyntaxException {
|
|
ne0(u, v);
|
|
cmp0(u, v, false);
|
|
}
|
|
|
|
static void lt(URI u, URI v) throws URISyntaxException {
|
|
ne0(u, v);
|
|
int c = u.compareTo(v);
|
|
if (c >= 0) {
|
|
show(u);
|
|
show(v);
|
|
throw new RuntimeException("Not less than: " + u + " " + v
|
|
+ " " + c);
|
|
}
|
|
out.println(u + " < " + v);
|
|
}
|
|
|
|
static void lt(String s, String t) throws URISyntaxException {
|
|
lt(new URI(s), new URI(t));
|
|
}
|
|
|
|
static void gt0(URI u, URI v) throws URISyntaxException {
|
|
ne0(u, v);
|
|
int c = u.compareTo(v);
|
|
if (c <= 0) {
|
|
show(u);
|
|
show(v);
|
|
throw new RuntimeException("Not greater than: " + u + " " + v
|
|
+ " " + c);
|
|
}
|
|
out.println(u + " < " + v);
|
|
}
|
|
|
|
static void gt(URI u, URI v) throws URISyntaxException {
|
|
lt(v, u);
|
|
}
|
|
|
|
static void eqHashComp() throws URISyntaxException {
|
|
|
|
header("Equality, hashing, and comparison");
|
|
|
|
URI o = new URI("mailto:foo@bar.com");
|
|
URI r = new URI("reg://some%20registry/b/c/d?q#f");
|
|
URI s = new URI("http://jag:cafebabe@java.sun.com:94/b/c/d?q#f");
|
|
URI t = new URI("http://example.com/%5bsegment%5d");
|
|
eq(o, o);
|
|
lt(o, r);
|
|
lt(s, o);
|
|
lt(s, r);
|
|
|
|
eq(o, new URI("MaILto:foo@bar.com"));
|
|
gt(o, new URI("mailto:foo@bar.COM"));
|
|
eq(r, new URI("rEg://some%20registry/b/c/d?q#f"));
|
|
gt(r, new URI("reg://Some%20Registry/b/c/d?q#f"));
|
|
gt(r, new URI("reg://some%20registry/b/c/D?q#f"));
|
|
eq(s, new URI("hTtP://jag:cafebabe@Java.Sun.COM:94/b/c/d?q#f"));
|
|
gt(s, new URI("http://jag:CafeBabe@java.sun.com:94/b/c/d?q#f"));
|
|
lt(s, new URI("http://jag:cafebabe@java.sun.com:94/b/c/d?r#f"));
|
|
lt(s, new URI("http://jag:cafebabe@java.sun.com:94/b/c/d?q#g"));
|
|
cmp0(t, new URI("http://example.com/%5Bsegment%5D"), true);
|
|
gt0(t, new URI("http://example.com/%5BSegment%5D"));
|
|
lt(new URI("http://example.com/%5Asegment%5D"), new URI("http://example.com/%5Bsegment%5D"));
|
|
eq(new URI("http://host/a%00bcd"), new URI("http://host/a%00bcd"));
|
|
ne(new URI("http://host/a%00bcd"), new URI("http://host/aZ00bcd"));
|
|
eq0(new URI("http://host/abc%e2def%C3ghi"),
|
|
new URI("http://host/abc%E2def%c3ghi"));
|
|
|
|
lt("p", "s:p");
|
|
lt("s:p", "T:p");
|
|
lt("S:p", "t:p");
|
|
lt("s:/p", "s:p");
|
|
lt("s:p", "s:q");
|
|
lt("s:p#f", "s:p#g");
|
|
lt("s://u@h:1", "s://v@h:1");
|
|
lt("s://u@h:1", "s://u@i:1");
|
|
lt("s://u@h:1", "s://v@h:2");
|
|
lt("s://a%20b", "s://a%20c");
|
|
lt("s://a%20b", "s://aab");
|
|
lt("s://AA", "s://A_");
|
|
lt("s:/p", "s:/q");
|
|
lt("s:/p?q", "s:/p?r");
|
|
lt("s:/p#f", "s:/p#g");
|
|
|
|
lt("s://h", "s://h/p");
|
|
lt("s://h/p", "s://h/p?q");
|
|
|
|
}
|
|
|
|
|
|
static void serial(URI u) throws IOException, URISyntaxException {
|
|
|
|
ByteArrayOutputStream bo = new ByteArrayOutputStream();
|
|
ObjectOutputStream oo = new ObjectOutputStream(bo);
|
|
|
|
oo.writeObject(u);
|
|
oo.close();
|
|
|
|
ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
|
|
ObjectInputStream oi = new ObjectInputStream(bi);
|
|
try {
|
|
Object o = oi.readObject();
|
|
eq(u, (URI)o);
|
|
} catch (ClassNotFoundException x) {
|
|
x.printStackTrace();
|
|
throw new RuntimeException(x.toString());
|
|
}
|
|
|
|
testCount++;
|
|
}
|
|
|
|
static void serial() throws IOException, URISyntaxException {
|
|
header("Serialization");
|
|
|
|
serial(URI.create("http://java.sun.com/jdk/1.4?release#beta"));
|
|
serial(URI.create("s://h/p").resolve("/long%20path/"));
|
|
}
|
|
|
|
|
|
static void urls() throws URISyntaxException {
|
|
|
|
header("URLs");
|
|
|
|
URI uri;
|
|
URL url;
|
|
boolean caught = false;
|
|
|
|
out.println();
|
|
uri = new URI("http://a/p?q#f");
|
|
try {
|
|
url = uri.toURL();
|
|
} catch (MalformedURLException x) {
|
|
throw new RuntimeException(x.toString());
|
|
}
|
|
if (!url.toString().equals("http://a/p?q#f"))
|
|
throw new RuntimeException("Incorrect URL: " + url);
|
|
out.println(uri + " url --> " + url);
|
|
|
|
out.println();
|
|
uri = new URI("a/b");
|
|
try {
|
|
out.println(uri + " url --> ");
|
|
url = uri.toURL();
|
|
} catch (IllegalArgumentException x) {
|
|
caught = true;
|
|
out.println("Correct exception: " + x);
|
|
} catch (MalformedURLException x) {
|
|
caught = true;
|
|
throw new RuntimeException("Incorrect exception: " + x);
|
|
}
|
|
if (!caught)
|
|
throw new RuntimeException("Incorrect URL: " + url);
|
|
|
|
out.println();
|
|
uri = new URI("foo://bar/baz");
|
|
caught = false;
|
|
try {
|
|
out.println(uri + " url --> ");
|
|
url = uri.toURL();
|
|
} catch (MalformedURLException x) {
|
|
caught = true;
|
|
out.println("Correct exception: " + x);
|
|
} catch (IllegalArgumentException x) {
|
|
caught = true;
|
|
throw new RuntimeException("Incorrect exception: " + x);
|
|
}
|
|
if (!caught)
|
|
throw new RuntimeException("Incorrect URL: " + url);
|
|
|
|
testCount += 3;
|
|
}
|
|
|
|
|
|
static void tests() throws IOException, URISyntaxException {
|
|
rfc2396();
|
|
ip();
|
|
misc();
|
|
chars();
|
|
eqHashComp();
|
|
serial();
|
|
urls();
|
|
npes();
|
|
bugs();
|
|
}
|
|
|
|
|
|
// -- Command-line invocation --
|
|
|
|
static void usage() {
|
|
out.println("Usage:");
|
|
out.println(" java Test -- Runs all tests in this file");
|
|
out.println(" java Test <uri> -- Parses uri, shows components");
|
|
out.println(" java Test <base> <uri> -- Parses uri and base, then resolves");
|
|
out.println(" uri against base");
|
|
}
|
|
|
|
static void clargs(String base, String uri) {
|
|
URI b = null, u;
|
|
try {
|
|
if (base != null) {
|
|
b = new URI(base);
|
|
out.println(base);
|
|
show(b);
|
|
}
|
|
u = new URI(uri);
|
|
out.println(uri);
|
|
show(u);
|
|
if (base != null) {
|
|
URI r = b.resolve(u);
|
|
out.println(r);
|
|
show(r);
|
|
}
|
|
} catch (URISyntaxException x) {
|
|
show("ERROR", x);
|
|
x.printStackTrace(out);
|
|
}
|
|
}
|
|
|
|
|
|
// miscellaneous bugs/rfes that don't fit in with the test framework
|
|
|
|
static void bugs() {
|
|
header("Bugs");
|
|
b6339649();
|
|
b6933879();
|
|
b8037396();
|
|
b8051627();
|
|
b8272072();
|
|
b8297687();
|
|
}
|
|
|
|
private static void b8297687() {
|
|
// constructors that take a hostname should fail
|
|
test("ftps", "p.e.local|SIT@p.e.local", "/path", null)
|
|
.x().z();
|
|
test("ftps", null,"p.e.local|SIT@p.e.local", -1, "/path", null, null)
|
|
.x().z();
|
|
// constructors that take an authority component should succeed
|
|
test("ftps", "p.e.local|SIT@p.e.local", "/path", null,null)
|
|
.s("ftps")
|
|
.sp("//p.e.local%7CSIT@p.e.local/path")
|
|
.spd("//p.e.local|SIT@p.e.local/path")
|
|
.u("p.e.local%7CSIT")
|
|
.ud("p.e.local|SIT")
|
|
.h("p.e.local")
|
|
.n(-1)
|
|
.p("/path")
|
|
.pd("/path")
|
|
.z();
|
|
|
|
// check index in exception for constructors that should fail
|
|
try {
|
|
URI uri = new URI("ftps", "p.e.local|SIT@p.e.local", "/path", null);
|
|
throw new AssertionError("Expected URISyntaxException not thrown for " + uri);
|
|
} catch (URISyntaxException ex) {
|
|
if (ex.getMessage().contains("at index 16")) {
|
|
System.out.println("Got expected exception: " + ex);
|
|
} else {
|
|
throw new AssertionError("Exception does not point at index 16", ex);
|
|
}
|
|
}
|
|
testCount++;
|
|
|
|
// check index in exception for constructors that should fail
|
|
try {
|
|
URI uri = new URI("ftps", null, "p.e.local|SIT@p.e.local", -1, "/path", null, null);
|
|
throw new AssertionError("Expected URISyntaxException not thrown for " + uri);
|
|
} catch (URISyntaxException ex) {
|
|
if (ex.getMessage().contains("at index 16")) {
|
|
System.out.println("Got expected exception: " + ex);
|
|
} else {
|
|
throw new AssertionError("Exception does not point at index 16", ex);
|
|
}
|
|
}
|
|
testCount++;
|
|
}
|
|
|
|
// 6339649 - include detail message from nested exception
|
|
private static void b6339649() {
|
|
try {
|
|
URI uri = URI.create("http://nowhere.net/should not be permitted");
|
|
} catch (IllegalArgumentException e) {
|
|
if ("".equals(e.getMessage()) || e.getMessage() == null) {
|
|
throw new RuntimeException ("No detail message");
|
|
}
|
|
}
|
|
testCount++;
|
|
}
|
|
|
|
// 6933879 - check that "." and "_" characters are allowed in IPv6 scope_id.
|
|
private static void b6933879() {
|
|
final String HOST = "fe80::c00:16fe:cebe:3214%eth1.12_55";
|
|
URI uri;
|
|
try {
|
|
uri = new URI("http", null, HOST, 10, "/", null, null);
|
|
} catch (URISyntaxException ex) {
|
|
throw new AssertionError("Should not happen", ex);
|
|
}
|
|
eq("[" + HOST + "]", uri.getHost());
|
|
}
|
|
|
|
private static void b8037396() {
|
|
|
|
// primary checks:
|
|
|
|
URI u;
|
|
try {
|
|
u = new URI("http", "example.org", "/[a b]", "[a b]", "[a b]");
|
|
} catch (URISyntaxException e) {
|
|
throw new AssertionError("shouldn't ever happen", e);
|
|
}
|
|
eq("/[a b]", u.getPath());
|
|
eq("[a b]", u.getQuery());
|
|
eq("[a b]", u.getFragment());
|
|
|
|
// additional checks:
|
|
// * '%' symbols are still decoded outside square brackets
|
|
// * the getRawXXX() functionality left intact
|
|
|
|
try {
|
|
u = new URI("http", "example.org", "/a b[c d]", "a b[c d]", "a b[c d]");
|
|
} catch (URISyntaxException e) {
|
|
throw new AssertionError("shouldn't ever happen", e);
|
|
}
|
|
|
|
eq("/a b[c d]", u.getPath());
|
|
eq("a b[c d]", u.getQuery());
|
|
eq("a b[c d]", u.getFragment());
|
|
|
|
eq("/a%20b%5Bc%20d%5D", u.getRawPath());
|
|
eq("a%20b[c%20d]", u.getRawQuery());
|
|
eq("a%20b[c%20d]", u.getRawFragment());
|
|
}
|
|
|
|
// 8051627 - Invariants about java.net.URI resolve and relativize are wrong
|
|
private static void b8051627() {
|
|
try {
|
|
// Let u be a normalized absolute URI u which ends with "/" and
|
|
// v be a normalized relative URI v which does not start with "." or "/", then
|
|
// u.relativize(u.resolve(v)).equals(v) should be true
|
|
reltivizeAfterResolveTest("http://a/b/", "c/d", "c/d");
|
|
reltivizeAfterResolveTest("http://a/b/", "g;x?y#s", "g;x?y#s");
|
|
|
|
// when the URI condition is not met, u.relativize(u.resolve(v)).equals(v) may be false
|
|
// In the following examples, that should be false
|
|
reltivizeAfterResolveTest("http://a/b", "c/d", "http://a/c/d");
|
|
reltivizeAfterResolveTest("http://a/b/", "../c/d", "http://a/c/d");
|
|
reltivizeAfterResolveTest("http://a/b/", "/c/d", "http://a/c/d");
|
|
reltivizeAfterResolveTest("http://a/b/", "http://a/b/c/d", "c/d");
|
|
|
|
// Let u be a normalized absolute URI u which ends with "/" and
|
|
// v be a normalized absolute URI v, then
|
|
// u.resolve(u.relativize(v)).equals(v) should be true
|
|
resolveAfterRelativizeTest("http://a/b/", "http://a/b/c/d", "http://a/b/c/d");
|
|
resolveAfterRelativizeTest("http://a/b/", "http://a/b/c/g;x?y#s", "http://a/b/c/g;x?y#s");
|
|
|
|
// when the URI condition is not met, u.resolve(u.relativize(v)).equals(v) may be false
|
|
// In the following examples, that should be false
|
|
resolveAfterRelativizeTest("http://a/b", "http://a/b/c/d", "http://a/c/d");
|
|
resolveAfterRelativizeTest("http://a/b/", "c/d", "http://a/b/c/d");
|
|
} catch (URISyntaxException e) {
|
|
throw new AssertionError("shouldn't ever happen", e);
|
|
}
|
|
}
|
|
private static void reltivizeAfterResolveTest(String base, String target, String expected)
|
|
throws URISyntaxException {
|
|
URI baseURI = URI.create(base);
|
|
URI targetURI = URI.create(target);
|
|
eq(URI.create(expected), baseURI.relativize(baseURI.resolve(targetURI)));
|
|
}
|
|
private static void resolveAfterRelativizeTest(String base, String target, String expected)
|
|
throws URISyntaxException {
|
|
URI baseURI = URI.create(base);
|
|
URI targetURI = URI.create(target);
|
|
eq(URI.create(expected), baseURI.resolve(baseURI.relativize(targetURI)));
|
|
}
|
|
|
|
// 8272072 - Resolving URI relative path with no "/" may lead to incorrect toString
|
|
private static void b8272072() {
|
|
try {
|
|
URI baseURI = new URI("http://example.com");
|
|
URI relativeURI = new URI("test");
|
|
URI resolvedURI = baseURI.resolve(relativeURI);
|
|
|
|
eq(new URI("http://example.com/test"), resolvedURI);
|
|
|
|
baseURI = new URI("relativeBase");
|
|
resolvedURI = baseURI.resolve(relativeURI);
|
|
|
|
eq(new URI("test"), resolvedURI);
|
|
} catch (URISyntaxException e) {
|
|
throw new AssertionError("shouldn't ever happen", e);
|
|
}
|
|
}
|
|
|
|
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;
|
|
|
|
}
|
|
}
|
|
|
|
}
|