happy-dom
Version:
Happy DOM is a JavaScript implementation of a web browser without its graphical user interface. It includes many web standards from WHATWG DOM and HTML.
169 lines (155 loc) • 4.44 kB
text/typescript
import DOMException from '../exception/DOMException.js';
import * as PropertySymbol from '../PropertySymbol.js';
import DOMExceptionNameEnum from '../exception/DOMExceptionNameEnum.js';
import IHeadersInit from './types/IHeadersInit.js';
/**
* Fetch headers.
*
* @see https://developer.mozilla.org/en-US/docs/Web/API/Headers
*/
export default class Headers {
public [PropertySymbol.entries]: { [k: string]: { name: string; value: string[] } } = {};
/**
* Constructor.
*
* @param init Headers init.
*/
constructor(init?: IHeadersInit) {
if (init) {
if (init instanceof Headers) {
this[PropertySymbol.entries] = JSON.parse(JSON.stringify(init[PropertySymbol.entries]));
} else if (Array.isArray(init)) {
for (const entry of init) {
if (entry.length !== 2) {
throw new DOMException(
'Failed to construct "Headers": The provided init is not a valid array.',
DOMExceptionNameEnum.invalidStateError
);
}
this.append(entry[0], entry[1]);
}
} else {
for (const name of Object.keys(init)) {
this.set(name, init[name]);
}
}
}
}
/**
* Appends a new value onto an existing header inside a Headers object, or adds the header if it does not already exist.
*
* @param name Name.
* @param value Value.
*/
public append(name: string, value: string): void {
const lowerName = name.toLowerCase();
if (this[PropertySymbol.entries][lowerName]) {
this[PropertySymbol.entries][lowerName].value.push(value);
} else {
this[PropertySymbol.entries][lowerName] = {
name,
value: [value]
};
}
}
/**
* Removes an header.
*
* @param name Name.
*/
public delete(name: string): void {
delete this[PropertySymbol.entries][name.toLowerCase()];
}
/**
* Returns header value.
*
* @param name Name.
* @returns Value.
*/
public get(name: string): string | null {
return this[PropertySymbol.entries][name.toLowerCase()]?.value.join(', ') ?? null;
}
/**
* Sets a new value for an existing header inside a Headers object, or adds the header if it does not already exist.
*
* @param name Name.
* @param value Value.
*/
public set(name: string, value: string): void {
this[PropertySymbol.entries][name.toLowerCase()] = {
name,
value: [value]
};
}
/**
* Returns an array containing the values of all Set-Cookie headers associated with a response.
*
* @returns An array of strings representing the values of all the different Set-Cookie headers.
*/
public getSetCookie(): string[] {
const entry = this[PropertySymbol.entries]['set-cookie'];
if (!entry) {
return [];
}
return entry.value;
}
/**
* Returns whether an Headers object contains a certain key.
*
* @param name Name.
* @returns "true" if the Headers object contains the key.
*/
public has(name: string): boolean {
return !!this[PropertySymbol.entries][name.toLowerCase()];
}
/**
* Executes a callback function once per each key/value pair in the Headers object.
*
* @param callback Callback.
*/
public forEach(callback: (name: string, value: string, thisArg: Headers) => void): void {
for (const header of Object.values(this[PropertySymbol.entries])) {
callback(header.value.join(', '), header.name, this);
}
}
/**
* Returns an iterator, allowing you to go through all keys of the key/value pairs contained in this object.
*
* @returns Iterator.
*/
public *keys(): IterableIterator<string> {
for (const header of Object.values(this[PropertySymbol.entries])) {
yield header.name;
}
}
/**
* Returns an iterator, allowing you to go through all values of the key/value pairs contained in this object.
*
* @returns Iterator.
*/
public *values(): IterableIterator<string> {
for (const header of Object.values(this[PropertySymbol.entries])) {
yield header.value.join(', ');
}
}
/**
* Returns an iterator, allowing you to go through all key/value pairs contained in this object.
*
* @returns Iterator.
*/
public *entries(): IterableIterator<[string, string]> {
for (const header of Object.values(this[PropertySymbol.entries])) {
yield [header.name, header.value.join(', ')];
}
}
/**
* Iterator.
*
* @returns Iterator.
*/
public *[Symbol.iterator](): IterableIterator<[string, string]> {
for (const header of Object.values(this[PropertySymbol.entries])) {
yield [header.name, header.value.join(', ')];
}
}
}