| Index: test/mjsunit/harmony/proxies-function.js
|
| diff --git a/test/mjsunit/harmony/proxies-function.js b/test/mjsunit/harmony/proxies-function.js
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..541bca8cc82258550e0ebb8310982357d3c35279
|
| --- /dev/null
|
| +++ b/test/mjsunit/harmony/proxies-function.js
|
| @@ -0,0 +1,382 @@
|
| +// Copyright 2011 the V8 project authors. All rights reserved.
|
| +// Redistribution and use in source and binary forms, with or without
|
| +// modification, are permitted provided that the following conditions are
|
| +// met:
|
| +//
|
| +// * Redistributions of source code must retain the above copyright
|
| +// notice, this list of conditions and the following disclaimer.
|
| +// * Redistributions in binary form must reproduce the above
|
| +// copyright notice, this list of conditions and the following
|
| +// disclaimer in the documentation and/or other materials provided
|
| +// with the distribution.
|
| +// * Neither the name of Google Inc. nor the names of its
|
| +// contributors may be used to endorse or promote products derived
|
| +// from this software without specific prior written permission.
|
| +//
|
| +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
| +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
| +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
| +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
| +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
| +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
| +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
| +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
| +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
| +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
| +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
| +
|
| +// Flags: --harmony-proxies
|
| +
|
| +
|
| +// Helper.
|
| +
|
| +function CreateFrozen(handler, callTrap, constructTrap) {
|
| + if (handler.fix === undefined) handler.fix = function() { return {} }
|
| + var f = Proxy.createFunction(handler, callTrap, constructTrap)
|
| + Object.freeze(f)
|
| + return f
|
| +}
|
| +
|
| +
|
| +// Calling (call, Function.prototype.call, Function.prototype.apply,
|
| +// Function.prototype.bind).
|
| +
|
| +var global_object = this
|
| +var receiver
|
| +
|
| +function TestCall(isStrict, callTrap) {
|
| + assertEquals(42, callTrap(5, 37))
|
| + // TODO(rossberg): unrelated bug: this does not succeed for optimized code:
|
| + // assertEquals(isStrict ? undefined : global_object, receiver)
|
| +
|
| + var f = Proxy.createFunction({}, callTrap)
|
| + receiver = 333
|
| + assertEquals(42, f(11, 31))
|
| + assertEquals(isStrict ? undefined : global_object, receiver)
|
| + var o = {}
|
| + assertEquals(42, Function.prototype.call.call(f, o, 20, 22))
|
| + assertEquals(o, receiver)
|
| + assertEquals(43, Function.prototype.call.call(f, null, 20, 23))
|
| + assertEquals(isStrict ? null : global_object, receiver)
|
| + assertEquals(44, Function.prototype.call.call(f, 2, 21, 23))
|
| + assertEquals(2, receiver.valueOf())
|
| + receiver = 333
|
| + assertEquals(32, Function.prototype.apply.call(f, o, [17, 15]))
|
| + assertEquals(o, receiver)
|
| + var ff = Function.prototype.bind.call(f, o, 12)
|
| + receiver = 333
|
| + assertEquals(42, ff(30))
|
| + assertEquals(o, receiver)
|
| + receiver = 333
|
| + assertEquals(32, Function.prototype.apply.call(ff, {}, [20]))
|
| + assertEquals(o, receiver)
|
| +
|
| + var f = CreateFrozen({}, callTrap)
|
| + receiver = 333
|
| + assertEquals(42, f(11, 31))
|
| + // TODO(rossberg): unrelated bug: this does not succeed for optimized code.
|
| + // assertEquals(isStrict ? undefined : global, receiver)
|
| + receiver = 333
|
| + assertEquals(42, Function.prototype.call.call(f, o, 20, 22))
|
| + assertEquals(o, receiver)
|
| + receiver = 333
|
| + assertEquals(32, Function.prototype.apply.call(f, o, [17, 15]))
|
| + assertEquals(o, receiver)
|
| + receiver = 333
|
| + assertEquals(42, ff(30))
|
| + assertEquals(o, receiver)
|
| + receiver = 333
|
| + assertEquals(32, Function.prototype.apply.call(ff, {}, [20]))
|
| + assertEquals(o, receiver)
|
| +}
|
| +
|
| +TestCall(false, function(x, y) {
|
| + receiver = this; return x + y
|
| +})
|
| +
|
| +TestCall(true, function(x, y) {
|
| + "use strict";
|
| + receiver = this; return x + y
|
| +})
|
| +
|
| +TestCall(false, Proxy.createFunction({}, function(x, y) {
|
| + receiver = this; return x + y
|
| +}))
|
| +
|
| +TestCall(true, Proxy.createFunction({}, function(x, y) {
|
| + "use strict";
|
| + receiver = this; return x + y
|
| +}))
|
| +
|
| +TestCall(false, CreateFrozen({}, function(x, y) {
|
| + receiver = this; return x + y
|
| +}))
|
| +
|
| +
|
| +function TestCallThrow(callTrap) {
|
| + var f = Proxy.createFunction({}, callTrap)
|
| + assertThrows(function(){ f(11) }, "myexn")
|
| + assertThrows(function(){ Function.prototype.call.call(f, {}, 2) }, "myexn")
|
| + assertThrows(function(){ Function.prototype.apply.call(f, {}, [1]) }, "myexn")
|
| +
|
| + var f = CreateFrozen({}, callTrap)
|
| + assertThrows(function(){ f(11) }, "myexn")
|
| + assertThrows(function(){ Function.prototype.call.call(f, {}, 2) }, "myexn")
|
| + assertThrows(function(){ Function.prototype.apply.call(f, {}, [1]) }, "myexn")
|
| +}
|
| +
|
| +TestCallThrow(function() { throw "myexn" })
|
| +TestCallThrow(Proxy.createFunction({}, function() { throw "myexn" }))
|
| +TestCallThrow(CreateFrozen({}, function() { throw "myexn" }))
|
| +
|
| +
|
| +
|
| +// Construction (new).
|
| +
|
| +var prototype = {}
|
| +var receiver
|
| +
|
| +var handlerWithPrototype = {
|
| + fix: function() { return {prototype: prototype} },
|
| + get: function(r, n) { assertEquals("prototype", n); return prototype }
|
| +}
|
| +
|
| +var handlerSansPrototype = {
|
| + fix: function() { return {} },
|
| + get: function(r, n) { assertEquals("prototype", n); return undefined }
|
| +}
|
| +
|
| +function ReturnUndef(x, y) { "use strict"; receiver = this; this.sum = x + y }
|
| +function ReturnThis(x, y) { "use strict"; receiver = this; this.sum = x + y; return this }
|
| +function ReturnNew(x, y) { "use strict"; receiver = this; return {sum: x + y} }
|
| +function ReturnNewWithProto(x, y) {
|
| + "use strict";
|
| + receiver = this;
|
| + var result = Object.create(prototype)
|
| + result.sum = x + y
|
| + return result
|
| +}
|
| +
|
| +function TestConstruct(proto, constructTrap) {
|
| + TestConstruct2(proto, constructTrap, handlerWithPrototype)
|
| + TestConstruct2(proto, constructTrap, handlerSansPrototype)
|
| +}
|
| +
|
| +function TestConstruct2(proto, constructTrap, handler) {
|
| + var f = Proxy.createFunction(handler, function() {}, constructTrap)
|
| + var o = new f(11, 31)
|
| + // TODO(rossberg): doesn't hold, due to unrelated bug.
|
| + // assertEquals(undefined, receiver)
|
| + assertEquals(42, o.sum)
|
| + assertSame(proto, Object.getPrototypeOf(o))
|
| +
|
| + var f = CreateFrozen(handler, function() {}, constructTrap)
|
| + var o = new f(11, 32)
|
| + // TODO(rossberg): doesn't hold, due to unrelated bug.
|
| + // assertEquals(undefined, receiver)
|
| + assertEquals(43, o.sum)
|
| + assertSame(proto, Object.getPrototypeOf(o))
|
| +}
|
| +
|
| +TestConstruct(Object.prototype, ReturnNew)
|
| +TestConstruct(prototype, ReturnNewWithProto)
|
| +
|
| +TestConstruct(Object.prototype, Proxy.createFunction({}, ReturnNew))
|
| +TestConstruct(prototype, Proxy.createFunction({}, ReturnNewWithProto))
|
| +
|
| +TestConstruct(Object.prototype, CreateFrozen({}, ReturnNew))
|
| +TestConstruct(prototype, CreateFrozen({}, ReturnNewWithProto))
|
| +
|
| +
|
| +function TestConstructFromCall(proto, returnsThis, callTrap) {
|
| + TestConstructFromCall2(proto, returnsThis, callTrap, handlerWithPrototype)
|
| + TestConstructFromCall2(proto, returnsThis, callTrap, handlerSansPrototype)
|
| +}
|
| +
|
| +function TestConstructFromCall2(proto, returnsThis, callTrap, handler) {
|
| + var f = Proxy.createFunction(handler, callTrap)
|
| + var o = new f(11, 31)
|
| + if (returnsThis) assertEquals(o, receiver)
|
| + assertEquals(42, o.sum)
|
| + assertSame(proto, Object.getPrototypeOf(o))
|
| +
|
| + var f = CreateFrozen(handler, callTrap)
|
| + var o = new f(11, 32)
|
| + if (returnsThis) assertEquals(o, receiver)
|
| + assertEquals(43, o.sum)
|
| + assertSame(proto, Object.getPrototypeOf(o))
|
| +}
|
| +
|
| +TestConstructFromCall(Object.prototype, true, ReturnUndef)
|
| +TestConstructFromCall(Object.prototype, true, ReturnThis)
|
| +TestConstructFromCall(Object.prototype, false, ReturnNew)
|
| +TestConstructFromCall(prototype, false, ReturnNewWithProto)
|
| +
|
| +TestConstructFromCall(Object.prototype, true, Proxy.createFunction({}, ReturnUndef))
|
| +TestConstructFromCall(Object.prototype, true, Proxy.createFunction({}, ReturnThis))
|
| +TestConstructFromCall(Object.prototype, false, Proxy.createFunction({}, ReturnNew))
|
| +TestConstructFromCall(prototype, false, Proxy.createFunction({}, ReturnNewWithProto))
|
| +
|
| +TestConstructFromCall(Object.prototype, true, CreateFrozen({}, ReturnUndef))
|
| +TestConstructFromCall(Object.prototype, true, CreateFrozen({}, ReturnThis))
|
| +TestConstructFromCall(Object.prototype, false, CreateFrozen({}, ReturnNew))
|
| +TestConstructFromCall(prototype, false, CreateFrozen({}, ReturnNewWithProto))
|
| +
|
| +ReturnUndef.prototype = prototype
|
| +ReturnThis.prototype = prototype
|
| +ReturnNew.prototype = prototype
|
| +ReturnNewWithProto.prototype = prototype
|
| +
|
| +TestConstructFromCall(prototype, true, ReturnUndef)
|
| +TestConstructFromCall(prototype, true, ReturnThis)
|
| +TestConstructFromCall(Object.prototype, false, ReturnNew)
|
| +TestConstructFromCall(prototype, false, ReturnNewWithProto)
|
| +
|
| +TestConstructFromCall(Object.prototype, true, Proxy.createFunction({}, ReturnUndef))
|
| +TestConstructFromCall(Object.prototype, true, Proxy.createFunction({}, ReturnThis))
|
| +TestConstructFromCall(Object.prototype, false, Proxy.createFunction({}, ReturnNew))
|
| +TestConstructFromCall(prototype, false, Proxy.createFunction({}, ReturnNewWithProto))
|
| +
|
| +TestConstructFromCall(prototype, true, Proxy.createFunction(handlerWithPrototype, ReturnUndef))
|
| +TestConstructFromCall(prototype, true, Proxy.createFunction(handlerWithPrototype, ReturnThis))
|
| +TestConstructFromCall(Object.prototype, false, Proxy.createFunction(handlerWithPrototype, ReturnNew))
|
| +TestConstructFromCall(prototype, false, Proxy.createFunction(handlerWithPrototype, ReturnNewWithProto))
|
| +
|
| +TestConstructFromCall(prototype, true, CreateFrozen(handlerWithPrototype, ReturnUndef))
|
| +TestConstructFromCall(prototype, true, CreateFrozen(handlerWithPrototype, ReturnThis))
|
| +TestConstructFromCall(Object.prototype, false, CreateFrozen(handlerWithPrototype, ReturnNew))
|
| +TestConstructFromCall(prototype, false, CreateFrozen(handlerWithPrototype, ReturnNewWithProto))
|
| +
|
| +
|
| +function TestConstructThrow(trap) {
|
| + TestConstructThrow2(Proxy.createFunction({fix: function() {return {}}}, trap))
|
| + TestConstructThrow2(Proxy.createFunction({fix: function() {return {}}},
|
| + function() {}, trap))
|
| +}
|
| +
|
| +function TestConstructThrow2(f) {
|
| + assertThrows(function(){ new f(11) }, "myexn")
|
| + Object.freeze(f)
|
| + assertThrows(function(){ new f(11) }, "myexn")
|
| +}
|
| +
|
| +TestConstructThrow(function() { throw "myexn" })
|
| +TestConstructThrow(Proxy.createFunction({}, function() { throw "myexn" }))
|
| +TestConstructThrow(CreateFrozen({}, function() { throw "myexn" }))
|
| +
|
| +
|
| +
|
| +// Getters and setters.
|
| +
|
| +var value
|
| +var receiver
|
| +
|
| +function TestAccessorCall(getterCallTrap, setterCallTrap) {
|
| + var handler = {fix: function() { return {} }}
|
| + var pgetter = Proxy.createFunction(handler, getterCallTrap)
|
| + var psetter = Proxy.createFunction(handler, setterCallTrap)
|
| +
|
| + var o = {}
|
| + var oo = Object.create(o)
|
| + Object.defineProperty(o, "a", {get: pgetter, set: psetter})
|
| + Object.defineProperty(o, "b", {get: pgetter})
|
| + Object.defineProperty(o, "c", {set: psetter})
|
| + Object.defineProperty(o, "3", {get: pgetter, set: psetter})
|
| + Object.defineProperty(oo, "a", {value: 43})
|
| +
|
| + receiver = ""
|
| + assertEquals(42, o.a)
|
| + assertSame(o, receiver)
|
| + receiver = ""
|
| + assertEquals(42, o.b)
|
| + assertSame(o, receiver)
|
| + receiver = ""
|
| + assertEquals(undefined, o.c)
|
| + assertEquals("", receiver)
|
| + receiver = ""
|
| + assertEquals(42, o["a"])
|
| + assertSame(o, receiver)
|
| + receiver = ""
|
| + assertEquals(42, o[3])
|
| + assertSame(o, receiver)
|
| +
|
| + receiver = ""
|
| + assertEquals(43, oo.a)
|
| + assertEquals("", receiver)
|
| + receiver = ""
|
| + assertEquals(42, oo.b)
|
| + assertSame(o, receiver)
|
| + receiver = ""
|
| + assertEquals(undefined, oo.c)
|
| + assertEquals("", receiver)
|
| + receiver = ""
|
| + assertEquals(43, oo["a"])
|
| + assertEquals("", receiver)
|
| + receiver = ""
|
| + assertEquals(42, oo[3])
|
| + assertSame(o, receiver)
|
| +
|
| + receiver = ""
|
| + assertEquals(50, o.a = 50)
|
| + assertSame(o, receiver)
|
| + assertEquals(50, value)
|
| + receiver = ""
|
| + assertEquals(51, o.b = 51)
|
| + assertEquals("", receiver)
|
| + assertEquals(50, value) // no setter
|
| + assertThrows(function() { "use strict"; o.b = 51 }, TypeError)
|
| + receiver = ""
|
| + assertEquals(52, o.c = 52)
|
| + assertSame(o, receiver)
|
| + assertEquals(52, value)
|
| + receiver = ""
|
| + assertEquals(53, o["a"] = 53)
|
| + assertSame(o, receiver)
|
| + assertEquals(53, value)
|
| + receiver = ""
|
| + assertEquals(54, o[3] = 54)
|
| + assertSame(o, receiver)
|
| + assertEquals(54, value)
|
| +
|
| + value = 0
|
| + receiver = ""
|
| + assertEquals(60, oo.a = 60)
|
| + assertEquals("", receiver)
|
| + assertEquals(0, value) // oo has own 'a'
|
| + assertEquals(61, oo.b = 61)
|
| + assertSame("", receiver)
|
| + assertEquals(0, value) // no setter
|
| + assertThrows(function() { "use strict"; oo.b = 61 }, TypeError)
|
| + receiver = ""
|
| + assertEquals(62, oo.c = 62)
|
| + assertSame(oo, receiver)
|
| + assertEquals(62, value)
|
| + receiver = ""
|
| + assertEquals(63, oo["c"] = 63)
|
| + assertSame(oo, receiver)
|
| + assertEquals(63, value)
|
| + receiver = ""
|
| + assertEquals(64, oo[3] = 64)
|
| + assertSame(oo, receiver)
|
| + assertEquals(64, value)
|
| +}
|
| +
|
| +TestAccessorCall(
|
| + function() { receiver = this; return 42 },
|
| + function(x) { receiver = this; value = x }
|
| +)
|
| +
|
| +TestAccessorCall(
|
| + function() { "use strict"; receiver = this; return 42 },
|
| + function(x) { "use strict"; receiver = this; value = x }
|
| +)
|
| +
|
| +TestAccessorCall(
|
| + Proxy.createFunction({}, function() { receiver = this; return 42 }),
|
| + Proxy.createFunction({}, function(x) { receiver = this; value = x })
|
| +)
|
| +
|
| +TestAccessorCall(
|
| + CreateFrozen({}, function() { receiver = this; return 42 }),
|
| + CreateFrozen({}, function(x) { receiver = this; value = x })
|
| +)
|
|
|