86 lines
4.7 KiB
JavaScript
86 lines
4.7 KiB
JavaScript
|
// Copyright 2013 the V8 project authors. All rights reserved.
|
||
|
// Copyright (C) 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved.
|
||
|
//
|
||
|
// Redistribution and use in source and binary forms, with or without
|
||
|
// modification, are permitted provided that the following conditions
|
||
|
// are met:
|
||
|
// 1. Redistributions of source code must retain the above copyright
|
||
|
// notice, this list of conditions and the following disclaimer.
|
||
|
// 2. 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.
|
||
|
//
|
||
|
// THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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.
|
||
|
|
||
|
description(
|
||
|
'This tests for caller property in functions. Only functions that are called from inside of other functions and have a parent should have this property set. Tests return true when caller is found and false when the caller is null.'
|
||
|
)
|
||
|
function child()
|
||
|
{
|
||
|
return (child.caller !== null);
|
||
|
}
|
||
|
|
||
|
function parent()
|
||
|
{
|
||
|
return child();
|
||
|
}
|
||
|
|
||
|
var childHasCallerWhenExecutingGlobalCode = (child.caller !== null);
|
||
|
var childHasCallerWhenCalledWithoutParent = child();
|
||
|
var childHasCallerWhenCalledFromWithinParent = parent();
|
||
|
|
||
|
shouldBe('childHasCallerWhenExecutingGlobalCode', 'false');
|
||
|
shouldBe('childHasCallerWhenCalledWithoutParent', 'false');
|
||
|
shouldBe('childHasCallerWhenCalledFromWithinParent', 'true')
|
||
|
|
||
|
// The caller property should throw in strict mode, and a non-strict function cannot use caller to reach a strict caller (see ES5.1 15.3.5.4).
|
||
|
function nonStrictCallee() { return nonStrictCallee.caller; }
|
||
|
function strictCallee() { "use strict"; return strictCallee.caller; }
|
||
|
function nonStrictCaller(x) { return x(); }
|
||
|
function strictCaller(x) { "use strict"; return x(); }
|
||
|
shouldBe("nonStrictCaller(nonStrictCallee)", "nonStrictCaller");
|
||
|
shouldThrow("nonStrictCaller(strictCallee)", '"TypeError: Type error"');
|
||
|
shouldThrow("strictCaller(nonStrictCallee)", '"TypeError: Function.caller used to retrieve strict caller"');
|
||
|
shouldThrow("strictCaller(strictCallee)", '"TypeError: Type error"');
|
||
|
|
||
|
// .caller within a bound function reaches the caller, ignoring the binding.
|
||
|
var boundNonStrictCallee = nonStrictCallee.bind();
|
||
|
var boundStrictCallee = strictCallee.bind();
|
||
|
shouldBe("nonStrictCaller(boundNonStrictCallee)", "nonStrictCaller");
|
||
|
shouldThrow("nonStrictCaller(boundStrictCallee)", '"TypeError: Type error"');
|
||
|
shouldThrow("strictCaller(boundNonStrictCallee)", '"TypeError: Function.caller used to retrieve strict caller"');
|
||
|
shouldThrow("strictCaller(boundStrictCallee)", '"TypeError: Type error"');
|
||
|
|
||
|
// Check that .caller works (or throws) as expected, over an accessor call.
|
||
|
function getFooGetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').get; }
|
||
|
function getFooSetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').set; }
|
||
|
var nonStrictAccessor = {
|
||
|
get foo() { return getFooGetter(nonStrictAccessor).caller; },
|
||
|
set foo(x) { if (getFooSetter(nonStrictAccessor).caller !==x) throw false; }
|
||
|
};
|
||
|
var strictAccessor = {
|
||
|
get foo() { "use strict"; return getFooGetter(strictAccessor).caller; },
|
||
|
set foo(x) { "use strict"; if (getFooSetter(strictAccessor).caller !==x) throw false; }
|
||
|
};
|
||
|
function nonStrictGetter(x) { return x.foo; }
|
||
|
function nonStrictSetter(x) { x.foo = nonStrictSetter; return true; }
|
||
|
function strictGetter(x) { "use strict"; return x.foo; }
|
||
|
function strictSetter(x) { "use strict"; x.foo = nonStrictSetter; return true; }
|
||
|
shouldBe("nonStrictGetter(nonStrictAccessor)", "nonStrictGetter");
|
||
|
shouldBeTrue("nonStrictSetter(nonStrictAccessor)");
|
||
|
shouldThrow("nonStrictGetter(strictAccessor)", '"TypeError: Type error"');
|
||
|
shouldThrow("nonStrictSetter(strictAccessor)", '"TypeError: Type error"');
|
||
|
shouldThrow("strictGetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"');
|
||
|
shouldThrow("strictSetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"');
|
||
|
shouldThrow("strictGetter(strictAccessor)", '"TypeError: Type error"');
|
||
|
shouldThrow("strictSetter(strictAccessor)", '"TypeError: Type error"');
|