Index: test/mjsunit/harmony/array-includes.js |
diff --git a/test/mjsunit/harmony/array-includes.js b/test/mjsunit/harmony/array-includes.js |
new file mode 100644 |
index 0000000000000000000000000000000000000000..2cdd1123d711b03ad76a6a252d26f386b6a6d1fe |
--- /dev/null |
+++ b/test/mjsunit/harmony/array-includes.js |
@@ -0,0 +1,677 @@ |
+// Copyright 2014 the V8 project authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+// Flags: --harmony-array-includes |
+ |
+// Largely ported from |
+// https://github.com/tc39/Array.prototype.includes/tree/master/test |
+// using https://www.npmjs.org/package/test262-to-mjsunit with further edits |
+ |
+ |
+// Array.prototype.includes sees a new element added by a getter that is hit |
+// during iteration |
+(function() { |
+ var arrayLike = { |
+ length: 5, |
+ 0: "a", |
+ |
+ get 1() { |
+ this[2] = "c"; |
+ return "b"; |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike, "c")); |
+})(); |
+ |
+ |
+// Array.prototype.includes works on array-like objects |
+(function() { |
+ var arrayLike1 = { |
+ length: 5, |
+ 0: "a", |
+ 1: "b" |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike1, "a")); |
+ assertFalse(Array.prototype.includes.call(arrayLike1, "c")); |
+ |
+ var arrayLike2 = { |
+ length: 2, |
+ 0: "a", |
+ 1: "b", |
+ 2: "c" |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike2, "b")); |
+ assertFalse(Array.prototype.includes.call(arrayLike2, "c")); |
+})(); |
+ |
+ |
+// Array.prototype.includes should fail if used on a null or undefined this |
+(function() { |
+ assertThrows(function() { |
+ Array.prototype.includes.call(null, "a"); |
+ }, TypeError); |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(undefined, "a"); |
+ }, TypeError); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if getting an index throws an |
+// exception |
+(function() { |
+ function Test262Error() {} |
+ |
+ var trappedZero = { |
+ length: 2, |
+ |
+ get 0() { |
+ throw new Test262Error(); |
+ }, |
+ |
+ get 1() { |
+ assertUnreachable("Should not try to get the first element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(trappedZero, "a"); |
+ }, Test262Error); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if ToNumber ends up being called on |
+// a symbol fromIndex |
+(function() { |
+ var trappedZero = { |
+ length: 1, |
+ |
+ get 0() { |
+ assertUnreachable("Should not try to get the zeroth element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(trappedZero, "a", Symbol()); |
+ }, TypeError); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if an exception occurs converting |
+// the fromIndex to a number |
+(function() { |
+ function Test262Error() {} |
+ |
+ var fromIndex = { |
+ valueOf: function() { |
+ throw new Test262Error(); |
+ } |
+ }; |
+ |
+ var trappedZero = { |
+ length: 1, |
+ |
+ get 0() { |
+ assertUnreachable("Should not try to get the zeroth element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(trappedZero, "a", fromIndex); |
+ }, Test262Error); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if an exception occurs getting the |
+// length |
+(function() { |
+ function Test262Error() {} |
+ |
+ var fromIndexTrap = { |
+ valueOf: function() { |
+ assertUnreachable("Should not try to call ToInteger on valueOf"); |
+ } |
+ }; |
+ |
+ var throwingLength = { |
+ get length() { |
+ throw new Test262Error(); |
+ }, |
+ |
+ get 0() { |
+ assertUnreachable("Should not try to get the zeroth element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(throwingLength, "a", fromIndexTrap); |
+ }, Test262Error); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if ToLength ends up being called on |
+// a symbol length |
+(function() { |
+ var fromIndexTrap = { |
+ valueOf: function() { |
+ assertUnreachable("Should not try to call ToInteger on valueOf"); |
+ } |
+ }; |
+ |
+ var badLength = { |
+ length: Symbol(), |
+ |
+ get 0() { |
+ assertUnreachable("Should not try to get the zeroth element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(badLength, "a", fromIndexTrap); |
+ }, TypeError); |
+})(); |
+ |
+ |
+// Array.prototype.includes should terminate if an exception occurs converting |
+// the length to a number |
+(function() { |
+ function Test262Error() {} |
+ |
+ var fromIndexTrap = { |
+ valueOf: function() { |
+ assertUnreachable("Should not try to call ToInteger on valueOf"); |
+ } |
+ }; |
+ |
+ var badLength = { |
+ length: { |
+ valueOf: function() { |
+ throw new Test262Error(); |
+ } |
+ }, |
+ |
+ get 0() { |
+ assertUnreachable("Should not try to get the zeroth element"); |
+ } |
+ }; |
+ |
+ assertThrows(function() { |
+ Array.prototype.includes.call(badLength, "a", fromIndexTrap); |
+ }, Test262Error); |
+})(); |
+ |
+ |
+// Array.prototype.includes should search the whole array, as the optional |
+// second argument fromIndex defaults to 0 |
+(function() { |
+ assertTrue([10, 11].includes(10)); |
+ assertTrue([10, 11].includes(11)); |
+ |
+ var arrayLike = { |
+ length: 2, |
+ |
+ get 0() { |
+ return "1"; |
+ }, |
+ |
+ get 1() { |
+ return "2"; |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike, "1")); |
+ assertTrue(Array.prototype.includes.call(arrayLike, "2")); |
+})(); |
+ |
+ |
+// Array.prototype.includes returns false if fromIndex is greater or equal to |
+// the length of the array |
+(function() { |
+ assertFalse([1, 2].includes(2, 3)); |
+ assertFalse([1, 2].includes(2, 2)); |
+ |
+ var arrayLikeWithTrap = { |
+ length: 2, |
+ |
+ get 0() { |
+ assertUnreachable("Getter for 0 was called"); |
+ }, |
+ |
+ get 1() { |
+ assertUnreachable("Getter for 1 was called"); |
+ } |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTrap, "c", 2)); |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTrap, "c", 3)); |
+})(); |
+ |
+ |
+// Array.prototype.includes searches the whole array if the computed index from |
+// the given negative fromIndex argument is less than 0 |
+(function() { |
+ assertTrue([1, 3].includes(1, -4)); |
+ assertTrue([1, 3].includes(3, -4)); |
+ |
+ var arrayLike = { |
+ length: 2, |
+ 0: "a", |
+ |
+ get 1() { |
+ return "b"; |
+ }, |
+ |
+ get "-1"() { |
+ assertUnreachable("Should not try to get the element at index -1"); |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike, "a", -4)); |
+ assertTrue(Array.prototype.includes.call(arrayLike, "b", -4)); |
+})(); |
+ |
+ |
+// Array.prototype.includes should use a negative value as the offset from the |
+// end of the array to compute fromIndex |
+(function() { |
+ assertTrue([12, 13].includes(13, -1)); |
+ assertFalse([12, 13].includes(12, -1)); |
+ assertTrue([12, 13].includes(12, -2)); |
+ |
+ var arrayLike = { |
+ length: 2, |
+ |
+ get 0() { |
+ return "a"; |
+ }, |
+ |
+ get 1() { |
+ return "b"; |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike, "b", -1)); |
+ assertFalse(Array.prototype.includes.call(arrayLike, "a", -1)); |
+ assertTrue(Array.prototype.includes.call(arrayLike, "a", -2)); |
+})(); |
+ |
+ |
+// Array.prototype.includes converts its fromIndex parameter to an integer |
+(function() { |
+ assertFalse(["a", "b"].includes("a", 2.3)); |
+ |
+ var arrayLikeWithTraps = { |
+ length: 2, |
+ |
+ get 0() { |
+ assertUnreachable("Getter for 0 was called"); |
+ }, |
+ |
+ get 1() { |
+ assertUnreachable("Getter for 1 was called"); |
+ } |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTraps, "c", 2.1)); |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTraps, "c", +Infinity)); |
+ assertTrue(["a", "b", "c"].includes("a", -Infinity)); |
+ assertTrue(["a", "b", "c"].includes("c", 2.9)); |
+ assertTrue(["a", "b", "c"].includes("c", NaN)); |
+ |
+ var arrayLikeWithTrapAfterZero = { |
+ length: 2, |
+ |
+ get 0() { |
+ return "a"; |
+ }, |
+ |
+ get 1() { |
+ assertUnreachable("Getter for 1 was called"); |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLikeWithTrapAfterZero, "a", NaN)); |
+ |
+ var numberLike = { |
+ valueOf: function() { |
+ return 2; |
+ } |
+ }; |
+ |
+ assertFalse(["a", "b", "c"].includes("a", numberLike)); |
+ assertFalse(["a", "b", "c"].includes("a", "2")); |
+ assertTrue(["a", "b", "c"].includes("c", numberLike)); |
+ assertTrue(["a", "b", "c"].includes("c", "2")); |
+})(); |
+ |
+ |
+// Array.prototype.includes should have length 1 |
+(function() { |
+ assertEquals(1, Array.prototype.includes.length); |
+})(); |
+ |
+ |
+// Array.prototype.includes should have name property with value 'includes' |
+(function() { |
+ assertEquals("includes", Array.prototype.includes.name); |
+})(); |
+ |
+ |
+// !!! Test failed to convert: |
+// Cannot convert tests with includes. |
+// !!! |
+ |
+ |
+// Array.prototype.includes does not skip holes; if the array has a prototype it |
+// gets from that |
+(function() { |
+ var holesEverywhere = [,,,]; |
+ |
+ holesEverywhere.__proto__ = { |
+ 1: "a" |
+ }; |
+ |
+ holesEverywhere.__proto__.__proto__ = Array.prototype; |
+ assertTrue(holesEverywhere.includes("a")); |
+ var oneHole = ["a", "b",, "d"]; |
+ |
+ oneHole.__proto__ = { |
+ get 2() { |
+ return "c"; |
+ } |
+ }; |
+ |
+ assertTrue(Array.prototype.includes.call(oneHole, "c")); |
+})(); |
+ |
+ |
+// Array.prototype.includes does not skip holes; instead it treates them as |
+// undefined |
+(function() { |
+ assertTrue([,,,].includes(undefined)); |
+ assertTrue(["a", "b",, "d"].includes(undefined)); |
+})(); |
+ |
+ |
+// Array.prototype.includes gets length property from the prototype if it's |
+// available |
+(function() { |
+ var proto = { |
+ length: 1 |
+ }; |
+ |
+ var arrayLike = Object.create(proto); |
+ arrayLike[0] = "a"; |
+ |
+ Object.defineProperty(arrayLike, "1", { |
+ get: function() { |
+ assertUnreachable("Getter for 1 was called"); |
+ } |
+ }); |
+ |
+ assertTrue(Array.prototype.includes.call(arrayLike, "a")); |
+})(); |
+ |
+ |
+// Array.prototype.includes treats a missing length property as zero |
+(function() { |
+ var arrayLikeWithTraps = { |
+ get 0() { |
+ assertUnreachable("Getter for 0 was called"); |
+ }, |
+ |
+ get 1() { |
+ assertUnreachable("Getter for 1 was called"); |
+ } |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTraps, "a")); |
+})(); |
+ |
+ |
+// Array.prototype.includes should always return false on negative-length |
+// objects |
+(function() { |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -1 |
+ }, 2)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -2 |
+ })); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -Infinity |
+ }, undefined)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -Math.pow(2, 53) |
+ }, NaN)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -1, |
+ "-1": 2 |
+ }, 2)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -3, |
+ "-1": 2 |
+ }, 2)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: -Infinity, |
+ "-1": 2 |
+ }, 2)); |
+ |
+ var arrayLikeWithTrap = { |
+ length: -1, |
+ |
+ get 0() { |
+ assertUnreachable("Getter for 0 was called"); |
+ } |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLikeWithTrap, 2)); |
+})(); |
+ |
+ |
+// Array.prototype.includes should clamp positive lengths to 2^53 - 1 |
+(function() { |
+ var fromIndexForLargeIndexTests = 9007199254740990; |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 1 |
+ }, 2)); |
+ |
+ assertTrue(Array.prototype.includes.call({ |
+ length: 1, |
+ 0: "a" |
+ }, "a")); |
+ |
+ assertTrue(Array.prototype.includes.call({ |
+ length: +Infinity, |
+ 0: "a" |
+ }, "a")); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: +Infinity |
+ }, "a", fromIndexForLargeIndexTests)); |
+ |
+ var arrayLikeWithTrap = { |
+ length: +Infinity, |
+ |
+ get 9007199254740992() { |
+ assertUnreachable("Getter for 9007199254740992 (i.e. 2^53) was called"); |
+ }, |
+ |
+ "9007199254740993": "a" |
+ }; |
+ |
+ assertFalse( |
+ Array.prototype.includes.call(arrayLikeWithTrap, "a", fromIndexForLargeIndexTests) |
+ ); |
+ |
+ var arrayLikeWithTooBigLength = { |
+ length: 9007199254740996, |
+ "9007199254740992": "a" |
+ }; |
+ |
+ assertFalse( |
+ Array.prototype.includes.call(arrayLikeWithTooBigLength, "a", fromIndexForLargeIndexTests) |
+ ); |
+})(); |
+ |
+ |
+// Array.prototype.includes should always return false on zero-length objects |
+(function() { |
+ assertFalse([].includes(2)); |
+ assertFalse([].includes()); |
+ assertFalse([].includes(undefined)); |
+ assertFalse([].includes(NaN)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0 |
+ }, 2)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0 |
+ })); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0 |
+ }, undefined)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0 |
+ }, NaN)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0, |
+ 0: 2 |
+ }, 2)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0, |
+ 0: undefined |
+ })); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0, |
+ 0: undefined |
+ }, undefined)); |
+ |
+ assertFalse(Array.prototype.includes.call({ |
+ length: 0, |
+ 0: NaN |
+ }, NaN)); |
+ |
+ var arrayLikeWithTrap = { |
+ length: 0, |
+ |
+ get 0() { |
+ assertUnreachable("Getter for 0 was called"); |
+ } |
+ }; |
+ |
+ Array.prototype.includes.call(arrayLikeWithTrap); |
+ |
+ var trappedFromIndex = { |
+ valueOf: function() { |
+ assertUnreachable("Should not try to convert fromIndex to a number on a zero-length array"); |
+ } |
+ }; |
+ |
+ [].includes("a", trappedFromIndex); |
+ |
+ Array.prototype.includes.call({ |
+ length: 0 |
+ }, trappedFromIndex); |
+})(); |
+ |
+ |
+// Array.prototype.includes works on objects |
+(function() { |
+ assertFalse(["a", "b", "c"].includes({})); |
+ assertFalse([{}, {}].includes({})); |
+ var obj = {}; |
+ assertTrue([obj].includes(obj)); |
+ assertFalse([obj].includes(obj, 1)); |
+ assertTrue([obj, obj].includes(obj, 1)); |
+ |
+ var stringyObject = { |
+ toString: function() { |
+ return "a"; |
+ } |
+ }; |
+ |
+ assertFalse(["a", "b", obj].includes(stringyObject)); |
+})(); |
+ |
+ |
+// Array.prototype.includes does not see an element removed by a getter that is |
+// hit during iteration |
+(function() { |
+ var arrayLike = { |
+ length: 5, |
+ 0: "a", |
+ |
+ get 1() { |
+ delete this[2]; |
+ return "b"; |
+ }, |
+ |
+ 2: "c" |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLike, "c")); |
+})(); |
+ |
+ |
+// Array.prototype.includes should use the SameValueZero algorithm to compare |
+(function() { |
+ assertTrue([1, 2, 3].includes(2)); |
+ assertFalse([1, 2, 3].includes(4)); |
+ assertTrue([1, 2, NaN].includes(NaN)); |
+ assertTrue([1, 2, -0].includes(+0)); |
+ assertTrue([1, 2, -0].includes(-0)); |
+ assertTrue([1, 2, +0].includes(-0)); |
+ assertTrue([1, 2, +0].includes(+0)); |
+ assertFalse([1, 2, -Infinity].includes(+Infinity)); |
+ assertTrue([1, 2, -Infinity].includes(-Infinity)); |
+ assertFalse([1, 2, +Infinity].includes(-Infinity)); |
+ assertTrue([1, 2, +Infinity].includes(+Infinity)); |
+})(); |
+ |
+ |
+// Array.prototype.includes stops once it hits the length of an array-like, even |
+// if there are more after |
+(function() { |
+ var arrayLike = { |
+ length: 2, |
+ 0: "a", |
+ 1: "b", |
+ |
+ get 2() { |
+ assertUnreachable("Should not try to get the second element"); |
+ } |
+ }; |
+ |
+ assertFalse(Array.prototype.includes.call(arrayLike, "c")); |
+})(); |
+ |
+ |
+// Array.prototype.includes works on typed arrays |
+(function() { |
+ assertTrue(Array.prototype.includes.call(new Uint8Array([1, 2, 3]), 2)); |
+ |
+ assertTrue( |
+ Array.prototype.includes.call(new Float32Array([2.5, 3.14, Math.PI]), 3.1415927410125732) |
+ ); |
+ |
+ assertFalse(Array.prototype.includes.call(new Uint8Array([1, 2, 3]), 4)); |
+ assertFalse(Array.prototype.includes.call(new Uint8Array([1, 2, 3]), 2, 2)); |
+})(); |