blob: f52ae1aa7e4258799fb368b2f82c0b92badcf0a4 [file] [log] [blame]
// 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: --allow-natives-syntax
(function testBasicFunctionality() {
var target = {
target_one: 1,
property: "value"
};
var handler = {handler:1};
var proxy = new Proxy(target, handler);
assertEquals("value", proxy.property);
assertEquals(undefined, proxy.nothing);
assertEquals(undefined, proxy.handler);
handler.get = function() { return "value 2" };
assertEquals("value 2", proxy.property);
assertEquals("value 2", proxy.nothing);
assertEquals("value 2", proxy.handler);
var handler2 = new Proxy({get: function() { return "value 3" }},{});
var proxy2 = new Proxy(target, handler2);
assertEquals("value 3", proxy2.property);
assertEquals("value 3", proxy2.nothing);
assertEquals("value 3", proxy2.handler);
})();
(function testThrowOnGettingTrap() {
var handler = new Proxy({}, {get: function(){ throw Error() }});
var proxy = new Proxy({}, handler);
assertThrows("proxy.property", Error);
})();
(function testThrowOnTrapNotCallable() {
var handler = new Proxy({}, {get: 'not_callable' });
var proxy = new Proxy({}, handler);
assertThrows("proxy.property", Error);
})();
(function testFallback() {
var target = {property:"value"};
var proxy = new Proxy(target, {});
assertEquals("value", proxy.property);
assertEquals(undefined, proxy.property2);
})();
(function testFallbackUndefinedTrap() {
var handler = new Proxy({}, {get: function(){ return undefined }});
var target = {property:"value"};
var proxy = new Proxy(target, handler);
assertEquals("value", proxy.property);
assertEquals(undefined, proxy.property2);
})();
(function testFailingInvariant() {
var target = {};
var handler = { get: function(r, p){ if (p != "key4") return "value" }}
var proxy = new Proxy(target, handler);
assertEquals("value", proxy.property);
assertEquals("value", proxy.key);
assertEquals("value", proxy.key2);
assertEquals("value", proxy.key3);
// Define a non-configurable, non-writeable property on the target for
// which the handler will return a different value.
Object.defineProperty(target, "key", {
configurable: false,
writable: false,
value: "different value"
});
assertEquals("value", proxy.property);
assertThrows(function(){ proxy.key }, TypeError);
assertEquals("value", proxy.key2);
assertEquals("value", proxy.key3);
// Define a non-configurable getter on the target for which the handler
// will return a value, according to the spec we do not throw.
Object.defineProperty(target, "key2", {
configurable: false,
get: function() { return "different value" }
});
assertEquals("value", proxy.property);
assertThrows(function(){ proxy.key }, TypeError);
assertEquals("value", proxy.key2);
assertEquals("value", proxy.key3);
// Define a non-configurable setter without a corresponding getter on the
// target for which the handler will return a value.
Object.defineProperty(target, "key3", {
configurable: false,
set: function() { }
});
assertEquals("value", proxy.property);
assertThrows(function(){ proxy.key }, TypeError);
assertEquals("value", proxy.key2);
assertThrows(function(){ proxy.key3 }, TypeError);
// Define a non-configurable setter without a corresponding getter on the
// target for which the handler will return undefined.
Object.defineProperty(target, "key4", {
configurable: false,
set: function() { }
});
assertSame(undefined, proxy.key4);
})();
(function testGetInternalIterators() {
var log = [];
var array = [1,2,3,4,5]
var origIt = array[Symbol.iterator]();
var it = new Proxy(origIt, {
get(t, name) {
log.push(`[[Get]](iterator, ${String(name)})`);
return Reflect.get(t, name);
},
set(t, name, val) {
log.push(`[[Set]](iterator, ${String(name)}, ${String(val)})`);
return Reflect.set(t, name, val);
}
});
assertThrows(function() {
for (var v of it) log.push(v);
}, TypeError);
assertEquals([
"[[Get]](iterator, Symbol(Symbol.iterator))",
"[[Get]](iterator, next)"
], log);
})();
(function testGetterWithSideEffect() {
var obj = {
key: 0
}
assertEquals(obj.key, 0);
var p = new Proxy(obj, {});
var q = new Proxy(p, {
get(target, name) {
if (name != 'key') return Reflect.get(target, name);
target.key++;
return target.key;
}
});
assertEquals(0, p.key);
// Assert the trap is not called twice.
assertEquals(1, q.key);
})();
(function testReceiverWithTrap() {
var obj = {};
var p = new Proxy(obj, {
get(target, name, receiver) {
if (name != 'key') return Reflect.get(target, name);
assertSame(target, obj);
assertSame(receiver, p);
return 42;
}
});
assertEquals(42, p.key);
})();
(function testReceiverWithoutTrap() {
var obj = {
get prop() {
assertSame(this, p);
return 42;
}
};
var p = new Proxy(obj, {});
assertEquals(42, p.prop);
})();
(function testGetPropertyDetailsBailout() {
var obj = {
}
var p = new Proxy(obj, {
getOwnPropertyDescriptor() {
throw new Error('Error from proxy getOwnPropertyDescriptor trap');
}
});
var q = new Proxy(p, {
get(target, name) {
return 42;
}
});
assertThrows(function(){ q.prop }, Error,
'Error from proxy getOwnPropertyDescriptor trap');
})();
(function testGetPropertyDetailsBailout2() {
var obj = {};
Object.defineProperty(obj, 'prop', {
value: 53,
configurable: false
});
var p = new Proxy(obj, {});
var q = new Proxy(p, {
get(target, name) {
return 42;
}
});
assertThrows(function(){ q.prop }, TypeError,
"'get' on proxy: property 'prop' is a read-only and non-configurable data" +
" property on the proxy target but the proxy did not return its actual" +
" value (expected '53' but got '42')");
})();
(function test32BitIndex() {
var index = (1 << 31) + 1;
var obj = {};
obj[index] = 42;
var p = new Proxy(obj, {});
for (var i = 0; i < 3; ++i) {
assertEquals(42, p[index]);
}
})();