Skip to content

Commit 4c4bcd6

Browse files
committed
feat: base (non-named) capture
1 parent 576f566 commit 4c4bcd6

File tree

5 files changed

+65
-1
lines changed

5 files changed

+65
-1
lines changed

src/__tests__/capture.test.tsx

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import { capture } from '../capture';
2+
import { buildPattern } from '../compiler';
3+
import { oneOrMore } from '../quantifiers/base';
4+
import { execRegex } from '../test-utils';
5+
6+
test('"capture" base cases', () => {
7+
expect(buildPattern(capture('a'))).toBe('(a)');
8+
expect(buildPattern(capture('abc'))).toBe('(abc)');
9+
expect(buildPattern(capture(oneOrMore('abc')))).toBe('((?:abc)+)');
10+
11+
// To be improved in optimized compiler
12+
expect(buildPattern(oneOrMore(capture('abc')))).toBe('(?:(abc))+');
13+
});
14+
15+
test('"capture" captures group', () => {
16+
expect(execRegex('ab', [capture('b')])).toEqual(['b', 'b']);
17+
expect(execRegex('ab', ['a', capture('b')])).toEqual(['ab', 'b']);
18+
expect(execRegex('abc', ['a', capture('b'), capture('c')])).toEqual([
19+
'abc',
20+
'b',
21+
'c',
22+
]);
23+
});

src/capture.ts

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
import type { Capture, RegexElement } from './types';
2+
3+
export function capture(...children: RegexElement[]): Capture {
4+
return {
5+
type: 'capture',
6+
children,
7+
};
8+
}
9+
10+
export function compileCapture(compiledChildren: string): string {
11+
return `(${compiledChildren})`;
12+
}

src/compiler.ts

+6
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import { compileCharacterClass } from './character-classes/compiler';
44
import { baseQuantifiers, isBaseQuantifier } from './quantifiers/base';
55
import { compileRepeat } from './quantifiers/repeat';
66
import { escapeText } from './utils';
7+
import { compileCapture } from './capture';
78

89
/**
910
* Generate RegExp object for elements.
@@ -55,6 +56,11 @@ function compileSingle(element: RegexElement): string {
5556
return compiler(compiledChildren);
5657
}
5758

59+
if (element.type === 'capture') {
60+
const compiledChildren = compileList(element.children);
61+
return compileCapture(compiledChildren);
62+
}
63+
5864
// @ts-expect-error User passed incorrect type
5965
throw new Error(`Unknown elements type ${element.type}`);
6066
}

src/test-utils.ts

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
import { buildRegex } from './compiler';
2+
import type { RegexElement } from './types';
3+
4+
export function execRegex(text: string, elements: RegexElement[]) {
5+
const regex = buildRegex(...elements);
6+
return [...regex.exec(text)!];
7+
}
8+
9+
export function execRegexFull(text: string, elements: RegexElement[]) {
10+
const regex = buildRegex(...elements);
11+
return regex.exec(text)!;
12+
}

src/types.ts

+12-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,9 @@
1-
export type RegexElement = string | ChoiceOf | CharacterClass | Quantifier;
1+
export type RegexElement =
2+
| string
3+
| CharacterClass
4+
| ChoiceOf
5+
| Quantifier
6+
| Capture;
27

38
export type Quantifier = One | OneOrMore | Optionally | ZeroOrMore | Repeat;
49

@@ -45,3 +50,9 @@ export type ZeroOrMore = {
4550
type: 'zeroOrMore';
4651
children: RegexElement[];
4752
};
53+
54+
// Captures
55+
export type Capture = {
56+
type: 'capture';
57+
children: RegexElement[];
58+
};

0 commit comments

Comments
 (0)