@remix-run/headers
Version:
A toolkit for working with HTTP headers in JavaScript
175 lines (165 loc) • 5.17 kB
text/typescript
import { type HeaderValue } from './header-value.ts';
import { parseParams, quote } from './param-values.ts';
import { capitalize, isValidDate } from './utils.ts';
type SameSiteValue = 'Strict' | 'Lax' | 'None';
export interface SetCookieInit {
/**
* The domain of the cookie. For example, `example.com`.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#domaindomain-value)
*/
domain?: string;
/**
* The expiration date of the cookie. If not specified, the cookie is a session cookie.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#expiresdate)
*/
expires?: Date;
/**
* Indicates this cookie should not be accessible via JavaScript.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#httponly)
*/
httpOnly?: true;
/**
* The maximum age of the cookie in seconds.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#max-age)
*/
maxAge?: number;
/**
* The name of the cookie.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#cookie-namecookie-value)
*/
name?: string;
/**
* The path of the cookie. For example, `/` or `/admin`.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#pathpath-value)
*/
path?: string;
/**
* The `SameSite` attribute of the cookie. This attribute lets servers require that a cookie shouldn't be sent with
* cross-site requests, which provides some protection against cross-site request forgery attacks.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#samesitesamesite-value)
*/
sameSite?: SameSiteValue;
/**
* Indicates the cookie should only be sent over HTTPS.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#secure)
*/
secure?: true;
/**
* The value of the cookie.
*
* [MDN Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#cookie-namecookie-value)
*/
value?: string;
}
/**
* The value of a `Set-Cookie` HTTP header.
*
* [MDN `Set-Cookie` Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie)
*
* [HTTP/1.1 Specification](https://datatracker.ietf.org/doc/html/rfc6265#section-4.1)
*/
export class SetCookie implements HeaderValue, SetCookieInit {
domain?: string;
expires?: Date;
httpOnly?: true;
maxAge?: number;
name?: string;
path?: string;
sameSite?: SameSiteValue;
secure?: true;
value?: string;
constructor(init?: string | SetCookieInit) {
if (init) {
if (typeof init === 'string') {
let params = parseParams(init);
if (params.length > 0) {
this.name = params[0][0];
this.value = params[0][1];
for (let [key, value] of params.slice(1)) {
switch (key.toLowerCase()) {
case 'domain':
this.domain = value;
break;
case 'expires': {
if (typeof value === 'string') {
let date = new Date(value);
if (isValidDate(date)) {
this.expires = date;
}
}
break;
}
case 'httponly':
this.httpOnly = true;
break;
case 'max-age': {
if (typeof value === 'string') {
let v = parseInt(value, 10);
if (!isNaN(v)) this.maxAge = v;
}
break;
}
case 'path':
this.path = value;
break;
case 'samesite':
if (typeof value === 'string' && /strict|lax|none/i.test(value)) {
this.sameSite = capitalize(value) as SameSiteValue;
}
break;
case 'secure':
this.secure = true;
break;
}
}
}
} else {
this.domain = init.domain;
this.expires = init.expires;
this.httpOnly = init.httpOnly;
this.maxAge = init.maxAge;
this.name = init.name;
this.path = init.path;
this.sameSite = init.sameSite;
this.secure = init.secure;
this.value = init.value;
}
}
}
toString(): string {
if (!this.name) {
return '';
}
let parts = [`${this.name}=${quote(this.value || '')}`];
if (this.domain) {
parts.push(`Domain=${this.domain}`);
}
if (this.path) {
parts.push(`Path=${this.path}`);
}
if (this.expires) {
parts.push(`Expires=${this.expires.toUTCString()}`);
}
if (this.maxAge) {
parts.push(`Max-Age=${this.maxAge}`);
}
if (this.secure) {
parts.push('Secure');
}
if (this.httpOnly) {
parts.push('HttpOnly');
}
if (this.sameSite) {
parts.push(`SameSite=${this.sameSite}`);
}
return parts.join('; ');
}
}