// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license. // @ts-check /// /// /// const core = globalThis.Deno.core; const ops = core.ops; import * as webidl from "ext:deno_webidl/00_webidl.js"; const primordials = globalThis.__bootstrap.primordials; const { ArrayIsArray, ArrayPrototypeMap, ArrayPrototypePush, ArrayPrototypeSome, ArrayPrototypeSort, ArrayPrototypeSplice, ObjectKeys, Uint32Array, SafeArrayIterator, StringPrototypeSlice, Symbol, SymbolFor, SymbolIterator, TypeError, } = primordials; const _list = Symbol("list"); const _urlObject = Symbol("url object"); // WARNING: must match rust code's UrlSetter::* const SET_HASH = 0; const SET_HOST = 1; const SET_HOSTNAME = 2; const SET_PASSWORD = 3; const SET_PATHNAME = 4; const SET_PORT = 5; const SET_PROTOCOL = 6; const SET_SEARCH = 7; const SET_USERNAME = 8; // Helper functions function opUrlReparse(href, setter, value) { const status = ops.op_url_reparse( href, setter, value, componentsBuf, ); return getSerialization(status, href); } function opUrlParse(href, maybeBase) { let status; if (maybeBase === undefined) { status = ops.op_url_parse(href, componentsBuf); } else { status = ops.op_url_parse_with_base( href, maybeBase, componentsBuf, ); } return getSerialization(status, href, maybeBase); } function getSerialization(status, href, maybeBase) { if (status === 0) { return href; } else if (status === 1) { return ops.op_url_get_serialization(); } else { throw new TypeError( `Invalid URL: '${href}'` + (maybeBase ? ` with base '${maybeBase}'` : ""), ); } } class URLSearchParams { [_list]; [_urlObject] = null; /** * @param {string | [string][] | Record} init */ constructor(init = "") { const prefix = "Failed to construct 'URL'"; init = webidl.converters ["sequence> or record or USVString"]( init, { prefix, context: "Argument 1" }, ); this[webidl.brand] = webidl.brand; if (!init) { // if there is no query string, return early this[_list] = []; return; } if (typeof init === "string") { // Overload: USVString // If init is a string and starts with U+003F (?), // remove the first code point from init. if (init[0] == "?") { init = StringPrototypeSlice(init, 1); } this[_list] = ops.op_url_parse_search_params(init); } else if (ArrayIsArray(init)) { // Overload: sequence> this[_list] = ArrayPrototypeMap(init, (pair, i) => { if (pair.length !== 2) { throw new TypeError( `${prefix}: Item ${ i + 0 } in the parameter list does have length 2 exactly.`, ); } return [pair[0], pair[1]]; }); } else { // Overload: record this[_list] = ArrayPrototypeMap( ObjectKeys(init), (key) => [key, init[key]], ); } } #updateUrlSearch() { const url = this[_urlObject]; if (url === null) { return; } url[_updateUrlSearch](this.toString()); } /** * @param {string} name * @param {string} value */ append(name, value) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'append' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 2, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); value = webidl.converters.USVString(value, { prefix, context: "Argument 2", }); ArrayPrototypePush(this[_list], [name, value]); this.#updateUrlSearch(); } /** * @param {string} name */ delete(name) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'append' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 1, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); const list = this[_list]; let i = 0; while (i < list.length) { if (list[i][0] === name) { ArrayPrototypeSplice(list, i, 1); } else { i++; } } this.#updateUrlSearch(); } /** * @param {string} name * @returns {string[]} */ getAll(name) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'getAll' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 1, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); const values = []; const entries = this[_list]; for (let i = 0; i < entries.length; ++i) { const entry = entries[i]; if (entry[0] === name) { ArrayPrototypePush(values, entry[1]); } } return values; } /** * @param {string} name * @return {string | null} */ get(name) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'get' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 1, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); const entries = this[_list]; for (let i = 0; i < entries.length; ++i) { const entry = entries[i]; if (entry[0] === name) { return entry[1]; } } return null; } /** * @param {string} name * @return {boolean} */ has(name) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'has' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 1, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); return ArrayPrototypeSome(this[_list], (entry) => entry[0] === name); } /** * @param {string} name * @param {string} value */ set(name, value) { webidl.assertBranded(this, URLSearchParamsPrototype); const prefix = "Failed to execute 'set' on 'URLSearchParams'"; webidl.requiredArguments(arguments.length, 2, prefix); name = webidl.converters.USVString(name, { prefix, context: "Argument 1", }); value = webidl.converters.USVString(value, { prefix, context: "Argument 2", }); const list = this[_list]; // If there are any name-value pairs whose name is name, in list, // set the value of the first such name-value pair to value // and remove the others. let found = false; let i = 0; while (i < list.length) { if (list[i][0] === name) { if (!found) { list[i][1] = value; found = true; i++; } else { ArrayPrototypeSplice(list, i, 1); } } else { i++; } } // Otherwise, append a new name-value pair whose name is name // and value is value, to list. if (!found) { ArrayPrototypePush(list, [name, value]); } this.#updateUrlSearch(); } sort() { webidl.assertBranded(this, URLSearchParamsPrototype); ArrayPrototypeSort( this[_list], (a, b) => (a[0] === b[0] ? 0 : a[0] > b[0] ? 1 : -1), ); this.#updateUrlSearch(); } /** * @return {string} */ toString() { webidl.assertBranded(this, URLSearchParamsPrototype); return ops.op_url_stringify_search_params(this[_list]); } get size() { webidl.assertBranded(this, URLSearchParamsPrototype); return this[_list].length; } } webidl.mixinPairIterable("URLSearchParams", URLSearchParams, _list, 0, 1); webidl.configurePrototype(URLSearchParams); const URLSearchParamsPrototype = URLSearchParams.prototype; webidl.converters["URLSearchParams"] = webidl.createInterfaceConverter( "URLSearchParams", URLSearchParamsPrototype, ); const _updateUrlSearch = Symbol("updateUrlSearch"); function trim(s) { if (s.length === 1) return ""; return s; } // Represents a "no port" value. A port in URL cannot be greater than 2^16 - 1 const NO_PORT = 65536; const componentsBuf = new Uint32Array(8); class URL { #queryObject = null; #serialization; #schemeEnd; #usernameEnd; #hostStart; #hostEnd; #port; #pathStart; #queryStart; #fragmentStart; [_updateUrlSearch](value) { this.#serialization = opUrlReparse( this.#serialization, SET_SEARCH, value, ); this.#updateComponents(); } /** * @param {string} url * @param {string} base */ constructor(url, base = undefined) { const prefix = "Failed to construct 'URL'"; url = webidl.converters.DOMString(url, { prefix, context: "Argument 1" }); if (base !== undefined) { base = webidl.converters.DOMString(base, { prefix, context: "Argument 2", }); } this[webidl.brand] = webidl.brand; this.#serialization = opUrlParse(url, base); this.#updateComponents(); } #updateComponents() { ({ 0: this.#schemeEnd, 1: this.#usernameEnd, 2: this.#hostStart, 3: this.#hostEnd, 4: this.#port, 5: this.#pathStart, 6: this.#queryStart, 7: this.#fragmentStart, } = componentsBuf); } [SymbolFor("Deno.privateCustomInspect")](inspect, inspectOptions) { const object = { href: this.href, origin: this.origin, protocol: this.protocol, username: this.username, password: this.password, host: this.host, hostname: this.hostname, port: this.port, pathname: this.pathname, hash: this.hash, search: this.search, }; return `${this.constructor.name} ${inspect(object, inspectOptions)}`; } #updateSearchParams() { if (this.#queryObject !== null) { const params = this.#queryObject[_list]; const newParams = ops.op_url_parse_search_params( StringPrototypeSlice(this.search, 1), ); ArrayPrototypeSplice( params, 0, params.length, ...new SafeArrayIterator(newParams), ); } } #hasAuthority() { // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/lib.rs#L824 return this.#serialization.slice(this.#schemeEnd).startsWith("://"); } /** @return {string} */ get hash() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/quirks.rs#L263 return this.#fragmentStart ? trim(this.#serialization.slice(this.#fragmentStart)) : ""; } /** @param {string} value */ set hash(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'hash' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_HASH, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get host() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/quirks.rs#L101 return this.#serialization.slice(this.#hostStart, this.#pathStart); } /** @param {string} value */ set host(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'host' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_HOST, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get hostname() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/lib.rs#L988 return this.#serialization.slice(this.#hostStart, this.#hostEnd); } /** @param {string} value */ set hostname(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'hostname' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_HOSTNAME, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get href() { webidl.assertBranded(this, URLPrototype); return this.#serialization; } /** @param {string} value */ set href(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'href' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); this.#serialization = opUrlParse(value); this.#updateComponents(); this.#updateSearchParams(); } /** @return {string} */ get origin() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/origin.rs#L14 const scheme = this.#serialization.slice(0, this.#schemeEnd); if ( scheme === "http" || scheme === "https" || scheme === "ftp" || scheme === "ws" || scheme === "wss" ) { return `${scheme}://${this.host}`; } if (scheme === "blob") { // TODO(@littledivy): Fast path. try { return new URL(this.pathname).origin; } catch { return "null"; } } return "null"; } /** @return {string} */ get password() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/lib.rs#L914 if ( this.#hasAuthority() && this.#usernameEnd !== this.#serialization.length && this.#serialization[this.#usernameEnd] === ":" ) { return this.#serialization.slice( this.#usernameEnd + 1, this.#hostStart - 1, ); } return ""; } /** @param {string} value */ set password(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'password' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_PASSWORD, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get pathname() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/lib.rs#L1203 if (!this.#queryStart && !this.#fragmentStart) { return this.#serialization.slice(this.#pathStart); } const nextComponentStart = this.#queryStart || this.#fragmentStart; return this.#serialization.slice(this.#pathStart, nextComponentStart); } /** @param {string} value */ set pathname(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'pathname' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_PATHNAME, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get port() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/quirks.rs#L196 if (this.#port === NO_PORT) { return this.#serialization.slice(this.#hostEnd, this.#pathStart); } else { return this.#serialization.slice( this.#hostEnd + 1, /* : */ this.#pathStart, ); } } /** @param {string} value */ set port(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'port' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_PORT, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get protocol() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/quirks.rs#L56 return this.#serialization.slice(0, this.#schemeEnd + 1 /* : */); } /** @param {string} value */ set protocol(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'protocol' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_PROTOCOL, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get search() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/quirks.rs#L249 const afterPath = this.#queryStart || this.#fragmentStart || this.#serialization.length; const afterQuery = this.#fragmentStart || this.#serialization.length; return trim(this.#serialization.slice(afterPath, afterQuery)); } /** @param {string} value */ set search(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'search' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_SEARCH, value, ); this.#updateComponents(); this.#updateSearchParams(); } catch { /* pass */ } } /** @return {string} */ get username() { webidl.assertBranded(this, URLPrototype); // https://github.com/servo/rust-url/blob/1d307ae51a28fecc630ecec03380788bfb03a643/url/src/lib.rs#L881 const schemeSeperatorLen = 3; /* :// */ if ( this.#hasAuthority() && this.#usernameEnd > this.#schemeEnd + schemeSeperatorLen ) { return this.#serialization.slice( this.#schemeEnd + schemeSeperatorLen, this.#usernameEnd, ); } else { return ""; } } /** @param {string} value */ set username(value) { webidl.assertBranded(this, URLPrototype); const prefix = "Failed to set 'username' on 'URL'"; webidl.requiredArguments(arguments.length, 1, prefix); value = webidl.converters.DOMString(value, { prefix, context: "Argument 1", }); try { this.#serialization = opUrlReparse( this.#serialization, SET_USERNAME, value, ); this.#updateComponents(); } catch { /* pass */ } } /** @return {string} */ get searchParams() { if (this.#queryObject == null) { this.#queryObject = new URLSearchParams(this.search); this.#queryObject[_urlObject] = this; } return this.#queryObject; } /** @return {string} */ toString() { webidl.assertBranded(this, URLPrototype); return this.#serialization; } /** @return {string} */ toJSON() { webidl.assertBranded(this, URLPrototype); return this.#serialization; } } webidl.configurePrototype(URL); const URLPrototype = URL.prototype; /** * This function implements application/x-www-form-urlencoded parsing. * https://url.spec.whatwg.org/#concept-urlencoded-parser * @param {Uint8Array} bytes * @returns {[string, string][]} */ function parseUrlEncoded(bytes) { return ops.op_url_parse_search_params(null, bytes); } webidl .converters[ "sequence> or record or USVString" ] = (V, opts) => { // Union for (sequence> or record or USVString) if (webidl.type(V) === "Object" && V !== null) { if (V[SymbolIterator] !== undefined) { return webidl.converters["sequence>"](V, opts); } return webidl.converters["record"](V, opts); } return webidl.converters.USVString(V, opts); }; export { parseUrlEncoded, URL, URLPrototype, URLSearchParams, URLSearchParamsPrototype, };