|
|
|
'use strict';
|
|
|
|
|
|
|
|
function getPunycode() {
|
|
|
|
try {
|
|
|
|
return process.binding('icu');
|
|
|
|
} catch (err) {
|
|
|
|
return require('punycode');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const punycode = getPunycode();
|
|
|
|
const binding = process.binding('url');
|
|
|
|
const context = Symbol('context');
|
|
|
|
const cannotBeBase = Symbol('cannot-be-base');
|
|
|
|
const special = Symbol('special');
|
|
|
|
const searchParams = Symbol('query');
|
|
|
|
const querystring = require('querystring');
|
|
|
|
|
|
|
|
const kScheme = Symbol('scheme');
|
|
|
|
const kHost = Symbol('host');
|
|
|
|
const kPort = Symbol('port');
|
|
|
|
const kDomain = Symbol('domain');
|
|
|
|
|
|
|
|
// https://tc39.github.io/ecma262/#sec-%iteratorprototype%-object
|
|
|
|
const IteratorPrototype = Object.getPrototypeOf(
|
|
|
|
Object.getPrototypeOf([][Symbol.iterator]())
|
|
|
|
);
|
|
|
|
|
|
|
|
function StorageObject() {}
|
|
|
|
StorageObject.prototype = Object.create(null);
|
|
|
|
|
|
|
|
class OpaqueOrigin {
|
|
|
|
toString() {
|
|
|
|
return 'null';
|
|
|
|
}
|
|
|
|
|
|
|
|
get effectiveDomain() {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class TupleOrigin {
|
|
|
|
constructor(scheme, host, port, domain) {
|
|
|
|
this[kScheme] = scheme;
|
|
|
|
this[kHost] = host;
|
|
|
|
this[kPort] = port;
|
|
|
|
this[kDomain] = domain;
|
|
|
|
}
|
|
|
|
|
|
|
|
get scheme() {
|
|
|
|
return this[kScheme];
|
|
|
|
}
|
|
|
|
|
|
|
|
get host() {
|
|
|
|
return this[kHost];
|
|
|
|
}
|
|
|
|
|
|
|
|
get port() {
|
|
|
|
return this[kPort];
|
|
|
|
}
|
|
|
|
|
|
|
|
get domain() {
|
|
|
|
return this[kDomain];
|
|
|
|
}
|
|
|
|
|
|
|
|
get effectiveDomain() {
|
|
|
|
return this[kDomain] || this[kHost];
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://url.spec.whatwg.org/#dom-url-origin
|
|
|
|
toString(unicode = true) {
|
|
|
|
var result = this[kScheme];
|
|
|
|
result += '://';
|
|
|
|
result += unicode ? domainToUnicode(this[kHost]) : this[kHost];
|
|
|
|
if (this[kPort] !== undefined && this[kPort] !== null)
|
|
|
|
result += `:${this[kPort]}`;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
inspect() {
|
|
|
|
return `TupleOrigin {
|
|
|
|
scheme: ${this[kScheme]},
|
|
|
|
host: ${this[kHost]},
|
|
|
|
port: ${this[kPort]},
|
|
|
|
domain: ${this[kDomain]}
|
|
|
|
}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
throw new TypeError('Invalid URL');
|
|
|
|
var ctx = this[context];
|
|
|
|
ctx.flags = flags;
|
|
|
|
ctx.scheme = protocol;
|
|
|
|
ctx.username = username;
|
|
|
|
ctx.password = password;
|
|
|
|
ctx.port = port;
|
|
|
|
ctx.path = path;
|
|
|
|
ctx.query = query;
|
|
|
|
ctx.fragment = fragment;
|
|
|
|
ctx.host = host;
|
|
|
|
if (this[searchParams]) { // invoked from href setter
|
|
|
|
initSearchParams(this[searchParams], query);
|
|
|
|
} else {
|
|
|
|
this[searchParams] = new URLSearchParams(query);
|
|
|
|
}
|
|
|
|
this[searchParams][context] = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reused by URL constructor and URL#href setter.
|
|
|
|
function parse(url, input, base) {
|
|
|
|
input = String(input);
|
|
|
|
const base_context = base ? base[context] : undefined;
|
|
|
|
url[context] = new StorageObject();
|
|
|
|
binding.parse(input.trim(), -1,
|
|
|
|
base_context, undefined,
|
|
|
|
onParseComplete.bind(url));
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseProtocolComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const newIsSpecial = (flags & binding.URL_FLAGS_SPECIAL) !== 0;
|
|
|
|
const s = this[special];
|
|
|
|
const ctx = this[context];
|
|
|
|
if ((s && !newIsSpecial) || (!s && newIsSpecial)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (newIsSpecial) {
|
|
|
|
ctx.flags |= binding.URL_FLAGS_SPECIAL;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_SPECIAL;
|
|
|
|
}
|
|
|
|
if (protocol) {
|
|
|
|
ctx.scheme = protocol;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_SCHEME;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_SCHEME;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseHostComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (host) {
|
|
|
|
ctx.host = host;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_HOST;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_HOST;
|
|
|
|
}
|
|
|
|
if (port !== undefined)
|
|
|
|
ctx.port = port;
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseHostnameComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (host) {
|
|
|
|
ctx.host = host;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_HOST;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_HOST;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParsePortComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
this[context].port = port;
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParsePathComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (path) {
|
|
|
|
ctx.path = path;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_PATH;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_PATH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseSearchComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (query) {
|
|
|
|
ctx.query = query;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_QUERY;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_QUERY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onParseHashComplete(flags, protocol, username, password,
|
|
|
|
host, port, path, query, fragment) {
|
|
|
|
if (flags & binding.URL_FLAGS_FAILED)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (fragment) {
|
|
|
|
ctx.fragment = fragment;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_FRAGMENT;
|
|
|
|
} else {
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_FRAGMENT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class URL {
|
|
|
|
constructor(input, base) {
|
|
|
|
if (base !== undefined && !(base instanceof URL))
|
|
|
|
base = new URL(String(base));
|
|
|
|
parse(this, input, base);
|
|
|
|
}
|
|
|
|
|
|
|
|
get [special]() {
|
|
|
|
return (this[context].flags & binding.URL_FLAGS_SPECIAL) !== 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
get [cannotBeBase]() {
|
|
|
|
return (this[context].flags & binding.URL_FLAGS_CANNOT_BE_BASE) !== 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inspect(depth, opts) {
|
|
|
|
const ctx = this[context];
|
|
|
|
var ret = 'URL {\n';
|
|
|
|
ret += ` href: ${this.href}\n`;
|
|
|
|
if (ctx.scheme !== undefined)
|
|
|
|
ret += ` protocol: ${this.protocol}\n`;
|
|
|
|
if (ctx.username !== undefined)
|
|
|
|
ret += ` username: ${this.username}\n`;
|
|
|
|
if (ctx.password !== undefined) {
|
|
|
|
const pwd = opts.showHidden ? ctx.password : '--------';
|
|
|
|
ret += ` password: ${pwd}\n`;
|
|
|
|
}
|
|
|
|
if (ctx.host !== undefined)
|
|
|
|
ret += ` hostname: ${this.hostname}\n`;
|
|
|
|
if (ctx.port !== undefined)
|
|
|
|
ret += ` port: ${this.port}\n`;
|
|
|
|
if (ctx.path !== undefined)
|
|
|
|
ret += ` pathname: ${this.pathname}\n`;
|
|
|
|
if (ctx.query !== undefined)
|
|
|
|
ret += ` search: ${this.search}\n`;
|
|
|
|
if (ctx.fragment !== undefined)
|
|
|
|
ret += ` hash: ${this.hash}\n`;
|
|
|
|
if (opts.showHidden) {
|
|
|
|
ret += ` cannot-be-base: ${this[cannotBeBase]}\n`;
|
|
|
|
ret += ` special: ${this[special]}\n`;
|
|
|
|
}
|
|
|
|
ret += '}';
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.defineProperties(URL.prototype, {
|
|
|
|
toString: {
|
|
|
|
// https://heycam.github.io/webidl/#es-stringifier
|
|
|
|
writable: true,
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
// eslint-disable-next-line func-name-matching
|
|
|
|
value: function toString(options) {
|
|
|
|
options = options || {};
|
|
|
|
const fragment =
|
|
|
|
options.fragment !== undefined ?
|
|
|
|
!!options.fragment : true;
|
|
|
|
const unicode = !!options.unicode;
|
|
|
|
const ctx = this[context];
|
|
|
|
var ret;
|
|
|
|
if (this.protocol)
|
|
|
|
ret = this.protocol;
|
|
|
|
if (ctx.host !== undefined) {
|
|
|
|
ret += '//';
|
|
|
|
const has_username = typeof ctx.username === 'string';
|
|
|
|
const has_password = typeof ctx.password === 'string';
|
|
|
|
if (has_username || has_password) {
|
|
|
|
if (has_username)
|
|
|
|
ret += ctx.username;
|
|
|
|
if (has_password)
|
|
|
|
ret += `:${ctx.password}`;
|
|
|
|
ret += '@';
|
|
|
|
}
|
|
|
|
if (unicode) {
|
|
|
|
ret += punycode.toUnicode(this.hostname);
|
|
|
|
if (this.port !== undefined)
|
|
|
|
ret += `:${this.port}`;
|
|
|
|
} else {
|
|
|
|
ret += this.host;
|
|
|
|
}
|
|
|
|
} else if (ctx.scheme === 'file:') {
|
|
|
|
ret += '//';
|
|
|
|
}
|
|
|
|
if (this.pathname)
|
|
|
|
ret += this.pathname;
|
|
|
|
if (typeof ctx.query === 'string')
|
|
|
|
ret += `?${ctx.query}`;
|
|
|
|
if (fragment & typeof ctx.fragment === 'string')
|
|
|
|
ret += `#${ctx.fragment}`;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
href: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this.toString();
|
|
|
|
},
|
|
|
|
set(input) {
|
|
|
|
parse(this, input);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
origin: { // readonly
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return originFor(this).toString();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
protocol: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this[context].scheme;
|
|
|
|
},
|
|
|
|
set(scheme) {
|
|
|
|
scheme = String(scheme);
|
|
|
|
if (scheme.length === 0)
|
|
|
|
return;
|
|
|
|
binding.parse(scheme, binding.kSchemeStart, null, this[context],
|
|
|
|
onParseProtocolComplete.bind(this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
username: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this[context].username || '';
|
|
|
|
},
|
|
|
|
set(username) {
|
|
|
|
username = String(username);
|
|
|
|
if (!this.hostname)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (!username) {
|
|
|
|
ctx.username = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_USERNAME;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ctx.username = binding.encodeAuth(username);
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_USERNAME;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
password: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this[context].password || '';
|
|
|
|
},
|
|
|
|
set(password) {
|
|
|
|
password = String(password);
|
|
|
|
if (!this.hostname)
|
|
|
|
return;
|
|
|
|
const ctx = this[context];
|
|
|
|
if (!password) {
|
|
|
|
ctx.password = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_PASSWORD;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ctx.password = binding.encodeAuth(password);
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_PASSWORD;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
host: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
const ctx = this[context];
|
|
|
|
var ret = ctx.host || '';
|
|
|
|
if (ctx.port !== undefined)
|
|
|
|
ret += `:${ctx.port}`;
|
|
|
|
return ret;
|
|
|
|
},
|
|
|
|
set(host) {
|
|
|
|
const ctx = this[context];
|
|
|
|
host = String(host);
|
|
|
|
if (this[cannotBeBase] ||
|
|
|
|
(this[special] && host.length === 0)) {
|
|
|
|
// Cannot set the host if cannot-be-base is set or
|
|
|
|
// scheme is special and host length is zero
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!host) {
|
|
|
|
ctx.host = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_HOST;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
binding.parse(host, binding.kHost, null, ctx,
|
|
|
|
onParseHostComplete.bind(this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
hostname: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this[context].host || '';
|
|
|
|
},
|
|
|
|
set(host) {
|
|
|
|
const ctx = this[context];
|
|
|
|
host = String(host);
|
|
|
|
if (this[cannotBeBase] ||
|
|
|
|
(this[special] && host.length === 0)) {
|
|
|
|
// Cannot set the host if cannot-be-base is set or
|
|
|
|
// scheme is special and host length is zero
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!host) {
|
|
|
|
ctx.host = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_HOST;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
binding.parse(host, binding.kHostname, null, ctx,
|
|
|
|
onParseHostnameComplete.bind(this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
port: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
const port = this[context].port;
|
|
|
|
return port === undefined ? '' : String(port);
|
|
|
|
},
|
|
|
|
set(port) {
|
|
|
|
const ctx = this[context];
|
|
|
|
if (!ctx.host || this[cannotBeBase] ||
|
|
|
|
this.protocol === 'file:')
|
|
|
|
return;
|
|
|
|
port = String(port);
|
|
|
|
if (port === '') {
|
|
|
|
ctx.port = undefined;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
binding.parse(port, binding.kPort, null, ctx,
|
|
|
|
onParsePortComplete.bind(this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
pathname: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
const ctx = this[context];
|
|
|
|
if (this[cannotBeBase])
|
|
|
|
return ctx.path[0];
|
|
|
|
return ctx.path !== undefined ? `/${ctx.path.join('/')}` : '';
|
|
|
|
},
|
|
|
|
set(path) {
|
|
|
|
if (this[cannotBeBase])
|
|
|
|
return;
|
|
|
|
binding.parse(String(path), binding.kPathStart, null, this[context],
|
|
|
|
onParsePathComplete.bind(this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
search: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
const ctx = this[context];
|
|
|
|
return !ctx.query ? '' : `?${ctx.query}`;
|
|
|
|
},
|
|
|
|
set(search) {
|
|
|
|
const ctx = this[context];
|
|
|
|
search = String(search);
|
|
|
|
if (!search) {
|
|
|
|
ctx.query = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_QUERY;
|
|
|
|
this[searchParams][searchParams] = {};
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (search[0] === '?') search = search.slice(1);
|
|
|
|
ctx.query = '';
|
|
|
|
binding.parse(search, binding.kQuery, null, ctx,
|
|
|
|
onParseSearchComplete.bind(this));
|
|
|
|
this[searchParams][searchParams] = querystring.parse(search);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
searchParams: { // readonly
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return this[searchParams];
|
|
|
|
}
|
|
|
|
},
|
|
|
|
hash: {
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
const ctx = this[context];
|
|
|
|
return !ctx.fragment ? '' : `#${ctx.fragment}`;
|
|
|
|
},
|
|
|
|
set(hash) {
|
|
|
|
const ctx = this[context];
|
|
|
|
hash = String(hash);
|
|
|
|
if (this.protocol === 'javascript:')
|
|
|
|
return;
|
|
|
|
if (!hash) {
|
|
|
|
ctx.fragment = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_FRAGMENT;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (hash[0] === '#') hash = hash.slice(1);
|
|
|
|
ctx.fragment = '';
|
|
|
|
binding.parse(hash, binding.kFragment, null, ctx,
|
|
|
|
onParseHashComplete.bind(this));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const hexTable = new Array(256);
|
|
|
|
|
|
|
|
for (var i = 0; i < 256; ++i)
|
|
|
|
hexTable[i] = '%' + ((i < 16 ? '0' : '') + i.toString(16)).toUpperCase();
|
|
|
|
function encodeAuth(str) {
|
|
|
|
// faster encodeURIComponent alternative for encoding auth uri components
|
|
|
|
var out = '';
|
|
|
|
var lastPos = 0;
|
|
|
|
for (var i = 0; i < str.length; ++i) {
|
|
|
|
var c = str.charCodeAt(i);
|
|
|
|
|
|
|
|
// These characters do not need escaping:
|
|
|
|
// ! - . _ ~
|
|
|
|
// ' ( ) * :
|
|
|
|
// digits
|
|
|
|
// alpha (uppercase)
|
|
|
|
// alpha (lowercase)
|
|
|
|
if (c === 0x21 || c === 0x2D || c === 0x2E || c === 0x5F || c === 0x7E ||
|
|
|
|
(c >= 0x27 && c <= 0x2A) ||
|
|
|
|
(c >= 0x30 && c <= 0x3A) ||
|
|
|
|
(c >= 0x41 && c <= 0x5A) ||
|
|
|
|
(c >= 0x61 && c <= 0x7A)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i - lastPos > 0)
|
|
|
|
out += str.slice(lastPos, i);
|
|
|
|
|
|
|
|
lastPos = i + 1;
|
|
|
|
|
|
|
|
// Other ASCII characters
|
|
|
|
if (c < 0x80) {
|
|
|
|
out += hexTable[c];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Multi-byte characters ...
|
|
|
|
if (c < 0x800) {
|
|
|
|
out += hexTable[0xC0 | (c >> 6)] + hexTable[0x80 | (c & 0x3F)];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (c < 0xD800 || c >= 0xE000) {
|
|
|
|
out += hexTable[0xE0 | (c >> 12)] +
|
|
|
|
hexTable[0x80 | ((c >> 6) & 0x3F)] +
|
|
|
|
hexTable[0x80 | (c & 0x3F)];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Surrogate pair
|
|
|
|
++i;
|
|
|
|
var c2;
|
|
|
|
if (i < str.length)
|
|
|
|
c2 = str.charCodeAt(i) & 0x3FF;
|
|
|
|
else
|
|
|
|
c2 = 0;
|
|
|
|
c = 0x10000 + (((c & 0x3FF) << 10) | c2);
|
|
|
|
out += hexTable[0xF0 | (c >> 18)] +
|
|
|
|
hexTable[0x80 | ((c >> 12) & 0x3F)] +
|
|
|
|
hexTable[0x80 | ((c >> 6) & 0x3F)] +
|
|
|
|
hexTable[0x80 | (c & 0x3F)];
|
|
|
|
}
|
|
|
|
if (lastPos === 0)
|
|
|
|
return str;
|
|
|
|
if (lastPos < str.length)
|
|
|
|
return out + str.slice(lastPos);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
function update(url, params) {
|
|
|
|
if (!url)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const ctx = url[context];
|
|
|
|
const serializedParams = params.toString();
|
|
|
|
if (serializedParams) {
|
|
|
|
ctx.query = serializedParams;
|
|
|
|
ctx.flags |= binding.URL_FLAGS_HAS_QUERY;
|
|
|
|
} else {
|
|
|
|
ctx.query = null;
|
|
|
|
ctx.flags &= ~binding.URL_FLAGS_HAS_QUERY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getSearchParamPairs(target) {
|
|
|
|
const obj = target[searchParams];
|
|
|
|
const keys = Object.keys(obj);
|
|
|
|
const values = [];
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
const name = keys[i];
|
|
|
|
const value = obj[name];
|
|
|
|
if (Array.isArray(value)) {
|
|
|
|
for (const item of value)
|
|
|
|
values.push([name, item]);
|
|
|
|
} else {
|
|
|
|
values.push([name, value]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return values;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reused by the URL parse function invoked by
|
|
|
|
// the href setter, and the URLSearchParams constructor
|
|
|
|
function initSearchParams(url, init) {
|
|
|
|
url[searchParams] = querystring.parse(init);
|
|
|
|
}
|
|
|
|
|
|
|
|
class URLSearchParams {
|
|
|
|
constructor(init = '') {
|
|
|
|
if (init instanceof URLSearchParams) {
|
|
|
|
const childParams = init[searchParams];
|
|
|
|
this[searchParams] = Object.assign(Object.create(null), childParams);
|
|
|
|
} else {
|
|
|
|
init = String(init);
|
|
|
|
if (init[0] === '?') init = init.slice(1);
|
|
|
|
initSearchParams(this, init);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "associated url object"
|
|
|
|
this[context] = null;
|
|
|
|
|
|
|
|
// Class string for an instance of URLSearchParams. This is different from
|
|
|
|
// the class string of the prototype object (set below).
|
|
|
|
Object.defineProperty(this, Symbol.toStringTag, {
|
|
|
|
value: 'URLSearchParams',
|
|
|
|
writable: false,
|
|
|
|
enumerable: false,
|
|
|
|
configurable: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
append(name, value) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 2) {
|
|
|
|
throw new TypeError(
|
|
|
|
'Both `name` and `value` arguments need to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
value = String(value);
|
|
|
|
var existing = obj[name];
|
|
|
|
if (existing === undefined) {
|
|
|
|
obj[name] = value;
|
|
|
|
} else if (Array.isArray(existing)) {
|
|
|
|
existing.push(value);
|
|
|
|
} else {
|
|
|
|
obj[name] = [existing, value];
|
|
|
|
}
|
|
|
|
update(this[context], this);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(name) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 1) {
|
|
|
|
throw new TypeError('The `name` argument needs to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
delete obj[name];
|
|
|
|
update(this[context], this);
|
|
|
|
}
|
|
|
|
|
|
|
|
set(name, value) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 2) {
|
|
|
|
throw new TypeError(
|
|
|
|
'Both `name` and `value` arguments need to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
value = String(value);
|
|
|
|
obj[name] = value;
|
|
|
|
update(this[context], this);
|
|
|
|
}
|
|
|
|
|
|
|
|
get(name) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 1) {
|
|
|
|
throw new TypeError('The `name` argument needs to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
var value = obj[name];
|
|
|
|
return value === undefined ? null : Array.isArray(value) ? value[0] : value;
|
|
|
|
}
|
|
|
|
|
|
|
|
getAll(name) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 1) {
|
|
|
|
throw new TypeError('The `name` argument needs to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
var value = obj[name];
|
|
|
|
return value === undefined ? [] : Array.isArray(value) ? value : [value];
|
|
|
|
}
|
|
|
|
|
|
|
|
has(name) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 1) {
|
|
|
|
throw new TypeError('The `name` argument needs to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = this[searchParams];
|
|
|
|
name = String(name);
|
|
|
|
return name in obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://heycam.github.io/webidl/#es-iterators
|
|
|
|
// Define entries here rather than [Symbol.iterator] as the function name
|
|
|
|
// must be set to `entries`.
|
|
|
|
entries() {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
|
|
|
|
return createSearchParamsIterator(this, 'key+value');
|
|
|
|
}
|
|
|
|
|
|
|
|
forEach(callback, thisArg = undefined) {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
if (arguments.length < 1) {
|
|
|
|
throw new TypeError('The `callback` argument needs to be specified');
|
|
|
|
}
|
|
|
|
|
|
|
|
let pairs = getSearchParamPairs(this);
|
|
|
|
|
|
|
|
var i = 0;
|
|
|
|
while (i < pairs.length) {
|
|
|
|
const [key, value] = pairs[i];
|
|
|
|
callback.call(thisArg, value, key, this);
|
|
|
|
pairs = getSearchParamPairs(this);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://heycam.github.io/webidl/#es-iterable
|
|
|
|
keys() {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
|
|
|
|
return createSearchParamsIterator(this, 'key');
|
|
|
|
}
|
|
|
|
|
|
|
|
values() {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
|
|
|
|
return createSearchParamsIterator(this, 'value');
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://url.spec.whatwg.org/#urlsearchparams-stringification-behavior
|
|
|
|
toString() {
|
|
|
|
if (!this || !(this instanceof URLSearchParams)) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
|
|
|
}
|
|
|
|
|
|
|
|
return querystring.stringify(this[searchParams]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// https://heycam.github.io/webidl/#es-iterable-entries
|
|
|
|
URLSearchParams.prototype[Symbol.iterator] = URLSearchParams.prototype.entries;
|
|
|
|
Object.defineProperty(URLSearchParams.prototype, Symbol.toStringTag, {
|
|
|
|
value: 'URLSearchParamsPrototype',
|
|
|
|
writable: false,
|
|
|
|
enumerable: false,
|
|
|
|
configurable: true
|
|
|
|
});
|
|
|
|
|
|
|
|
// https://heycam.github.io/webidl/#dfn-default-iterator-object
|
|
|
|
function createSearchParamsIterator(target, kind) {
|
|
|
|
const iterator = Object.create(URLSearchParamsIteratorPrototype);
|
|
|
|
iterator[context] = {
|
|
|
|
target,
|
|
|
|
kind,
|
|
|
|
index: 0
|
|
|
|
};
|
|
|
|
return iterator;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://heycam.github.io/webidl/#dfn-iterator-prototype-object
|
|
|
|
const URLSearchParamsIteratorPrototype = Object.setPrototypeOf({
|
|
|
|
next() {
|
|
|
|
if (!this ||
|
|
|
|
Object.getPrototypeOf(this) !== URLSearchParamsIteratorPrototype) {
|
|
|
|
throw new TypeError('Value of `this` is not a URLSearchParamsIterator');
|
|
|
|
}
|
|
|
|
|
|
|
|
const {
|
|
|
|
target,
|
|
|
|
kind,
|
|
|
|
index
|
|
|
|
} = this[context];
|
|
|
|
const values = getSearchParamPairs(target);
|
|
|
|
const len = values.length;
|
|
|
|
if (index >= len) {
|
|
|
|
return {
|
|
|
|
value: undefined,
|
|
|
|
done: true
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const pair = values[index];
|
|
|
|
this[context].index = index + 1;
|
|
|
|
|
|
|
|
let result;
|
|
|
|
if (kind === 'key') {
|
|
|
|
result = pair[0];
|
|
|
|
} else if (kind === 'value') {
|
|
|
|
result = pair[1];
|
|
|
|
} else {
|
|
|
|
result = pair;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
value: result,
|
|
|
|
done: false
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}, IteratorPrototype);
|
|
|
|
|
|
|
|
// Unlike interface and its prototype object, both default iterator object and
|
|
|
|
// iterator prototype object of an interface have the same class string.
|
|
|
|
Object.defineProperty(URLSearchParamsIteratorPrototype, Symbol.toStringTag, {
|
|
|
|
value: 'URLSearchParamsIterator',
|
|
|
|
writable: false,
|
|
|
|
enumerable: false,
|
|
|
|
configurable: true
|
|
|
|
});
|
|
|
|
|
|
|
|
function originFor(url, base) {
|
|
|
|
if (!(url instanceof URL))
|
|
|
|
url = new URL(url, base);
|
|
|
|
var origin;
|
|
|
|
const protocol = url.protocol;
|
|
|
|
switch (protocol) {
|
|
|
|
case 'blob:':
|
|
|
|
if (url[context].path && url[context].path.length > 0) {
|
|
|
|
try {
|
|
|
|
return (new URL(url[context].path[0])).origin;
|
|
|
|
} catch (err) {
|
|
|
|
// fall through... do nothing
|
|
|
|
}
|
|
|
|
}
|
|
|
|
origin = new OpaqueOrigin();
|
|
|
|
break;
|
|
|
|
case 'ftp:':
|
|
|
|
case 'gopher:':
|
|
|
|
case 'http:':
|
|
|
|
case 'https:':
|
|
|
|
case 'ws:':
|
|
|
|
case 'wss:':
|
|
|
|
case 'file':
|
|
|
|
origin = new TupleOrigin(protocol.slice(0, -1),
|
|
|
|
url[context].host,
|
|
|
|
url[context].port,
|
|
|
|
null);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
origin = new OpaqueOrigin();
|
|
|
|
}
|
|
|
|
return origin;
|
|
|
|
}
|
|
|
|
|
|
|
|
function domainToASCII(domain) {
|
|
|
|
return binding.domainToASCII(String(domain));
|
|
|
|
}
|
|
|
|
|
|
|
|
function domainToUnicode(domain) {
|
|
|
|
return binding.domainToUnicode(String(domain));
|
|
|
|
}
|
|
|
|
|
|
|
|
exports.URL = URL;
|
|
|
|
exports.originFor = originFor;
|
|
|
|
exports.domainToASCII = domainToASCII;
|
|
|
|
exports.domainToUnicode = domainToUnicode;
|
|
|
|
exports.encodeAuth = encodeAuth;
|