summaryrefslogtreecommitdiff
path: root/ext/url/00_url.js
diff options
context:
space:
mode:
Diffstat (limited to 'ext/url/00_url.js')
-rw-r--r--ext/url/00_url.js623
1 files changed, 623 insertions, 0 deletions
diff --git a/ext/url/00_url.js b/ext/url/00_url.js
new file mode 100644
index 000000000..f3c12d0c2
--- /dev/null
+++ b/ext/url/00_url.js
@@ -0,0 +1,623 @@
+// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
+
+// @ts-check
+/// <reference path="../../core/internal.d.ts" />
+/// <reference path="../../core/lib.deno_core.d.ts" />
+/// <reference path="../webidl/internal.d.ts" />
+
+"use strict";
+
+((window) => {
+ const core = window.Deno.core;
+ const webidl = window.__bootstrap.webidl;
+ const {
+ ArrayIsArray,
+ ArrayPrototypeMap,
+ ArrayPrototypePush,
+ ArrayPrototypeSome,
+ ArrayPrototypeSort,
+ ArrayPrototypeSplice,
+ ObjectKeys,
+ StringPrototypeSlice,
+ Symbol,
+ SymbolFor,
+ SymbolIterator,
+ SymbolToStringTag,
+ TypeError,
+ } = window.__bootstrap.primordials;
+
+ const _list = Symbol("list");
+ const _urlObject = Symbol("url object");
+
+ class URLSearchParams {
+ [_list];
+ [_urlObject] = null;
+
+ /**
+ * @param {string | [string][] | Record<string, string>} init
+ */
+ constructor(init = "") {
+ const prefix = "Failed to construct 'URL'";
+ init = webidl.converters
+ ["sequence<sequence<USVString>> or record<USVString, USVString> or USVString"](
+ init,
+ { prefix, context: "Argument 1" },
+ );
+ this[webidl.brand] = webidl.brand;
+
+ 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] = core.opSync("op_url_parse_search_params", init);
+ } else if (ArrayIsArray(init)) {
+ // Overload: sequence<sequence<USVString>>
+ 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<USVString, USVString>
+ this[_list] = ArrayPrototypeMap(
+ ObjectKeys(init),
+ (key) => [key, init[key]],
+ );
+ }
+ }
+
+ #updateUrlSearch() {
+ const url = this[_urlObject];
+ if (url === null) {
+ return;
+ }
+ const parts = core.opSync("op_url_parse", {
+ href: url.href,
+ setSearch: this.toString(),
+ });
+ url[_url] = parts;
+ }
+
+ /**
+ * @param {string} name
+ * @param {string} value
+ */
+ append(name, value) {
+ webidl.assertBranded(this, URLSearchParams);
+ 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, URLSearchParams);
+ 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, URLSearchParams);
+ 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 = [];
+ for (const entry of this[_list]) {
+ if (entry[0] === name) {
+ ArrayPrototypePush(values, entry[1]);
+ }
+ }
+ return values;
+ }
+
+ /**
+ * @param {string} name
+ * @return {string | null}
+ */
+ get(name) {
+ webidl.assertBranded(this, URLSearchParams);
+ const prefix = "Failed to execute 'get' on 'URLSearchParams'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ name = webidl.converters.USVString(name, {
+ prefix,
+ context: "Argument 1",
+ });
+ for (const entry of this[_list]) {
+ if (entry[0] === name) {
+ return entry[1];
+ }
+ }
+ return null;
+ }
+
+ /**
+ * @param {string} name
+ * @return {boolean}
+ */
+ has(name) {
+ webidl.assertBranded(this, URLSearchParams);
+ 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, URLSearchParams);
+ 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, URLSearchParams);
+ ArrayPrototypeSort(
+ this[_list],
+ (a, b) => (a[0] === b[0] ? 0 : a[0] > b[0] ? 1 : -1),
+ );
+ this.#updateUrlSearch();
+ }
+
+ /**
+ * @return {string}
+ */
+ toString() {
+ webidl.assertBranded(this, URLSearchParams);
+ return core.opSync("op_url_stringify_search_params", this[_list]);
+ }
+
+ get [SymbolToStringTag]() {
+ return "URLSearchParams";
+ }
+ }
+
+ webidl.mixinPairIterable("URLSearchParams", URLSearchParams, _list, 0, 1);
+
+ webidl.configurePrototype(URLSearchParams);
+
+ const _url = Symbol("url");
+
+ class URL {
+ [_url];
+ #queryObject = null;
+
+ /**
+ * @param {string} url
+ * @param {string} base
+ */
+ constructor(url, base = undefined) {
+ const prefix = "Failed to construct 'URL'";
+ url = webidl.converters.USVString(url, { prefix, context: "Argument 1" });
+ if (base !== undefined) {
+ base = webidl.converters.USVString(base, {
+ prefix,
+ context: "Argument 2",
+ });
+ }
+ this[webidl.brand] = webidl.brand;
+
+ const parts = core.opSync("op_url_parse", { href: url, baseHref: base });
+ this[_url] = parts;
+ }
+
+ [SymbolFor("Deno.privateCustomInspect")](inspect) {
+ 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)}`;
+ }
+
+ #updateSearchParams() {
+ if (this.#queryObject !== null) {
+ const params = this.#queryObject[_list];
+ const newParams = core.opSync(
+ "op_url_parse_search_params",
+ StringPrototypeSlice(this.search, 1),
+ );
+ ArrayPrototypeSplice(params, 0, params.length, ...newParams);
+ }
+ }
+
+ /** @return {string} */
+ get hash() {
+ webidl.assertBranded(this, URL);
+ return this[_url].hash;
+ }
+
+ /** @param {string} value */
+ set hash(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'hash' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setHash: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get host() {
+ webidl.assertBranded(this, URL);
+ return this[_url].host;
+ }
+
+ /** @param {string} value */
+ set host(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'host' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setHost: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get hostname() {
+ webidl.assertBranded(this, URL);
+ return this[_url].hostname;
+ }
+
+ /** @param {string} value */
+ set hostname(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'hostname' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setHostname: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get href() {
+ webidl.assertBranded(this, URL);
+ return this[_url].href;
+ }
+
+ /** @param {string} value */
+ set href(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'href' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ this[_url] = core.opSync("op_url_parse", {
+ href: value,
+ });
+ this.#updateSearchParams();
+ }
+
+ /** @return {string} */
+ get origin() {
+ webidl.assertBranded(this, URL);
+ return this[_url].origin;
+ }
+
+ /** @return {string} */
+ get password() {
+ webidl.assertBranded(this, URL);
+ return this[_url].password;
+ }
+
+ /** @param {string} value */
+ set password(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'password' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setPassword: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get pathname() {
+ webidl.assertBranded(this, URL);
+ return this[_url].pathname;
+ }
+
+ /** @param {string} value */
+ set pathname(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'pathname' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setPathname: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get port() {
+ webidl.assertBranded(this, URL);
+ return this[_url].port;
+ }
+
+ /** @param {string} value */
+ set port(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'port' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setPort: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get protocol() {
+ webidl.assertBranded(this, URL);
+ return this[_url].protocol;
+ }
+
+ /** @param {string} value */
+ set protocol(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'protocol' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setProtocol: value,
+ });
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get search() {
+ webidl.assertBranded(this, URL);
+ return this[_url].search;
+ }
+
+ /** @param {string} value */
+ set search(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'search' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setSearch: value,
+ });
+ this.#updateSearchParams();
+ } catch {
+ /* pass */
+ }
+ }
+
+ /** @return {string} */
+ get username() {
+ webidl.assertBranded(this, URL);
+ return this[_url].username;
+ }
+
+ /** @param {string} value */
+ set username(value) {
+ webidl.assertBranded(this, URL);
+ const prefix = "Failed to set 'username' on 'URL'";
+ webidl.requiredArguments(arguments.length, 1, { prefix });
+ value = webidl.converters.USVString(value, {
+ prefix,
+ context: "Argument 1",
+ });
+ try {
+ this[_url] = core.opSync("op_url_parse", {
+ href: this[_url].href,
+ setUsername: value,
+ });
+ } 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, URL);
+ return this[_url].href;
+ }
+
+ /** @return {string} */
+ toJSON() {
+ webidl.assertBranded(this, URL);
+ return this[_url].href;
+ }
+
+ get [SymbolToStringTag]() {
+ return "URL";
+ }
+ }
+
+ webidl.configurePrototype(URL);
+
+ /**
+ * 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 core.opSync("op_url_parse_search_params", null, bytes);
+ }
+
+ webidl
+ .converters[
+ "sequence<sequence<USVString>> or record<USVString, USVString> or USVString"
+ ] = (V, opts) => {
+ // Union for (sequence<sequence<USVString>> or record<USVString, USVString> or USVString)
+ if (webidl.type(V) === "Object" && V !== null) {
+ if (V[SymbolIterator] !== undefined) {
+ return webidl.converters["sequence<sequence<USVString>>"](V, opts);
+ }
+ return webidl.converters["record<USVString, USVString>"](V, opts);
+ }
+ return webidl.converters.USVString(V, opts);
+ };
+
+ window.__bootstrap.url = {
+ URL,
+ URLSearchParams,
+ parseUrlEncoded,
+ };
+})(this);