gecko-dev/addon-sdk/source/test/test-url.js

355 lines
13 KiB
JavaScript

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
var url = require("sdk/url");
exports.testResolve = function(test) {
test.assertEqual(url.URL("bar", "http://www.foo.com/").toString(),
"http://www.foo.com/bar");
test.assertEqual(url.URL("bar", "http://www.foo.com"),
"http://www.foo.com/bar");
test.assertEqual(url.URL("http://bar.com/", "http://foo.com/"),
"http://bar.com/",
"relative should override base");
test.assertRaises(function() { url.URL("blah"); },
"malformed URI: blah",
"url.resolve() should throw malformed URI on base");
test.assertRaises(function() { url.URL("chrome://global"); },
"invalid URI: chrome://global",
"url.resolve() should throw invalid URI on base");
test.assertRaises(function() { url.URL("chrome://foo/bar"); },
"invalid URI: chrome://foo/bar",
"url.resolve() should throw on bad chrome URI");
test.assertEqual(url.URL("", "http://www.foo.com"),
"http://www.foo.com/",
"url.resolve() should add slash to end of domain");
};
exports.testParseHttp = function(test) {
var info = url.URL("http://foo.com/bar");
test.assertEqual(info.scheme, "http");
test.assertEqual(info.host, "foo.com");
test.assertEqual(info.port, null);
test.assertEqual(info.userPass, null);
test.assertEqual(info.path, "/bar");
};
exports.testParseHttpWithPort = function(test) {
var info = url.URL("http://foo.com:5/bar");
test.assertEqual(info.port, 5);
};
exports.testParseChrome = function(test) {
var info = url.URL("chrome://global/content/blah");
test.assertEqual(info.scheme, "chrome");
test.assertEqual(info.host, "global");
test.assertEqual(info.port, null);
test.assertEqual(info.userPass, null);
test.assertEqual(info.path, "/content/blah");
};
exports.testParseAbout = function(test) {
var info = url.URL("about:boop");
test.assertEqual(info.scheme, "about");
test.assertEqual(info.host, null);
test.assertEqual(info.port, null);
test.assertEqual(info.userPass, null);
test.assertEqual(info.path, "boop");
};
exports.testParseFTP = function(test) {
var info = url.URL("ftp://1.2.3.4/foo");
test.assertEqual(info.scheme, "ftp");
test.assertEqual(info.host, "1.2.3.4");
test.assertEqual(info.port, null);
test.assertEqual(info.userPass, null);
test.assertEqual(info.path, "/foo");
};
exports.testParseFTPWithUserPass = function(test) {
var info = url.URL("ftp://user:pass@1.2.3.4/foo");
test.assertEqual(info.userPass, "user:pass");
};
exports.testToFilename = function(test) {
test.assertRaises(
function() { url.toFilename("resource://nonexistent"); },
"resource does not exist: resource://nonexistent/",
"url.toFilename() on nonexistent resources should throw"
);
test.assertRaises(
function() { url.toFilename("http://foo.com/"); },
"cannot map to filename: http://foo.com/",
"url.toFilename() on http: URIs should raise error"
);
try {
test.assertMatches(
url.toFilename("chrome://global/content/console.xul"),
/.*console\.xul$/,
"url.toFilename() w/ console.xul works when it maps to filesystem"
);
} catch (e) {
if (/chrome url isn\'t on filesystem/.test(e.message))
test.pass("accessing console.xul in jar raises exception");
else
test.fail("accessing console.xul raises " + e);
}
// TODO: Are there any chrome URLs that we're certain exist on the
// filesystem?
// test.assertMatches(url.toFilename("chrome://myapp/content/main.js"),
// /.*main\.js$/);
};
exports.testFromFilename = function(test) {
var profileDirName = require("sdk/system").pathFor("ProfD");
var fileUrl = url.fromFilename(profileDirName);
test.assertEqual(url.URL(fileUrl).scheme, 'file',
'url.toFilename() should return a file: url');
test.assertEqual(url.fromFilename(url.toFilename(fileUrl)),
fileUrl);
};
exports.testURL = function(test) {
let URL = url.URL;
test.assert(URL("h:foo") instanceof URL, "instance is of correct type");
test.assertRaises(function() URL(),
"malformed URI: undefined",
"url.URL should throw on undefined");
test.assertRaises(function() URL(""),
"malformed URI: ",
"url.URL should throw on empty string");
test.assertRaises(function() URL("foo"),
"malformed URI: foo",
"url.URL should throw on invalid URI");
test.assert(URL("h:foo").scheme, "has scheme");
test.assertEqual(URL("h:foo").toString(),
"h:foo",
"toString should roundtrip");
// test relative + base
test.assertEqual(URL("mypath", "http://foo").toString(),
"http://foo/mypath",
"relative URL resolved to base");
// test relative + no base
test.assertRaises(function() URL("path").toString(),
"malformed URI: path",
"no base for relative URI should throw");
let a = URL("h:foo");
let b = URL(a);
test.assertEqual(b.toString(),
"h:foo",
"a URL can be initialized from another URL");
test.assertNotStrictEqual(a, b,
"a URL initialized from another URL is not the same object");
test.assert(a == "h:foo",
"toString is implicit when a URL is compared to a string via ==");
test.assertStrictEqual(a + "", "h:foo",
"toString is implicit when a URL is concatenated to a string");
};
exports.testStringInterface = function(test) {
let URL = url.URL;
var EM = "about:addons";
var a = URL(EM);
// make sure the standard URL properties are enumerable and not the String interface bits
test.assertEqual(Object.keys(a), "scheme,userPass,host,port,path", "enumerable key list check for URL.");
test.assertEqual(
JSON.stringify(a),
"{\"scheme\":\"about\",\"userPass\":null,\"host\":null,\"port\":null,\"path\":\"addons\"}",
"JSON.stringify should return a object with correct props and vals.");
// make sure that the String interface exists and works as expected
test.assertEqual(a.indexOf(":"), EM.indexOf(":"), "indexOf on URL works");
test.assertEqual(a.valueOf(), EM.valueOf(), "valueOf on URL works.");
test.assertEqual(a.toSource(), EM.toSource(), "toSource on URL works.");
test.assertEqual(a.lastIndexOf("a"), EM.lastIndexOf("a"), "lastIndexOf on URL works.");
test.assertEqual(a.match("t:").toString(), EM.match("t:").toString(), "match on URL works.");
test.assertEqual(a.toUpperCase(), EM.toUpperCase(), "toUpperCase on URL works.");
test.assertEqual(a.toLowerCase(), EM.toLowerCase(), "toLowerCase on URL works.");
test.assertEqual(a.split(":").toString(), EM.split(":").toString(), "split on URL works.");
test.assertEqual(a.charAt(2), EM.charAt(2), "charAt on URL works.");
test.assertEqual(a.charCodeAt(2), EM.charCodeAt(2), "charCodeAt on URL works.");
test.assertEqual(a.concat(EM), EM.concat(a), "concat on URL works.");
test.assertEqual(a.substr(2,3), EM.substr(2,3), "substr on URL works.");
test.assertEqual(a.substring(2,3), EM.substring(2,3), "substring on URL works.");
test.assertEqual(a.trim(), EM.trim(), "trim on URL works.");
test.assertEqual(a.trimRight(), EM.trimRight(), "trimRight on URL works.");
test.assertEqual(a.trimLeft(), EM.trimLeft(), "trimLeft on URL works.");
}
exports.testDataURLwithouthURI = function (test) {
const { DataURL } = url;
let dataURL = new DataURL();
test.assertEqual(dataURL.base64, false, "base64 is false for empty uri")
test.assertEqual(dataURL.data, "", "data is an empty string for empty uri")
test.assertEqual(dataURL.mimeType, "", "mimeType is an empty string for empty uri")
test.assertEqual(Object.keys(dataURL.parameters).length, 0, "parameters is an empty object for empty uri");
test.assertEqual(dataURL.toString(), "data:,");
}
exports.testDataURLwithMalformedURI = function (test) {
const { DataURL } = url;
test.assertRaises(function() {
let dataURL = new DataURL("http://www.mozilla.com/");
},
"Malformed Data URL: http://www.mozilla.com/",
"DataURL raises an exception for malformed data uri"
);
}
exports.testDataURLparse = function (test) {
const { DataURL } = url;
let dataURL = new DataURL("data:text/html;charset=US-ASCII,%3Ch1%3EHello!%3C%2Fh1%3E");
test.assertEqual(dataURL.base64, false, "base64 is false for non base64 data uri")
test.assertEqual(dataURL.data, "<h1>Hello!</h1>", "data is properly decoded")
test.assertEqual(dataURL.mimeType, "text/html", "mimeType is set properly")
test.assertEqual(Object.keys(dataURL.parameters).length, 1, "one parameters specified");
test.assertEqual(dataURL.parameters["charset"], "US-ASCII", "charset parsed");
test.assertEqual(dataURL.toString(), "data:text/html;charset=US-ASCII,%3Ch1%3EHello!%3C%2Fh1%3E");
}
exports.testDataURLparseBase64 = function (test) {
const { DataURL } = url;
const { decode } = require("sdk/base64");
let text = "Awesome!";
let b64text = "QXdlc29tZSE=";
let dataURL = new DataURL("data:text/plain;base64," + b64text);
test.assertEqual(dataURL.base64, true, "base64 is true for base64 encoded data uri")
test.assertEqual(dataURL.data, text, "data is properly decoded")
test.assertEqual(dataURL.mimeType, "text/plain", "mimeType is set properly")
test.assertEqual(Object.keys(dataURL.parameters).length, 1, "one parameters specified");
test.assertEqual(dataURL.parameters["base64"], "", "parameter set without value");
test.assertEqual(dataURL.toString(), "data:text/plain;base64," + encodeURIComponent(b64text));
}
exports.testIsValidURI = function (test) {
validURIs().forEach(function (aUri) {
test.assertEqual(url.isValidURI(aUri), true, aUri + ' is a valid URL');
});
};
exports.testIsInvalidURI = function (test) {
invalidURIs().forEach(function (aUri) {
test.assertEqual(url.isValidURI(aUri), false, aUri + ' is an invalid URL');
});
};
exports.testURLFromURL = function(test) {
let aURL = url.URL('http://mozilla.org');
let bURL = url.URL(aURL);
test.assertEqual(aURL.toString(), bURL.toString(), 'Making a URL from a URL works');
};
function validURIs() {
return [
'http://foo.com/blah_blah',
'http://foo.com/blah_blah/',
'http://foo.com/blah_blah_(wikipedia)',
'http://foo.com/blah_blah_(wikipedia)_(again)',
'http://www.example.com/wpstyle/?p=364',
'https://www.example.com/foo/?bar=baz&amp;inga=42&amp;quux',
'http://✪df.ws/123',
'http://userid:password@example.com:8080',
'http://userid:password@example.com:8080/',
'http://userid@example.com',
'http://userid@example.com/',
'http://userid@example.com:8080',
'http://userid@example.com:8080/',
'http://userid:password@example.com',
'http://userid:password@example.com/',
'http://142.42.1.1/',
'http://142.42.1.1:8080/',
'http://➡.ws/䨹',
'http://⌘.ws',
'http://⌘.ws/',
'http://foo.com/blah_(wikipedia)#cite-1',
'http://foo.com/blah_(wikipedia)_blah#cite-1',
'http://foo.com/unicode_(✪)_in_parens',
'http://foo.com/(something)?after=parens',
'http://☺.damowmow.com/',
'http://code.google.com/events/#&amp;product=browser',
'http://j.mp',
'ftp://foo.bar/baz',
'http://foo.bar/?q=Test%20URL-encoded%20stuff',
'http://مثال.إختبار',
'http://例子.测试',
'http://उदाहरण.परीक्षा',
'http://-.~_!$&amp;\'()*+,;=:%40:80%2f::::::@example.com',
'http://1337.net',
'http://a.b-c.de',
'http://223.255.255.254',
// Also want to validate data-uris, localhost
'http://localhost:8432/some-file.js',
'data:text/plain;base64,',
'data:text/html;charset=US-ASCII,%3Ch1%3EHello!%3C%2Fh1%3E',
'data:text/html;charset=utf-8,'
];
}
// Some invalidURIs are valid according to the regex used,
// can be improved in the future, but better to pass some
// invalid URLs than prevent valid URLs
function invalidURIs () {
return [
// 'http://',
// 'http://.',
// 'http://..',
// 'http://../',
// 'http://?',
// 'http://??',
// 'http://??/',
// 'http://#',
// 'http://##',
// 'http://##/',
// 'http://foo.bar?q=Spaces should be encoded',
'not a url',
'//',
'//a',
'///a',
'///',
// 'http:///a',
'foo.com',
'http:// shouldfail.com',
':// should fail',
// 'http://foo.bar/foo(bar)baz quux',
// 'http://-error-.invalid/',
// 'http://a.b--c.de/',
// 'http://-a.b.co',
// 'http://a.b-.co',
// 'http://0.0.0.0',
// 'http://10.1.1.0',
// 'http://10.1.1.255',
// 'http://224.1.1.1',
// 'http://1.1.1.1.1',
// 'http://123.123.123',
// 'http://3628126748',
// 'http://.www.foo.bar/',
// 'http://www.foo.bar./',
// 'http://.www.foo.bar./',
// 'http://10.1.1.1',
// 'http://10.1.1.254'
];
}