1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
// RegExp.prototype[Symbol.search]: Test lastIndex changes for ES2017.
// RegExp-like class to test the RegExp method slow paths.
class DuckRegExp extends RegExp {
constructor(pattern, flags) {
return Object.create(DuckRegExp.prototype, {
regExp: {
value: new RegExp(pattern, flags)
},
lastIndex: {
value: 0, writable: true, enumerable: false, configurable: false
}
});
}
exec(...args) {
this.regExp.lastIndex = this.lastIndex;
try {
return this.regExp.exec(...args);
} finally {
if (this.global || this.sticky)
this.lastIndex = this.regExp.lastIndex;
}
}
get source() { return this.regExp.source; }
get global() { return this.regExp.global; }
get ignoreCase() { return this.regExp.ignoreCase; }
get multiline() { return this.regExp.multiline; }
get sticky() { return this.regExp.sticky; }
get unicode() { return this.regExp.unicode; }
}
// Test various combinations of:
// - Pattern matches or doesn't match
// - Global and/or sticky flag is set.
// - lastIndex exceeds the input string length
// - lastIndex is +-0
const testCasesNotPositiveZero = [
{ regExp: /a/, lastIndex: -1, input: "a" },
{ regExp: /a/g, lastIndex: -1, input: "a" },
{ regExp: /a/y, lastIndex: -1, input: "a" },
{ regExp: /a/, lastIndex: 100, input: "a" },
{ regExp: /a/g, lastIndex: 100, input: "a" },
{ regExp: /a/y, lastIndex: 100, input: "a" },
{ regExp: /a/, lastIndex: -1, input: "b" },
{ regExp: /a/g, lastIndex: -1, input: "b" },
{ regExp: /a/y, lastIndex: -1, input: "b" },
{ regExp: /a/, lastIndex: -0, input: "a" },
{ regExp: /a/g, lastIndex: -0, input: "a" },
{ regExp: /a/y, lastIndex: -0, input: "a" },
{ regExp: /a/, lastIndex: -0, input: "b" },
{ regExp: /a/g, lastIndex: -0, input: "b" },
{ regExp: /a/y, lastIndex: -0, input: "b" },
];
const testCasesPositiveZero = [
{ regExp: /a/, lastIndex: 0, input: "a" },
{ regExp: /a/g, lastIndex: 0, input: "a" },
{ regExp: /a/y, lastIndex: 0, input: "a" },
{ regExp: /a/, lastIndex: 0, input: "b" },
{ regExp: /a/g, lastIndex: 0, input: "b" },
{ regExp: /a/y, lastIndex: 0, input: "b" },
];
const testCases = [...testCasesNotPositiveZero, ...testCasesPositiveZero];
for (let Constructor of [RegExp, DuckRegExp]) {
// Basic test.
for (let {regExp, lastIndex, input} of testCases) {
let re = new Constructor(regExp);
re.lastIndex = lastIndex;
re[Symbol.search](input);
assertEq(re.lastIndex, lastIndex);
}
// Test when lastIndex is non-writable and not positive zero.
for (let {regExp, lastIndex, input} of testCasesNotPositiveZero) {
let re = new Constructor(regExp);
Object.defineProperty(re, "lastIndex", { value: lastIndex, writable: false });
assertThrowsInstanceOf(() => re[Symbol.search](input), TypeError);
assertEq(re.lastIndex, lastIndex);
}
// Test when lastIndex is non-writable and positive zero.
for (let {regExp, lastIndex, input} of testCasesPositiveZero) {
let re = new Constructor(regExp);
Object.defineProperty(re, "lastIndex", { value: lastIndex, writable: false });
if (re.global || re.sticky) {
assertThrowsInstanceOf(() => re[Symbol.search](input), TypeError);
} else {
re[Symbol.search](input);
}
assertEq(re.lastIndex, lastIndex);
}
// Test lastIndex isn't converted to a number.
for (let {regExp, lastIndex, input} of testCases) {
let re = new RegExp(regExp);
let badIndex = {
valueOf() {
assertEq(false, true);
}
};
re.lastIndex = badIndex;
re[Symbol.search](input);
assertEq(re.lastIndex, badIndex);
}
}
if (typeof reportCompare === "function")
reportCompare(true, true);
|