regexp_test.go 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960
  1. package goja
  2. import (
  3. "testing"
  4. )
  5. func TestRegexp1(t *testing.T) {
  6. const SCRIPT = `
  7. var r = new RegExp("(['\"])(.*?)\\1");
  8. var m = r.exec("'test'");
  9. m !== null && m.length == 3 && m[2] === "test";
  10. `
  11. testScript(SCRIPT, valueTrue, t)
  12. }
  13. func TestRegexp2(t *testing.T) {
  14. const SCRIPT = `
  15. var r = new RegExp("(['\"])(.*?)['\"]");
  16. var m = r.exec("'test'");
  17. m !== null && m.length == 3 && m[2] === "test";
  18. `
  19. testScript(SCRIPT, valueTrue, t)
  20. }
  21. func TestRegexpLiteral(t *testing.T) {
  22. const SCRIPT = `
  23. var r = /(['\"])(.*?)\1/;
  24. var m = r.exec("'test'");
  25. m !== null && m.length == 3 && m[2] === "test";
  26. `
  27. testScript(SCRIPT, valueTrue, t)
  28. }
  29. func TestRegexpRe2Unicode(t *testing.T) {
  30. const SCRIPT = `
  31. var r = /(тест)/i;
  32. var m = r.exec("'Тест'");
  33. m !== null && m.length == 2 && m[1] === "Тест";
  34. `
  35. testScript(SCRIPT, valueTrue, t)
  36. }
  37. func TestRegexpRe2UnicodeTarget(t *testing.T) {
  38. const SCRIPT = `
  39. var r = /(['\"])(.*?)['\"]/i;
  40. var m = r.exec("'Тест'");
  41. m !== null && m.length == 3 && m[2] === "Тест";
  42. `
  43. testScript(SCRIPT, valueTrue, t)
  44. }
  45. func TestRegexpRegexp2Unicode(t *testing.T) {
  46. const SCRIPT = `
  47. var r = /(['\"])(тест)\1/i;
  48. var m = r.exec("'Тест'");
  49. m !== null && m.length == 3 && m[2] === "Тест";
  50. `
  51. testScript(SCRIPT, valueTrue, t)
  52. }
  53. func TestRegexpRegexp2UnicodeTarget(t *testing.T) {
  54. const SCRIPT = `
  55. var r = /(['\"])(.*?)\1/;
  56. var m = r.exec("'Тест'");
  57. m !== null && m.length == 3 && m[2] === "Тест";
  58. `
  59. testScript(SCRIPT, valueTrue, t)
  60. }
  61. func TestRegexpRe2Whitespace(t *testing.T) {
  62. const SCRIPT = `
  63. "\u2000\u2001\u2002\u200b".replace(/\s+/g, "") === "\u200b";
  64. `
  65. testScript(SCRIPT, valueTrue, t)
  66. }
  67. func TestRegexpRegexp2Whitespace(t *testing.T) {
  68. const SCRIPT = `
  69. "A\u2000\u2001\u2002A\u200b".replace(/(A)\s+\1/g, "") === "\u200b"
  70. `
  71. testScript(SCRIPT, valueTrue, t)
  72. }
  73. func TestEmptyCharClassRe2(t *testing.T) {
  74. const SCRIPT = `
  75. /[]/.test("\u0000");
  76. `
  77. testScript(SCRIPT, valueFalse, t)
  78. }
  79. func TestNegatedEmptyCharClassRe2(t *testing.T) {
  80. const SCRIPT = `
  81. /[^]/.test("\u0000");
  82. `
  83. testScript(SCRIPT, valueTrue, t)
  84. }
  85. func TestEmptyCharClassRegexp2(t *testing.T) {
  86. const SCRIPT = `
  87. /([])\1/.test("\u0000\u0000");
  88. `
  89. testScript(SCRIPT, valueFalse, t)
  90. }
  91. func TestRegexp2Negate(t *testing.T) {
  92. const SCRIPT = `
  93. /([\D1])\1/.test("aa");
  94. `
  95. testScript(SCRIPT, valueTrue, t)
  96. }
  97. func TestAlternativeRe2(t *testing.T) {
  98. const SCRIPT = `
  99. /()|/.exec("") !== null;
  100. `
  101. testScript(SCRIPT, valueTrue, t)
  102. }
  103. func TestRegexpReplaceGlobal(t *testing.T) {
  104. const SCRIPT = `
  105. "QBZPbage\ny_cynprubyqre".replace(/^\s*|\s*$/g, '')
  106. `
  107. testScript(SCRIPT, asciiString("QBZPbage\ny_cynprubyqre"), t)
  108. }
  109. func TestRegexpNumCaptures(t *testing.T) {
  110. const SCRIPT = `
  111. "Fubpxjnir Synfu 9.0 e115".replace(/([a-zA-Z]|\s)+/, '')
  112. `
  113. testScript(SCRIPT, asciiString("9.0 e115"), t)
  114. }
  115. func TestRegexpNumCaptures1(t *testing.T) {
  116. const SCRIPT = `
  117. "Fubpxjnir Sy\tfu 9.0 e115".replace(/^.*\s+(\S+\s+\S+$)/, '')
  118. `
  119. testScript(SCRIPT, asciiString(""), t)
  120. }
  121. func TestRegexpSInClass(t *testing.T) {
  122. const SCRIPT = `
  123. /[\S]/.test("\u2028");
  124. `
  125. testScript(SCRIPT, valueFalse, t)
  126. }
  127. func TestRegexpDotMatchCR(t *testing.T) {
  128. const SCRIPT = `
  129. /./.test("\r");
  130. `
  131. testScript(SCRIPT, valueFalse, t)
  132. }
  133. func TestRegexpDotMatchCRInGroup(t *testing.T) {
  134. const SCRIPT = `
  135. /(.)/.test("\r");
  136. `
  137. testScript(SCRIPT, valueFalse, t)
  138. }
  139. func TestRegexpDotMatchLF(t *testing.T) {
  140. const SCRIPT = `
  141. /./.test("\n");
  142. `
  143. testScript(SCRIPT, valueFalse, t)
  144. }
  145. func TestRegexpSplitWithBackRef(t *testing.T) {
  146. const SCRIPT = `
  147. "a++b+-c".split(/([+-])\1/).join(" $$ ")
  148. `
  149. testScript(SCRIPT, asciiString("a $$ + $$ b+-c"), t)
  150. }
  151. func TestEscapeNonASCII(t *testing.T) {
  152. const SCRIPT = `
  153. /\⩓/.test("⩓")
  154. `
  155. testScript(SCRIPT, valueTrue, t)
  156. }
  157. func TestRegexpUTF16(t *testing.T) {
  158. const SCRIPT = `
  159. var str = "\uD800\uDC00";
  160. assert(/\uD800/g.test(str), "#1");
  161. assert(/\uD800/.test(str), "#2");
  162. assert(/𐀀/.test(str), "#3");
  163. var re = /\uD800/;
  164. assert(compareArray(str.replace(re, "X"), ["X", "\uDC00"]), "#4");
  165. assert(compareArray(str.split(re), ["", "\uDC00"]), "#5");
  166. assert(compareArray("a\uD800\uDC00b".split(/\uD800/g), ["a", "\uDC00b"]), "#6");
  167. assert(compareArray("a\uD800\uDC00b".split(/(?:)/g), ["a", "\uD800", "\uDC00", "b"]), "#7");
  168. assert(compareArray("0\x80".split(/(0){0}/g), ["0", undefined, "\x80"]), "#7+");
  169. re = /(?=)a/; // a hack to use regexp2
  170. assert.sameValue(re.exec('\ud83d\ude02a').index, 2, "#8");
  171. assert.sameValue(/./.exec('\ud83d\ude02')[0], '\ud83d', "#9");
  172. assert(RegExp("\uD800").test("\uD800"), "#10");
  173. var cu = 0xD800;
  174. var xx = "a\\" + String.fromCharCode(cu);
  175. var pattern = eval("/" + xx + "/");
  176. assert.sameValue(pattern.source, "a\\\\\\ud800", "Code unit: " + cu.toString(16), "#11");
  177. assert(pattern.test("a\\\uD800"), "#12");
  178. `
  179. testScriptWithTestLib(SCRIPT, _undefined, t)
  180. }
  181. func TestRegexpUnicode(t *testing.T) {
  182. const SCRIPT = `
  183. assert(!/\uD800/u.test("\uD800\uDC00"), "#1");
  184. assert(!/\uFFFD/u.test("\uD800\uDC00"), "#2");
  185. assert(/\uD800\uDC00/u.test("\uD800\uDC00"), "#3");
  186. assert(/\uD800/u.test("\uD800"), "#4");
  187. assert(compareArray("a\uD800\uDC00b".split(/\uD800/gu), ["a\uD800\uDC00b"]), "#5");
  188. assert(compareArray("a\uD800\uDC00b".split(/(?:)/gu), ["a", "𐀀", "b"]), "#6");
  189. assert(compareArray("0\x80".split(/(0){0}/gu), ["0", undefined, "\x80"]), "#7");
  190. var re = eval('/' + /\ud834\udf06/u.source + '/u');
  191. assert(re.test('\ud834\udf06'), "#9");
  192. /*re = RegExp("\\p{L}", "u");
  193. if (!re.test("A")) {
  194. throw new Error("Test 9 failed");
  195. }*/
  196. `
  197. testScriptWithTestLib(SCRIPT, _undefined, t)
  198. }
  199. func TestConvertRegexpToUnicode(t *testing.T) {
  200. if s := convertRegexpToUnicode(`test\uD800\u0C00passed`); s != `test\uD800\u0C00passed` {
  201. t.Fatal(s)
  202. }
  203. if s := convertRegexpToUnicode(`test\uD800\uDC00passed`); s != `test𐀀passed` {
  204. t.Fatal(s)
  205. }
  206. if s := convertRegexpToUnicode(`test\u0023passed`); s != `test\u0023passed` {
  207. t.Fatal(s)
  208. }
  209. if s := convertRegexpToUnicode(`test\u0passed`); s != `test\u0passed` {
  210. t.Fatal(s)
  211. }
  212. if s := convertRegexpToUnicode(`test\uD800passed`); s != `test\uD800passed` {
  213. t.Fatal(s)
  214. }
  215. if s := convertRegexpToUnicode(`test\uD800`); s != `test\uD800` {
  216. t.Fatal(s)
  217. }
  218. if s := convertRegexpToUnicode(`test\uD80`); s != `test\uD80` {
  219. t.Fatal(s)
  220. }
  221. if s := convertRegexpToUnicode(`\\uD800\uDC00passed`); s != `\\uD800\uDC00passed` {
  222. t.Fatal(s)
  223. }
  224. if s := convertRegexpToUnicode(`testpassed`); s != `testpassed` {
  225. t.Fatal(s)
  226. }
  227. }
  228. func TestConvertRegexpToUtf16(t *testing.T) {
  229. if s := convertRegexpToUtf16(`𐀀`); s != `\ud800\udc00` {
  230. t.Fatal(s)
  231. }
  232. if s := convertRegexpToUtf16(`\𐀀`); s != `\\\ud800\udc00` {
  233. t.Fatal(s)
  234. }
  235. }
  236. func TestEscapeInvalidUtf16(t *testing.T) {
  237. if s := escapeInvalidUtf16(asciiString("test")); s != "test" {
  238. t.Fatal(s)
  239. }
  240. if s := escapeInvalidUtf16(newStringValue("test\U00010000")); s != "test\U00010000" {
  241. t.Fatal(s)
  242. }
  243. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', 0xD800})); s != "t\\ud800" {
  244. t.Fatal(s)
  245. }
  246. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', 0xD800, 'p'})); s != "t\\ud800p" {
  247. t.Fatal(s)
  248. }
  249. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{0xD800, 'p'})); s != "\\ud800p" {
  250. t.Fatal(s)
  251. }
  252. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', '\\', 0xD800, 'p'})); s != `t\\\ud800p` {
  253. t.Fatal(s)
  254. }
  255. }
  256. func TestRegexpAssertion(t *testing.T) {
  257. const SCRIPT = `
  258. var res = 'aaa'.match(/^a/g);
  259. res.length === 1 || res[0] === 'a';
  260. `
  261. testScript(SCRIPT, valueTrue, t)
  262. }
  263. func TestRegexpUnicodeAdvanceStringIndex(t *testing.T) {
  264. const SCRIPT = `
  265. // deoptimise RegExp
  266. var origExec = RegExp.prototype.exec;
  267. RegExp.prototype.exec = function(s) {
  268. return origExec.call(this, s);
  269. };
  270. var re = /(?:)/gu;
  271. var str = "a\uD800\uDC00b";
  272. assert(compareArray(str.split(re), ["a", "𐀀", "b"]), "#1");
  273. re.lastIndex = 3;
  274. assert.sameValue(re.exec(str).index, 3, "#2");
  275. re.lastIndex = 2;
  276. assert.sameValue(re.exec(str).index, 1, "#3");
  277. re.lastIndex = 4;
  278. assert.sameValue(re.exec(str).index, 4, "#4");
  279. re.lastIndex = 5;
  280. assert.sameValue(re.exec(str), null, "#5");
  281. var iterator = str.matchAll(re); // regexp is copied by matchAll, but resets lastIndex
  282. var matches = [];
  283. for (var v of iterator) {matches.push(v);}
  284. assert.sameValue(matches.length, 4, "#6");
  285. assert.sameValue(matches[0].index, 0, "#7 index");
  286. assert.sameValue(matches[0][0], "", "#7 value");
  287. assert.sameValue(matches[1].index, 1, "#8 index");
  288. assert.sameValue(matches[1][0], "", "#8 value");
  289. assert.sameValue(matches[2].index, 3, "#9 index");
  290. assert.sameValue(matches[2][0], "", "#9 value");
  291. assert.sameValue(matches[3].index, 4, "#10 index");
  292. assert.sameValue(matches[3][0], "", "#10 value");
  293. `
  294. testScriptWithTestLib(SCRIPT, _undefined, t)
  295. }
  296. func TestRegexpInit(t *testing.T) {
  297. const SCRIPT = `
  298. RegExp(".").lastIndex;
  299. `
  300. testScript(SCRIPT, intToValue(0), t)
  301. }
  302. func TestRegexpToString(t *testing.T) {
  303. const SCRIPT = `
  304. RegExp.prototype.toString.call({
  305. source: 'foo',
  306. flags: 'bar'});
  307. `
  308. testScript(SCRIPT, asciiString("/foo/bar"), t)
  309. }
  310. func TestRegexpEscapeSource(t *testing.T) {
  311. const SCRIPT = `
  312. /href="(.+?)(\/.*\/\S+?)\/"/.source;
  313. `
  314. testScript(SCRIPT, asciiString(`href="(.+?)(\/.*\/\S+?)\/"`), t)
  315. }
  316. func TestRegexpConsecutiveMatchCache(t *testing.T) {
  317. const SCRIPT = `
  318. (function test(unicode) {
  319. var regex = new RegExp('t(e)(st(\\d?))', unicode?'gu':'g');
  320. var string = 'test1test2';
  321. var match;
  322. var matches = [];
  323. while (match = regex.exec(string)) {
  324. matches.push(match);
  325. }
  326. var expectedMatches = [
  327. [
  328. 'test1',
  329. 'e',
  330. 'st1',
  331. '1'
  332. ],
  333. [
  334. 'test2',
  335. 'e',
  336. 'st2',
  337. '2'
  338. ]
  339. ];
  340. expectedMatches[0].index = 0;
  341. expectedMatches[0].input = 'test1test2';
  342. expectedMatches[1].index = 5;
  343. expectedMatches[1].input = 'test1test2';
  344. assert(deepEqual(matches, expectedMatches), "#1");
  345. // try the same regexp with a different string
  346. regex.lastIndex = 0;
  347. match = regex.exec(' test5');
  348. var expectedMatch = [
  349. 'test5',
  350. 'e',
  351. 'st5',
  352. '5'
  353. ];
  354. expectedMatch.index = 1;
  355. expectedMatch.input = ' test5';
  356. assert(deepEqual(match, expectedMatch), "#2");
  357. assert.sameValue(regex.lastIndex, 6, "#3");
  358. // continue matching with a different string
  359. match = regex.exec(' test5test6');
  360. expectedMatch = [
  361. 'test6',
  362. 'e',
  363. 'st6',
  364. '6'
  365. ];
  366. expectedMatch.index = 6;
  367. expectedMatch.input = ' test5test6';
  368. assert(deepEqual(match, expectedMatch), "#4");
  369. assert.sameValue(regex.lastIndex, 11, "#5");
  370. match = regex.exec(' test5test6');
  371. assert.sameValue(match, null, "#6");
  372. return regex;
  373. });
  374. `
  375. vm := New()
  376. _, _ = vm.RunProgram(testLib())
  377. _, _ = vm.RunProgram(testLibX())
  378. v, err := vm.RunString(SCRIPT)
  379. if err != nil {
  380. t.Fatal(err)
  381. }
  382. var f func(bool) (*Object, error)
  383. err = vm.ExportTo(v, &f)
  384. if err != nil {
  385. t.Fatal(err)
  386. }
  387. regex, err := f(false)
  388. if err != nil {
  389. t.Fatal(err)
  390. }
  391. if regex.self.(*regexpObject).pattern.regexp2Wrapper.cache != nil {
  392. t.Fatal("Cache is not nil (non-unicode)")
  393. }
  394. regex, err = f(true)
  395. if err != nil {
  396. t.Fatal(err)
  397. }
  398. if regex.self.(*regexpObject).pattern.regexp2Wrapper.cache != nil {
  399. t.Fatal("Cache is not nil (unicode)")
  400. }
  401. }
  402. func TestRegexpMatchAll(t *testing.T) {
  403. const SCRIPT = `
  404. (function test(unicode) {
  405. var regex = new RegExp('t(e)(st(\\d?))', unicode?'gu':'g');
  406. var string = 'test1test2';
  407. var matches = [];
  408. for (var match of string.matchAll(regex)) {
  409. matches.push(match);
  410. }
  411. var expectedMatches = [
  412. [
  413. 'test1',
  414. 'e',
  415. 'st1',
  416. '1'
  417. ],
  418. [
  419. 'test2',
  420. 'e',
  421. 'st2',
  422. '2'
  423. ]
  424. ];
  425. expectedMatches[0].index = 0;
  426. expectedMatches[0].input = 'test1test2';
  427. expectedMatches[1].index = 5;
  428. expectedMatches[1].input = 'test1test2';
  429. assert(deepEqual(matches, expectedMatches), "#1");
  430. assert.sameValue(regex.lastIndex, 0, "#1 lastIndex");
  431. // try the same regexp with a different string
  432. string = ' test5';
  433. matches = [];
  434. for (var match of string.matchAll(regex)) {
  435. matches.push(match);
  436. }
  437. expectedMatches = [
  438. [
  439. 'test5',
  440. 'e',
  441. 'st5',
  442. '5'
  443. ]
  444. ];
  445. expectedMatches[0].index = 1;
  446. expectedMatches[0].input = ' test5';
  447. assert(deepEqual(matches, expectedMatches), "#2");
  448. assert.sameValue(regex.lastIndex, 0, "#2 lastIndex");
  449. // continue matching with a different string
  450. string = ' test5test6';
  451. matches = [];
  452. for (var match of string.matchAll(regex)) {
  453. matches.push(match);
  454. }
  455. var expectedMatches = [
  456. [
  457. 'test5',
  458. 'e',
  459. 'st5',
  460. '5'
  461. ],
  462. [
  463. 'test6',
  464. 'e',
  465. 'st6',
  466. '6'
  467. ]
  468. ];
  469. expectedMatches[0].index = 1;
  470. expectedMatches[0].input = ' test5test6';
  471. expectedMatches[1].index = 6;
  472. expectedMatches[1].input = ' test5test6';
  473. assert(deepEqual(matches, expectedMatches), "#3");
  474. assert.sameValue(regex.lastIndex, 0, "#3 lastindex");
  475. });
  476. `
  477. vm := New()
  478. _, _ = vm.RunProgram(testLib())
  479. _, _ = vm.RunProgram(testLibX())
  480. v, err := vm.RunString(SCRIPT)
  481. if err != nil {
  482. t.Fatal(err)
  483. }
  484. var f func(bool) (*Object, error)
  485. err = vm.ExportTo(v, &f)
  486. if err != nil {
  487. t.Fatal(err)
  488. }
  489. _, err = f(false)
  490. if err != nil {
  491. t.Fatal(err)
  492. }
  493. _, err = f(true)
  494. if err != nil {
  495. t.Fatal(err)
  496. }
  497. }
  498. func TestRegexpOverrideSpecies(t *testing.T) {
  499. const SCRIPT = `
  500. Object.defineProperty(RegExp, Symbol.species, {
  501. configurable: true,
  502. value: function() {
  503. throw "passed";
  504. }
  505. });
  506. try {
  507. "ab".split(/a/);
  508. throw new Error("Expected error");
  509. } catch(e) {
  510. if (e !== "passed") {
  511. throw e;
  512. }
  513. }
  514. `
  515. testScript(SCRIPT, _undefined, t)
  516. }
  517. func TestRegexpSymbolMatchAllCallsIsRegexp(t *testing.T) {
  518. // This is tc39's test/built-ins/RegExp/prototype/Symbol.matchAll/isregexp-this-throws.js
  519. const SCRIPT = `
  520. var a = new Object();
  521. Object.defineProperty(a, Symbol.match, {
  522. get: function() {
  523. throw "passed";
  524. }
  525. });
  526. try {
  527. RegExp.prototype[Symbol.matchAll].call(a, '');
  528. throw new Error("Expected error");
  529. } catch(e) {
  530. if (e !== "passed") {
  531. throw e;
  532. }
  533. }
  534. `
  535. testScript(SCRIPT, _undefined, t)
  536. }
  537. func TestRegexpMatchAllConstructor(t *testing.T) {
  538. // This is tc39's test/built-ins/RegExp/prototype/Symbol.matchAll/species-constuctor.js
  539. const SCRIPT = `
  540. var callCount = 0;
  541. var callArgs;
  542. var regexp = /\d/u;
  543. var obj = {}
  544. Object.defineProperty(obj, Symbol.species, {
  545. value: function() {
  546. callCount++;
  547. callArgs = arguments;
  548. return /\w/g;
  549. }
  550. });
  551. regexp.constructor = obj;
  552. var str = 'a*b';
  553. var iter = regexp[Symbol.matchAll](str);
  554. assert.sameValue(callCount, 1);
  555. assert.sameValue(callArgs.length, 2);
  556. assert.sameValue(callArgs[0], regexp);
  557. assert.sameValue(callArgs[1], 'u');
  558. var first = iter.next()
  559. assert.sameValue(first.done, false);
  560. assert.sameValue(first.value.length, 1);
  561. assert.sameValue(first.value[0], "a");
  562. var second = iter.next()
  563. assert.sameValue(second.done, true);
  564. `
  565. testScriptWithTestLib(SCRIPT, _undefined, t)
  566. }
  567. func TestRegexp2InvalidEscape(t *testing.T) {
  568. testScript(`/(?=)\x0/.test("x0")`, valueTrue, t)
  569. }
  570. func TestRegexpUnicodeEmptyMatch(t *testing.T) {
  571. testScript(`/(0)0|/gu.exec("0\xef").length === 2`, valueTrue, t)
  572. }
  573. func TestRegexpInvalidGroup(t *testing.T) {
  574. const SCRIPT = `
  575. ["?", "(?)"].forEach(function(s) {
  576. assert.throws(SyntaxError, function() {new RegExp(s)}, s);
  577. });
  578. `
  579. testScriptWithTestLib(SCRIPT, _undefined, t)
  580. }
  581. func TestRegexpLookbehindAssertion(t *testing.T) {
  582. const SCRIPT = `
  583. var re = /(?<=Jack|Tom)Sprat/;
  584. assert(re.test("JackSprat"), "#1");
  585. assert(!re.test("JohnSprat"), "#2");
  586. re = /(?<!-)\d+/;
  587. assert(re.test("3"), "#3");
  588. assert(!re.test("-3"), "#4");
  589. `
  590. testScriptWithTestLib(SCRIPT, _undefined, t)
  591. }
  592. func TestRegexpInvalidUTF8(t *testing.T) {
  593. vm := New()
  594. // Note that normally vm.ToValue() would replace invalid UTF-8 sequences with RuneError
  595. _, err := vm.New(vm.Get("RegExp"), asciiString([]byte{0xAD}))
  596. if err == nil {
  597. t.Fatal("Expected error")
  598. }
  599. }
  600. // this should not cause data races when run with -race
  601. func TestRegexpConcurrentLiterals(t *testing.T) {
  602. prg := MustCompile("test.js", `var r = /(?<!-)\d+/; r.test("");`, false)
  603. go func() {
  604. vm := New()
  605. _, err := vm.RunProgram(prg)
  606. if err != nil {
  607. panic(err)
  608. }
  609. }()
  610. vm := New()
  611. _, _ = vm.RunProgram(prg)
  612. }
  613. func TestRegexpDotAll(t *testing.T) {
  614. const SCRIPT = `
  615. var re = /./s;
  616. re.test("\r") && re.test("\n")
  617. `
  618. testScript(SCRIPT, valueTrue, t)
  619. }
  620. func TestRegexpDotAllInGroup(t *testing.T) {
  621. const SCRIPT = `
  622. var re = /(.)/s;
  623. re.test("\r") && re.test("\n")
  624. `
  625. testScript(SCRIPT, valueTrue, t)
  626. }
  627. func TestRegexpNumSeparators(t *testing.T) {
  628. const SCRIPT = `
  629. const re = /(?<=a)\u{65}_/u;
  630. assert(re.test("ae_") && !re.test("e_"));
  631. assert.throws(SyntaxError, () => {
  632. new RegExp("(?<=a)\\u{6_5}", "u");
  633. });
  634. assert.throws(SyntaxError, () => {
  635. new RegExp("a\\u{6_5}", "u");
  636. });
  637. `
  638. testScriptWithTestLib(SCRIPT, _undefined, t)
  639. }
  640. func TestRegexpUnicodeEscape(t *testing.T) {
  641. const SCRIPT = `
  642. assert.sameValue("u{0_2}".match(/\u{0_2}/)[0], "u{0_2}");
  643. assert.sameValue("uu\x02".match(/\u{2}/u)[0], '\x02');
  644. assert.sameValue("uu\x02".match(/\u{2}/)[0], "uu");
  645. `
  646. testScriptWithTestLib(SCRIPT, _undefined, t)
  647. }
  648. func BenchmarkRegexpSplitWithBackRef(b *testing.B) {
  649. const SCRIPT = `
  650. "aaaaaaaaaaaaaaaaaaaaaaaaa++bbbbbbbbbbbbbbbbbbbbbb+-ccccccccccccccccccccccc".split(/([+-])\1/)
  651. `
  652. b.StopTimer()
  653. prg, err := Compile("test.js", SCRIPT, false)
  654. if err != nil {
  655. b.Fatal(err)
  656. }
  657. vm := New()
  658. b.StartTimer()
  659. for i := 0; i < b.N; i++ {
  660. vm.RunProgram(prg)
  661. }
  662. }
  663. func BenchmarkRegexpMatch(b *testing.B) {
  664. const SCRIPT = `
  665. "a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  666. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  667. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  668. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  669. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  670. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  671. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  672. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  673. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  674. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  675. ".match(/[^\r\n]+/g)
  676. `
  677. b.StopTimer()
  678. prg, err := Compile("test.js", SCRIPT, false)
  679. if err != nil {
  680. b.Fatal(err)
  681. }
  682. vm := New()
  683. b.StartTimer()
  684. for i := 0; i < b.N; i++ {
  685. vm.RunProgram(prg)
  686. }
  687. }
  688. func BenchmarkRegexpMatchCache(b *testing.B) {
  689. const SCRIPT = `
  690. (function() {
  691. var s = "a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  692. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  693. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  694. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  695. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  696. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  697. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  698. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  699. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  700. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  701. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  702. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  703. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  704. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  705. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  706. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  707. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  708. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  709. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  710. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  711. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  712. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  713. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  714. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  715. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  716. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  717. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  718. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  719. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  720. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  721. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  722. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  723. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  724. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  725. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  726. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  727. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  728. "
  729. var r = /[^\r\n]+/g
  730. while(r.exec(s)) {};
  731. });
  732. `
  733. vm := New()
  734. v, err := vm.RunString(SCRIPT)
  735. if err != nil {
  736. b.Fatal(err)
  737. }
  738. if fn, ok := AssertFunction(v); ok {
  739. b.ResetTimer()
  740. b.ReportAllocs()
  741. for i := 0; i < b.N; i++ {
  742. fn(_undefined)
  743. }
  744. } else {
  745. b.Fatal("not a function")
  746. }
  747. }
  748. func BenchmarkRegexpMatchAll(b *testing.B) {
  749. const SCRIPT = `
  750. (function() {
  751. var s = "a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  752. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  753. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  754. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  755. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  756. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  757. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  758. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  759. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  760. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  761. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  762. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  763. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  764. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  765. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  766. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  767. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  768. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  769. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  770. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  771. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  772. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  773. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  774. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  775. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  776. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  777. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  778. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  779. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  780. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  781. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  782. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  783. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  784. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  785. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  786. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  787. a\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\ra\nb\r\c\nd\r\e\n\f\rg\nh\r\
  788. "
  789. var r = /[^\r\n]+/g
  790. for (var v of s.matchAll(r)) {}
  791. });
  792. `
  793. vm := New()
  794. v, err := vm.RunString(SCRIPT)
  795. if err != nil {
  796. b.Fatal(err)
  797. }
  798. if fn, ok := AssertFunction(v); ok {
  799. b.ResetTimer()
  800. b.ReportAllocs()
  801. for i := 0; i < b.N; i++ {
  802. fn(_undefined)
  803. }
  804. } else {
  805. b.Fatal("not a function")
  806. }
  807. }
  808. func BenchmarkRegexpSingleExec(b *testing.B) {
  809. vm := New()
  810. regexp := vm.Get("RegExp")
  811. f := func(reStr, str string, b *testing.B) {
  812. r, err := vm.New(regexp, vm.ToValue(reStr))
  813. if err != nil {
  814. b.Fatal(err)
  815. }
  816. exec, ok := AssertFunction(r.Get("exec"))
  817. if !ok {
  818. b.Fatal("RegExp.exec is not a function")
  819. }
  820. arg := vm.ToValue(str)
  821. b.ResetTimer()
  822. b.ReportAllocs()
  823. for i := 0; i < b.N; i++ {
  824. _, err := exec(r, arg)
  825. if err != nil {
  826. b.Fatal(err)
  827. }
  828. }
  829. }
  830. b.Run("Re-ASCII", func(b *testing.B) {
  831. f("test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing", b)
  832. })
  833. b.Run("Re2-ASCII", func(b *testing.B) {
  834. f("(?=)test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing", b)
  835. })
  836. b.Run("Re-Unicode", func(b *testing.B) {
  837. f("test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing 😀", b)
  838. })
  839. b.Run("Re2-Unicode", func(b *testing.B) {
  840. f("(?=)test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing 😀", b)
  841. })
  842. }