blob: f2cbd7346f7f465a4293fef0aecfe3e68aaa2128 [file] [log] [blame]
//-------------------------------------------------------------------------------------------------------
// Copyright (C) Microsoft. All rights reserved.
// Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
//-------------------------------------------------------------------------------------------------------
// Tests for...in behavior when child object shadows a prototype property with a non-enumerable shadow
// See OS bug #850013
if (this.WScript && this.WScript.LoadScriptFile) { // Check for running in ch
this.WScript.LoadScriptFile("..\\UnitTestFramework\\UnitTestFramework.js");
}
function forInKeysToArray(obj) {
var s = [];
for (key in obj) {
s.push(key);
}
return s;
}
var tests = [
{
name: "Simple test of prototype property shadowed by non-enumerable property",
body: function () {
var proto = { x: 1 };
var child = Object.create(proto, { x: { value: 2, enumerable: false} });
var result = forInKeysToArray(child);
assert.areEqual([], result, "for...in does not enumerate a key which is enumerable in a prototype but shadowed by a non-enumerable property");
}
},
{
name: "Multiple properties on an object with some prototype properties shadowed by non-enumerable versions",
body: function () {
var proto = { a: 1, b: 2, c: 3, d: 4, e: 5 };
var child = Object.create(proto, { b: { value: 20, enumerable: false} });
Object.defineProperty(child, 'c', { enumerable: false, value: 30 });
child['d'] = 4;
var result = forInKeysToArray(child);
assert.areEqual(['d','a','e'], result, "for...in does not enumerate a key which is enumerable in a prototype but shadowed by a non-enumerable property");
}
},
{
name: "Array indices which are non-enumerable (force ES5Array object)",
body: function () {
var o = [0,1,2];
o[4] = 4;
Object.defineProperty(o, 3, { enumerable: false, value: '3' })
var result = forInKeysToArray(o);
assert.areEqual(['0','1','2','4'], result, "for...in does not enumerate non-enumerable properties, even for array indices");
}
},
{
name: "Explicitly test for...in fast path",
body: function () {
function test(obj, expected) {
var result = forInKeysToArray(obj);
result = result.concat(forInKeysToArray(obj));
result = result.concat(forInKeysToArray(obj));
assert.areEqual(expected, result, "for...in does not enumerate non-enumerable properties, even from the fast-path");
}
var o = Object.create(null);
Object.defineProperty(o, 'a', { value: 1, enumerable: false });
Object.defineProperty(o, 'b', { value: 2, enumerable: false });
Object.defineProperty(o, 'c', { value: 3, enumerable: false });
test(o, []);
var o = Object.create(null);
Object.defineProperty(o, 'a', { value: 1, enumerable: true });
Object.defineProperty(o, 'b', { value: 2, enumerable: false });
Object.defineProperty(o, 'c', { value: 3, enumerable: false });
test(o, ['a','a','a']);
var o = Object.create(null);
Object.defineProperty(o, 'a', { value: 1, enumerable: false });
Object.defineProperty(o, 'b', { value: 2, enumerable: false });
Object.defineProperty(o, 'c', { value: 3, enumerable: true });
test(o, ['c','c','c']);
var o = Object.create(null);
Object.defineProperty(o, 'a', { value: 1, enumerable: false });
Object.defineProperty(o, 'b', { value: 2, enumerable: true });
Object.defineProperty(o, 'c', { value: 3, enumerable: false });
test(o, ['b','b','b']);
var o = Object.create(null);
Object.defineProperty(o, 'a', { value: 1, enumerable: true });
Object.defineProperty(o, 'b', { value: 2, enumerable: false });
Object.defineProperty(o, 'c', { value: 3, enumerable: true });
test(o, ['a','c','a','c','a','c']);
// JSON is a delay-load object
test(JSON, []);
}
},
{
name: "Shadowing non-enumerable prototype property with an enumerable version",
body: function () {
var proto = Object.create(null, { x: { value: 1, enumerable: false} });
var child = Object.create(proto, { x: { value: 2, enumerable: true} });
var result = forInKeysToArray(child);
assert.areEqual(['x'], result, "Child property shadows proto property");
}
},
{
name: "Shadowing non-enumerable prototype property with another non-enumerable version",
body: function () {
var proto = Object.create(null, { x: { value: 1, enumerable: false} });
var child = Object.create(proto, { x: { value: 2, enumerable: false} });
var result = forInKeysToArray(child);
assert.areEqual([], result, "Child property shadows proto property with another non-enumerable property");
}
},
{
name: "Enumerating RegExp constructor is a bit of a special case",
body: function() {
var result = forInKeysToArray(RegExp);
assert.areEqual(['$1','$2','$3','$4','$5','$6','$7','$8','$9','input','rightContext','leftContext','lastParen','lastMatch'], result, "for..in of RegExp constructor returns some special properties");
var result = Object.keys(RegExp);
assert.areEqual(['$1','$2','$3','$4','$5','$6','$7','$8','$9','input','rightContext','leftContext','lastParen','lastMatch'], result, "Object.keys returns the same set of properties for RegExp as for..in");
var result = Object.getOwnPropertyNames(RegExp);
assert.areEqual(['$1','$2','$3','$4','$5','$6','$7','$8','$9','input','rightContext','leftContext','lastParen','lastMatch','length','prototype','name','$_','$&','$+','$`',"$'",'index'], result, "Object.getOwnPropertyNames returns special non-enumerable properties too");
}
},
{
name: "Multiple objects in prototype chain with enum and non-enum property shadowing",
body: function() {
var proto = Object.create(null, {
a: { value: 1, enumerable: false},
b: { value: 1, enumerable: true},
c: { value: 1, enumerable: false},
d: { value: 1, enumerable: false},
w: { value: 1, enumerable: true},
x: { value: 1, enumerable: false},
y: { value: 1, enumerable: true},
z: { value: 1, enumerable: true},
});
var child = Object.create(proto, {
a: { value: 2, enumerable: false},
b: { value: 2, enumerable: false},
c: { value: 2, enumerable: true},
d: { value: 2, enumerable: false},
w: { value: 2, enumerable: true},
x: { value: 2, enumerable: true},
y: { value: 2, enumerable: false},
z: { value: 2, enumerable: true},
});
var childchild = Object.create(child, {
a: { value: 3, enumerable: false},
b: { value: 3, enumerable: false},
c: { value: 3, enumerable: false},
d: { value: 3, enumerable: true},
w: { value: 3, enumerable: false},
x: { value: 3, enumerable: true},
y: { value: 3, enumerable: true},
z: { value: 3, enumerable: true},
});
var result = forInKeysToArray(childchild);
assert.areEqual(['d','x','y','z'], result, "childchild should shadow all properties and disable enumerable properties from the prototype chain leaking out");
var result = forInKeysToArray(child);
assert.areEqual(['c','w','x','z'], result, "child should shadow all properties and disable enumerable properties from the prototype chain leaking out");
var result = forInKeysToArray(proto);
assert.areEqual(['b','w','y','z'], result, "proto doesn't shadow any properties but non-enumerable properties should not show up in for..in loop");
}
},
{
name: "OS: 1905906 - Enumerating a type and alternating non-enumerable properties causes assert",
body: function() {
function foo() { JSON.stringify(arguments); }
foo();
var arr = [];
function foo2() {
for(var i in arguments) {
arr.push(i);
}
}
foo2('a','b');
assert.areEqual(['0','1'], arr, "Correct values are enumerated via for...in loop");
}
},
];
testRunner.runTests(tests, { verbose: WScript.Arguments[0] != "summary" });