2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-03-05 13:05:41 +01:00
|
|
|
import { DomIterableMixin } from "./dom_iterable.ts";
|
2020-03-11 15:49:53 +01:00
|
|
|
import { requiredArguments } from "./util.ts";
|
2020-03-11 10:53:06 +01:00
|
|
|
import { customInspect } from "./console.ts";
|
2018-11-02 21:43:37 -04:00
|
|
|
|
2018-12-19 02:57:23 -05:00
|
|
|
// From node-fetch
|
|
|
|
// Copyright (c) 2016 David Frank. MIT License.
|
|
|
|
const invalidTokenRegex = /[^\^_`a-zA-Z\-0-9!#$%&'*+.|~]/;
|
|
|
|
const invalidHeaderCharRegex = /[^\t\x20-\x7e\x80-\xff]/;
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
2020-04-13 22:46:23 -04:00
|
|
|
function isHeaders(value: any): value is Headers {
|
2019-03-10 04:30:38 +11:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-use-before-define
|
2018-11-02 21:43:37 -04:00
|
|
|
return value instanceof Headers;
|
|
|
|
}
|
|
|
|
|
|
|
|
const headerMap = Symbol("header map");
|
|
|
|
|
2020-03-29 04:03:49 +11:00
|
|
|
// TODO: headerGuard? Investigate if it is needed
|
|
|
|
// node-fetch did not implement this but it is in the spec
|
|
|
|
function normalizeParams(name: string, value?: string): string[] {
|
|
|
|
name = String(name).toLowerCase();
|
|
|
|
value = String(value).trim();
|
|
|
|
return [name, value];
|
|
|
|
}
|
2018-11-02 21:43:37 -04:00
|
|
|
|
2020-03-29 04:03:49 +11:00
|
|
|
// The following name/value validations are copied from
|
|
|
|
// https://github.com/bitinn/node-fetch/blob/master/src/headers.js
|
|
|
|
// Copyright (c) 2016 David Frank. MIT License.
|
|
|
|
function validateName(name: string): void {
|
|
|
|
if (invalidTokenRegex.test(name) || name === "") {
|
|
|
|
throw new TypeError(`${name} is not a legal HTTP header name`);
|
2018-12-19 02:57:23 -05:00
|
|
|
}
|
2020-03-29 04:03:49 +11:00
|
|
|
}
|
2018-12-29 20:30:11 +08:00
|
|
|
|
2020-03-29 04:03:49 +11:00
|
|
|
function validateValue(value: string): void {
|
|
|
|
if (invalidHeaderCharRegex.test(value)) {
|
|
|
|
throw new TypeError(`${value} is not a legal HTTP header value`);
|
2018-12-19 02:57:23 -05:00
|
|
|
}
|
2020-03-29 04:03:49 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// ref: https://fetch.spec.whatwg.org/#dom-headers
|
|
|
|
class HeadersBase {
|
|
|
|
[headerMap]: Map<string, string>;
|
2018-12-19 02:57:23 -05:00
|
|
|
|
2020-04-13 22:46:23 -04:00
|
|
|
constructor(init?: HeadersInit) {
|
2018-11-02 21:43:37 -04:00
|
|
|
if (init === null) {
|
|
|
|
throw new TypeError(
|
|
|
|
"Failed to construct 'Headers'; The provided value was not valid"
|
|
|
|
);
|
|
|
|
} else if (isHeaders(init)) {
|
|
|
|
this[headerMap] = new Map(init);
|
|
|
|
} else {
|
|
|
|
this[headerMap] = new Map();
|
|
|
|
if (Array.isArray(init)) {
|
2018-12-29 20:30:11 +08:00
|
|
|
for (const tuple of init) {
|
|
|
|
// If header does not contain exactly two items,
|
|
|
|
// then throw a TypeError.
|
|
|
|
// ref: https://fetch.spec.whatwg.org/#concept-headers-fill
|
2019-04-21 20:40:15 +01:00
|
|
|
requiredArguments(
|
|
|
|
"Headers.constructor tuple array argument",
|
|
|
|
tuple.length,
|
|
|
|
2
|
|
|
|
);
|
2018-12-29 20:30:11 +08:00
|
|
|
|
2020-03-29 04:03:49 +11:00
|
|
|
const [name, value] = normalizeParams(tuple[0], tuple[1]);
|
|
|
|
validateName(name);
|
|
|
|
validateValue(value);
|
2018-11-02 21:43:37 -04:00
|
|
|
const existingValue = this[headerMap].get(name);
|
|
|
|
this[headerMap].set(
|
|
|
|
name,
|
|
|
|
existingValue ? `${existingValue}, ${value}` : value
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else if (init) {
|
|
|
|
const names = Object.keys(init);
|
|
|
|
for (const rawName of names) {
|
|
|
|
const rawValue = init[rawName];
|
2020-03-29 04:03:49 +11:00
|
|
|
const [name, value] = normalizeParams(rawName, rawValue);
|
|
|
|
validateName(name);
|
|
|
|
validateValue(value);
|
2018-11-02 21:43:37 -04:00
|
|
|
this[headerMap].set(name, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-28 16:23:39 +00:00
|
|
|
[customInspect](): string {
|
|
|
|
let headerSize = this[headerMap].size;
|
|
|
|
let output = "";
|
|
|
|
this[headerMap].forEach((value, key) => {
|
|
|
|
const prefix = headerSize === this[headerMap].size ? " " : "";
|
|
|
|
const postfix = headerSize === 1 ? " " : ", ";
|
|
|
|
output = output + `${prefix}${key}: ${value}${postfix}`;
|
|
|
|
headerSize--;
|
|
|
|
});
|
|
|
|
return `Headers {${output}}`;
|
|
|
|
}
|
|
|
|
|
2018-12-19 02:57:23 -05:00
|
|
|
// ref: https://fetch.spec.whatwg.org/#concept-headers-append
|
2018-11-02 21:43:37 -04:00
|
|
|
append(name: string, value: string): void {
|
2018-12-27 10:12:55 +08:00
|
|
|
requiredArguments("Headers.append", arguments.length, 2);
|
2020-03-29 04:03:49 +11:00
|
|
|
const [newname, newvalue] = normalizeParams(name, value);
|
|
|
|
validateName(newname);
|
|
|
|
validateValue(newvalue);
|
2018-11-02 21:43:37 -04:00
|
|
|
const v = this[headerMap].get(newname);
|
|
|
|
const str = v ? `${v}, ${newvalue}` : newvalue;
|
|
|
|
this[headerMap].set(newname, str);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(name: string): void {
|
2018-12-27 10:12:55 +08:00
|
|
|
requiredArguments("Headers.delete", arguments.length, 1);
|
2020-03-29 04:03:49 +11:00
|
|
|
const [newname] = normalizeParams(name);
|
|
|
|
validateName(newname);
|
2018-11-02 21:43:37 -04:00
|
|
|
this[headerMap].delete(newname);
|
|
|
|
}
|
|
|
|
|
|
|
|
get(name: string): string | null {
|
2018-12-27 10:12:55 +08:00
|
|
|
requiredArguments("Headers.get", arguments.length, 1);
|
2020-03-29 04:03:49 +11:00
|
|
|
const [newname] = normalizeParams(name);
|
|
|
|
validateName(newname);
|
2018-11-02 21:43:37 -04:00
|
|
|
const value = this[headerMap].get(newname);
|
|
|
|
return value || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
has(name: string): boolean {
|
2018-12-27 10:12:55 +08:00
|
|
|
requiredArguments("Headers.has", arguments.length, 1);
|
2020-03-29 04:03:49 +11:00
|
|
|
const [newname] = normalizeParams(name);
|
|
|
|
validateName(newname);
|
2018-11-02 21:43:37 -04:00
|
|
|
return this[headerMap].has(newname);
|
|
|
|
}
|
|
|
|
|
|
|
|
set(name: string, value: string): void {
|
2018-12-27 10:12:55 +08:00
|
|
|
requiredArguments("Headers.set", arguments.length, 2);
|
2020-03-29 04:03:49 +11:00
|
|
|
const [newname, newvalue] = normalizeParams(name, value);
|
|
|
|
validateName(newname);
|
|
|
|
validateValue(newvalue);
|
2018-11-02 21:43:37 -04:00
|
|
|
this[headerMap].set(newname, newvalue);
|
|
|
|
}
|
2019-04-03 20:41:05 +08:00
|
|
|
|
|
|
|
get [Symbol.toStringTag](): string {
|
|
|
|
return "Headers";
|
|
|
|
}
|
2018-11-02 21:43:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// @internal
|
2020-04-13 22:46:23 -04:00
|
|
|
export class HeadersImpl extends DomIterableMixin<
|
2018-11-14 17:36:34 -08:00
|
|
|
string,
|
|
|
|
string,
|
|
|
|
typeof HeadersBase
|
|
|
|
>(HeadersBase, headerMap) {}
|