regexp_test.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  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. testScript1(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. testScript1(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. testScript1(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. testScript1(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. testScript1(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. testScript1(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. testScript1(SCRIPT, valueTrue, t)
  60. }
  61. func TestRegexpRe2Whitespace(t *testing.T) {
  62. const SCRIPT = `
  63. "\u2000\u2001\u2002\u200b".replace(/\s+/g, "") === "\u200b";
  64. `
  65. testScript1(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. testScript1(SCRIPT, valueTrue, t)
  72. }
  73. func TestEmptyCharClassRe2(t *testing.T) {
  74. const SCRIPT = `
  75. /[]/.test("\u0000");
  76. `
  77. testScript1(SCRIPT, valueFalse, t)
  78. }
  79. func TestNegatedEmptyCharClassRe2(t *testing.T) {
  80. const SCRIPT = `
  81. /[^]/.test("\u0000");
  82. `
  83. testScript1(SCRIPT, valueTrue, t)
  84. }
  85. func TestEmptyCharClassRegexp2(t *testing.T) {
  86. const SCRIPT = `
  87. /([])\1/.test("\u0000\u0000");
  88. `
  89. testScript1(SCRIPT, valueFalse, t)
  90. }
  91. func TestRegexp2Negate(t *testing.T) {
  92. const SCRIPT = `
  93. /([\D1])\1/.test("aa");
  94. `
  95. testScript1(SCRIPT, valueTrue, t)
  96. }
  97. func TestAlternativeRe2(t *testing.T) {
  98. const SCRIPT = `
  99. /()|/.exec("") !== null;
  100. `
  101. testScript1(SCRIPT, valueTrue, t)
  102. }
  103. func TestRegexpReplaceGlobal(t *testing.T) {
  104. const SCRIPT = `
  105. "QBZPbage\ny_cynprubyqre".replace(/^\s*|\s*$/g, '')
  106. `
  107. testScript1(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. testScript1(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. testScript1(SCRIPT, asciiString(""), t)
  120. }
  121. func TestRegexpSInClass(t *testing.T) {
  122. const SCRIPT = `
  123. /[\S]/.test("\u2028");
  124. `
  125. testScript1(SCRIPT, valueFalse, t)
  126. }
  127. func TestRegexpDotMatchSlashR(t *testing.T) {
  128. const SCRIPT = `
  129. /./.test("\r");
  130. `
  131. testScript1(SCRIPT, valueFalse, t)
  132. }
  133. func TestRegexpDotMatchSlashRInGroup(t *testing.T) {
  134. const SCRIPT = `
  135. /(.)/.test("\r");
  136. `
  137. testScript1(SCRIPT, valueFalse, t)
  138. }
  139. func TestRegexpSplitWithBackRef(t *testing.T) {
  140. const SCRIPT = `
  141. "a++b+-c".split(/([+-])\1/).join(" $$ ")
  142. `
  143. testScript1(SCRIPT, asciiString("a $$ + $$ b+-c"), t)
  144. }
  145. func TestEscapeNonASCII(t *testing.T) {
  146. const SCRIPT = `
  147. /\⩓/.test("⩓")
  148. `
  149. testScript1(SCRIPT, valueTrue, t)
  150. }
  151. func TestRegexpUTF16(t *testing.T) {
  152. const SCRIPT = `
  153. var str = "\uD800\uDC00";
  154. assert(/\uD800/g.test(str), "#1");
  155. assert(/\uD800/.test(str), "#2");
  156. assert(/𐀀/.test(str), "#3");
  157. var re = /\uD800/;
  158. assert(compareArray(str.replace(re, "X"), ["X", "\uDC00"]), "#4");
  159. assert(compareArray(str.split(re), ["", "\uDC00"]), "#5");
  160. assert(compareArray("a\uD800\uDC00b".split(/\uD800/g), ["a", "\uDC00b"]), "#6");
  161. assert(compareArray("a\uD800\uDC00b".split(/(?:)/g), ["a", "\uD800", "\uDC00", "b"]), "#7");
  162. assert(compareArray("0\x80".split(/(0){0}/g), ["0", undefined, "\x80"]), "#7+");
  163. re = /(?=)a/; // a hack to use regexp2
  164. assert.sameValue(re.exec('\ud83d\ude02a').index, 2, "#8");
  165. assert.sameValue(/./.exec('\ud83d\ude02')[0], '\ud83d', "#9");
  166. assert(RegExp("\uD800").test("\uD800"), "#10");
  167. var cu = 0xD800;
  168. var xx = "a\\" + String.fromCharCode(cu);
  169. var pattern = eval("/" + xx + "/");
  170. assert.sameValue(pattern.source, "a\\\\\\ud800", "Code unit: " + cu.toString(16), "#11");
  171. assert(pattern.test("a\\\uD800"), "#12");
  172. `
  173. testScript1(TESTLIB+SCRIPT, _undefined, t)
  174. }
  175. func TestRegexpUnicode(t *testing.T) {
  176. const SCRIPT = `
  177. assert(!/\uD800/u.test("\uD800\uDC00"), "#1");
  178. assert(!/\uFFFD/u.test("\uD800\uDC00"), "#2");
  179. assert(/\uD800\uDC00/u.test("\uD800\uDC00"), "#3");
  180. assert(/\uD800/u.test("\uD800"), "#4");
  181. assert(compareArray("a\uD800\uDC00b".split(/\uD800/gu), ["a\uD800\uDC00b"]), "#5");
  182. assert(compareArray("a\uD800\uDC00b".split(/(?:)/gu), ["a", "𐀀", "b"]), "#6");
  183. assert(compareArray("0\x80".split(/(0){0}/gu), ["0", undefined, "\x80"]), "#7");
  184. var re = eval('/' + /\ud834\udf06/u.source + '/u');
  185. assert(re.test('\ud834\udf06'), "#9");
  186. /*re = RegExp("\\p{L}", "u");
  187. if (!re.test("A")) {
  188. throw new Error("Test 9 failed");
  189. }*/
  190. `
  191. testScript1(TESTLIB+SCRIPT, _undefined, t)
  192. }
  193. func TestConvertRegexpToUnicode(t *testing.T) {
  194. if s := convertRegexpToUnicode(`test\uD800\u0C00passed`); s != `test\uD800\u0C00passed` {
  195. t.Fatal(s)
  196. }
  197. if s := convertRegexpToUnicode(`test\uD800\uDC00passed`); s != `test𐀀passed` {
  198. t.Fatal(s)
  199. }
  200. if s := convertRegexpToUnicode(`test\u0023passed`); s != `test\u0023passed` {
  201. t.Fatal(s)
  202. }
  203. if s := convertRegexpToUnicode(`test\u0passed`); s != `test\u0passed` {
  204. t.Fatal(s)
  205. }
  206. if s := convertRegexpToUnicode(`test\uD800passed`); s != `test\uD800passed` {
  207. t.Fatal(s)
  208. }
  209. if s := convertRegexpToUnicode(`test\uD800`); s != `test\uD800` {
  210. t.Fatal(s)
  211. }
  212. if s := convertRegexpToUnicode(`test\uD80`); s != `test\uD80` {
  213. t.Fatal(s)
  214. }
  215. if s := convertRegexpToUnicode(`\\uD800\uDC00passed`); s != `\\uD800\uDC00passed` {
  216. t.Fatal(s)
  217. }
  218. if s := convertRegexpToUnicode(`testpassed`); s != `testpassed` {
  219. t.Fatal(s)
  220. }
  221. }
  222. func TestConvertRegexpToUtf16(t *testing.T) {
  223. if s := convertRegexpToUtf16(`𐀀`); s != `\ud800\udc00` {
  224. t.Fatal(s)
  225. }
  226. if s := convertRegexpToUtf16(`\𐀀`); s != `\\\ud800\udc00` {
  227. t.Fatal(s)
  228. }
  229. }
  230. func TestEscapeInvalidUtf16(t *testing.T) {
  231. if s := escapeInvalidUtf16(asciiString("test")); s != "test" {
  232. t.Fatal(s)
  233. }
  234. if s := escapeInvalidUtf16(newStringValue("test\U00010000")); s != "test\U00010000" {
  235. t.Fatal(s)
  236. }
  237. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', 0xD800})); s != "t\\ud800" {
  238. t.Fatal(s)
  239. }
  240. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', 0xD800, 'p'})); s != "t\\ud800p" {
  241. t.Fatal(s)
  242. }
  243. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{0xD800, 'p'})); s != "\\ud800p" {
  244. t.Fatal(s)
  245. }
  246. if s := escapeInvalidUtf16(unicodeStringFromRunes([]rune{'t', '\\', 0xD800, 'p'})); s != `t\\\ud800p` {
  247. t.Fatal(s)
  248. }
  249. }
  250. func TestRegexpAssertion(t *testing.T) {
  251. const SCRIPT = `
  252. var res = 'aaa'.match(/^a/g);
  253. res.length === 1 || res[0] === 'a';
  254. `
  255. testScript1(SCRIPT, valueTrue, t)
  256. }
  257. func TestRegexpUnicodeAdvanceStringIndex(t *testing.T) {
  258. const SCRIPT = `
  259. // deoptimise RegExp
  260. var origExec = RegExp.prototype.exec;
  261. RegExp.prototype.exec = function(s) {
  262. return origExec.call(this, s);
  263. };
  264. var re = /(?:)/gu;
  265. var str = "a\uD800\uDC00b";
  266. assert(compareArray(str.split(re), ["a", "𐀀", "b"]), "#1");
  267. re.lastIndex = 3;
  268. assert.sameValue(re.exec(str).index, 3, "#2");
  269. re.lastIndex = 2;
  270. assert.sameValue(re.exec(str).index, 1, "#3");
  271. re.lastIndex = 4;
  272. assert.sameValue(re.exec(str).index, 4, "#4");
  273. re.lastIndex = 5;
  274. assert.sameValue(re.exec(str), null, "#5");
  275. `
  276. testScript1(TESTLIB+SCRIPT, _undefined, t)
  277. }
  278. func TestRegexpInit(t *testing.T) {
  279. const SCRIPT = `
  280. RegExp(".").lastIndex;
  281. `
  282. testScript1(SCRIPT, intToValue(0), t)
  283. }
  284. func TestRegexpToString(t *testing.T) {
  285. const SCRIPT = `
  286. RegExp.prototype.toString.call({
  287. source: 'foo',
  288. flags: 'bar'});
  289. `
  290. testScript1(SCRIPT, asciiString("/foo/bar"), t)
  291. }
  292. func TestRegexpEscapeSource(t *testing.T) {
  293. const SCRIPT = `
  294. /href="(.+?)(\/.*\/\S+?)\/"/.source;
  295. `
  296. testScript1(SCRIPT, asciiString(`href="(.+?)(\/.*\/\S+?)\/"`), t)
  297. }
  298. func TestRegexpConsecutiveMatchCache(t *testing.T) {
  299. const SCRIPT = `
  300. (function test(unicode) {
  301. var regex = new RegExp('t(e)(st(\\d?))', unicode?'gu':'g');
  302. var string = 'test1test2';
  303. var match;
  304. var matches = [];
  305. while (match = regex.exec(string)) {
  306. matches.push(match);
  307. }
  308. var expectedMatches = [
  309. [
  310. 'test1',
  311. 'e',
  312. 'st1',
  313. '1'
  314. ],
  315. [
  316. 'test2',
  317. 'e',
  318. 'st2',
  319. '2'
  320. ]
  321. ];
  322. expectedMatches[0].index = 0;
  323. expectedMatches[0].input = 'test1test2';
  324. expectedMatches[1].index = 5;
  325. expectedMatches[1].input = 'test1test2';
  326. assert(deepEqual(matches, expectedMatches), "#1");
  327. // try the same regexp with a different string
  328. regex.lastIndex = 0;
  329. match = regex.exec(' test5');
  330. var expectedMatch = [
  331. 'test5',
  332. 'e',
  333. 'st5',
  334. '5'
  335. ];
  336. expectedMatch.index = 1;
  337. expectedMatch.input = ' test5';
  338. assert(deepEqual(match, expectedMatch), "#2");
  339. assert.sameValue(regex.lastIndex, 6, "#3");
  340. // continue matching with a different string
  341. match = regex.exec(' test5test6');
  342. expectedMatch = [
  343. 'test6',
  344. 'e',
  345. 'st6',
  346. '6'
  347. ];
  348. expectedMatch.index = 6;
  349. expectedMatch.input = ' test5test6';
  350. assert(deepEqual(match, expectedMatch), "#4");
  351. assert.sameValue(regex.lastIndex, 11, "#5");
  352. match = regex.exec(' test5test6');
  353. assert.sameValue(match, null, "#6");
  354. return regex;
  355. });
  356. `
  357. vm := New()
  358. v, err := vm.RunString(TESTLIBX + SCRIPT)
  359. if err != nil {
  360. t.Fatal(err)
  361. }
  362. var f func(bool) (*Object, error)
  363. err = vm.ExportTo(v, &f)
  364. if err != nil {
  365. t.Fatal(err)
  366. }
  367. regex, err := f(false)
  368. if err != nil {
  369. t.Fatal(err)
  370. }
  371. if regex.self.(*regexpObject).pattern.regexp2Wrapper.cache != nil {
  372. t.Fatal("Cache is not nil (non-unicode)")
  373. }
  374. regex, err = f(true)
  375. if err != nil {
  376. t.Fatal(err)
  377. }
  378. if regex.self.(*regexpObject).pattern.regexp2Wrapper.cache != nil {
  379. t.Fatal("Cache is not nil (unicode)")
  380. }
  381. }
  382. func TestRegexpOverrideSpecies(t *testing.T) {
  383. const SCRIPT = `
  384. Object.defineProperty(RegExp, Symbol.species, {
  385. configurable: true,
  386. value: function() {
  387. throw "passed";
  388. }
  389. });
  390. try {
  391. "ab".split(/a/);
  392. throw new Error("Expected error");
  393. } catch(e) {
  394. if (e !== "passed") {
  395. throw e;
  396. }
  397. }
  398. `
  399. testScript1(SCRIPT, _undefined, t)
  400. }
  401. func TestRegexp2InvalidEscape(t *testing.T) {
  402. testScript1(`/(?=)\x0/.test("x0")`, valueTrue, t)
  403. }
  404. func TestRegexpUnicodeEmptyMatch(t *testing.T) {
  405. testScript1(`/(0)0|/gu.exec("0\xef").length === 2`, valueTrue, t)
  406. }
  407. func BenchmarkRegexpSplitWithBackRef(b *testing.B) {
  408. const SCRIPT = `
  409. "aaaaaaaaaaaaaaaaaaaaaaaaa++bbbbbbbbbbbbbbbbbbbbbb+-ccccccccccccccccccccccc".split(/([+-])\1/)
  410. `
  411. b.StopTimer()
  412. prg, err := Compile("test.js", SCRIPT, false)
  413. if err != nil {
  414. b.Fatal(err)
  415. }
  416. vm := New()
  417. b.StartTimer()
  418. for i := 0; i < b.N; i++ {
  419. vm.RunProgram(prg)
  420. }
  421. }
  422. func BenchmarkRegexpMatch(b *testing.B) {
  423. const SCRIPT = `
  424. "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\
  425. 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\
  426. 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\
  427. 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\
  428. 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\
  429. 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\
  430. 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\
  431. 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\
  432. 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\
  433. 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\
  434. ".match(/[^\r\n]+/g)
  435. `
  436. b.StopTimer()
  437. prg, err := Compile("test.js", SCRIPT, false)
  438. if err != nil {
  439. b.Fatal(err)
  440. }
  441. vm := New()
  442. b.StartTimer()
  443. for i := 0; i < b.N; i++ {
  444. vm.RunProgram(prg)
  445. }
  446. }
  447. func BenchmarkRegexpMatchCache(b *testing.B) {
  448. const SCRIPT = `
  449. (function() {
  450. 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\
  451. 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\
  452. 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\
  453. 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\
  454. 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\
  455. 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\
  456. 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\
  457. 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\
  458. 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\
  459. 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\
  460. 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\
  461. 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\
  462. 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\
  463. 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\
  464. 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\
  465. 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\
  466. 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\
  467. 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\
  468. 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\
  469. 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\
  470. 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\
  471. 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\
  472. 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\
  473. 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\
  474. 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\
  475. 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\
  476. 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\
  477. 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\
  478. 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\
  479. 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\
  480. 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\
  481. 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\
  482. 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\
  483. 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\
  484. 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\
  485. 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\
  486. 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\
  487. "
  488. var r = /[^\r\n]+/g
  489. while(r.exec(s)) {};
  490. });
  491. `
  492. vm := New()
  493. v, err := vm.RunString(SCRIPT)
  494. if err != nil {
  495. b.Fatal(err)
  496. }
  497. if fn, ok := AssertFunction(v); ok {
  498. b.ResetTimer()
  499. b.ReportAllocs()
  500. for i := 0; i < b.N; i++ {
  501. fn(_undefined)
  502. }
  503. } else {
  504. b.Fatal("not a function")
  505. }
  506. }
  507. func BenchmarkRegexpSingleExec(b *testing.B) {
  508. vm := New()
  509. regexp := vm.Get("RegExp")
  510. f := func(reStr, str string, b *testing.B) {
  511. r, err := vm.New(regexp, vm.ToValue(reStr))
  512. if err != nil {
  513. b.Fatal(err)
  514. }
  515. exec, ok := AssertFunction(r.Get("exec"))
  516. if !ok {
  517. b.Fatal("RegExp.exec is not a function")
  518. }
  519. arg := vm.ToValue(str)
  520. b.ResetTimer()
  521. b.ReportAllocs()
  522. for i := 0; i < b.N; i++ {
  523. _, err := exec(r, arg)
  524. if err != nil {
  525. b.Fatal(err)
  526. }
  527. }
  528. }
  529. b.Run("Re-ASCII", func(b *testing.B) {
  530. f("test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing", b)
  531. })
  532. b.Run("Re2-ASCII", func(b *testing.B) {
  533. f("(?=)test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing", b)
  534. })
  535. b.Run("Re-Unicode", func(b *testing.B) {
  536. f("test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing 😀", b)
  537. })
  538. b.Run("Re2-Unicode", func(b *testing.B) {
  539. f("(?=)test", "aaaaaaaaaaaaaaaaaaaaaaaaa testing 😀", b)
  540. })
  541. }