// Copyright 2015 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-proxies --harmony-reflect
var properties =
["bla", "0", 1, Symbol(), {[Symbol.toPrimitive]() {return "a"}}];
function TestForwarding(handler, myDelete, shouldThrow) {
var target = {};
var proxy = new Proxy(target, handler);
assertFalse(target.hasOwnProperty("doesnotexist"));
assertTrue(myDelete(proxy, "doesnotexist"));
for (p of properties) {
target[p] = 42;
assertTrue(myDelete(proxy, p));
assertFalse(target.hasOwnProperty(p));
}
for (p of properties) {
Object.defineProperty(target, p, {value: 42, configurable: false});
if (shouldThrow) {
assertThrows(() => myDelete(proxy, p), TypeError);
} else {
assertFalse(myDelete(proxy, p));
}
assertTrue(target.hasOwnProperty(p));
}
};
(function () {
// No trap.
var handler = {};
TestForwarding(handler,
(o, p) => delete o[p], false);
TestForwarding(handler,
(o, p) => Reflect.deleteProperty(o, p), false);
TestForwarding(handler,
(o, p) => {"use strict"; return delete o[p]}, true);
TestForwarding(handler,
(o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);
})();
(function () {
// "Undefined" trap.
var handler = { deleteProperty: null };
TestForwarding(handler,
(o, p) => delete o[p], false);
TestForwarding(handler,
(o, p) => Reflect.deleteProperty(o, p), false);
TestForwarding(handler,
(o, p) => {"use strict"; return delete o[p]}, true);
TestForwarding(handler,
(o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);
})();
(function () {
// Invalid trap.
var target = {};
var handler = { deleteProperty: true };
var proxy = new Proxy(target, handler);
assertThrows(() => delete proxy[0], TypeError);
assertThrows(() => Reflect.deleteProperty(proxy, 0), TypeError);
})();
function TestTrappingTrueish(myDelete) {
var handler = { deleteProperty() {return 42} };
var target = {};
var proxy = new Proxy(target, handler);
// Trap returns trueish and target doesn't own property.
for (p of properties) {
assertTrue(myDelete(proxy, p));
}
// Trap returns trueish and target property is configurable.
for (p of properties) {
target[p] = 42;
assertTrue(myDelete(proxy, p));
}
// Trap returns trueish but target property is not configurable.
for (p of properties) {
Object.defineProperty(target, p, {value: 42, configurable: false});
assertThrows(() => myDelete(proxy, p), TypeError);
}
};
TestTrappingTrueish(
(o, p) => delete o[p]);
TestTrappingTrueish(
(o, p) => Reflect.deleteProperty(o, p));
TestTrappingTrueish(
(o, p) => {"use strict"; return delete o[p]});
TestTrappingTrueish(
(o, p) => {"use strict"; return Reflect.deleteProperty(o, p)});
function TestTrappingTrueish2(myDelete) {
var handler = {
deleteProperty(target, p) {
Object.defineProperty(target, p, {configurable: false});
return 42
}
};
var target = {};
var proxy = new Proxy(target, handler);
// Trap returns trueish but target property is not configurable. In contrast
// to above, here the target property was configurable before the trap call.
for (p of properties) {
target[p] = 42;
assertThrows(() => myDelete(proxy, p), TypeError);
}
};
TestTrappingTrueish2(
(o, p) => delete o[p]);
TestTrappingTrueish2(
(o, p) => Reflect.deleteProperty(o, p));
TestTrappingTrueish2(
(o, p) => {"use strict"; return delete o[p]});
TestTrappingTrueish2(
(o, p) => {"use strict"; return Reflect.deleteProperty(o, p)});
function TestTrappingFalsish(myDelete, shouldThrow) {
var handler = { deleteProperty() {return ""} };
var target = {};
var proxy = new Proxy(target, handler);
var properties =
["bla", "0", 1, Symbol(), {[Symbol.toPrimitive]() {return "a"}}];
// Trap returns falsish and target doesn't own property.
for (p of properties) {
if (shouldThrow) {
assertThrows(() => myDelete(proxy, p), TypeError);
} else {
assertFalse(myDelete(proxy, p));
}
}
// Trap returns falsish and target property is configurable.
for (p of properties) {
target[p] = 42;
if (shouldThrow) {
assertThrows(() => myDelete(proxy, p), TypeError);
} else {
assertFalse(myDelete(proxy, p));
}
}
// Trap returns falsish and target property is not configurable.
for (p of properties) {
Object.defineProperty(target, p, {value: 42, configurable: false});
if (shouldThrow) {
assertThrows(() => myDelete(proxy, p), TypeError);
} else {
assertFalse(myDelete(proxy, p));
}
}
};
TestTrappingFalsish(
(o, p) => delete o[p], false);
TestTrappingFalsish(
(o, p) => Reflect.deleteProperty(o, p), false);
TestTrappingFalsish(
(o, p) => {"use strict"; return delete o[p]}, true);
TestTrappingFalsish(
(o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);