1// Copyright 2013 the V8 project authors. All rights reserved. 2// Redistribution and use in source and binary forms, with or without 3// modification, are permitted provided that the following conditions are 4// met: 5// 6// * Redistributions of source code must retain the above copyright 7// notice, this list of conditions and the following disclaimer. 8// * Redistributions in binary form must reproduce the above 9// copyright notice, this list of conditions and the following 10// disclaimer in the documentation and/or other materials provided 11// with the distribution. 12// * Neither the name of Google Inc. nor the names of its 13// contributors may be used to endorse or promote products derived 14// from this software without specific prior written permission. 15// 16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 27 28// Flags: --allow-natives-syntax --harmony-tostring 29 30 31var NONE = 0; 32var READ_ONLY = 1; 33var DONT_ENUM = 2; 34var DONT_DELETE = 4; 35 36 37function assertHasOwnProperty(object, name, attrs) { 38 assertTrue(object.hasOwnProperty(name)); 39 var desc = Object.getOwnPropertyDescriptor(object, name); 40 assertEquals(desc.writable, !(attrs & READ_ONLY)); 41 assertEquals(desc.enumerable, !(attrs & DONT_ENUM)); 42 assertEquals(desc.configurable, !(attrs & DONT_DELETE)); 43} 44 45 46function TestArrayPrototype() { 47 assertHasOwnProperty(Array.prototype, 'entries', DONT_ENUM); 48 assertHasOwnProperty(Array.prototype, 'keys', DONT_ENUM); 49 assertHasOwnProperty(Array.prototype, Symbol.iterator, DONT_ENUM); 50 assertEquals('entries', Array.prototype.entries.name); 51 assertEquals('keys', Array.prototype.keys.name); 52 assertEquals('values', Array.prototype[Symbol.iterator].name); 53} 54TestArrayPrototype(); 55 56 57function assertIteratorResult(value, done, result) { 58 assertEquals({value: value, done: done}, result); 59} 60 61 62function TestValues() { 63 var array = ['a', 'b', 'c']; 64 var iterator = array[Symbol.iterator](); 65 assertIteratorResult('a', false, iterator.next()); 66 assertIteratorResult('b', false, iterator.next()); 67 assertIteratorResult('c', false, iterator.next()); 68 assertIteratorResult(void 0, true, iterator.next()); 69 70 array.push('d'); 71 assertIteratorResult(void 0, true, iterator.next()); 72} 73TestValues(); 74 75 76function TestValuesMutate() { 77 var array = ['a', 'b', 'c']; 78 var iterator = array[Symbol.iterator](); 79 assertIteratorResult('a', false, iterator.next()); 80 assertIteratorResult('b', false, iterator.next()); 81 assertIteratorResult('c', false, iterator.next()); 82 array.push('d'); 83 assertIteratorResult('d', false, iterator.next()); 84 assertIteratorResult(void 0, true, iterator.next()); 85} 86TestValuesMutate(); 87 88 89function TestKeys() { 90 var array = ['a', 'b', 'c']; 91 var iterator = array.keys(); 92 assertIteratorResult(0, false, iterator.next()); 93 assertIteratorResult(1, false, iterator.next()); 94 assertIteratorResult(2, false, iterator.next()); 95 assertIteratorResult(void 0, true, iterator.next()); 96 97 array.push('d'); 98 assertIteratorResult(void 0, true, iterator.next()); 99} 100TestKeys(); 101 102 103function TestKeysMutate() { 104 var array = ['a', 'b', 'c']; 105 var iterator = array.keys(); 106 assertIteratorResult(0, false, iterator.next()); 107 assertIteratorResult(1, false, iterator.next()); 108 assertIteratorResult(2, false, iterator.next()); 109 array.push('d'); 110 assertIteratorResult(3, false, iterator.next()); 111 assertIteratorResult(void 0, true, iterator.next()); 112} 113TestKeysMutate(); 114 115 116function TestEntries() { 117 var array = ['a', 'b', 'c']; 118 var iterator = array.entries(); 119 assertIteratorResult([0, 'a'], false, iterator.next()); 120 assertIteratorResult([1, 'b'], false, iterator.next()); 121 assertIteratorResult([2, 'c'], false, iterator.next()); 122 assertIteratorResult(void 0, true, iterator.next()); 123 124 array.push('d'); 125 assertIteratorResult(void 0, true, iterator.next()); 126} 127TestEntries(); 128 129 130function TestEntriesMutate() { 131 var array = ['a', 'b', 'c']; 132 var iterator = array.entries(); 133 assertIteratorResult([0, 'a'], false, iterator.next()); 134 assertIteratorResult([1, 'b'], false, iterator.next()); 135 assertIteratorResult([2, 'c'], false, iterator.next()); 136 array.push('d'); 137 assertIteratorResult([3, 'd'], false, iterator.next()); 138 assertIteratorResult(void 0, true, iterator.next()); 139} 140TestEntriesMutate(); 141 142 143function TestArrayIteratorPrototype() { 144 var array = []; 145 var iterator = array.keys(); 146 147 var ArrayIteratorPrototype = iterator.__proto__; 148 149 assertEquals(ArrayIteratorPrototype, array[Symbol.iterator]().__proto__); 150 assertEquals(ArrayIteratorPrototype, array.keys().__proto__); 151 assertEquals(ArrayIteratorPrototype, array.entries().__proto__); 152 153 assertEquals(Object.prototype, ArrayIteratorPrototype.__proto__); 154 155 assertEquals('Array Iterator', %_ClassOf(array[Symbol.iterator]())); 156 assertEquals('Array Iterator', %_ClassOf(array.keys())); 157 assertEquals('Array Iterator', %_ClassOf(array.entries())); 158 159 assertFalse(ArrayIteratorPrototype.hasOwnProperty('constructor')); 160 assertArrayEquals(['next'], 161 Object.getOwnPropertyNames(ArrayIteratorPrototype)); 162 assertHasOwnProperty(ArrayIteratorPrototype, 'next', DONT_ENUM); 163 assertHasOwnProperty(ArrayIteratorPrototype, Symbol.iterator, DONT_ENUM); 164 165 assertEquals("[object Array Iterator]", 166 Object.prototype.toString.call(iterator)); 167 assertEquals("Array Iterator", ArrayIteratorPrototype[Symbol.toStringTag]); 168 var desc = Object.getOwnPropertyDescriptor( 169 ArrayIteratorPrototype, Symbol.toStringTag); 170 assertTrue(desc.configurable); 171 assertFalse(desc.writable); 172 assertEquals("Array Iterator", desc.value); 173} 174TestArrayIteratorPrototype(); 175 176 177function TestForArrayValues() { 178 var buffer = []; 179 var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; 180 var i = 0; 181 for (var value of array[Symbol.iterator]()) { 182 buffer[i++] = value; 183 } 184 185 assertEquals(8, buffer.length); 186 187 for (var i = 0; i < buffer.length; i++) { 188 assertSame(array[i], buffer[i]); 189 } 190} 191TestForArrayValues(); 192 193 194function TestForArrayKeys() { 195 var buffer = []; 196 var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; 197 var i = 0; 198 for (var key of array.keys()) { 199 buffer[i++] = key; 200 } 201 202 assertEquals(8, buffer.length); 203 204 for (var i = 0; i < buffer.length; i++) { 205 assertEquals(i, buffer[i]); 206 } 207} 208TestForArrayKeys(); 209 210 211function TestForArrayEntries() { 212 var buffer = []; 213 var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; 214 var i = 0; 215 for (var entry of array.entries()) { 216 buffer[i++] = entry; 217 } 218 219 assertEquals(8, buffer.length); 220 221 for (var i = 0; i < buffer.length; i++) { 222 assertSame(array[i], buffer[i][1]); 223 } 224 225 for (var i = 0; i < buffer.length; i++) { 226 assertEquals(i, buffer[i][0]); 227 } 228} 229TestForArrayEntries(); 230 231 232function TestForArray() { 233 var buffer = []; 234 var array = [0, 'a', true, false, null, /* hole */, undefined, NaN]; 235 var i = 0; 236 for (var value of array) { 237 buffer[i++] = value; 238 } 239 240 assertEquals(8, buffer.length); 241 242 for (var i = 0; i < buffer.length; i++) { 243 assertSame(array[i], buffer[i]); 244 } 245} 246TestForArrayValues(); 247 248 249function TestNonOwnSlots() { 250 var array = [0]; 251 var iterator = array[Symbol.iterator](); 252 var object = {__proto__: iterator}; 253 254 assertThrows(function() { 255 object.next(); 256 }, TypeError); 257} 258TestNonOwnSlots(); 259