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
|
// This test case checks that some code can be optimized for non-proxy object,
// and than we can correctly fallback if a proxy object ever flow into this
// code.
// Similar functions are part of popular framework such as React and Angular.
function shallowEqual(o1, o2) {
var k1 = Object.keys(o1);
var k2 = Object.keys(o2);
if (k1.length != k2.length) {
return false;
}
for (var k = 0; k < k1.length; k++) {
if (!Object.hasOwnProperty.call(o2, k1[k]) || !Object.is(o1[k1[k]], o2[k1[k]])) {
return false;
}
}
return true;
}
let sideEffectCounter = 0;
const payload = {x: 5, a: 1, b: 2, c: 3, d: 4};
const handler = {
ownKeys(target) {
// side-effect that should not be removed.
sideEffectCounter++;
// answer returned.
return Reflect.ownKeys(target);
},
};
const proxy = new Proxy(payload, handler);
let objs = [
{x:0, a: 1, b: 2},
{x:1, b: 1, c: 2},
{x:2, c: 1, d: 2},
{x:3, a: 1, b: 2, c: 3},
{x:4, b: 1, c: 2, d: 3},
{x:5, a: 1, b: 2, c: 3, d: 4}
];
// Ion compile shallowEqual ...
with({}) {}
let iterMax = 1000;
for (let i = 0; i < iterMax; i++) {
for (let o1 of objs) {
for (let o2 of objs) {
assertEq(shallowEqual(o1, o2), Object.is(o1, o2));
}
}
}
assertEq(sideEffectCounter, 0);
// ... before calling it with a proxy.
// This should bailout with a guard failure.
shallowEqual(objs[0], proxy);
// Assert that the proxy's ownKeys function has been called.
assertEq(sideEffectCounter, 1);
|