'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;