jscript: Fixed String.match implementation for non-global regexps.
[wine] / dlls / jscript / tests / regexp.js
1 /*
2  * Copyright 2008 Jacek Caban for CodeWeavers
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  */
18
19
20 var m, re, b, i, obj;
21
22 re = /a+/;
23 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
24
25 m = re.exec(" aabaaa");
26 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
27 ok(m.index === 1, "m.index = " + m.index);
28 ok(m.input === " aabaaa", "m.input = " + m.input);
29 ok(m.length === 1, "m.length = " + m.length);
30 ok(m[0] === "aa", "m[0] = " + m[0]);
31
32 m = re.exec(" aabaaa");
33 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
34 ok(m.index === 1, "m.index = " + m.index);
35 ok(m.input === " aabaaa", "m.input = " + m.input);
36 ok(m.length === 1, "m.length = " + m.length);
37 ok(m[0] === "aa", "m[0] = " + m[0]);
38
39 re = /a+/g;
40 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
41
42 m = re.exec(" aabaaa");
43 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
44 ok(m.index === 1, "m.index = " + m.index);
45 ok(m.input === " aabaaa", "m.input = " + m.input);
46 ok(m.length === 1, "m.length = " + m.length);
47 ok(m[0] === "aa", "m[0] = " + m[0]);
48
49 m = re.exec(" aabaaa");
50 ok(re.lastIndex === 7, "re.lastIndex = " + re.lastIndex);
51 ok(m.index === 4, "m.index = " + m.index);
52 ok(m.input === " aabaaa", "m.input = " + m.input);
53 ok(m.length === 1, "m.length = " + m.length);
54 ok(m[0] === "aaa", "m[0] = " + m[0]);
55
56 m = re.exec(" aabaaa");
57 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
58 ok(m === null, "m is not null");
59
60 re.exec("               a");
61 ok(re.lastIndex === 16, "re.lastIndex = " + re.lastIndex);
62
63 m = re.exec(" a");
64 ok(m === null, "m is not null");
65 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
66
67 m = re.exec(" a");
68 ok(re.lastIndex === 2, "re.lastIndex = " + re.lastIndex);
69
70 m = re.exec();
71 ok(m === null, "m is not null");
72 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
73
74 m = /(a|b)+|(c)/.exec("aa");
75 ok(m[0] === "aa", "m[0] = " + m[0]);
76 ok(m[1] === "a", "m[1] = " + m[1]);
77 ok(m[2] === "", "m[2] = " + m[2]);
78
79 b = re.test("  a ");
80 ok(b === true, "re.test('  a ') returned " + b);
81 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
82
83 b = re.test(" a ");
84 ok(b === false, "re.test(' a ') returned " + b);
85 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
86
87 re = /\[([^\[]+)\]/g;
88 m = re.exec(" [test]  ");
89 ok(re.lastIndex === 7, "re.lastIndex = " + re.lastIndex);
90 ok(m.index === 1, "m.index = " + m.index);
91 ok(m.input === " [test]  ", "m.input = " + m.input);
92 ok(m.length === 2, "m.length = " + m.length);
93 ok(m[0] === "[test]", "m[0] = " + m[0]);
94 ok(m[1] === "test", "m[1] = " + m[1]);
95
96 b = /a*/.test();
97 ok(b === true, "/a*/.test() returned " + b);
98
99 m = "abcabc".match(re = /ca/);
100 ok(typeof(m) === "object", "typeof m is not object");
101 ok(m.length === 1, "m.length is not 1");
102 ok(m["0"] === "ca", "m[0] is not \"ca\"");
103 ok(m.constructor === Array, "unexpected m.constructor");
104 ok(re.lastIndex === 4, "re.lastIndex = " + re.lastIndex);
105
106 m = "abcabc".match(/ab/);
107 ok(typeof(m) === "object", "typeof m is not object");
108 ok(m.length === 1, "m.length is not 1");
109 ok(m["0"] === "ab", "m[0] is not \"ab\"");
110
111 m = "abcabc".match(/ab/g);
112 ok(typeof(m) === "object", "typeof m is not object");
113 ok(m.length === 2, "m.length is not 2");
114 ok(m["0"] === "ab", "m[0] is not \"ab\"");
115 ok(m["1"] === "ab", "m[1] is not \"ab\"");
116 /* ok(m.input === "abcabc", "m.input = " + m.input); */
117
118 m = "abcabc".match(/Ab/g);
119 ok(typeof(m) === "object", "typeof m is not object");
120 ok(m === null, "m is not null");
121
122 m = "abcabc".match(/Ab/gi);
123 ok(typeof(m) === "object", "typeof m is not object");
124 ok(m.length === 2, "m.length is not 2");
125 ok(m["0"] === "ab", "m[0] is not \"ab\"");
126 ok(m["1"] === "ab", "m[1] is not \"ab\"");
127
128 m = "aaabcabc".match(/a+b/g);
129 ok(typeof(m) === "object", "typeof m is not object");
130 ok(m.length === 2, "m.length is not 2");
131 ok(m["0"] === "aaab", "m[0] is not \"ab\"");
132 ok(m["1"] === "ab", "m[1] is not \"ab\"");
133
134 m = "aaa\\\\cabc".match(/\\/g);
135 ok(typeof(m) === "object", "typeof m is not object");
136 ok(m.length === 2, "m.length is not 2");
137 ok(m["0"] === "\\", "m[0] is not \"\\\"");
138 ok(m["1"] === "\\", "m[1] is not \"\\\"");
139
140 m = "abcabc".match(new RegExp("ab"));
141 ok(typeof(m) === "object", "typeof m is not object");
142 ok(m.length === 1, "m.length is not 1");
143 ok(m["0"] === "ab", "m[0] is not \"ab\"");
144
145 m = "abcabc".match(new RegExp("ab","g"));
146 ok(typeof(m) === "object", "typeof m is not object");
147 ok(m.length === 2, "m.length is not 2");
148 ok(m["0"] === "ab", "m[0] is not \"ab\"");
149 ok(m["1"] === "ab", "m[1] is not \"ab\"");
150
151 m = "abcabc".match(new RegExp(/ab/g));
152 ok(typeof(m) === "object", "typeof m is not object");
153 ok(m.length === 2, "m.length is not 2");
154 ok(m["0"] === "ab", "m[0] is not \"ab\"");
155 ok(m["1"] === "ab", "m[1] is not \"ab\"");
156
157 m = "abcabc".match(new RegExp("ab","g", "test"));
158 ok(typeof(m) === "object", "typeof m is not object");
159 ok(m.length === 2, "m.length is not 2");
160 ok(m["0"] === "ab", "m[0] is not \"ab\"");
161 ok(m["1"] === "ab", "m[1] is not \"ab\"");
162
163 m = "abcabcg".match("ab", "g");
164 ok(typeof(m) === "object", "typeof m is not object");
165 ok(m.length === 1, "m.length is not 1");
166 ok(m["0"] === "ab", "m[0] is not \"ab\"");
167
168 m = "abcabc".match();
169 ok(m === null, "m is not null");
170
171 m = "abcabc".match(/(a)(b)cabc/);
172 ok(typeof(m) === "object", "typeof m is not object");
173 ok(m.length === 3, "m.length is not 3");
174 ok(m[0] === "abcabc", "m[0] is not \"abc\"");
175 ok(m[1] === "a", "m[1] is not \"a\"");
176 ok(m[2] === "b", "m[2] is not \"b\"");
177
178 re = /(a)bcabc/;
179 re.lastIndex = -3;
180 m = "abcabc".match(re);
181 ok(typeof(m) === "object", "typeof m is not object");
182 ok(m.length === 2, "m.length = " + m.length + "expected 3");
183 ok(m[0] === "abcabc", "m[0] is not \"abc\"");
184 ok(m[1] === "a", "m[1] is not \"a\"");
185 ok(re.lastIndex === 6, "re.lastIndex = " + re.lastIndex);
186
187 re = /(a)bcabc/;
188 re.lastIndex = 2;
189 m = "abcabcxxx".match(re);
190 ok(typeof(m) === "object", "typeof m is not object");
191 ok(m.length === 2, "m.length = " + m.length + "expected 3");
192 ok(m[0] === "abcabc", "m[0] is not \"abc\"");
193 ok(m[1] === "a", "m[1] is not \"a\"");
194 ok(m.input === "abcabcxxx", "m.input = " + m.input);
195 ok(re.lastIndex === 6, "re.lastIndex = " + re.lastIndex);
196
197 r = "- [test] -".replace(re = /\[([^\[]+)\]/g, "success");
198 ok(r === "- success -", "r = " + r + " expected '- success -'");
199 ok(re.lastIndex === 8, "re.lastIndex = " + re.lastIndex);
200
201 r = "[test] [test]".replace(/\[([^\[]+)\]/g, "aa");
202 ok(r === "aa aa", "r = " + r + "aa aa");
203
204 r = "[test] [test]".replace(/\[([^\[]+)\]/, "aa");
205 ok(r === "aa [test]", "r = " + r + " expected 'aa [test]'");
206
207 r = "- [test] -".replace(/\[([^\[]+)\]/g);
208 ok(r === "- undefined -", "r = " + r + " expected '- undefined -'");
209
210 r = "- [test] -".replace(/\[([^\[]+)\]/g, true);
211 ok(r === "- true -", "r = " + r + " expected '- true -'");
212
213 r = "- [test] -".replace(/\[([^\[]+)\]/g, true, "test");
214 ok(r === "- true -", "r = " + r + " expected '- true -'");
215
216 var tmp = 0;
217
218 function replaceFunc1(m, off, str) {
219     ok(arguments.length === 3, "arguments.length = " + arguments.length);
220
221     switch(tmp) {
222     case 0:
223         ok(m === "[test1]", "m = " + m + " expected [test1]");
224         ok(off === 0, "off = " + off + " expected 0");
225         break;
226     case 1:
227         ok(m === "[test2]", "m = " + m + " expected [test2]");
228         ok(off === 8, "off = " + off + " expected 8");
229         break;
230     default:
231         ok(false, "unexpected call");
232     }
233
234     ok(str === "[test1] [test2]", "str = " + arguments[3]);
235     return "r" + tmp++;
236 }
237
238 r = "[test1] [test2]".replace(/\[[^\[]+\]/g, replaceFunc1);
239 ok(r === "r0 r1", "r = " + r + " expected 'r0 r1'");
240
241 tmp = 0;
242
243 function replaceFunc2(m, subm, off, str) {
244     ok(arguments.length === 4, "arguments.length = " + arguments.length);
245
246     switch(tmp) {
247     case 0:
248         ok(subm === "test1", "subm = " + subm);
249         ok(m === "[test1]", "m = " + m + " expected [test1]");
250         ok(off === 0, "off = " + off + " expected 0");
251         break;
252     case 1:
253         ok(subm === "test2", "subm = " + subm);
254         ok(m === "[test2]", "m = " + m + " expected [test2]");
255         ok(off === 8, "off = " + off + " expected 8");
256         break;
257     default:
258         ok(false, "unexpected call");
259     }
260
261     ok(str === "[test1] [test2]", "str = " + arguments[3]);
262     return "r" + tmp++;
263 }
264
265 r = "[test1] [test2]".replace(/\[([^\[]+)\]/g, replaceFunc2);
266 ok(r === "r0 r1", "r = '" + r + "' expected 'r0 r1'");
267
268 r = "$1,$2".replace(/(\$(\d))/g, "$$1-$1$2");
269 ok(r === "$1-$11,$1-$22", "r = '" + r + "' expected '$1-$11,$1-$22'");
270
271 r = "abc &1 123".replace(/(\&(\d))/g, "$&");
272 ok(r === "abc &1 123", "r = '" + r + "' expected 'abc &1 123'");
273
274 r = "abc &1 123".replace(/(\&(\d))/g, "$'");
275 ok(r === "abc  123 123", "r = '" + r + "' expected 'abc  123 123'");
276
277 r = "abc &1 123".replace(/(\&(\d))/g, "$`");
278 ok(r === "abc abc  123", "r = '" + r + "' expected 'abc abc  123'");
279
280 r = "abc &1 123".replace(/(\&(\d))/g, "$3");
281 ok(r === "abc $3 123", "r = '" + r + "' expected 'abc $3 123'");
282
283 r = "abc &1 123".replace(/(\&(\d))/g, "$");
284 ok(r === "abc $ 123", "r = '" + r + "' expected 'abc $ 123'");
285
286 r = "abc &1 123".replace(/(\&(\d))/g, "$a");
287 ok(r === "abc $a 123", "r = '" + r + "' expected 'abc $a 123'");
288
289 r = "abc &1 123".replace(/(\&(\d))/g, "$11");
290 ok(r === "abc &11 123", "r = '" + r + "' expected 'abc &11 123'");
291
292 r = "abc &1 123".replace(/(\&(\d))/g, "$0");
293 ok(r === "abc $0 123", "r = '" + r + "' expected 'abc $0 123'");
294
295 r = "1 2 3".replace("2", "$&");
296 ok(r === "1 $& 3", "r = '" + r + "' expected '1 $& 3'");
297
298 r = "1 2 3".replace("2", "$'");
299 ok(r === "1 $' 3", "r = '" + r + "' expected '1 $' 3'");
300
301 r = "1,,2,3".split(/,+/g);
302 ok(r.length === 3, "r.length = " + r.length);
303 ok(r[0] === "1", "r[0] = " + r[0]);
304 ok(r[1] === "2", "r[1] = " + r[1]);
305 ok(r[2] === "3", "r[2] = " + r[2]);
306
307 r = "1,,2,3".split(/,+/);
308 ok(r.length === 3, "r.length = " + r.length);
309 ok(r[0] === "1", "r[0] = " + r[0]);
310 ok(r[1] === "2", "r[1] = " + r[1]);
311 ok(r[2] === "3", "r[2] = " + r[2]);
312
313 r = "1,,2,".split(/,+/);
314 ok(r.length === 2, "r.length = " + r.length);
315 ok(r[0] === "1", "r[0] = " + r[0]);
316 ok(r[1] === "2", "r[1] = " + r[1]);
317
318 re = /,+/;
319 r = "1,,2,".split(re);
320 ok(r.length === 2, "r.length = " + r.length);
321 ok(r[0] === "1", "r[0] = " + r[0]);
322 ok(r[1] === "2", "r[1] = " + r[1]);
323 ok(re.lastIndex === 5, "re.lastIndex = " + re.lastIndex);
324
325 re = /,+/g;
326 r = "1,,2,".split(re);
327 ok(r.length === 2, "r.length = " + r.length);
328 ok(r[0] === "1", "r[0] = " + r[0]);
329 ok(r[1] === "2", "r[1] = " + r[1]);
330 ok(re.lastIndex === 5, "re.lastIndex = " + re.lastIndex);
331
332 r = "1 12 \t3".split(re = /\s+/).join(";");
333 ok(r === "1;12;3", "r = " + r);
334 ok(re.lastIndex === 6, "re.lastIndex = " + re.lastIndex);
335
336 r = "123".split(re = /\s+/).join(";");
337 ok(r === "123", "r = " + r);
338 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
339
340 /* another standard violation */
341 r = "1 12 \t3".split(re = /(\s)+/g).join(";");
342 ok(r === "1;12;3", "r = " + r);
343 ok(re.lastIndex === 6, "re.lastIndex = " + re.lastIndex);
344
345 re = /,+/;
346 re.lastIndex = 4;
347 r = "1,,2,".split(re);
348 ok(r.length === 2, "r.length = " + r.length);
349 ok(r[0] === "1", "r[0] = " + r[0]);
350 ok(r[1] === "2", "r[1] = " + r[1]);
351 ok(re.lastIndex === 5, "re.lastIndex = " + re.lastIndex);
352
353 re = /abc[^d]/g;
354 ok(re.source === "abc[^d]", "re.source = '" + re.source + "', expected 'abc[^d]'");
355
356 re = /a\bc[^d]/g;
357 ok(re.source === "a\\bc[^d]", "re.source = '" + re.source + "', expected 'a\\bc[^d]'");
358
359 re = /abc/;
360 ok(re === RegExp(re), "re !== RegExp(re)");
361
362 re = RegExp("abc[^d]", "g");
363 ok(re.source === "abc[^d]", "re.source = '" + re.source + "', expected 'abc[^d]'");
364
365 re = /abc/;
366 ok(re === RegExp(re, undefined), "re !== RegExp(re, undefined)");
367
368 re = /abc/;
369 ok(re === RegExp(re, undefined, 1), "re !== RegExp(re, undefined, 1)");
370
371 re = /a/g;
372 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 0");
373
374 m = re.exec(" a   ");
375 ok(re.lastIndex === 2, "re.lastIndex = " + re.lastIndex + " expected 2");
376 ok(m.index === 1, "m.index = " + m.index + " expected 1");
377
378 m = re.exec(" a   ");
379 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 0");
380 ok(m === null, "m = " + m + " expected null");
381
382 re.lastIndex = 2;
383 m = re.exec(" a a ");
384 ok(re.lastIndex === 4, "re.lastIndex = " + re.lastIndex + " expected 4");
385 ok(m.index === 3, "m.index = " + m.index + " expected 3");
386
387 re.lastIndex = "2";
388 ok(re.lastIndex === "2", "re.lastIndex = " + re.lastIndex + " expected '2'");
389 m = re.exec(" a a ");
390 ok(re.lastIndex === 4, "re.lastIndex = " + re.lastIndex + " expected 4");
391 ok(m.index === 3, "m.index = " + m.index + " expected 3");
392
393 var li = 0;
394 var obj = new Object();
395 obj.valueOf = function() { return li; };
396
397 re.lastIndex = obj;
398 ok(re.lastIndex === obj, "re.lastIndex = " + re.lastIndex + " expected obj");
399 li = 2;
400 m = re.exec(" a a ");
401 ok(re.lastIndex === 2, "re.lastIndex = " + re.lastIndex + " expected 2");
402 ok(m.index === 1, "m.index = " + m.index + " expected 1");
403
404 re.lastIndex = 3;
405 re.lastIndex = "test";
406 ok(re.lastIndex === "test", "re.lastIndex = " + re.lastIndex + " expected 'test'");
407 m = re.exec(" a a ");
408 ok(re.lastIndex === 2 || re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 2 or 0");
409 if(re.lastIndex != 0)
410     ok(m.index === 1, "m.index = " + m.index + " expected 1");
411 else
412     ok(m === null, "m = " + m + " expected null");
413
414 re.lastIndex = 0;
415 re.lastIndex = 3.9;
416 ok(re.lastIndex === 3.9, "re.lastIndex = " + re.lastIndex + " expected 3.9");
417 m = re.exec(" a a ");
418 ok(re.lastIndex === 4, "re.lastIndex = " + re.lastIndex + " expected 4");
419 ok(m.index === 3, "m.index = " + m.index + " expected 3");
420
421 obj.valueOf = function() { throw 0; }
422 re.lastIndex = obj;
423 ok(re.lastIndex === obj, "unexpected re.lastIndex");
424 m = re.exec(" a a ");
425 ok(re.lastIndex === 2, "re.lastIndex = " + re.lastIndex + " expected 2");
426 ok(m.index === 1, "m.index = " + m.index + " expected 1");
427
428 re.lastIndex = -3;
429 ok(re.lastIndex === -3, "re.lastIndex = " + re.lastIndex + " expected -3");
430 m = re.exec(" a a ");
431 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 0");
432 ok(m === null, "m = " + m + " expected null");
433
434 re.lastIndex = -1;
435 ok(re.lastIndex === -1, "re.lastIndex = " + re.lastIndex + " expected -1");
436 m = re.exec("  ");
437 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 0");
438 ok(m === null, "m = " + m + " expected null");
439
440 re = /a/;
441 re.lastIndex = -3;
442 ok(re.lastIndex === -3, "re.lastIndex = " + re.lastIndex + " expected -3");
443 m = re.exec(" a a ");
444 ok(re.lastIndex === 2, "re.lastIndex = " + re.lastIndex + " expected 0");
445 ok(m.index === 1, "m = " + m + " expected 1");
446
447 re.lastIndex = -1;
448 ok(re.lastIndex === -1, "re.lastIndex = " + re.lastIndex + " expected -1");
449 m = re.exec("  ");
450 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex + " expected 0");
451 ok(m === null, "m = " + m + " expected null");
452
453 re = /aa/g;
454 i = 'baacd'.search(re);
455 ok(i === 1, "'baacd'.search(re) = " + i);
456 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
457
458 re.lastIndex = 2;
459 i = 'baacdaa'.search(re);
460 ok(i === 1, "'baacd'.search(re) = " + i);
461 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
462
463 re = /aa/;
464 i = 'baacd'.search(re);
465 ok(i === 1, "'baacd'.search(re) = " + i);
466 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
467
468 re.lastIndex = 2;
469 i = 'baacdaa'.search(re);
470 ok(i === 1, "'baacd'.search(re) = " + i);
471 ok(re.lastIndex === 3, "re.lastIndex = " + re.lastIndex);
472
473 re = /d/g;
474 re.lastIndex = 1;
475 i = 'abc'.search(re);
476 ok(i === -1, "'abc'.search(/d/g) = " + i);
477 ok(re.lastIndex === 0, "re.lastIndex = " + re.lastIndex);
478
479 i = 'abcdde'.search(/[df]/);
480 ok(i === 3, "'abc'.search(/[df]/) = " + i);
481
482 i = 'abcdde'.search(/[df]/, "a");
483 ok(i === 3, "'abc'.search(/[df]/) = " + i);
484
485 i = 'abcdde'.search("[df]");
486 ok(i === 3, "'abc'.search(/d*/) = " + i);
487
488 obj = {
489     toString: function() { return "abc"; }
490 };
491 i = String.prototype.search.call(obj, "b");
492 ok(i === 1, "String.prototype.seatch.apply(obj, 'b') = " + i);
493
494 i = " undefined ".search();
495 ok(i === null, "' undefined '.search() = " + i);
496
497 reportSuccess();