| // 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]); |
| } |
| })(); |