Skip to content

Commit 1cc64ff

Browse files
authoredOct 7, 2024··
Use modern JS features, ship TS defs (#175)
1 parent e206fd5 commit 1cc64ff

21 files changed

+5680
-1369
lines changed
 

‎.eslintignore

-2
This file was deleted.

‎.eslintrc.yml

-11
This file was deleted.

‎.github/workflows/ci.yml

+17-244
Original file line numberDiff line numberDiff line change
@@ -1,253 +1,26 @@
11
name: ci
22

33
on:
4-
- pull_request
5-
- push
4+
- pull_request
5+
- push
66

77
jobs:
88
test:
9-
runs-on: ubuntu-20.04
9+
name: Node.js ${{ matrix.node-version }}
10+
runs-on: ubuntu-latest
1011
strategy:
1112
matrix:
12-
name:
13-
- Node.js 0.6
14-
- Node.js 0.8
15-
- Node.js 0.10
16-
- Node.js 0.12
17-
- io.js 1.x
18-
- io.js 2.x
19-
- io.js 3.x
20-
- Node.js 4.x
21-
- Node.js 5.x
22-
- Node.js 6.x
23-
- Node.js 7.x
24-
- Node.js 8.x
25-
- Node.js 9.x
26-
- Node.js 10.x
27-
- Node.js 11.x
28-
- Node.js 12.x
29-
- Node.js 13.x
30-
- Node.js 14.x
31-
- Node.js 15.x
32-
- Node.js 16.x
33-
- Node.js 17.x
34-
- Node.js 18.x
35-
- Node.js 19.x
36-
- Node.js 20.x
37-
- Node.js 21.x
38-
39-
include:
40-
- name: Node.js 0.6
41-
node-version: "0.6"
42-
npm-i: mocha@1.21.5
43-
npm-rm: beautify-benchmark benchmark nyc top-sites
44-
45-
- name: Node.js 0.8
46-
node-version: "0.8"
47-
npm-i: mocha@2.5.3
48-
npm-rm: beautify-benchmark benchmark nyc top-sites
49-
50-
- name: Node.js 0.10
51-
node-version: "0.10"
52-
npm-i: mocha@3.5.3 nyc@10.3.2
53-
npm-rm: beautify-benchmark benchmark top-sites
54-
55-
- name: Node.js 0.12
56-
node-version: "0.12"
57-
npm-i: mocha@3.5.3 nyc@10.3.2
58-
npm-rm: beautify-benchmark benchmark top-sites
59-
60-
- name: io.js 1.x
61-
node-version: "1.8"
62-
npm-i: mocha@3.5.3 nyc@10.3.2
63-
npm-rm: beautify-benchmark benchmark top-sites
64-
65-
- name: io.js 2.x
66-
node-version: "2.5"
67-
npm-i: mocha@3.5.3 nyc@10.3.2
68-
npm-rm: beautify-benchmark benchmark top-sites
69-
70-
- name: io.js 3.x
71-
node-version: "3.3"
72-
npm-i: mocha@3.5.3 nyc@10.3.2
73-
npm-rm: beautify-benchmark benchmark top-sites
74-
75-
- name: Node.js 4.x
76-
node-version: "4.9"
77-
npm-i: mocha@5.2.0 nyc@11.9.0
78-
npm-rm: beautify-benchmark benchmark top-sites
79-
80-
- name: Node.js 5.x
81-
node-version: "5.12"
82-
npm-i: mocha@5.2.0 nyc@11.9.0
83-
npm-rm: beautify-benchmark benchmark top-sites
84-
85-
- name: Node.js 6.x
86-
node-version: "6.17"
87-
npm-i: mocha@6.2.2 nyc@14.1.1
88-
npm-rm: beautify-benchmark benchmark top-sites
89-
90-
- name: Node.js 7.x
91-
node-version: "7.10"
92-
npm-i: mocha@6.2.2 nyc@14.1.1
93-
npm-rm: beautify-benchmark benchmark top-sites
94-
95-
- name: Node.js 8.x
96-
node-version: "8.17"
97-
npm-i: mocha@7.1.2 nyc@14.1.1
98-
npm-rm: beautify-benchmark benchmark top-sites
99-
100-
- name: Node.js 9.x
101-
node-version: "9.11"
102-
npm-i: mocha@7.1.2 nyc@14.1.1
103-
npm-rm: beautify-benchmark benchmark top-sites
104-
105-
- name: Node.js 10.x
106-
node-version: "10.24"
107-
npm-i: mocha@8.4.0
108-
npm-rm: beautify-benchmark benchmark top-sites
109-
110-
- name: Node.js 11.x
111-
node-version: "11.15"
112-
npm-i: mocha@8.4.0
113-
npm-rm: beautify-benchmark benchmark top-sites
114-
115-
- name: Node.js 12.x
116-
node-version: "12.22"
117-
npm-i: mocha@9.2.2
118-
npm-rm: beautify-benchmark benchmark top-sites
119-
120-
- name: Node.js 13.x
121-
node-version: "13.14"
122-
npm-i: mocha@9.2.2
123-
npm-rm: beautify-benchmark benchmark top-sites
124-
125-
- name: Node.js 14.x
126-
node-version: "14.21"
127-
npm-rm: beautify-benchmark benchmark top-sites
128-
129-
- name: Node.js 15.x
130-
node-version: "15.14"
131-
npm-rm: beautify-benchmark benchmark top-sites
132-
133-
- name: Node.js 16.x
134-
node-version: "16.20"
135-
npm-rm: beautify-benchmark benchmark top-sites
136-
137-
- name: Node.js 17.x
138-
node-version: "17.9"
139-
npm-rm: beautify-benchmark benchmark top-sites
140-
141-
- name: Node.js 18.x
142-
node-version: "18.18"
143-
npm-rm: beautify-benchmark benchmark top-sites
144-
145-
- name: Node.js 19.x
146-
node-version: "19.9"
147-
npm-rm: beautify-benchmark benchmark top-sites
148-
149-
- name: Node.js 20.x
150-
node-version: "20.9"
151-
npm-rm: beautify-benchmark benchmark top-sites
152-
153-
- name: Node.js 21.x
154-
node-version: "21.1"
155-
156-
steps:
157-
- uses: actions/checkout@v3
158-
159-
- name: Install Node.js ${{ matrix.node-version }}
160-
shell: bash -eo pipefail -l {0}
161-
run: |
162-
if [[ "${{ matrix.node-version }}" == 0.6* ]]; then
163-
sudo sh -c 'echo "deb http://us.archive.ubuntu.com/ubuntu/ bionic universe" >> /etc/apt/sources.list'
164-
sudo sh -c 'echo "deb http://security.ubuntu.com/ubuntu bionic-security main" >> /etc/apt/sources.list'
165-
sudo apt-get update
166-
sudo apt-get install g++-4.8 gcc-4.8 libssl1.0-dev python2 python-is-python2
167-
export CC=/usr/bin/gcc-4.8
168-
export CXX=/usr/bin/g++-4.8
169-
fi
170-
nvm install --default ${{ matrix.node-version }}
171-
if [[ "${{ matrix.node-version }}" == 0.* && "$(cut -d. -f2 <<< "${{ matrix.node-version }}")" -lt 10 ]]; then
172-
nvm install --alias=npm 0.10
173-
nvm use ${{ matrix.node-version }}
174-
if [[ "$(npm -v)" == 1.1.* ]]; then
175-
nvm exec npm npm install -g npm@1.1
176-
ln -fs "$(which npm)" "$(dirname "$(nvm which npm)")/npm"
177-
else
178-
sed -i '1s;^.*$;'"$(printf '#!%q' "$(nvm which npm)")"';' "$(readlink -f "$(which npm)")"
179-
fi
180-
npm config set strict-ssl false
181-
fi
182-
dirname "$(nvm which ${{ matrix.node-version }})" >> "$GITHUB_PATH"
183-
184-
- name: Configure npm
185-
run: |
186-
if [[ "$(npm config get package-lock)" == "true" ]]; then
187-
npm config set package-lock false
188-
else
189-
npm config set shrinkwrap false
190-
fi
191-
192-
- name: Remove npm module(s) ${{ matrix.npm-rm }}
193-
run: npm rm --silent --save-dev ${{ matrix.npm-rm }}
194-
if: matrix.npm-rm != ''
195-
196-
- name: Install npm module(s) ${{ matrix.npm-i }}
197-
run: npm install --save-dev ${{ matrix.npm-i }}
198-
if: matrix.npm-i != ''
199-
200-
- name: Setup Node.js version-specific dependencies
201-
shell: bash
202-
run: |
203-
# eslint for linting
204-
# - remove on Node.js < 12
205-
if [[ "$(cut -d. -f1 <<< "${{ matrix.node-version }}")" -lt 12 ]]; then
206-
node -pe 'Object.keys(require("./package").devDependencies).join("\n")' | \
207-
grep -E '^eslint(-|$)' | \
208-
sort -r | \
209-
xargs -n1 npm rm --silent --save-dev
210-
fi
211-
212-
- name: Install Node.js dependencies
213-
run: npm install
214-
215-
- name: List environment
216-
id: list_env
217-
shell: bash
218-
run: |
219-
echo "node@$(node -v)"
220-
echo "npm@$(npm -v)"
221-
npm -s ls ||:
222-
(npm -s ls --depth=0 ||:) | awk -F'[ @]' 'NR>1 && $2 { print $2 "=" $3 }' >> "$GITHUB_OUTPUT"
223-
224-
- name: Run tests
225-
shell: bash
226-
run: |
227-
if npm -ps ls nyc | grep -q nyc; then
228-
npm run test-ci
229-
else
230-
npm test
231-
fi
232-
233-
- name: Lint code
234-
if: steps.list_env.outputs.eslint != ''
235-
run: npm run lint
236-
237-
- name: Collect code coverage
238-
uses: coverallsapp/github-action@master
239-
if: steps.list_env.outputs.nyc != ''
240-
with:
241-
github-token: ${{ secrets.GITHUB_TOKEN }}
242-
flag-name: run-${{ matrix.test_number }}
243-
parallel: true
244-
245-
coverage:
246-
needs: test
247-
runs-on: ubuntu-latest
13+
node-version:
14+
- "18"
15+
- "20"
16+
- "22"
24817
steps:
249-
- name: Upload code coverage
250-
uses: coverallsapp/github-action@master
251-
with:
252-
github-token: ${{ secrets.github_token }}
253-
parallel-finished: true
18+
- uses: actions/checkout@v4
19+
- uses: actions/setup-node@v4
20+
with:
21+
node-version: ${{ matrix.node-version }}
22+
- run: npm ci
23+
- run: npm test
24+
- uses: codecov/codecov-action@v4
25+
with:
26+
name: Node.js ${{ matrix.node-version }}

‎.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,5 @@
22
coverage/
33
node_modules/
44
npm-debug.log
5-
package-lock.json
5+
dist/
6+
*.tsbuildinfo

‎README.md

+99-152
Large diffs are not rendered by default.

‎benchmark/index.js

-34
This file was deleted.

‎benchmark/parse-top.js

-37
This file was deleted.

‎benchmark/parse.js

-74
This file was deleted.

‎index.js

-335
This file was deleted.

‎package-lock.json

+4,585
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+28-30
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,42 @@
11
{
22
"name": "cookie",
3-
"description": "HTTP server cookie parsing and serialization",
43
"version": "0.7.2",
5-
"author": "Roman Shtylman <shtylman@gmail.com>",
6-
"contributors": [
7-
"Douglas Christopher Wilson <doug@somethingdoug.com>"
8-
],
9-
"license": "MIT",
4+
"description": "HTTP server cookie parsing and serialization",
105
"keywords": [
116
"cookie",
127
"cookies"
138
],
149
"repository": "jshttp/cookie",
15-
"devDependencies": {
16-
"beautify-benchmark": "0.2.4",
17-
"benchmark": "2.1.4",
18-
"eslint": "8.53.0",
19-
"eslint-plugin-markdown": "3.0.1",
20-
"mocha": "10.2.0",
21-
"nyc": "15.1.0",
22-
"safe-buffer": "5.2.1",
23-
"top-sites": "1.1.194"
24-
},
10+
"license": "MIT",
11+
"author": "Roman Shtylman <shtylman@gmail.com>",
12+
"contributors": [
13+
"Douglas Christopher Wilson <doug@somethingdoug.com>"
14+
],
15+
"main": "dist/index.js",
16+
"types": "dist/index.d.ts",
2517
"files": [
26-
"HISTORY.md",
27-
"LICENSE",
28-
"README.md",
29-
"SECURITY.md",
30-
"index.js"
18+
"dist/"
3119
],
32-
"main": "index.js",
20+
"scripts": {
21+
"bench": "vitest bench",
22+
"build": "ts-scripts build",
23+
"format": "ts-scripts format",
24+
"prepare": "ts-scripts install",
25+
"prepublishOnly": "npm run build",
26+
"specs": "ts-scripts specs",
27+
"test": "ts-scripts test"
28+
},
29+
"devDependencies": {
30+
"@borderless/ts-scripts": "^0.15.0",
31+
"@vitest/coverage-v8": "^2.1.2",
32+
"top-sites": "1.1.194",
33+
"typescript": "^5.6.2",
34+
"vitest": "^2.1.2"
35+
},
3336
"engines": {
34-
"node": ">= 0.6"
37+
"node": ">=18"
3538
},
36-
"scripts": {
37-
"bench": "node benchmark/index.js",
38-
"lint": "eslint .",
39-
"test": "mocha --reporter spec --bail --check-leaks test/",
40-
"test-ci": "nyc --reporter=lcov --reporter=text npm test",
41-
"test-cov": "nyc --reporter=html --reporter=text npm test",
42-
"update-bench": "node scripts/update-benchmark.js"
39+
"ts-scripts": {
40+
"project": "tsconfig.build.json"
4341
}
4442
}
File renamed without changes.

‎scripts/update-benchmark.js

+66-45
Original file line numberDiff line numberDiff line change
@@ -1,69 +1,90 @@
1-
'use strict'
1+
"use strict";
22

3-
var fs = require('fs')
4-
var http = require('http')
5-
var https = require('https')
6-
var path = require('path')
7-
var topSites = require('top-sites')
8-
var url = require('url')
3+
var fs = require("fs");
4+
var http = require("http");
5+
var https = require("https");
6+
var path = require("path");
7+
var topSites = require("top-sites");
8+
var url = require("url");
99

10-
var BENCH_COOKIES_FILE = path.join(__dirname, '..', 'benchmark', 'parse-top.json')
10+
var BENCH_COOKIES_FILE = path.join(__dirname, "parse-top.json");
1111

1212
getAllCookies(topSites.slice(0, 20), function (err, cookies) {
13-
if (err) throw err
14-
var str = '{\n' +
15-
Object.keys(cookies).sort().map(function (key) {
16-
return ' ' + JSON.stringify(key) + ': ' + JSON.stringify(cookies[key])
17-
}).join(',\n') +
18-
'\n}\n'
19-
fs.writeFileSync(BENCH_COOKIES_FILE, str)
20-
})
13+
if (err) throw err;
14+
var str =
15+
"{\n" +
16+
Object.keys(cookies)
17+
.sort()
18+
.map(function (key) {
19+
return " " + JSON.stringify(key) + ": " + JSON.stringify(cookies[key]);
20+
})
21+
.join(",\n") +
22+
"\n}\n";
23+
fs.writeFileSync(BENCH_COOKIES_FILE, str);
24+
});
2125

22-
function get (href, callback) {
23-
var protocol = url.parse(href, false, true).protocol
24-
var proto = protocol === 'https:' ? https : http
26+
function get(href, callback) {
27+
var protocol = url.parse(href, false, true).protocol;
28+
var proto = protocol === "https:" ? https : http;
2529

26-
proto.get(href)
27-
.on('error', callback)
28-
.on('response', function (res) {
29-
if (res.headers.location && res.statusCode >= 300 && res.statusCode < 400) {
30-
get(url.resolve(href, res.headers.location), callback)
30+
proto
31+
.get(href)
32+
.on("error", callback)
33+
.on("response", function (res) {
34+
if (
35+
res.headers.location &&
36+
res.statusCode >= 300 &&
37+
res.statusCode < 400
38+
) {
39+
get(url.resolve(href, res.headers.location), callback);
3140
} else {
32-
callback(null, res)
41+
callback(null, res);
3342
}
34-
})
43+
});
3544
}
3645

37-
function getAllCookies (sites, callback) {
38-
var all = Object.create(null)
39-
var wait = sites.length
46+
function getAllCookies(sites, callback) {
47+
var all = Object.create(null);
48+
var wait = sites.length;
4049

4150
sites.forEach(function (site) {
4251
getCookies(site, function (err, cookies) {
4352
if (!err && cookies.length) {
44-
all[site.rootDomain] = cookies.map(obfuscate).join('; ')
53+
all[site.rootDomain] = cookies.map(obfuscate).join("; ");
4554
}
4655
if (!--wait) {
47-
callback(null, all)
56+
callback(null, all);
4857
}
49-
})
50-
})
58+
});
59+
});
5160
}
5261

53-
function getCookies (site, callback) {
54-
var href = url.format({ hostname: site.rootDomain, protocol: 'http' })
62+
function getCookies(site, callback) {
63+
var href = url.format({ hostname: site.rootDomain, protocol: "http" });
5564
get(href, function (err, res) {
56-
if (err) return callback(err)
57-
var cookies = (res.headers['set-cookie'] || []).map(function (c) { return c.split(';')[0] })
58-
callback(null, cookies)
59-
})
65+
if (err) return callback(err);
66+
var cookies = (res.headers["set-cookie"] || []).map(function (c) {
67+
return c.split(";")[0];
68+
});
69+
callback(null, cookies);
70+
});
6071
}
6172

62-
function obfuscate (str) {
73+
function obfuscate(str) {
6374
return str
64-
.replace(/%[0-9a-f]{2}/gi, function () { return '%__' })
65-
.replace(/[a-z]/g, function () { return 'l' })
66-
.replace(/[A-Z]/g, function () { return 'U' })
67-
.replace(/[0-9]/g, function () { return '0' })
68-
.replace(/%__/g, function () { return '%22' })
75+
.replace(/%[0-9a-f]{2}/gi, function () {
76+
return "%__";
77+
})
78+
.replace(/[a-z]/g, function () {
79+
return "l";
80+
})
81+
.replace(/[A-Z]/g, function () {
82+
return "U";
83+
})
84+
.replace(/[0-9]/g, function () {
85+
return "0";
86+
})
87+
.replace(/%__/g, function () {
88+
return "%22";
89+
});
6990
}

‎src/index.ts

+389
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,389 @@
1+
/**
2+
* RegExp to match cookie-name in RFC 6265 sec 4.1.1
3+
* This refers out to the obsoleted definition of token in RFC 2616 sec 2.2
4+
* which has been replaced by the token definition in RFC 7230 appendix B.
5+
*
6+
* cookie-name = token
7+
* token = 1*tchar
8+
* tchar = "!" / "#" / "$" / "%" / "&" / "'" /
9+
* "*" / "+" / "-" / "." / "^" / "_" /
10+
* "`" / "|" / "~" / DIGIT / ALPHA
11+
*/
12+
const cookieNameRegExp = /^[!#$%&'*+\-.^_`|~0-9A-Za-z]+$/;
13+
14+
/**
15+
* RegExp to match cookie-value in RFC 6265 sec 4.1.1
16+
*
17+
* cookie-value = *cookie-octet / ( DQUOTE *cookie-octet DQUOTE )
18+
* cookie-octet = %x21 / %x23-2B / %x2D-3A / %x3C-5B / %x5D-7E
19+
* ; US-ASCII characters excluding CTLs,
20+
* ; whitespace DQUOTE, comma, semicolon,
21+
* ; and backslash
22+
*/
23+
const cookieValueRegExp =
24+
/^("?)[\u0021\u0023-\u002B\u002D-\u003A\u003C-\u005B\u005D-\u007E]*\1$/;
25+
26+
/**
27+
* RegExp to match domain-value in RFC 6265 sec 4.1.1
28+
*
29+
* domain-value = <subdomain>
30+
* ; defined in [RFC1034], Section 3.5, as
31+
* ; enhanced by [RFC1123], Section 2.1
32+
* <subdomain> = <label> | <subdomain> "." <label>
33+
* <label> = <let-dig> [ [ <ldh-str> ] <let-dig> ]
34+
* Labels must be 63 characters or less.
35+
* 'let-dig' not 'letter' in the first char, per RFC1123
36+
* <ldh-str> = <let-dig-hyp> | <let-dig-hyp> <ldh-str>
37+
* <let-dig-hyp> = <let-dig> | "-"
38+
* <let-dig> = <letter> | <digit>
39+
* <letter> = any one of the 52 alphabetic characters A through Z in
40+
* upper case and a through z in lower case
41+
* <digit> = any one of the ten digits 0 through 9
42+
*
43+
* Keep support for leading dot: https://github.com/jshttp/cookie/issues/173
44+
*
45+
* > (Note that a leading %x2E ("."), if present, is ignored even though that
46+
* character is not permitted, but a trailing %x2E ("."), if present, will
47+
* cause the user agent to ignore the attribute.)
48+
*/
49+
const domainValueRegExp =
50+
/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i;
51+
52+
/**
53+
* RegExp to match path-value in RFC 6265 sec 4.1.1
54+
*
55+
* path-value = <any CHAR except CTLs or ";">
56+
* CHAR = %x01-7F
57+
* ; defined in RFC 5234 appendix B.1
58+
*/
59+
const pathValueRegExp = /^[\u0020-\u003A\u003D-\u007E]*$/;
60+
61+
const __toString = Object.prototype.toString;
62+
const __hasOwnProperty = Object.prototype.hasOwnProperty;
63+
64+
const NullObject = /* @__PURE__ */ (() => {
65+
const C = function () {};
66+
C.prototype = Object.create(null);
67+
return C;
68+
})() as unknown as { new (): any };
69+
70+
/**
71+
* Parse options.
72+
*/
73+
export interface ParseOptions {
74+
/**
75+
* Specifies a function that will be used to decode a cookie's value. Since
76+
* the value of a cookie has a limited character set (and must be a simple
77+
* string), this function can be used to decode a previously-encoded cookie
78+
* value into a JavaScript string.
79+
*
80+
* Note: if an error is thrown from this function, the original, non-decoded
81+
* cookie value will be returned as the cookie's value.
82+
*
83+
* @default decodeURIComponent
84+
*/
85+
decode?: (str: string) => string;
86+
}
87+
88+
/**
89+
* Parse a cookie header.
90+
*
91+
* Parse the given cookie header string into an object
92+
* The object has the various cookies as keys(names) => values
93+
*/
94+
export function parse(
95+
str: string,
96+
options?: ParseOptions,
97+
): Record<string, string> {
98+
const obj: Record<string, string> = new NullObject();
99+
const len = str.length;
100+
// RFC 6265 sec 4.1.1, RFC 2616 2.2 defines a cookie name consists of one char minimum, plus '='.
101+
if (len < 2) return obj;
102+
103+
const dec = options?.decode || decode;
104+
let index = 0;
105+
106+
do {
107+
const eqIdx = str.indexOf("=", index);
108+
if (eqIdx === -1) break; // No more cookie pairs.
109+
110+
const colonIdx = str.indexOf(";", index);
111+
const endIdx = colonIdx === -1 ? len : colonIdx;
112+
113+
if (eqIdx > endIdx) {
114+
// backtrack on prior semicolon
115+
index = str.lastIndexOf(";", eqIdx - 1) + 1;
116+
continue;
117+
}
118+
119+
const keyStartIdx = startIndex(str, index, eqIdx);
120+
const keyEndIdx = endIndex(str, eqIdx, keyStartIdx);
121+
const key = str.slice(keyStartIdx, keyEndIdx);
122+
123+
// only assign once
124+
if (!__hasOwnProperty.call(obj, key)) {
125+
let valStartIdx = startIndex(str, eqIdx + 1, endIdx);
126+
let valEndIdx = endIndex(str, endIdx, valStartIdx);
127+
128+
if (
129+
str.charCodeAt(valStartIdx) === 0x22 /* " */ &&
130+
str.charCodeAt(valEndIdx - 1) === 0x22 /* " */
131+
) {
132+
valStartIdx++;
133+
valEndIdx--;
134+
}
135+
136+
const val = str.slice(valStartIdx, valEndIdx);
137+
obj[key] = tryDecode(val, dec);
138+
}
139+
140+
index = endIdx + 1;
141+
} while (index < len);
142+
143+
return obj;
144+
}
145+
146+
function startIndex(str: string, index: number, max: number) {
147+
do {
148+
const code = str.charCodeAt(index);
149+
if (code !== 0x20 /* */ && code !== 0x09 /* \t */) return index;
150+
} while (++index < max);
151+
return max;
152+
}
153+
154+
function endIndex(str: string, index: number, min: number) {
155+
while (index > min) {
156+
const code = str.charCodeAt(--index);
157+
if (code !== 0x20 /* */ && code !== 0x09 /* \t */) return index + 1;
158+
}
159+
return min;
160+
}
161+
162+
/**
163+
* Serialize options.
164+
*/
165+
export interface SerializeOptions {
166+
/**
167+
* Specifies a function that will be used to encode a cookie's value. Since
168+
* value of a cookie has a limited character set (and must be a simple string),
169+
* this function can be used to encode a value into a string suited for a cookie's value.
170+
*
171+
* @default encodeURIComponent
172+
*/
173+
encode?: (str: string) => string;
174+
/**
175+
* Specifies the `number` (in seconds) to be the value for the [`Max-Age` `Set-Cookie` attribute](https://tools.ietf.org/html/rfc6265#section-5.2.2).
176+
* The given number will be converted to an integer by rounding down. By default, no maximum age is set.
177+
*
178+
* The [cookie storage model specification](https://tools.ietf.org/html/rfc6265#section-5.3) states that if both `expires` and
179+
* `maxAge` are set, then `maxAge` takes precedence, but it is possible not all clients by obey this,
180+
* so if both are set, they should point to the same date and time.
181+
*/
182+
maxAge?: number;
183+
/**
184+
* Specifies the `Date` object to be the value for the [`Expires` `Set-Cookie` attribute](https://tools.ietf.org/html/rfc6265#section-5.2.1).
185+
* By default, no expiration is set, and most clients will consider this a "non-persistent cookie" and
186+
* will delete it on a condition like exiting a web browser application.
187+
*
188+
* The [cookie storage model specification](https://tools.ietf.org/html/rfc6265#section-5.3) states that if both `expires` and
189+
* `maxAge` are set, then `maxAge` takes precedence, but it is possible not all clients by obey this,
190+
* so if both are set, they should point to the same date and time.
191+
*/
192+
expires?: Date;
193+
/**
194+
* Specifies the value for the [`Domain` `Set-Cookie` attribute](https://tools.ietf.org/html/rfc6265#section-5.2.3).
195+
* By default, no domain is set, and most clients will consider the cookie to apply to only the current domain.
196+
*/
197+
domain?: string;
198+
/**
199+
* Specifies the value for the [`Path` `Set-Cookie` attribute](https://tools.ietf.org/html/rfc6265#section-5.2.4). By default, the path
200+
* is considered the ["default path"](https://tools.ietf.org/html/rfc6265#section-5.1.4).
201+
*/
202+
path?: string;
203+
/**
204+
* Specifies the `boolean` value for the [`HttpOnly` `Set-Cookie` attribute][rfc-6265-5.2.6]. When truthy,
205+
* the `HttpOnly` attribute is set, otherwise it is not. By default, the `HttpOnly` attribute is not set.
206+
*
207+
* Be careful when setting this to `true`, as compliant clients will not allow client-side
208+
* JavaScript to see the cookie in `document.cookie`.
209+
*/
210+
httpOnly?: boolean;
211+
/**
212+
* Specifies the `boolean` value for the [`Secure` `Set-Cookie` attribute](https://tools.ietf.org/html/rfc6265#section-5.2.5). When truthy,
213+
* the `Secure` attribute is set, otherwise it is not. By default, the `Secure` attribute is not set.
214+
*
215+
* Be careful when setting this to `true`, as compliant clients will not send the cookie back to
216+
* the server in the future if the browser does not have an HTTPS connection.
217+
*/
218+
secure?: boolean;
219+
/**
220+
* Specifies the `boolean` value for the [`Partitioned` `Set-Cookie`](https://tools.ietf.org/html/draft-cutler-httpbis-partitioned-cookies/)
221+
* attribute. When truthy, the `Partitioned` attribute is set, otherwise it is not. By default, the
222+
* `Partitioned` attribute is not set.
223+
*
224+
* This is an attribute that has not yet been fully standardized, and may change in the future.
225+
* This also means many clients may ignore this attribute until they understand it. More information
226+
* about can be found in [the proposal](https://github.com/privacycg/CHIPS).
227+
*/
228+
partitioned?: boolean;
229+
/**
230+
* Specifies the value for the [`Priority` `Set-Cookie` attribute](https://tools.ietf.org/html/draft-west-cookie-priority-00#section-4.1).
231+
*
232+
* - `'low'` will set the `Priority` attribute to `Low`.
233+
* - `'medium'` will set the `Priority` attribute to `Medium`, the default priority when not set.
234+
* - `'high'` will set the `Priority` attribute to `High`.
235+
*
236+
* More information about the different priority levels can be found in
237+
* [the specification](https://tools.ietf.org/html/draft-west-cookie-priority-00#section-4.1).
238+
*/
239+
priority?: "low" | "medium" | "high";
240+
/**
241+
* Specifies the value for the [`SameSite` `Set-Cookie` attribute](https://tools.ietf.org/html/draft-ietf-httpbis-rfc6265bis-09#section-5.4.7).
242+
*
243+
* - `true` will set the `SameSite` attribute to `Strict` for strict same site enforcement.
244+
* - `'lax'` will set the `SameSite` attribute to `Lax` for lax same site enforcement.
245+
* - `'none'` will set the `SameSite` attribute to `None` for an explicit cross-site cookie.
246+
* - `'strict'` will set the `SameSite` attribute to `Strict` for strict same site enforcement.
247+
*
248+
* More information about the different enforcement levels can be found in
249+
* [the specification](https://tools.ietf.org/html/draft-ietf-httpbis-rfc6265bis-09#section-5.4.7).
250+
*/
251+
sameSite?: boolean | "lax" | "strict" | "none";
252+
}
253+
254+
/**
255+
* Serialize data into a cookie header.
256+
*
257+
* Serialize a name value pair into a cookie string suitable for
258+
* http headers. An optional options object specifies cookie parameters.
259+
*
260+
* serialize('foo', 'bar', { httpOnly: true })
261+
* => "foo=bar; httpOnly"
262+
*/
263+
export function serialize(
264+
name: string,
265+
val: string,
266+
options?: SerializeOptions,
267+
): string {
268+
const enc = options?.encode || encodeURIComponent;
269+
270+
if (!cookieNameRegExp.test(name)) {
271+
throw new TypeError(`argument name is invalid: ${name}`);
272+
}
273+
274+
const value = enc(val);
275+
276+
if (!cookieValueRegExp.test(value)) {
277+
throw new TypeError(`argument val is invalid: ${val}`);
278+
}
279+
280+
let str = name + "=" + value;
281+
if (!options) return str;
282+
283+
if (options.maxAge !== undefined) {
284+
if (!Number.isInteger(options.maxAge)) {
285+
throw new TypeError(`option maxAge is invalid: ${options.maxAge}`);
286+
}
287+
288+
str += "; Max-Age=" + options.maxAge;
289+
}
290+
291+
if (options.domain) {
292+
if (!domainValueRegExp.test(options.domain)) {
293+
throw new TypeError(`option domain is invalid: ${options.domain}`);
294+
}
295+
296+
str += "; Domain=" + options.domain;
297+
}
298+
299+
if (options.path) {
300+
if (!pathValueRegExp.test(options.path)) {
301+
throw new TypeError(`option path is invalid: ${options.path}`);
302+
}
303+
304+
str += "; Path=" + options.path;
305+
}
306+
307+
if (options.expires) {
308+
if (
309+
!isDate(options.expires) ||
310+
!Number.isFinite(options.expires.valueOf())
311+
) {
312+
throw new TypeError(`option expires is invalid: ${options.expires}`);
313+
}
314+
315+
str += "; Expires=" + options.expires.toUTCString();
316+
}
317+
318+
if (options.httpOnly) {
319+
str += "; HttpOnly";
320+
}
321+
322+
if (options.secure) {
323+
str += "; Secure";
324+
}
325+
326+
if (options.partitioned) {
327+
str += "; Partitioned";
328+
}
329+
330+
if (options.priority) {
331+
switch (options.priority) {
332+
case "low":
333+
str += "; Priority=Low";
334+
break;
335+
case "medium":
336+
str += "; Priority=Medium";
337+
break;
338+
case "high":
339+
str += "; Priority=High";
340+
break;
341+
default:
342+
throw new TypeError(`option priority is invalid: ${options.priority}`);
343+
}
344+
}
345+
346+
if (options.sameSite) {
347+
switch (options.sameSite) {
348+
case true:
349+
case "strict":
350+
str += "; SameSite=Strict";
351+
break;
352+
case "lax":
353+
str += "; SameSite=Lax";
354+
break;
355+
case "none":
356+
str += "; SameSite=None";
357+
break;
358+
default:
359+
throw new TypeError(`option sameSite is invalid: ${options.sameSite}`);
360+
}
361+
}
362+
363+
return str;
364+
}
365+
366+
/**
367+
* URL-decode string value. Optimized to skip native call when no %.
368+
*/
369+
function decode(str: string): string {
370+
return str.indexOf("%") !== -1 ? decodeURIComponent(str) : str;
371+
}
372+
373+
/**
374+
* Determine if value is a Date.
375+
*/
376+
function isDate(val: any): val is Date {
377+
return __toString.call(val) === "[object Date]";
378+
}
379+
380+
/**
381+
* Try decoding a string using a decoding function.
382+
*/
383+
function tryDecode(str: string, decode: (str: string) => string): string {
384+
try {
385+
return decode(str);
386+
} catch (e) {
387+
return str;
388+
}
389+
}

‎src/parse.bench.ts

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { describe, bench } from "vitest";
2+
import * as cookie from "./index.js";
3+
import top from "../scripts/parse-top.json";
4+
5+
describe("parse", () => {
6+
bench("simple", () => {
7+
cookie.parse("foo=bar");
8+
});
9+
10+
bench("decode", () => {
11+
cookie.parse("foo=hello%20there!");
12+
});
13+
14+
bench("unquote", () => {
15+
cookie.parse('foo="foo bar"');
16+
});
17+
18+
bench("duplicates", () => {
19+
cookie.parse(genCookies(2) + "; " + genCookies(2));
20+
});
21+
22+
bench("10 cookies", () => {
23+
cookie.parse(genCookies(10));
24+
});
25+
26+
bench("100 cookies", () => {
27+
cookie.parse(genCookies(100));
28+
});
29+
});
30+
31+
describe("parse top-sites", () => {
32+
Object.entries(top).forEach(function ([domain, value]) {
33+
bench("parse " + domain, () => {
34+
cookie.parse(value);
35+
});
36+
});
37+
});
38+
39+
function genCookies(num: number) {
40+
let str = "";
41+
42+
for (let i = 0; i < num; i++) {
43+
str += "; foo" + i + "=bar";
44+
}
45+
46+
return str.slice(2);
47+
}

‎src/parse.spec.ts

+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
import { describe, it, expect } from "vitest";
2+
import * as cookie from "./index.js";
3+
4+
describe("cookie.parse(str)", function () {
5+
it("should parse cookie string to object", function () {
6+
expect(cookie.parse("foo=bar")).toEqual({ foo: "bar" });
7+
expect(cookie.parse("foo=123")).toEqual({ foo: "123" });
8+
});
9+
10+
it("should ignore OWS", function () {
11+
expect(cookie.parse("FOO = bar; baz = raz")).toEqual({
12+
FOO: "bar",
13+
baz: "raz",
14+
});
15+
});
16+
17+
it("should parse cookie with empty value", function () {
18+
expect(cookie.parse("foo=; bar=")).toEqual({ foo: "", bar: "" });
19+
});
20+
21+
it("should parse cookie with minimum length", function () {
22+
expect(cookie.parse("f=")).toEqual({ f: "" });
23+
expect(cookie.parse("f=;b=")).toEqual({ f: "", b: "" });
24+
});
25+
26+
it("should URL-decode values", function () {
27+
expect(cookie.parse('foo="bar=123456789&name=Magic+Mouse"')).toEqual({
28+
foo: "bar=123456789&name=Magic+Mouse",
29+
});
30+
31+
expect(cookie.parse("email=%20%22%2c%3b%2f")).toEqual({ email: ' ",;/' });
32+
});
33+
34+
it("should parse quoted values", function () {
35+
expect(cookie.parse('foo="bar"')).toEqual({ foo: "bar" });
36+
expect(cookie.parse('foo=" a b c "')).toEqual({ foo: " a b c " });
37+
});
38+
39+
it("should trim whitespace around key and value", function () {
40+
expect(cookie.parse(' foo = "bar" ')).toEqual({ foo: "bar" });
41+
expect(cookie.parse(" foo = bar ; fizz = buzz ")).toEqual({
42+
foo: "bar",
43+
fizz: "buzz",
44+
});
45+
expect(cookie.parse(' foo = " a b c " ')).toEqual({ foo: " a b c " });
46+
expect(cookie.parse(" = bar ")).toEqual({ "": "bar" });
47+
expect(cookie.parse(" foo = ")).toEqual({ foo: "" });
48+
expect(cookie.parse(" = ")).toEqual({ "": "" });
49+
expect(cookie.parse("\tfoo\t=\tbar\t")).toEqual({ foo: "bar" });
50+
});
51+
52+
it("should return original value on escape error", function () {
53+
expect(cookie.parse("foo=%1;bar=bar")).toEqual({ foo: "%1", bar: "bar" });
54+
});
55+
56+
it("should ignore cookies without value", function () {
57+
expect(cookie.parse("foo=bar;fizz ; buzz")).toEqual({ foo: "bar" });
58+
expect(cookie.parse(" fizz; foo= bar")).toEqual({ foo: "bar" });
59+
});
60+
61+
it("should ignore duplicate cookies", function () {
62+
expect(cookie.parse("foo=%1;bar=bar;foo=boo")).toEqual({
63+
foo: "%1",
64+
bar: "bar",
65+
});
66+
expect(cookie.parse("foo=false;bar=bar;foo=true")).toEqual({
67+
foo: "false",
68+
bar: "bar",
69+
});
70+
expect(cookie.parse("foo=;bar=bar;foo=boo")).toEqual({
71+
foo: "",
72+
bar: "bar",
73+
});
74+
});
75+
76+
it("should parse native properties", function () {
77+
expect(cookie.parse("toString=foo;valueOf=bar")).toEqual({
78+
toString: "foo",
79+
valueOf: "bar",
80+
});
81+
});
82+
});
83+
84+
describe("cookie.parse(str, options)", function () {
85+
describe('with "decode" option', function () {
86+
it("should specify alternative value decoder", function () {
87+
expect(
88+
cookie.parse('foo="YmFy"', {
89+
decode: function (v) {
90+
return Buffer.from(v, "base64").toString();
91+
},
92+
}),
93+
).toEqual({ foo: "bar" });
94+
});
95+
});
96+
});

‎src/serialize.spec.ts

+338
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,338 @@
1+
import { describe, it, expect } from "vitest";
2+
import * as cookie from "./index.js";
3+
4+
describe("cookie.serialize(name, value)", function () {
5+
it("should serialize name and value", function () {
6+
expect(cookie.serialize("foo", "bar")).toEqual("foo=bar");
7+
});
8+
9+
it("should URL-encode value", function () {
10+
expect(cookie.serialize("foo", "bar +baz")).toEqual("foo=bar%20%2Bbaz");
11+
});
12+
13+
it("should serialize empty value", function () {
14+
expect(cookie.serialize("foo", "")).toEqual("foo=");
15+
});
16+
17+
it("should serialize valid name", function () {
18+
var validNames = [
19+
"foo",
20+
"foo!bar",
21+
"foo#bar",
22+
"foo$bar",
23+
"foo'bar",
24+
"foo*bar",
25+
"foo+bar",
26+
"foo-bar",
27+
"foo.bar",
28+
"foo^bar",
29+
"foo_bar",
30+
"foo`bar",
31+
"foo|bar",
32+
"foo~bar",
33+
"foo7bar",
34+
];
35+
36+
validNames.forEach(function (name) {
37+
expect(cookie.serialize(name, "baz")).toEqual(name + "=baz");
38+
});
39+
});
40+
41+
it("should throw for invalid name", function () {
42+
var invalidNames = [
43+
"foo\n",
44+
"foo\u280a",
45+
"foo/foo",
46+
"foo,foo",
47+
"foo;foo",
48+
"foo@foo",
49+
"foo[foo]",
50+
"foo?foo",
51+
"foo:foo",
52+
"foo{foo}",
53+
"foo foo",
54+
"foo\tfoo",
55+
'foo"foo',
56+
"foo<script>foo",
57+
];
58+
59+
invalidNames.forEach(function (name) {
60+
expect(cookie.serialize.bind(cookie, name, "bar")).toThrow(
61+
/argument name is invalid/,
62+
);
63+
});
64+
});
65+
});
66+
67+
describe("cookie.serialize(name, value, options)", function () {
68+
describe('with "domain" option', function () {
69+
it("should serialize valid domain", function () {
70+
var validDomains = [
71+
"example.com",
72+
"sub.example.com",
73+
".example.com",
74+
"localhost",
75+
".localhost",
76+
"my-site.org",
77+
"localhost",
78+
];
79+
80+
validDomains.forEach(function (domain) {
81+
expect(cookie.serialize("foo", "bar", { domain: domain })).toEqual(
82+
"foo=bar; Domain=" + domain,
83+
);
84+
});
85+
});
86+
87+
it("should throw for invalid domain", function () {
88+
var invalidDomains = [
89+
"example.com\n",
90+
"sub.example.com\u0000",
91+
"my site.org",
92+
"domain..com",
93+
"example.com; Path=/",
94+
"example.com /* inject a comment */",
95+
];
96+
97+
invalidDomains.forEach(function (domain) {
98+
expect(
99+
cookie.serialize.bind(cookie, "foo", "bar", { domain: domain }),
100+
).toThrow(/option domain is invalid/);
101+
});
102+
});
103+
});
104+
105+
describe('with "encode" option', function () {
106+
it("should specify alternative value encoder", function () {
107+
expect(
108+
cookie.serialize("foo", "bar", {
109+
encode: function (v) {
110+
return Buffer.from(v, "utf8").toString("base64");
111+
},
112+
}),
113+
).toEqual("foo=YmFy");
114+
});
115+
116+
it("should throw when returned value is invalid", function () {
117+
expect(
118+
cookie.serialize.bind(cookie, "foo", "+ \n", {
119+
encode: function (v) {
120+
return v;
121+
},
122+
}),
123+
).toThrow(/argument val is invalid/);
124+
expect(
125+
cookie.serialize.bind(cookie, "foo", "foo bar", {
126+
encode: function (v) {
127+
return v;
128+
},
129+
}),
130+
).toThrow(/argument val is invalid/);
131+
});
132+
});
133+
134+
describe('with "expires" option', function () {
135+
it("should throw on invalid date", function () {
136+
expect(
137+
cookie.serialize.bind(cookie, "foo", "bar", { expires: new Date(NaN) }),
138+
).toThrow(/option expires is invalid/);
139+
});
140+
141+
it("should set expires to given date", function () {
142+
expect(
143+
cookie.serialize("foo", "bar", {
144+
expires: new Date(Date.UTC(2000, 11, 24, 10, 30, 59, 900)),
145+
}),
146+
).toEqual("foo=bar; Expires=Sun, 24 Dec 2000 10:30:59 GMT");
147+
});
148+
});
149+
150+
describe('with "httpOnly" option', function () {
151+
it("should include httpOnly flag when true", function () {
152+
expect(cookie.serialize("foo", "bar", { httpOnly: true })).toEqual(
153+
"foo=bar; HttpOnly",
154+
);
155+
});
156+
157+
it("should not include httpOnly flag when false", function () {
158+
expect(cookie.serialize("foo", "bar", { httpOnly: false })).toEqual(
159+
"foo=bar",
160+
);
161+
});
162+
});
163+
164+
describe('with "maxAge" option', function () {
165+
it("should throw when not a number", function () {
166+
expect(function () {
167+
cookie.serialize("foo", "bar", { maxAge: "buzz" as any });
168+
}).toThrow(/option maxAge is invalid/);
169+
});
170+
171+
it("should throw when Infinity", function () {
172+
expect(function () {
173+
cookie.serialize("foo", "bar", { maxAge: Infinity });
174+
}).toThrow(/option maxAge is invalid/);
175+
});
176+
177+
it("should throw when max-age is not an integer", function () {
178+
expect(function () {
179+
cookie.serialize("foo", "bar", { maxAge: 3.14 });
180+
}).toThrow(/option maxAge is invalid/);
181+
});
182+
183+
it("should set max-age to value", function () {
184+
expect(cookie.serialize("foo", "bar", { maxAge: 1000 })).toEqual(
185+
"foo=bar; Max-Age=1000",
186+
);
187+
expect(cookie.serialize("foo", "bar", { maxAge: 0 })).toEqual(
188+
"foo=bar; Max-Age=0",
189+
);
190+
});
191+
192+
it("should not set when undefined", function () {
193+
expect(cookie.serialize("foo", "bar", { maxAge: undefined })).toEqual(
194+
"foo=bar",
195+
);
196+
});
197+
});
198+
199+
describe('with "partitioned" option', function () {
200+
it("should include partitioned flag when true", function () {
201+
expect(cookie.serialize("foo", "bar", { partitioned: true })).toEqual(
202+
"foo=bar; Partitioned",
203+
);
204+
});
205+
206+
it("should not include partitioned flag when false", function () {
207+
expect(cookie.serialize("foo", "bar", { partitioned: false })).toEqual(
208+
"foo=bar",
209+
);
210+
});
211+
212+
it("should not include partitioned flag when not defined", function () {
213+
expect(cookie.serialize("foo", "bar", {})).toEqual("foo=bar");
214+
});
215+
});
216+
217+
describe('with "path" option', function () {
218+
it("should serialize path", function () {
219+
var validPaths = [
220+
"/",
221+
"/login",
222+
"/foo.bar/baz",
223+
"/foo-bar",
224+
"/foo=bar?baz",
225+
'/foo"bar"',
226+
"/../foo/bar",
227+
"../foo/",
228+
"./",
229+
];
230+
231+
validPaths.forEach(function (path) {
232+
expect(cookie.serialize("foo", "bar", { path: path })).toEqual(
233+
"foo=bar; Path=" + path,
234+
);
235+
});
236+
});
237+
238+
it("should throw for invalid value", function () {
239+
var invalidPaths = [
240+
"/\n",
241+
"/foo\u0000",
242+
"/path/with\rnewline",
243+
"/; Path=/sensitive-data",
244+
'/login"><script>alert(1)</script>',
245+
];
246+
247+
invalidPaths.forEach(function (path) {
248+
expect(
249+
cookie.serialize.bind(cookie, "foo", "bar", { path: path }),
250+
).toThrow(/option path is invalid/);
251+
});
252+
});
253+
});
254+
255+
describe('with "priority" option', function () {
256+
it("should throw on invalid priority", function () {
257+
expect(function () {
258+
cookie.serialize("foo", "bar", { priority: "foo" as any });
259+
}).toThrow(/option priority is invalid/);
260+
});
261+
262+
it("should throw on non-string", function () {
263+
expect(function () {
264+
cookie.serialize("foo", "bar", { priority: 42 as any });
265+
}).toThrow(/option priority is invalid/);
266+
});
267+
268+
it("should set priority low", function () {
269+
expect(cookie.serialize("foo", "bar", { priority: "low" })).toEqual(
270+
"foo=bar; Priority=Low",
271+
);
272+
});
273+
274+
it("should set priority medium", function () {
275+
expect(cookie.serialize("foo", "bar", { priority: "medium" })).toEqual(
276+
"foo=bar; Priority=Medium",
277+
);
278+
});
279+
280+
it("should set priority high", function () {
281+
expect(cookie.serialize("foo", "bar", { priority: "high" })).toEqual(
282+
"foo=bar; Priority=High",
283+
);
284+
});
285+
});
286+
287+
describe('with "sameSite" option', function () {
288+
it("should throw on invalid sameSite", function () {
289+
expect(() => {
290+
cookie.serialize("foo", "bar", { sameSite: "foo" as any });
291+
}).toThrow(/option sameSite is invalid/);
292+
});
293+
294+
it("should set sameSite strict", function () {
295+
expect(cookie.serialize("foo", "bar", { sameSite: "strict" })).toEqual(
296+
"foo=bar; SameSite=Strict",
297+
);
298+
});
299+
300+
it("should set sameSite lax", function () {
301+
expect(cookie.serialize("foo", "bar", { sameSite: "lax" })).toEqual(
302+
"foo=bar; SameSite=Lax",
303+
);
304+
});
305+
306+
it("should set sameSite none", function () {
307+
expect(cookie.serialize("foo", "bar", { sameSite: "none" })).toEqual(
308+
"foo=bar; SameSite=None",
309+
);
310+
});
311+
312+
it("should set sameSite strict when true", function () {
313+
expect(cookie.serialize("foo", "bar", { sameSite: true })).toEqual(
314+
"foo=bar; SameSite=Strict",
315+
);
316+
});
317+
318+
it("should not set sameSite when false", function () {
319+
expect(cookie.serialize("foo", "bar", { sameSite: false })).toEqual(
320+
"foo=bar",
321+
);
322+
});
323+
});
324+
325+
describe('with "secure" option', function () {
326+
it("should include secure flag when true", function () {
327+
expect(cookie.serialize("foo", "bar", { secure: true })).toEqual(
328+
"foo=bar; Secure",
329+
);
330+
});
331+
332+
it("should not include secure flag when false", function () {
333+
expect(cookie.serialize("foo", "bar", { secure: false })).toEqual(
334+
"foo=bar",
335+
);
336+
});
337+
});
338+
});

‎test/parse.js

-85
This file was deleted.

‎test/serialize.js

-319
This file was deleted.

‎tsconfig.build.json

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
{
2+
"extends": "./tsconfig.json",
3+
"exclude": ["src/**/*.spec.ts", "src/**/*.bench.ts"]
4+
}

‎tsconfig.json

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
{
2+
"extends": "@borderless/ts-scripts/configs/tsconfig.json",
3+
"compilerOptions": {
4+
"outDir": "dist",
5+
"rootDir": "src",
6+
"resolveJsonModule": true
7+
},
8+
"include": ["src/**/*"]
9+
}

0 commit comments

Comments
 (0)
Please sign in to comment.