1 // Copyright 2014 the V8 project authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 // Flags: --harmony-concat-spreadable --harmony-proxies --harmony-reflect 6 7 (function testArrayConcatArity() { 8 "use strict"; 9 assertEquals(1, Array.prototype.concat.length); 10 })(); 11 12 13 (function testArrayConcatNoPrototype() { 14 "use strict"; 15 assertEquals(void 0, Array.prototype.concat.prototype); 16 })(); 17 18 19 (function testArrayConcatDescriptor() { 20 "use strict"; 21 var desc = Object.getOwnPropertyDescriptor(Array.prototype, 'concat'); 22 assertEquals(false, desc.enumerable); 23 })(); 24 25 26 (function testConcatArrayLike() { 27 "use strict"; 28 var obj = { 29 "length": 6, 30 "1": "A", 31 "3": "B", 32 "5": "C" 33 }; 34 obj[Symbol.isConcatSpreadable] = true; 35 var obj2 = { length: 3, "0": "0", "1": "1", "2": "2" }; 36 var arr = ["X", "Y", "Z"]; 37 assertEquals([void 0, "A", void 0, "B", void 0, "C", 38 { "length": 3, "0": "0", "1": "1", "2": "2" }, 39 "X", "Y", "Z"], Array.prototype.concat.call(obj, obj2, arr)); 40 })(); 41 42 43 (function testConcatArrayLikeStringLength() { 44 "use strict"; 45 var obj = { 46 "length": "6", 47 "1": "A", 48 "3": "B", 49 "5": "C" 50 }; 51 obj[Symbol.isConcatSpreadable] = true; 52 var obj2 = { length: 3, "0": "0", "1": "1", "2": "2" }; 53 var arr = ["X", "Y", "Z"]; 54 assertEquals([void 0, "A", void 0, "B", void 0, "C", 55 { "length": 3, "0": "0", "1": "1", "2": "2" }, 56 "X", "Y", "Z"], Array.prototype.concat.call(obj, obj2, arr)); 57 })(); 58 59 60 (function testConcatArrayLikeNegativeLength() { 61 "use strict"; 62 var obj = { 63 "length": -6, 64 "1": "A", 65 "3": "B", 66 "5": "C" 67 }; 68 obj[Symbol.isConcatSpreadable] = true; 69 assertEquals([], [].concat(obj)); 70 obj.length = -6.7; 71 assertEquals([], [].concat(obj)); 72 obj.length = "-6"; 73 assertEquals([], [].concat(obj)); 74 })(); 75 76 77 (function testConcatArrayLikeToLengthThrows() { 78 "use strict"; 79 var obj = { 80 "length": {valueOf: null, toString: null}, 81 "1": "A", 82 "3": "B", 83 "5": "C" 84 }; 85 obj[Symbol.isConcatSpreadable] = true; 86 var obj2 = { length: 3, "0": "0", "1": "1", "2": "2" }; 87 var arr = ["X", "Y", "Z"]; 88 assertThrows(function() { 89 Array.prototype.concat.call(obj, obj2, arr); 90 }, TypeError); 91 })(); 92 93 94 (function testConcatArrayLikePrimitiveNonNumberLength() { 95 "use strict"; 96 var obj = { 97 "1": "A", 98 "3": "B", 99 "5": "C" 100 }; 101 obj[Symbol.isConcatSpreadable] = true; 102 obj.length = {toString: function() { return "SIX"; }, valueOf: null }; 103 assertEquals([], [].concat(obj)); 104 obj.length = {toString: null, valueOf: function() { return "SIX"; } }; 105 assertEquals([], [].concat(obj)); 106 })(); 107 108 109 (function testConcatArrayLikeLengthToStringThrows() { 110 "use strict"; 111 function MyError() {} 112 var obj = { 113 "length": { toString: function() { 114 throw new MyError(); 115 }, valueOf: null 116 }, 117 "1": "A", 118 "3": "B", 119 "5": "C" 120 }; 121 obj[Symbol.isConcatSpreadable] = true; 122 assertThrows(function() { 123 [].concat(obj); 124 }, MyError); 125 })(); 126 127 128 (function testConcatArrayLikeLengthValueOfThrows() { 129 "use strict"; 130 function MyError() {} 131 var obj = { 132 "length": { valueOf: function() { 133 throw new MyError(); 134 }, toString: null 135 }, 136 "1": "A", 137 "3": "B", 138 "5": "C" 139 }; 140 obj[Symbol.isConcatSpreadable] = true; 141 assertThrows(function() { 142 [].concat(obj); 143 }, MyError); 144 })(); 145 146 147 (function testConcatHoleyArray() { 148 "use strict"; 149 var arr = []; 150 arr[4] = "Item 4"; 151 arr[8] = "Item 8"; 152 var arr2 = [".", "!", "?"]; 153 assertEquals([void 0, void 0, void 0, void 0, "Item 4", void 0, void 0, 154 void 0, "Item 8", ".", "!", "?"], arr.concat(arr2)); 155 })(); 156 157 158 (function testIsConcatSpreadableGetterThrows() { 159 "use strict"; 160 function MyError() {} 161 var obj = {}; 162 Object.defineProperty(obj, Symbol.isConcatSpreadable, { 163 get: function() { throw new MyError(); } 164 }); 165 166 assertThrows(function() { 167 [].concat(obj); 168 }, MyError); 169 170 assertThrows(function() { 171 Array.prototype.concat.call(obj, 1, 2, 3); 172 }, MyError); 173 })(); 174 175 176 (function testConcatLengthThrows() { 177 "use strict"; 178 function MyError() {} 179 var obj = {}; 180 obj[Symbol.isConcatSpreadable] = true; 181 Object.defineProperty(obj, "length", { 182 get: function() { throw new MyError(); } 183 }); 184 185 assertThrows(function() { 186 [].concat(obj); 187 }, MyError); 188 189 assertThrows(function() { 190 Array.prototype.concat.call(obj, 1, 2, 3); 191 }, MyError); 192 })(); 193 194 195 (function testConcatArraySubclass() { 196 "use strict"; 197 // If @@isConcatSpreadable is not used, the value of IsArray(O) 198 // is used to determine the spreadable property. 199 class A extends Array {} 200 var obj = [].concat(new A(1, 2, 3), new A(4, 5, 6), new A(7, 8, 9)); 201 assertEquals(9, obj.length); 202 for (var i = 0; i < obj.length; ++i) { 203 assertEquals(i + 1, obj[i]); 204 } 205 206 // TODO(caitp): when concat is called on instances of classes which extend 207 // Array, they should: 208 // 209 // - return an instance of the class, rather than an Array instance (if from 210 // same Realm) 211 // - always treat such classes as concat-spreadable 212 })(); 213 214 215 (function testConcatArraySubclassOptOut() { 216 "use strict"; 217 class A extends Array { 218 get [Symbol.isConcatSpreadable]() { return false; } 219 } 220 var obj = [].concat(new A(1, 2, 3), new A(4, 5, 6), new A(7, 8, 9)); 221 assertEquals(3, obj.length); 222 assertEquals(3, obj[0].length); 223 assertEquals(3, obj[1].length); 224 assertEquals(3, obj[2].length); 225 })(); 226 227 228 (function testConcatNonArray() { 229 "use strict"; 230 class NonArray { 231 constructor() { Array.apply(this, arguments); } 232 }; 233 234 var obj = new NonArray(1,2,3); 235 var result = Array.prototype.concat.call(obj, 4, 5, 6); 236 assertEquals(Array, result.constructor); 237 assertEquals([obj,4,5,6], result); 238 assertFalse(result instanceof NonArray); 239 })(); 240 241 242 function testConcatTypedArray(type, elems, modulo) { 243 "use strict"; 244 var items = new Array(elems); 245 var ta_by_len = new type(elems); 246 for (var i = 0; i < elems; ++i) { 247 ta_by_len[i] = items[i] = modulo === false ? i : elems % modulo; 248 } 249 var ta = new type(items); 250 assertEquals([ta, ta], [].concat(ta, ta)); 251 ta[Symbol.isConcatSpreadable] = true; 252 assertEquals(items, [].concat(ta)); 253 254 assertEquals([ta_by_len, ta_by_len], [].concat(ta_by_len, ta_by_len)); 255 ta_by_len[Symbol.isConcatSpreadable] = true; 256 assertEquals(items, [].concat(ta_by_len)); 257 258 // TypedArray with fake `length`. 259 ta = new type(1); 260 var defValue = ta[0]; 261 var expected = new Array(4000); 262 expected[0] = defValue; 263 264 Object.defineProperty(ta, "length", { value: 4000 }); 265 ta[Symbol.isConcatSpreadable] = true; 266 assertEquals(expected, [].concat(ta)); 267 } 268 269 (function testConcatSmallTypedArray() { 270 var max = [Math.pow(2, 8), Math.pow(2, 16), Math.pow(2, 32), false, false]; 271 [ 272 Uint8Array, 273 Uint16Array, 274 Uint32Array, 275 Float32Array, 276 Float64Array 277 ].forEach(function(ctor, i) { 278 testConcatTypedArray(ctor, 1, max[i]); 279 }); 280 })(); 281 282 283 (function testConcatLargeTypedArray() { 284 var max = [Math.pow(2, 8), Math.pow(2, 16), Math.pow(2, 32), false, false]; 285 [ 286 Uint8Array, 287 Uint16Array, 288 Uint32Array, 289 Float32Array, 290 Float64Array 291 ].forEach(function(ctor, i) { 292 testConcatTypedArray(ctor, 4000, max[i]); 293 }); 294 })(); 295 296 297 (function testConcatStrictArguments() { 298 var args = (function(a, b, c) { "use strict"; return arguments; })(1,2,3); 299 args[Symbol.isConcatSpreadable] = true; 300 assertEquals([1, 2, 3, 1, 2, 3], [].concat(args, args)); 301 302 Object.defineProperty(args, "length", { value: 6 }); 303 assertEquals([1, 2, 3, void 0, void 0, void 0], [].concat(args)); 304 })(); 305 306 307 (function testConcatSloppyArguments() { 308 var args = (function(a, b, c) { return arguments; })(1,2,3); 309 args[Symbol.isConcatSpreadable] = true; 310 assertEquals([1, 2, 3, 1, 2, 3], [].concat(args, args)); 311 312 Object.defineProperty(args, "length", { value: 6 }); 313 assertEquals([1, 2, 3, void 0, void 0, void 0], [].concat(args)); 314 })(); 315 316 317 (function testConcatSloppyArgumentsWithDupes() { 318 var args = (function(a, a, a) { return arguments; })(1,2,3); 319 args[Symbol.isConcatSpreadable] = true; 320 assertEquals([1, 2, 3, 1, 2, 3], [].concat(args, args)); 321 322 Object.defineProperty(args, "length", { value: 6 }); 323 assertEquals([1, 2, 3, void 0, void 0, void 0], [].concat(args)); 324 })(); 325 326 327 (function testConcatSloppyArgumentsThrows() { 328 function MyError() {} 329 var args = (function(a) { return arguments; })(1,2,3); 330 Object.defineProperty(args, 0, { 331 get: function() { throw new MyError(); } 332 }); 333 args[Symbol.isConcatSpreadable] = true; 334 assertThrows(function() { 335 return [].concat(args, args); 336 }, MyError); 337 })(); 338 339 340 (function testConcatHoleySloppyArguments() { 341 var args = (function(a) { return arguments; })(1,2,3); 342 delete args[1]; 343 args[Symbol.isConcatSpreadable] = true; 344 assertEquals([1, void 0, 3, 1, void 0, 3], [].concat(args, args)); 345 })(); 346 347 348 (function testConcatSpreadableStringWrapper() { 349 "use strict"; 350 var str1 = new String("yuck\uD83D\uDCA9") 351 // String wrapper objects are not concat-spreadable by default 352 assertEquals([str1], [].concat(str1)); 353 354 // String wrapper objects may be individually concat-spreadable 355 str1[Symbol.isConcatSpreadable] = true; 356 assertEquals(["y", "u", "c", "k", "\uD83D", "\uDCA9"], 357 [].concat(str1)); 358 359 String.prototype[Symbol.isConcatSpreadable] = true; 360 // String wrapper objects may be concat-spreadable 361 assertEquals(["y", "u", "c", "k", "\uD83D", "\uDCA9"], 362 [].concat(new String("yuck\uD83D\uDCA9"))); 363 364 // String values are never concat-spreadable 365 assertEquals(["yuck\uD83D\uDCA9"], [].concat("yuck\uD83D\uDCA9")); 366 delete String.prototype[Symbol.isConcatSpreadable]; 367 })(); 368 369 370 (function testConcatSpreadableBooleanWrapper() { 371 "use strict"; 372 var bool = new Boolean(true) 373 // Boolean wrapper objects are not concat-spreadable by default 374 assertEquals([bool], [].concat(bool)); 375 376 // Boolean wrapper objects may be individually concat-spreadable 377 bool[Symbol.isConcatSpreadable] = true; 378 bool.length = 3; 379 bool[0] = 1, bool[1] = 2, bool[2] = 3; 380 assertEquals([1, 2, 3], [].concat(bool)); 381 382 Boolean.prototype[Symbol.isConcatSpreadable] = true; 383 // Boolean wrapper objects may be concat-spreadable 384 assertEquals([], [].concat(new Boolean(true))); 385 Boolean.prototype[0] = 1; 386 Boolean.prototype[1] = 2; 387 Boolean.prototype[2] = 3; 388 Boolean.prototype.length = 3; 389 assertEquals([1,2,3], [].concat(new Boolean(true))); 390 391 // Boolean values are never concat-spreadable 392 assertEquals([true], [].concat(true)); 393 delete Boolean.prototype[Symbol.isConcatSpreadable]; 394 delete Boolean.prototype[0]; 395 delete Boolean.prototype[1]; 396 delete Boolean.prototype[2]; 397 delete Boolean.prototype.length; 398 })(); 399 400 401 (function testConcatSpreadableNumberWrapper() { 402 "use strict"; 403 var num = new Number(true) 404 // Number wrapper objects are not concat-spreadable by default 405 assertEquals([num], [].concat(num)); 406 407 // Number wrapper objects may be individually concat-spreadable 408 num[Symbol.isConcatSpreadable] = true; 409 num.length = 3; 410 num[0] = 1, num[1] = 2, num[2] = 3; 411 assertEquals([1, 2, 3], [].concat(num)); 412 413 Number.prototype[Symbol.isConcatSpreadable] = true; 414 // Number wrapper objects may be concat-spreadable 415 assertEquals([], [].concat(new Number(123))); 416 Number.prototype[0] = 1; 417 Number.prototype[1] = 2; 418 Number.prototype[2] = 3; 419 Number.prototype.length = 3; 420 assertEquals([1,2,3], [].concat(new Number(123))); 421 422 // Number values are never concat-spreadable 423 assertEquals([true], [].concat(true)); 424 delete Number.prototype[Symbol.isConcatSpreadable]; 425 delete Number.prototype[0]; 426 delete Number.prototype[1]; 427 delete Number.prototype[2]; 428 delete Number.prototype.length; 429 })(); 430 431 432 (function testConcatSpreadableFunction() { 433 "use strict"; 434 var fn = function(a, b, c) {} 435 // Functions are not concat-spreadable by default 436 assertEquals([fn], [].concat(fn)); 437 438 // Functions may be individually concat-spreadable 439 fn[Symbol.isConcatSpreadable] = true; 440 fn[0] = 1, fn[1] = 2, fn[2] = 3; 441 assertEquals([1, 2, 3], [].concat(fn)); 442 443 Function.prototype[Symbol.isConcatSpreadable] = true; 444 // Functions may be concat-spreadable 445 assertEquals([void 0, void 0, void 0], [].concat(function(a,b,c) {})); 446 Function.prototype[0] = 1; 447 Function.prototype[1] = 2; 448 Function.prototype[2] = 3; 449 assertEquals([1,2,3], [].concat(function(a, b, c) {})); 450 451 delete Function.prototype[Symbol.isConcatSpreadable]; 452 delete Function.prototype[0]; 453 delete Function.prototype[1]; 454 delete Function.prototype[2]; 455 })(); 456 457 458 (function testConcatSpreadableRegExp() { 459 "use strict"; 460 var re = /abc/; 461 // RegExps are not concat-spreadable by default 462 assertEquals([re], [].concat(re)); 463 464 // RegExps may be individually concat-spreadable 465 re[Symbol.isConcatSpreadable] = true; 466 re[0] = 1, re[1] = 2, re[2] = 3, re.length = 3; 467 assertEquals([1, 2, 3], [].concat(re)); 468 469 // RegExps may be concat-spreadable 470 RegExp.prototype[Symbol.isConcatSpreadable] = true; 471 RegExp.prototype.length = 3; 472 473 assertEquals([void 0, void 0, void 0], [].concat(/abc/)); 474 RegExp.prototype[0] = 1; 475 RegExp.prototype[1] = 2; 476 RegExp.prototype[2] = 3; 477 assertEquals([1,2,3], [].concat(/abc/)); 478 479 delete RegExp.prototype[Symbol.isConcatSpreadable]; 480 delete RegExp.prototype[0]; 481 delete RegExp.prototype[1]; 482 delete RegExp.prototype[2]; 483 delete RegExp.prototype.length; 484 })(); 485 486 487 (function testArrayConcatSpreadableSparseObject() { 488 "use strict"; 489 var obj = { length: 5 }; 490 obj[Symbol.isConcatSpreadable] = true; 491 assertEquals([void 0, void 0, void 0, void 0, void 0], [].concat(obj)); 492 493 obj.length = 4000; 494 assertEquals(new Array(4000), [].concat(obj)); 495 })(); 496 497 498 // ES5 tests 499 (function testArrayConcatES5() { 500 "use strict"; 501 var poses; 502 var pos; 503 504 poses = [140, 4000000000]; 505 while (pos = poses.shift()) { 506 var a = new Array(pos); 507 var array_proto = []; 508 a.__proto__ = array_proto; 509 assertEquals(pos, a.length); 510 a.push('foo'); 511 assertEquals(pos + 1, a.length); 512 var b = ['bar']; 513 var c = a.concat(b); 514 assertEquals(pos + 2, c.length); 515 assertEquals("undefined", typeof(c[pos - 1])); 516 assertEquals("foo", c[pos]); 517 assertEquals("bar", c[pos + 1]); 518 519 // Can we fool the system by putting a number in a string? 520 var onetwofour = "124"; 521 a[onetwofour] = 'doo'; 522 assertEquals(a[124], 'doo'); 523 c = a.concat(b); 524 assertEquals(c[124], 'doo'); 525 526 // If we put a number in the prototype, then the spec says it should be 527 // copied on concat. 528 array_proto["123"] = 'baz'; 529 assertEquals(a[123], 'baz'); 530 531 c = a.concat(b); 532 assertEquals(pos + 2, c.length); 533 assertEquals("baz", c[123]); 534 assertEquals("undefined", typeof(c[pos - 1])); 535 assertEquals("foo", c[pos]); 536 assertEquals("bar", c[pos + 1]); 537 538 // When we take the number off the prototype it disappears from a, but 539 // the concat put it in c itself. 540 array_proto["123"] = undefined; 541 assertEquals("undefined", typeof(a[123])); 542 assertEquals("baz", c[123]); 543 544 // If the element of prototype is shadowed, the element on the instance 545 // should be copied, but not the one on the prototype. 546 array_proto[123] = 'baz'; 547 a[123] = 'xyz'; 548 assertEquals('xyz', a[123]); 549 c = a.concat(b); 550 assertEquals('xyz', c[123]); 551 552 // Non-numeric properties on the prototype or the array shouldn't get 553 // copied. 554 array_proto.moe = 'joe'; 555 a.ben = 'jerry'; 556 assertEquals(a["moe"], 'joe'); 557 assertEquals(a["ben"], 'jerry'); 558 c = a.concat(b); 559 // ben was not copied 560 assertEquals("undefined", typeof(c.ben)); 561 562 // When we take moe off the prototype it disappears from all arrays. 563 array_proto.moe = undefined; 564 assertEquals("undefined", typeof(c.moe)); 565 566 // Negative indices don't get concated. 567 a[-1] = 'minus1'; 568 assertEquals("minus1", a[-1]); 569 assertEquals("undefined", typeof(a[0xffffffff])); 570 c = a.concat(b); 571 assertEquals("undefined", typeof(c[-1])); 572 assertEquals("undefined", typeof(c[0xffffffff])); 573 assertEquals(c.length, a.length + 1); 574 } 575 576 poses = [140, 4000000000]; 577 while (pos = poses.shift()) { 578 var a = new Array(pos); 579 assertEquals(pos, a.length); 580 a.push('foo'); 581 assertEquals(pos + 1, a.length); 582 var b = ['bar']; 583 var c = a.concat(b); 584 assertEquals(pos + 2, c.length); 585 assertEquals("undefined", typeof(c[pos - 1])); 586 assertEquals("foo", c[pos]); 587 assertEquals("bar", c[pos + 1]); 588 589 // Can we fool the system by putting a number in a string? 590 var onetwofour = "124"; 591 a[onetwofour] = 'doo'; 592 assertEquals(a[124], 'doo'); 593 c = a.concat(b); 594 assertEquals(c[124], 'doo'); 595 596 // If we put a number in the prototype, then the spec says it should be 597 // copied on concat. 598 Array.prototype["123"] = 'baz'; 599 assertEquals(a[123], 'baz'); 600 601 c = a.concat(b); 602 assertEquals(pos + 2, c.length); 603 assertEquals("baz", c[123]); 604 assertEquals("undefined", typeof(c[pos - 1])); 605 assertEquals("foo", c[pos]); 606 assertEquals("bar", c[pos + 1]); 607 608 // When we take the number off the prototype it disappears from a, but 609 // the concat put it in c itself. 610 Array.prototype["123"] = undefined; 611 assertEquals("undefined", typeof(a[123])); 612 assertEquals("baz", c[123]); 613 614 // If the element of prototype is shadowed, the element on the instance 615 // should be copied, but not the one on the prototype. 616 Array.prototype[123] = 'baz'; 617 a[123] = 'xyz'; 618 assertEquals('xyz', a[123]); 619 c = a.concat(b); 620 assertEquals('xyz', c[123]); 621 622 // Non-numeric properties on the prototype or the array shouldn't get 623 // copied. 624 Array.prototype.moe = 'joe'; 625 a.ben = 'jerry'; 626 assertEquals(a["moe"], 'joe'); 627 assertEquals(a["ben"], 'jerry'); 628 c = a.concat(b); 629 // ben was not copied 630 assertEquals("undefined", typeof(c.ben)); 631 // moe was not copied, but we can see it through the prototype 632 assertEquals("joe", c.moe); 633 634 // When we take moe off the prototype it disappears from all arrays. 635 Array.prototype.moe = undefined; 636 assertEquals("undefined", typeof(c.moe)); 637 638 // Negative indices don't get concated. 639 a[-1] = 'minus1'; 640 assertEquals("minus1", a[-1]); 641 assertEquals("undefined", typeof(a[0xffffffff])); 642 c = a.concat(b); 643 assertEquals("undefined", typeof(c[-1])); 644 assertEquals("undefined", typeof(c[0xffffffff])); 645 assertEquals(c.length, a.length + 1); 646 647 } 648 649 a = []; 650 c = a.concat('Hello'); 651 assertEquals(1, c.length); 652 assertEquals("Hello", c[0]); 653 assertEquals("Hello", c.toString()); 654 655 // Check that concat preserves holes. 656 var holey = [void 0,'a',,'c'].concat(['d',,'f',[0,,2],void 0]) 657 assertEquals(9, holey.length); // hole in embedded array is ignored 658 for (var i = 0; i < holey.length; i++) { 659 if (i == 2 || i == 5) { 660 assertFalse(i in holey); 661 } else { 662 assertTrue(i in holey); 663 } 664 } 665 666 // Polluted prototype from prior tests. 667 delete Array.prototype[123]; 668 669 // Check that concat reads getters in the correct order. 670 var arr1 = [,2]; 671 var arr2 = [1,3]; 672 var r1 = [].concat(arr1, arr2); // [,2,1,3] 673 assertEquals([,2,1,3], r1); 674 675 // Make first array change length of second array. 676 Object.defineProperty(arr1, 0, {get: function() { 677 arr2.push("X"); 678 return undefined; 679 }, configurable: true}) 680 var r2 = [].concat(arr1, arr2); // [undefined,2,1,3,"X"] 681 assertEquals([undefined,2,1,3,"X"], r2); 682 683 // Make first array change length of second array massively. 684 arr2.length = 2; 685 Object.defineProperty(arr1, 0, {get: function() { 686 arr2[500000] = "X"; 687 return undefined; 688 }, configurable: true}) 689 var r3 = [].concat(arr1, arr2); // [undefined,2,1,3,"X"] 690 var expected = [undefined,2,1,3]; 691 expected[500000 + 2] = "X"; 692 693 assertEquals(expected, r3); 694 695 var arr3 = []; 696 var trace = []; 697 var expectedTrace = [] 698 function mkGetter(i) { return function() { trace.push(i); }; } 699 arr3.length = 10000; 700 for (var i = 0; i < 100; i++) { 701 Object.defineProperty(arr3, i * i, {get: mkGetter(i)}); 702 expectedTrace[i] = i; 703 expectedTrace[100 + i] = i; 704 } 705 var r4 = [0].concat(arr3, arr3); 706 assertEquals(1 + arr3.length * 2, r4.length); 707 assertEquals(expectedTrace, trace); 708 709 // Clean up. 710 delete Array.prototype[123]; 711 delete Array.prototype["123"]; 712 delete Array.prototype["moe"]; 713 })(); 714 715 716 717 718 //////////////////////////////////////////////////////////////////////////////// 719 // Tests with proxies 720 721 // Note: concat does not currently support species so there is no difference 722 // between [].concat(foo) and Array.prototype.concat.apply(foo). 723 724 725 var log = []; 726 var logger = {}; 727 var handler = new Proxy({}, logger); 728 729 logger.get = function(t, trap, r) { 730 return function(...args) { 731 log.push([trap, ...args]); 732 return Reflect[trap](...args); 733 } 734 }; 735 736 737 (function testUnspreadableNonArrayLikeProxy() { 738 var target = {0: "a", 1: "b"}; 739 var obj = new Proxy(target, handler); 740 741 log.length = 0; 742 assertEquals([obj], [].concat(obj)); 743 assertEquals(1, log.length); 744 for (var i in log) assertSame(target, log[i][1]); 745 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 746 747 log.length = 0; 748 assertEquals([obj], Array.prototype.concat.apply(obj)); 749 assertEquals(1, log.length); 750 for (var i in log) assertSame(target, log[i][1]); 751 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 752 })(); 753 754 755 (function testSpreadableNonArrayLikeProxy() { 756 var target = {0: "a", 1: "b", [Symbol.isConcatSpreadable]: "truish"}; 757 var obj = new Proxy(target, handler); 758 759 log.length = 0; 760 assertEquals([], [].concat(obj)); 761 assertEquals(2, log.length); 762 for (var i in log) assertSame(target, log[i][1]); 763 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 764 assertEquals(["get", target, "length", obj], log[1]); 765 766 log.length = 0; 767 assertEquals([], Array.prototype.concat.apply(obj)); 768 assertEquals(2, log.length); 769 for (var i in log) assertSame(target, log[i][1]); 770 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 771 assertEquals(["get", target, "length", obj], log[1]); 772 773 target.length = 3; 774 775 log.length = 0; 776 assertEquals(["a", "b", undefined], [].concat(obj)); 777 assertEquals(7, log.length); 778 for (var i in log) assertSame(target, log[i][1]); 779 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 780 assertEquals(["get", target, "length", obj], log[1]); 781 assertEquals(["has", target, "0"], log[2]); 782 assertEquals(["get", target, "0", obj], log[3]); 783 assertEquals(["has", target, "1"], log[4]); 784 assertEquals(["get", target, "1", obj], log[5]); 785 assertEquals(["has", target, "2"], log[6]); 786 787 log.length = 0; 788 assertEquals(["a", "b", undefined], Array.prototype.concat.apply(obj)); 789 assertEquals(7, log.length); 790 for (var i in log) assertSame(target, log[i][1]); 791 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 792 assertEquals(["get", target, "length", obj], log[1]); 793 assertEquals(["has", target, "0"], log[2]); 794 assertEquals(["get", target, "0", obj], log[3]); 795 assertEquals(["has", target, "1"], log[4]); 796 assertEquals(["get", target, "1", obj], log[5]); 797 assertEquals(["has", target, "2"], log[6]); 798 })(); 799 800 801 (function testUnspreadableArrayLikeProxy() { 802 var target = ["a", "b"]; 803 target[Symbol.isConcatSpreadable] = ""; 804 var obj = new Proxy(target, handler); 805 806 log.length = 0; 807 assertEquals([obj], [].concat(obj)); 808 assertEquals(1, log.length); 809 for (var i in log) assertSame(target, log[i][1]); 810 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 811 812 log.length = 0; 813 assertEquals([obj], Array.prototype.concat.apply(obj)); 814 assertEquals(1, log.length); 815 for (var i in log) assertSame(target, log[i][1]); 816 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 817 })(); 818 819 820 (function testSpreadableArrayLikeProxy() { 821 var target = ["a", "b"]; 822 target[Symbol.isConcatSpreadable] = undefined; 823 var obj = new Proxy(target, handler); 824 825 log.length = 0; 826 assertEquals(["a", "b"], [].concat(obj)); 827 assertEquals(6, log.length); 828 for (var i in log) assertSame(target, log[i][1]); 829 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 830 assertEquals(["get", target, "length", obj], log[1]); 831 assertEquals(["has", target, "0"], log[2]); 832 assertEquals(["get", target, "0", obj], log[3]); 833 assertEquals(["has", target, "1"], log[4]); 834 assertEquals(["get", target, "1", obj], log[5]); 835 836 log.length = 0; 837 assertEquals(["a", "b"], Array.prototype.concat.apply(obj)); 838 assertEquals(6, log.length); 839 for (var i in log) assertSame(target, log[i][1]); 840 assertEquals(["get", target, Symbol.isConcatSpreadable, obj], log[0]); 841 assertEquals(["get", target, "length", obj], log[1]); 842 assertEquals(["has", target, "0"], log[2]); 843 assertEquals(["get", target, "0", obj], log[3]); 844 assertEquals(["has", target, "1"], log[4]); 845 assertEquals(["get", target, "1", obj], log[5]); 846 })(); 847 848 849 (function testSpreadableArrayLikeProxyWithNontrivialLength() { 850 var getTrap = function(t, key) { 851 if (key === "length") return {[Symbol.toPrimitive]() {return 3}}; 852 if (key === "2") return "baz"; 853 if (key === "3") return "bar"; 854 }; 855 var target = []; 856 var obj = new Proxy(target, {get: getTrap, has: () => true}); 857 858 assertEquals([undefined, undefined, "baz"], [].concat(obj)); 859 assertEquals([undefined, undefined, "baz"], Array.prototype.concat.apply(obj)) 860 })(); 861 862 863 (function testSpreadableArrayLikeProxyWithBogusLength() { 864 var getTrap = function(t, key) { 865 if (key === "length") return Symbol(); 866 if (key === "2") return "baz"; 867 if (key === "3") return "bar"; 868 }; 869 var target = []; 870 var obj = new Proxy(target, {get: getTrap, has: () => true}); 871 872 assertThrows(() => [].concat(obj), TypeError); 873 assertThrows(() => Array.prototype.concat.apply(obj), TypeError); 874 })(); 875