// Copyright 2013 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // Flags: --allow-natives-syntax var NONE = 0; var READ_ONLY = 1; var DONT_ENUM = 2; var DONT_DELETE = 4; function assertHasOwnProperty(object, name, attrs) { assertTrue(object.hasOwnProperty(name)); var desc = Object.getOwnPropertyDescriptor(object, name); assertEquals(desc.writable, !(attrs & READ_ONLY)); assertEquals(desc.enumerable, !(attrs & DONT_ENUM)); assertEquals(desc.configurable, !(attrs & DONT_DELETE)); } function TestArrayPrototype() { assertHasOwnProperty(Array.prototype, 'entries', DONT_ENUM); assertHasOwnProperty(Array.prototype, 'keys', DONT_ENUM); assertHasOwnProperty(Array.prototype, Symbol.iterator, DONT_ENUM); assertEquals('entries', Array.prototype.entries.name); assertEquals('keys', Array.prototype.keys.name); assertEquals('values', Array.prototype[Symbol.iterator].name); } TestArrayPrototype(); function assertIteratorResult(value, done, result) { assertEquals({value: value, done: done}, result); } function TestValues() { var array = ['a', 'b', 'c']; var iterator = array[Symbol.iterator](); assertIteratorResult('a', false, iterator.next()); assertIteratorResult('b', false, iterator.next()); assertIteratorResult('c', false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); array.push('d'); assertIteratorResult(void 0, true, iterator.next()); } TestValues(); function TestValuesMutate() { var array = ['a', 'b', 'c']; var iterator = array[Symbol.iterator](); assertIteratorResult('a', false, iterator.next()); assertIteratorResult('b', false, iterator.next()); assertIteratorResult('c', false, iterator.next()); array.push('d'); assertIteratorResult('d', false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); } TestValuesMutate(); function TestKeys() { var array = ['a', 'b', 'c']; var iterator = array.keys(); assertIteratorResult(0, false, iterator.next()); assertIteratorResult(1, false, iterator.next()); assertIteratorResult(2, false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); array.push('d'); assertIteratorResult(void 0, true, iterator.next()); } TestKeys(); function TestKeysMutate() { var array = ['a', 'b', 'c']; var iterator = array.keys(); assertIteratorResult(0, false, iterator.next()); assertIteratorResult(1, false, iterator.next()); assertIteratorResult(2, false, iterator.next()); array.push('d'); assertIteratorResult(3, false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); } TestKeysMutate(); function TestEntries() { var array = ['a', 'b', 'c']; var iterator = array.entries(); assertIteratorResult([0, 'a'], false, iterator.next()); assertIteratorResult([1, 'b'], false, iterator.next()); assertIteratorResult([2, 'c'], false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); array.push('d'); assertIteratorResult(void 0, true, iterator.next()); } TestEntries(); function TestEntriesMutate() { var array = ['a', 'b', 'c']; var iterator = array.entries(); assertIteratorResult([0, 'a'], false, iterator.next()); assertIteratorResult([1, 'b'], false, iterator.next()); assertIteratorResult([2, 'c'], false, iterator.next()); array.push('d'); assertIteratorResult([3, 'd'], false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); } TestEntriesMutate(); function TestArrayIteratorPrototype() { var array = []; var iterator = array.keys(); var ArrayIteratorPrototype = iterator.__proto__; assertSame(ArrayIteratorPrototype, array[Symbol.iterator]().__proto__); assertSame(ArrayIteratorPrototype, array.keys().__proto__); assertSame(ArrayIteratorPrototype, array.entries().__proto__); assertSame(Object.prototype, ArrayIteratorPrototype.__proto__.__proto__); assertFalse(ArrayIteratorPrototype.hasOwnProperty('constructor')); assertArrayEquals(['next'], Object.getOwnPropertyNames(ArrayIteratorPrototype)); assertHasOwnProperty(ArrayIteratorPrototype, 'next', DONT_ENUM); assertFalse(ArrayIteratorPrototype.hasOwnProperty(Symbol.iterator)); assertEquals("[object Array Iterator]", Object.prototype.toString.call(iterator)); assertEquals("Array Iterator", ArrayIteratorPrototype[Symbol.toStringTag]); var desc = Object.getOwnPropertyDescriptor( ArrayIteratorPrototype, Symbol.toStringTag); assertTrue(desc.configurable); assertFalse(desc.writable); assertEquals("Array Iterator", desc.value); } TestArrayIteratorPrototype(); function TestForArrayValues() { var buffer = []; var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; var i = 0; for (var value of array[Symbol.iterator]()) { buffer[i++] = value; } assertEquals(8, buffer.length); for (var i = 0; i < buffer.length; i++) { assertSame(array[i], buffer[i]); } } TestForArrayValues(); function TestForArrayKeys() { var buffer = []; var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; var i = 0; for (var key of array.keys()) { buffer[i++] = key; } assertEquals(8, buffer.length); for (var i = 0; i < buffer.length; i++) { assertEquals(i, buffer[i]); } } TestForArrayKeys(); function TestForArrayEntries() { var buffer = []; var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; var i = 0; for (var entry of array.entries()) { buffer[i++] = entry; } assertEquals(8, buffer.length); for (var i = 0; i < buffer.length; i++) { assertSame(array[i], buffer[i][1]); } for (var i = 0; i < buffer.length; i++) { assertEquals(i, buffer[i][0]); } } TestForArrayEntries(); function TestForArray() { var buffer = []; var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; var i = 0; for (var value of array) { buffer[i++] = value; } assertEquals(8, buffer.length); for (var i = 0; i < buffer.length; i++) { assertSame(array[i], buffer[i]); } } TestForArrayValues(); function TestNonOwnSlots() { var array = [0]; var iterator = array[Symbol.iterator](); var object = {__proto__: iterator}; assertThrows(function() { object.next(); }, TypeError); } TestNonOwnSlots(); function TestForDictionaryArray() { var array = []; array[1024] = 'c'; assertTrue(%HasDictionaryElements(array)); var iterator = array[Symbol.iterator](); for (var i = 0; i < 1024; ++i) { assertIteratorResult(void 0, false, iterator.next()); } assertIteratorResult('c', false, iterator.next()); assertIteratorResult(void 0, true, iterator.next()); } TestForDictionaryArray();