1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
// Make sure we get the proper side effects.
// |delete super.prop| and |delete super[expr]| throw universally.
class base {
constructor() { }
}
class derived extends base {
constructor() { super(); }
testDeleteProp() { delete super.prop; }
testDeleteElem() {
let sideEffect = 0;
assertThrowsInstanceOf(() => delete super[sideEffect = 1], ReferenceError);
assertEq(sideEffect, 1);
}
testDeleteElemPropValFirst() {
// The deletion error is a reference error, but by munging the prototype
// chain, we can force a typeerror from JSOP_SUPERBASE
delete super[Object.setPrototypeOf(derived.prototype, null)];
}
}
var d = new derived();
assertThrowsInstanceOf(() => d.testDeleteProp(), ReferenceError);
d.testDeleteElem();
assertThrowsInstanceOf(() => d.testDeleteElemPropValFirst(), TypeError);
// |delete super.x| does not delete anything before throwing.
var thing1 = {
go() { delete super.toString; }
};
let saved = Object.prototype.toString;
assertThrowsInstanceOf(() => thing1.go(), ReferenceError);
assertEq(Object.prototype.toString, saved);
// |delete super.x| does not tell the prototype to delete anything, when it's a proxy.
var thing2 = {
go() { delete super.prop; }
};
Object.setPrototypeOf(thing2, new Proxy({}, {
deleteProperty(x) { throw "FAIL"; }
}));
assertThrowsInstanceOf(() => thing2.go(), ReferenceError);
class derivedTestDeleteProp extends base {
constructor() {
// The deletion error is a reference error, even after munging the prototype
// chain.
Object.setPrototypeOf(derivedTestDeleteProp.prototype, null);
assertThrowsInstanceOf(() => delete super.prop, ReferenceError);
super();
assertThrowsInstanceOf(() => delete super.prop, ReferenceError);
return {};
}
}
new derivedTestDeleteProp();
if (typeof reportCompare === 'function')
reportCompare(0,0,"OK");
|