Skip to content

Commit 60f2121

Browse files
committedSep 1, 2024··
Rewrite and simplify API
1 parent 74f97b5 commit 60f2121

8 files changed

+819
-1767
lines changed
 

‎Readme.md

+49-182
Original file line numberDiff line numberDiff line change
@@ -24,29 +24,9 @@ const { match, compile, parse } = require("path-to-regexp");
2424
// parse(path, options?)
2525
```
2626

27-
### Match
28-
29-
The `match` function returns a function for transforming paths into parameters:
30-
31-
- **path** A string.
32-
- **options** _(optional)_ (See [parse](#parse) for more options)
33-
- **sensitive** Regexp will be case sensitive. (default: `false`)
34-
- **end** Validate the match reaches the end of the string. (default: `true`)
35-
- **decode** Function for decoding strings to params, or `false` to disable all processing. (default: `decodeURIComponent`)
36-
37-
```js
38-
const fn = match("/foo/:bar");
39-
```
40-
41-
**Please note:** `path-to-regexp` is intended for ordered data (e.g. pathnames, hostnames). It can not handle arbitrarily ordered data (e.g. query strings, URL fragments, JSON, etc).
42-
4327
### Parameters
4428

45-
Parameters match arbitrary strings in a path by matching up to the end of the segment, or up to any proceeding tokens.
46-
47-
#### Named parameters
48-
49-
Named parameters are defined by prefixing a colon to the parameter name (`:foo`). Parameter names can use any valid unicode identifier characters, similar to JavaScript.
29+
Parameters match arbitrary strings in a path by matching up to the end of the segment, or up to any proceeding tokens. They are defined by prefixing a colon to the parameter name (`:foo`). Parameter names can use any valid JavaScript identifier, or be double quoted to use other characters (`:"param-name"`).
5030

5131
```js
5232
const fn = match("/:foo/:bar");
@@ -55,137 +35,54 @@ fn("/test/route");
5535
//=> { path: '/test/route', params: { foo: 'test', bar: 'route' } }
5636
```
5737

58-
##### Custom matching parameters
59-
60-
Parameters can have a custom regexp, which overrides the default match (`[^/]+`). For example, you can match digits or names in a path:
61-
62-
```js
63-
const exampleNumbers = match("/icon-:foo(\\d+).png");
64-
65-
exampleNumbers("/icon-123.png");
66-
//=> { path: '/icon-123.png', params: { foo: '123' } }
67-
68-
exampleNumbers("/icon-abc.png");
69-
//=> false
70-
71-
const exampleWord = pathToRegexp("/(user|u)");
72-
73-
exampleWord("/u");
74-
//=> { path: '/u', params: { '0': 'u' } }
75-
76-
exampleWord("/users");
77-
//=> false
78-
```
79-
80-
**Tip:** Backslashes need to be escaped with another backslash in JavaScript strings.
81-
82-
#### Unnamed parameters
83-
84-
It is possible to define a parameter without a name. The name will be numerically indexed:
85-
86-
```js
87-
const fn = match("/:foo/(.*)");
88-
89-
fn("/test/route");
90-
//=> { path: '/test/route', params: { '0': 'route', foo: 'test' } }
91-
```
92-
93-
#### Custom prefix and suffix
94-
95-
Parameters can be wrapped in `{}` to create custom prefixes or suffixes for your segment:
96-
97-
```js
98-
const fn = match("{/:attr1}?{-:attr2}?{-:attr3}?");
99-
100-
fn("/test");
101-
//=> { path: '/test', params: { attr1: 'test' } }
102-
103-
fn("/test-test");
104-
//=> { path: '/test-test', params: { attr1: 'test', attr2: 'test' } }
105-
```
106-
107-
#### Modifiers
108-
109-
Modifiers are used after parameters with custom prefixes and suffixes (`{}`).
110-
111-
##### Optional
112-
113-
Parameters can be suffixed with a question mark (`?`) to make the parameter optional.
114-
115-
```js
116-
const fn = match("/:foo{/:bar}?");
117-
118-
fn("/test");
119-
//=> { path: '/test', params: { foo: 'test' } }
120-
121-
fn("/test/route");
122-
//=> { path: '/test/route', params: { foo: 'test', bar: 'route' } }
123-
```
124-
125-
##### Zero or more
38+
### Wildcard
12639

127-
Parameters can be suffixed with an asterisk (`*`) to denote a zero or more parameter matches.
40+
Wildcard parameters match one or more characters across multiple segments. They are defined the same way as regular parameters, but are prefixed with an asterisk (`*foo`).
12841

12942
```js
130-
const fn = match("{/:foo}*");
131-
132-
fn("/foo");
133-
//=> { path: '/foo', params: { foo: [ 'foo' ] } }
43+
const fn = match("/*splat");
13444

13545
fn("/bar/baz");
136-
//=> { path: '/bar/baz', params: { foo: [ 'bar', 'baz' ] } }
46+
//=> { path: '/bar/baz', params: { splat: [ 'bar', 'baz' ] } }
13747
```
13848

139-
##### One or more
49+
### Optional
14050

141-
Parameters can be suffixed with a plus sign (`+`) to denote a one or more parameter matches.
51+
Braces can be used to define parts of the path that are optional.
14252

14353
```js
144-
const fn = match("{/:foo}+");
54+
const fn = match("/users{/:id}/delete");
14555

146-
fn("/");
147-
//=> false
56+
fn("/users/delete");
57+
//=> { path: '/users/delete', params: {} }
14858

149-
fn("/bar/baz");
150-
//=> { path: '/bar/baz', params: { foo: [ 'bar', 'baz' ] } }
59+
fn("/users/123/delete");
60+
//=> { path: '/users/123/delete', params: { id: '123' } }
15161
```
15262

153-
##### Custom separator
154-
155-
By default, parameters set the separator as the `prefix + suffix` of the token. Using `;` you can modify this:
156-
157-
```js
158-
const fn = match("/name{/:parts;-}+");
63+
## Match
15964

160-
fn("/name");
161-
//=> false
65+
The `match` function returns a function for matching strings against a path:
16266

163-
fn("/bar/1-2-3");
164-
//=> { path: '/name/1-2-3', params: { parts: [ '1', '2', '3' ] } }
165-
```
166-
167-
#### Wildcard
168-
169-
A wildcard is also supported. It is roughly equivalent to `(.*)`.
67+
- **path** String or array of strings.
68+
- **options** _(optional)_ (See [parse](#parse) for more options)
69+
- **sensitive** Regexp will be case sensitive. (default: `false`)
70+
- **end** Validate the match reaches the end of the string. (default: `true`)
71+
- **trailing** Allows optional trailing delimiter to match. (default: `true`)
72+
- **decode** Function for decoding strings to params, or `false` to disable all processing. (default: `decodeURIComponent`)
17073

17174
```js
172-
const fn = match("/*");
173-
174-
fn("/");
175-
//=> { path: '/', params: {} }
176-
177-
fn("/bar/baz");
178-
//=> { path: '/bar/baz', params: { '0': [ 'bar', 'baz' ] } }
75+
const fn = match("/foo/:bar");
17976
```
18077

181-
### Compile ("Reverse" Path-To-RegExp)
78+
**Please note:** `path-to-regexp` is intended for ordered data (e.g. pathnames, hostnames). It can not handle arbitrarily ordered data (e.g. query strings, URL fragments, JSON, etc).
79+
80+
## Compile ("Reverse" Path-To-RegExp)
18281

18382
The `compile` function will return a function for transforming parameters into a valid path:
18483

18584
- **path** A string.
18685
- **options** (See [parse](#parse) for more options)
187-
- **sensitive** Regexp will be case sensitive. (default: `false`)
188-
- **validate** When `false` the function can produce an invalid (unmatched) path. (default: `true`)
18986
- **encode** Function for encoding input strings for output into the path, or `false` to disable entirely. (default: `encodeURIComponent`)
19087

19188
```js
@@ -194,46 +91,34 @@ const toPath = compile("/user/:id");
19491
toPath({ id: "name" }); //=> "/user/name"
19592
toPath({ id: "café" }); //=> "/user/caf%C3%A9"
19693

197-
// When disabling `encode`, you need to make sure inputs are encoded correctly. No arrays are accepted.
198-
const toPathRaw = compile("/user/:id", { encode: false });
199-
200-
toPathRaw({ id: "%3A%2F" }); //=> "/user/%3A%2F"
201-
toPathRaw({ id: ":/" }); //=> Throws, "/user/:/" when `validate` is `false`.
202-
203-
const toPathRepeated = compile("{/:segment}+");
94+
const toPathRepeated = compile("/*segment");
20495

20596
toPathRepeated({ segment: ["foo"] }); //=> "/foo"
20697
toPathRepeated({ segment: ["a", "b", "c"] }); //=> "/a/b/c"
20798

208-
const toPathRegexp = compile("/user/:id(\\d+)");
99+
// When disabling `encode`, you need to make sure inputs are encoded correctly. No arrays are accepted.
100+
const toPathRaw = compile("/user/:id", { encode: false });
209101

210-
toPathRegexp({ id: "123" }); //=> "/user/123"
102+
toPathRaw({ id: "%3A%2F" }); //=> "/user/%3A%2F"
211103
```
212104

213105
## Developers
214106

215107
- If you are rewriting paths with match and compile, consider using `encode: false` and `decode: false` to keep raw paths passed around.
216-
- To ensure matches work on paths containing characters usually encoded, consider using [encodeurl](https://github.com/pillarjs/encodeurl) for `encodePath`.
108+
- To ensure matches work on paths containing characters usually encoded, such as emoji, consider using [encodeurl](https://github.com/pillarjs/encodeurl) for `encodePath`.
217109

218110
### Parse
219111

220-
The `parse` function accepts a string and returns `TokenData`, the set of tokens and other metadata parsed from the input string. `TokenData` is can used with `$match` and `$compile`.
112+
The `parse` function accepts a string and returns `TokenData`, the set of tokens and other metadata parsed from the input string. `TokenData` is can used with `match` and `compile`.
221113

222114
- **path** A string.
223115
- **options** _(optional)_
224116
- **delimiter** The default delimiter for segments, e.g. `[^/]` for `:named` parameters. (default: `'/'`)
225-
- **encodePath** A function for encoding input strings. (default: `x => x`, recommended: [`encodeurl`](https://github.com/pillarjs/encodeurl) for unicode encoding)
117+
- **encodePath** A function for encoding input strings. (default: `x => x`, recommended: [`encodeurl`](https://github.com/pillarjs/encodeurl))
226118

227119
### Tokens
228120

229-
The `tokens` returned by `TokenData` is an array of strings or keys, represented as objects, with the following properties:
230-
231-
- `name` The name of the token
232-
- `prefix` _(optional)_ The prefix string for the segment (e.g. `"/"`)
233-
- `suffix` _(optional)_ The suffix string for the segment (e.g. `""`)
234-
- `pattern` _(optional)_ The pattern defined to match this token
235-
- `modifier` _(optional)_ The modifier character used for the segment (e.g. `?`)
236-
- `separator` _(optional)_ The string used to separate repeated parameters
121+
`TokenData` is a sequence of tokens, currently of types `text`, `parameter`, `wildcard`, or `group`.
237122

238123
### Custom path
239124

@@ -242,9 +127,12 @@ In some applications, you may not be able to use the `path-to-regexp` syntax, bu
242127
```js
243128
import { TokenData, match } from "path-to-regexp";
244129

245-
const tokens = ["/", { name: "foo" }];
246-
const path = new TokenData(tokens, "/");
247-
const fn = $match(path);
130+
const tokens = [
131+
{ type: "text", value: "/" },
132+
{ type: "parameter", name: "foo" },
133+
];
134+
const path = new TokenData(tokens);
135+
const fn = match(path);
248136

249137
fn("/test"); //=> { path: '/test', index: 0, params: { foo: 'test' } }
250138
```
@@ -253,55 +141,34 @@ fn("/test"); //=> { path: '/test', index: 0, params: { foo: 'test' } }
253141

254142
An effort has been made to ensure ambiguous paths from previous releases throw an error. This means you might be seeing an error when things worked before.
255143

256-
### Unexpected `?`, `*`, or `+`
257-
258-
In previous major versions `/` and `.` were used as implicit prefixes of parameters. So `/:key?` was implicitly `{/:key}?`. For example:
259-
260-
- `/:key?``{/:key}?` or `/:key*``{/:key}*` or `/:key+``{/:key}+`
261-
- `.:key?``{.:key}?` or `.:key*``{.:key}*` or `.:key+``{.:key}+`
262-
- `:key?``{:key}?` or `:key*``{:key}*` or `:key+``{:key}+`
144+
### Unexpected `?` or `+`
263145

264-
### Unexpected `;`
146+
In past releases, `?`, `*`, and `+` were used to denote optional or repeating parameters. As an alternative, try these:
265147

266-
Used as a [custom separator](#custom-separator) for repeated parameters.
148+
- For optional (`?`), use an empty segment in a group such as `/:file{.:ext}`.
149+
- For repeating (`+`), only wildcard matching is supported, such as `/*path`.
150+
- For optional repeating (`*`), use a group and a wildcard parameter such as `/files{/*path}`.
267151

268-
### Unexpected `!`, `@`, or `,`
152+
### Unexpected `(`, `)`, `[`, `]`, etc.
269153

270-
These characters have been reserved for future use.
271-
272-
### Missing separator
273-
274-
Repeated parameters must have a separator to be valid. For example, `{:foo}*` can't be used. Separators can be defined manually, such as `{:foo;/}*`, or they default to the suffix and prefix with the parameter, such as `{/:foo}*`.
154+
Previous versions of Path-to-RegExp used these for RegExp features. This version no longer supports them so they've been reserved to avoid ambiguity. To use these characters literally, escape them with a backslash, e.g. `"\\("`.
275155

276156
### Missing parameter name
277157

278-
Parameter names, the part after `:`, must be a valid JavaScript identifier. For example, it cannot start with a number or dash. If you want a parameter name that uses these characters you can wrap the name in quotes, e.g. `:"my-name"`.
158+
Parameter names, the part after `:` or `*`, must be a valid JavaScript identifier. For example, it cannot start with a number or contain a dash. If you want a parameter name that uses these characters you can wrap the name in quotes, e.g. `:"my-name"`.
279159

280160
### Unterminated quote
281161

282162
Parameter names can be wrapped in double quote characters, and this error means you forgot to close the quote character.
283163

284-
### Pattern cannot start with "?"
285-
286-
Parameters in `path-to-regexp` must be basic groups. However, you can use features that require the `?` nested within the pattern. For example, `:foo((?!login)[^/]+)` is valid, but `:foo(?!login)` is not.
287-
288-
### Capturing groups are not allowed
289-
290-
A parameter pattern can not contain nested capturing groups.
291-
292-
### Unbalanced or missing pattern
293-
294-
A parameter pattern must have the expected number of parentheses. An unbalanced amount, such as `((?!login)` implies something has been written that is invalid. Check you didn't forget any parentheses.
295-
296164
### Express <= 4.x
297165

298166
Path-To-RegExp breaks compatibility with Express <= `4.x` in the following ways:
299167

300-
- The only part of the string that is a regex is within `()`.
301-
- In Express.js 4.x, everything was passed as-is after a simple replacement, so you could write `/[a-z]+` to match `/test`.
302-
- The `?` optional character must be used after `{}`.
168+
- Regexp characters can no longer be provided.
169+
- The optional character `?` is no longer supported, use braces instead: `/:file{.:ext}`.
303170
- Some characters have new meaning or have been reserved (`{}?*+@!;`).
304-
- The parameter name now supports all unicode identifier characters, previously it was only `[a-z0-9]`.
171+
- The parameter name now supports all JavaScript identifier characters, previously it was only `[a-z0-9]`.
305172

306173
## License
307174

‎package.json

+1
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
"dist/"
2121
],
2222
"scripts": {
23+
"bench": "vitest bench",
2324
"build": "ts-scripts build",
2425
"format": "ts-scripts format",
2526
"lint": "ts-scripts lint",

‎scripts/redos.ts

+1-7
Original file line numberDiff line numberDiff line change
@@ -5,13 +5,7 @@ import { MATCH_TESTS } from "../src/cases.spec.js";
55
let safe = 0;
66
let fail = 0;
77

8-
const TESTS = new Set(MATCH_TESTS.map((test) => test.path));
9-
// const TESTS = [
10-
// ":path([^\\.]+).:ext",
11-
// ":path.:ext(\\w+)",
12-
// ":path{.:ext([^\\.]+)}",
13-
// "/:path.:ext(\\\\w+)",
14-
// ];
8+
const TESTS = MATCH_TESTS.map((x) => x.path);
159

1610
for (const path of TESTS) {
1711
const { re } = match(path) as any;

‎src/cases.spec.ts

+329-1,036
Large diffs are not rendered by default.

‎src/index.bench.ts

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
import { bench } from "vitest";
2+
import { match } from "./index.js";
3+
4+
const PATHS: string[] = [
5+
"/xyz",
6+
"/user",
7+
"/user/123",
8+
"/" + "a".repeat(32_000),
9+
"/-" + "-a".repeat(8_000) + "/-",
10+
"/||||\x00|" + "||".repeat(27387) + "|\x00".repeat(27387) + "/||/",
11+
];
12+
13+
const STATIC_PATH_MATCH = match("/user");
14+
const SIMPLE_PATH_MATCH = match("/user/:id");
15+
const MULTI_SEGMENT_MATCH = match("/:x/:y");
16+
const MULTI_PATTERN_MATCH = match("/:x-:y");
17+
const TRICKY_PATTERN_MATCH = match("/:foo|:bar|");
18+
const ASTERISK_MATCH = match("/*foo");
19+
20+
bench("static path", () => {
21+
for (const path of PATHS) STATIC_PATH_MATCH(path);
22+
});
23+
24+
bench("simple path", () => {
25+
for (const path of PATHS) SIMPLE_PATH_MATCH(path);
26+
});
27+
28+
bench("multi segment", () => {
29+
for (const path of PATHS) MULTI_SEGMENT_MATCH(path);
30+
});
31+
32+
bench("multi pattern", () => {
33+
for (const path of PATHS) MULTI_PATTERN_MATCH(path);
34+
});
35+
36+
bench("tricky pattern", () => {
37+
for (const path of PATHS) TRICKY_PATTERN_MATCH(path);
38+
});
39+
40+
bench("asterisk", () => {
41+
for (const path of PATHS) ASTERISK_MATCH(path);
42+
});

‎src/index.spec.ts

+50-124
Original file line numberDiff line numberDiff line change
@@ -6,89 +6,86 @@ import { PARSER_TESTS, COMPILE_TESTS, MATCH_TESTS } from "./cases.spec.js";
66
* Dynamically generate the entire test suite.
77
*/
88
describe("path-to-regexp", () => {
9-
describe("arguments", () => {
10-
it("should throw on non-capturing pattern", () => {
11-
expect(() => match("/:foo(?:\\d+(\\.\\d+)?)")).toThrow(
9+
describe("parse errors", () => {
10+
it("should throw on unbalanced group", () => {
11+
expect(() => parse("/{:foo,")).toThrow(
1212
new TypeError(
13-
'Pattern cannot start with "?" at 6: https://git.new/pathToRegexpError',
13+
"Unexpected END at 7, expected }: https://git.new/pathToRegexpError",
1414
),
1515
);
1616
});
17-
18-
it("should throw on nested capturing group", () => {
19-
expect(() => match("/:foo(\\d+(\\.\\d+)?)")).toThrow(
17+
it("should throw on nested unbalanced group", () => {
18+
expect(() => parse("/{:foo/{x,y}")).toThrow(
2019
new TypeError(
21-
"Capturing groups are not allowed at 9: https://git.new/pathToRegexpError",
20+
"Unexpected END at 12, expected }: https://git.new/pathToRegexpError",
2221
),
2322
);
2423
});
2524

26-
it("should throw on unbalanced pattern", () => {
27-
expect(() => match("/:foo(abc")).toThrow(
25+
it("should throw on missing param name", () => {
26+
expect(() => parse("/:/")).toThrow(
2827
new TypeError(
29-
"Unbalanced pattern at 5: https://git.new/pathToRegexpError",
28+
"Missing parameter name at 2: https://git.new/pathToRegexpError",
3029
),
3130
);
3231
});
3332

34-
it("should throw on unmatched )", function () {
35-
expect(() => match("/:fooab)c")).toThrow(
36-
new TypeError("Unmatched ) at 7: https://git.new/pathToRegexpError"),
37-
);
38-
});
39-
40-
it("should throw on unmatched ) after other patterns", function () {
41-
expect(() => match("/:test(\\w+)/:foo(\\d+))")).toThrow(
42-
new TypeError("Unmatched ) at 21: https://git.new/pathToRegexpError"),
43-
);
44-
});
45-
46-
it("should throw on missing pattern", () => {
47-
expect(() => match("/:foo()")).toThrow(
33+
it("should throw on missing wildcard name", () => {
34+
expect(() => parse("/*/")).toThrow(
4835
new TypeError(
49-
"Missing pattern at 5: https://git.new/pathToRegexpError",
36+
"Missing parameter name at 2: https://git.new/pathToRegexpError",
5037
),
5138
);
5239
});
5340

54-
it("should throw on missing name", () => {
55-
expect(() => match("/:(test)")).toThrow(
41+
it("should throw on unterminated quote", () => {
42+
expect(() => parse('/:"foo')).toThrow(
5643
new TypeError(
57-
"Missing parameter name at 2: https://git.new/pathToRegexpError",
44+
"Unterminated quote at 2: https://git.new/pathToRegexpError",
5845
),
5946
);
6047
});
48+
});
6149

62-
it("should throw on nested groups", () => {
63-
expect(() => match("/{a{b:foo}}")).toThrow(
64-
new TypeError(
65-
"Unexpected { at 3, expected }: https://git.new/pathToRegexpError",
66-
),
67-
);
50+
describe("compile errors", () => {
51+
it("should throw when a param is missing", () => {
52+
const toPath = compile("/a/:b/c");
53+
54+
expect(() => {
55+
toPath();
56+
}).toThrow(new TypeError("Missing parameters: b"));
6857
});
6958

70-
it("should throw on repeat parameters without a separator", () => {
71-
expect(() => match("{:x}*")).toThrow(
72-
new TypeError(
73-
`Missing separator for "x": https://git.new/pathToRegexpError`,
74-
),
75-
);
59+
it("should throw when expecting a repeated value", () => {
60+
const toPath = compile("/*foo");
61+
62+
expect(() => {
63+
toPath({ foo: [] });
64+
}).toThrow(new TypeError('Expected "foo" to be a non-empty array'));
7665
});
7766

78-
it("should throw on unterminated quote", () => {
79-
expect(() => match('/:"foo')).toThrow(
80-
new TypeError(
81-
"Unterminated quote at 2: https://git.new/pathToRegexpError",
82-
),
83-
);
67+
it("should throw when param gets an array", () => {
68+
const toPath = compile("/:foo");
69+
70+
expect(() => {
71+
toPath({ foo: [] });
72+
}).toThrow(new TypeError('Expected "foo" to be a string'));
8473
});
8574

86-
it("should throw on invalid *", () => {
87-
expect(() => match("/:foo*")).toThrow(
88-
new TypeError(
89-
"Unexpected * at 5, you probably want `/*` or `{/:foo}*`: https://git.new/pathToRegexpError",
90-
),
91-
);
75+
it("should throw when a wildcard is not an array", () => {
76+
const toPath = compile("/*foo");
77+
78+
expect(() => {
79+
toPath({ foo: "a" });
80+
}).toThrow(new TypeError('Expected "foo" to be a non-empty array'));
81+
});
82+
83+
it("should throw when a wildcard array value is not a string", () => {
84+
const toPath = compile("/*foo");
85+
86+
expect(() => {
87+
toPath({ foo: [1, "a"] as any });
88+
}).toThrow(new TypeError('Expected "foo/0" to be a string'));
9289
});
9390
});
9491

@@ -126,75 +123,4 @@ describe("path-to-regexp", () => {
126123
});
127124
},
128125
);
129-
130-
describe("compile errors", () => {
131-
it("should throw when a required param is undefined", () => {
132-
const toPath = compile("/a/:b/c");
133-
134-
expect(() => {
135-
toPath();
136-
}).toThrow(new TypeError('Expected "b" to be a string'));
137-
});
138-
139-
it("should throw when it does not match the pattern", () => {
140-
const toPath = compile("/:foo(\\d+)");
141-
142-
expect(() => {
143-
toPath({ foo: "abc" });
144-
}).toThrow(new TypeError('Invalid value for "foo": "abc"'));
145-
});
146-
147-
it("should throw when expecting a repeated value", () => {
148-
const toPath = compile("{/:foo}+");
149-
150-
expect(() => {
151-
toPath({ foo: [] });
152-
}).toThrow(new TypeError('Invalid value for "foo": ""'));
153-
});
154-
155-
it("should throw when not expecting a repeated value", () => {
156-
const toPath = compile("/:foo");
157-
158-
expect(() => {
159-
toPath({ foo: [] });
160-
}).toThrow(new TypeError('Expected "foo" to be a string'));
161-
});
162-
163-
it("should throw when a repeated param is not an array", () => {
164-
const toPath = compile("{/:foo}+");
165-
166-
expect(() => {
167-
toPath({ foo: "a" });
168-
}).toThrow(new TypeError('Expected "foo" to be an array'));
169-
});
170-
171-
it("should throw when an array value is not a string", () => {
172-
const toPath = compile("{/:foo}+");
173-
174-
expect(() => {
175-
toPath({ foo: [1, "a"] as any });
176-
}).toThrow(new TypeError('Expected "foo/0" to be a string'));
177-
});
178-
179-
it("should throw when repeated value does not match", () => {
180-
const toPath = compile("{/:foo(\\d+)}+");
181-
182-
expect(() => {
183-
toPath({ foo: ["1", "2", "3", "a"] });
184-
}).toThrow(new TypeError('Invalid value for "foo": "/1/2/3/a"'));
185-
});
186-
});
187126
});
188-
189-
/**
190-
* Execute a regular expression and return a flat array for comparison.
191-
*
192-
* @param {RegExp} re
193-
* @param {String} str
194-
* @return {Array}
195-
*/
196-
function exec(re: RegExp, str: string) {
197-
const match = re.exec(str);
198-
199-
return match && Array.prototype.slice.call(match);
200-
}

‎src/index.ts

+346-417
Large diffs are not rendered by default.

‎tsconfig.build.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,5 @@
33
"compilerOptions": {
44
"types": []
55
},
6-
"exclude": ["src/**/*.spec.ts"]
6+
"exclude": ["src/**/*.spec.ts", "src/**/*.bench.ts"]
77
}

0 commit comments

Comments
 (0)
Please sign in to comment.