| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610 |
- //-------------------------------------------------------------------------------------------------------
- // Copyright (C) Microsoft. All rights reserved.
- // Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
- //-------------------------------------------------------------------------------------------------------
- WScript.LoadScriptFile("..\\UnitTestFramework\\UnitTestFramework.js");
- var tests = [
- {
- name: "Default argument parsing",
- body: function () {
- // Incomplete expressions
- assert.throws(function () { eval("function foo(a =) { return a; }"); }, SyntaxError, "Incomplete default expression throws in a function", "Syntax error");
- assert.throws(function () { eval("var x = function(a =) { return a; }"); }, SyntaxError, "Incomplete default expression throws in a function expression", "Syntax error");
- assert.throws(function () { eval("(a =) => a"); }, SyntaxError, "Incomplete default expression throws in a lambda", "Syntax error");
- assert.throws(function () { eval("var x = { foo(a =) { return a; } }"); }, SyntaxError, "Incomplete default expression throws in an object method", "Syntax error");
- assert.throws(function () { eval("var x = class { foo(a =) { return a; } }"); }, SyntaxError, "Incomplete default expression throws in a class method", "Syntax error");
- assert.throws(function () { eval("var x = { foo: function (a =) { return a; } }"); }, SyntaxError, "Incomplete default expression throws in an object member function", "Syntax error");
- assert.throws(function () { eval("function * foo(a =) { return a; }"); }, SyntaxError, "Incomplete default expression throws in a generator function", "Syntax error");
- assert.throws(function () { eval("var x = function*(a =) { return a; }"); }, SyntaxError, "Incomplete default expression throws in a generator function", "Syntax error");
- assert.throws(function () { eval("var x = class { * foo(a =) { return a; } }"); }, SyntaxError, "Incomplete default expression throws in a class generator method", "Syntax error");
- // Duplicate parameters
- assert.throws(function () { eval("function f(a, b, a, c = 10) { }"); }, SyntaxError, "Duplicate parameters are not allowed before the default argument", "Duplicate formal parameter names not allowed in this context");
- assert.throws(function () { eval("function f(a, b = 10, a) { }"); }, SyntaxError, "Duplicate parameters are not allolwed after the default argument", "Duplicate formal parameter names not allowed in this context");
- assert.throws(function () { eval("function f(a, b, a, c) { \"use strict\"; }"); }, SyntaxError, "When function is in strict mode duplicate parameters are not allowed for simple parameter list", "Duplicate formal parameter names not allowed in strict mode");
- assert.throws(function () { eval("function f(a, b = 1) { \"use strict\"; }"); }, SyntaxError, "Strict mode cannot be applied to functions with default parameters", "Cannot apply strict mode on functions with non-simple parameter list");
- assert.throws(function () { eval("function f() { \"use strict\"; function g(a, b, a) { } }"); }, SyntaxError, "When a function is already in strict mode duplicate parameters are not allowed for simple parameter list", "Duplicate formal parameter names not allowed in strict mode");
- assert.throws(function () { eval("function f() { \"use strict\"; function g(a, b, a = 10) { } }"); }, SyntaxError, "When a function is already in strict mode duplicate parameters are not allowed for formal parameter list", "Duplicate formal parameter names not allowed in strict mode");
- assert.doesNotThrow(function f() { "use strict"; function g(a, b = 10) { } }, "Default arguments are allowed for functions which are already in strict mode");
- assert.doesNotThrow(function f(a, b, a, c) { return a + b + c; }, "In non-strict mode duplicate parameters are allowed");
-
- assert.doesNotThrow(function () { var obj = { set f(a = 1) {} }; }, "Default parameters can be used with setters inside an object literal");
- assert.doesNotThrow(function () { class c { set f(a = 1) {} }; }, "Default parameters can be used with setters inside a class");
- assert.doesNotThrow(function () { var obj = { set f({a}) {} }; }, "Setter can have destructured param list");
- assert.doesNotThrow(function () { var obj = { set f({a, b}) {} }; }, "Setter can have destructured param list with more than one parameter");
- assert.doesNotThrow(function () { var obj = { set f([a, b]) {} }; }, "Setter can have destructured array pattern with more than one parameter");
- assert.doesNotThrow(function () { var obj = { set f([a, ...b]) {} }; }, "Setter can have destructured array pattern with rest");
- assert.throws(function () { eval("var obj = { set f(...a) {} };"); }, SyntaxError, "Rest parameter cannot be used with setters inside an object literal", "Unexpected ... operator");
- assert.throws(function () { eval("var obj = { set f(a, b = 1) {} };"); }, SyntaxError, "Setters can have only one parameter even if one of them is default parameter", "Setter functions must have exactly one parameter");
- assert.throws(function () { eval("var obj = { set f(a = 1, b) {} };"); }, SyntaxError, "Setters can have only one parameter even if one of them is default parameter", "Setter functions must have exactly one parameter");
- assert.throws(function () { eval("var obj = { set f(a = 1, ...b) {} };") }, SyntaxError, "Setters can have only one parameter even if one of them is rest parameter", "Setter functions must have exactly one parameter");
- assert.throws(function () { eval("var obj = { set f(a = 1, {b}) {} };"); }, SyntaxError, "Setters can have only one parameter even if one of them is destructured parameter", "Setter functions must have exactly one parameter");
- assert.throws(function () { eval("var obj = { set f({a}, b = 1) {} };"); }, SyntaxError, "Setters can have only one parameter even if one of them is default parameter", "Setter functions must have exactly one parameter");
- assert.throws(function () { eval("var obj = { get f(a = 1) {} };"); }, SyntaxError, "Getter cannnot have any parameter even if it is default parameter", "Getter functions must have no parameters");
- assert.throws(function () { eval("var obj = { get f(...a) {} };"); }, SyntaxError, "Getter cannot have any parameter even if it is rest parameter", "Getter functions must have no parameters");
- assert.throws(function () { eval("var obj = { get f({a}) {} };"); }, SyntaxError, "Getter cannot have any parameter even if it is destructured parameter", "Getter functions must have no parameters");
- assert.throws(function () { eval("function foo(a *= 5)"); }, SyntaxError, "Other assignment operators do not work");
- // Redeclaration errors - non-simple in this case means any parameter list with a default expression
- assert.doesNotThrow(function () { eval("function foo(a = 1) { var a; }"); }, "Var redeclaration with a non-simple parameter list");
- assert.doesNotThrow(function () { eval("function foo(a = 1, b) { var b; }"); }, "Var redeclaration does not throw with a non-simple parameter list on a non-default parameter");
- assert.throws(function () { function foo(a = 1) { eval('var a;'); }; foo() }, ReferenceError, "Var redeclaration throws with a non-simple parameter list inside an eval", "Let/Const redeclaration");
- assert.throws(function () { function foo(a = 1, b) { eval('var b;'); }; foo(); }, ReferenceError, "Var redeclaration throws with a non-simple parameter list on a non-default parameter inside eval", "Let/Const redeclaration");
- assert.doesNotThrow(function () { function foo(a = 1) { eval('let a;'); }; foo() }, "Let redeclaration inside an eval does not throw with a non-simple parameter list");
- assert.doesNotThrow(function () { function foo(a = 1) { eval('const a = "str";'); }; foo() }, "Const redeclaration inside an eval does not throw with a non-simple parameter list");
- assert.doesNotThrow(function () { function foo(a = 1) { eval('let a;'); }; foo() }, "Let redeclaration of a non-default parameter inside an eval does not throw with a non-simple parameter list");
- assert.doesNotThrow(function () { function foo(a = 1) { eval('const a = 0;'); }; foo() }, "Const redeclaration of a non-default parameter inside an eval does not throw with a non-simple parameter list");
- assert.throws(function () { eval("x = 3 => x"); }, SyntaxError, "Lambda formals without parentheses cannot have default expressions", "Expected \'(\'");
- assert.throws(function () { eval("var a = 0, b = 0; (x = ++a,++b) => x"); }, SyntaxError, "Default expressions cannot have comma separated expressions", "Expected identifier");
- assert.doesNotThrow(function f(a = 1, b = class c { f() { return 2; }}) { }, "Class methods that do not refer to a formal are allowed in the param scope");
- }
- },
- {
- name: "Default argument sanity checks and expressions",
- body: function () {
- function foo(a, b = 2, c = 1, d = a + b + c, e) { return d; }
- assert.areEqual(foo(), foo(undefined, undefined, undefined, undefined, undefined), "Passing undefined has the same behavior as a default expression");
- assert.areEqual(foo(1), foo(1, 2), "Passing some arguments leaves default values for the rest");
- assert.areEqual(foo(1, 2), foo(1, 2, 1), "Passing some arguments leaves default values for the rest");
- assert.areEqual(foo(3, 5), foo(3, 5, 1), "Overriding default values leaves other default values in tact");
- function sideEffects(a = 1, b = ++a, c = ++b, d = ++c) { return [a,b,c,d]; }
- assert.areEqual([2,3,4,4], sideEffects(), "Side effects chain properly");
- assert.areEqual([1,1,1,1], sideEffects(0,undefined,0), "Side effects chain properly with some arguments passed");
- function argsObj1(a = 15, b = arguments[1], c = arguments[0]) { return [a,b,c]; }
- assert.areEqual([15,undefined,undefined], argsObj1(), "Arguments object uses original parameters passed");
- function argsObj2(a, b = 5 * arguments[0]) { return arguments[1]; }
- assert.areEqual(undefined, argsObj2(5), "Arguments object does not return default expression");
- this.val = { test : "test" };
- function thisTest(a = this.val, b = this.val = 1.1, c = this.val, d = this.val2 = 2, e = this.val2) { return [a,b,c,d,e]; }
- assert.areEqual([{test:"test"}, 1.1, 1.1, 2, 2], thisTest(), "'this' is able to be referenced and modified");
- var expAValue, expBValue, expCValue;
- function f(a = 10, b = 20, c) {
- assert.areEqual(a, expAValue, "First argument");
- assert.areEqual(b, expBValue, "Second argument");
- assert.areEqual(c, expCValue, "Third argument");
- }
- expAValue = null, expBValue = 20, expCValue = 1;
- f(null, undefined, 1);
- expAValue = null, expBValue = null, expCValue = null;
- f(null, null, null);
- expAValue = 10, expBValue = null, expCValue = 3;
- f(undefined, null, 3);
- function lambdaCapture() {
- this.propA = 1;
- var lambda = (a = this.propA++) => {
- assert.areEqual(a, 1, "Lambda default parameters use 'this' correctly");
- assert.areEqual(this.propA, 2, "Lambda default parameters using 'this' support side effects");
- };
- lambda();
- }
- lambdaCapture();
- // Function length with and without default
- function length1(a, b, c) {}
- function length2(a, b, c = 1) {}
- function length3(a, b = 1, c = 1) {}
- function length4(a, b = 1, c) {}
- function length5(a = 2, b, c) {}
- function length6(a = 2, b = 5, c = "str") {}
- assert.areEqual(3, length1.length, "No default parameters gives correct length");
- assert.areEqual(2, length2.length, "One trailing default parameter gives correct length");
- assert.areEqual(1, length3.length, "Two trailing default parameters gives correct length");
- assert.areEqual(1, length4.length, "One default parameter with following non-default gives correct length");
- assert.areEqual(0, length5.length, "One default parameter with two following non-defaults gives correct length");
- assert.areEqual(0, length6.length, "All default parameters gives correct length");
- }
- },
- {
- name: "Use before declaration in parameter lists",
- body: function () {
- assert.throws(function () { eval("function foo(a = b, b = a) {}; foo();"); }, ReferenceError, "Unevaluated parameters cannot be referenced in a default initializer", "Use before declaration");
- assert.throws(function () { eval("function foo(a = a, b = b) {}; foo();"); }, ReferenceError, "Unevaluated parameters cannot be referenced in a default initializer", "Use before declaration");
- assert.throws(function () { eval("function foo(a = (b = 5), b) {}; foo();"); }, ReferenceError, "Unevaluated parameters cannot be modified in a default initializer", "Use before declaration");
- function argsFoo(a = (arguments[1] = 5), b) { return b };
- assert.areEqual(undefined, argsFoo(), "Unevaluated parameters are referenceable using the arguments object");
- assert.areEqual(undefined, argsFoo(1), "Side effects on the arguments object are allowed but has no effect on default parameter initialization");
- assert.areEqual(2, argsFoo(1, 2), "Side effects on the arguments object are allowed but has no effect on default parameter initialization");
- }
- },
- {
- name: "Parameter scope does not have visibility of body declarations",
- body: function () {
- function foo1(a = x) { var x = 1; return a; }
- assert.throws(function() { foo1(); },
- ReferenceError,
- "Shadowed var in parameter scope is not affected by body initialization when setting the default value",
- "'x' is not defined");
- function foo2(a = () => x) { var x = 1; return a(); }
- assert.throws(function () { foo2(); },
- ReferenceError,
- "Arrow function capturing var at parameter scope is not affected by body declaration",
- "'x' is not defined");
- function foo3(a = () => x) { var x = 1; return a; } // a() undefined
- assert.throws(function () { foo3()(); },
- ReferenceError,
- "Attempted closure capture of body scoped var throws in an arrow function default expression",
- "'x' is not defined");
- function foo4(a = function() { return x; }) { var x = 1; return a(); }
- assert.throws(function () { foo4(); },
- ReferenceError,
- "Attempted closure capture of body scoped var throws in an anonymous function default expression",
- "'x' is not defined");
- function foo5(a = function bar() { return 1; }, b = bar()) { return [a(), b]; }
- assert.throws(function () { foo5(); },
- ReferenceError,
- "Named function expression does not leak name into subsequent default expressions",
- "'bar' is not defined");
- function foo6(a = b1) {
- {
- function b1() {
- return 2;
- }
- }
- assert.areEqual(1, a, "First argument should get the initial value from outer variable");
- assert.areEqual(2, b1(), "Block scoped function should be visible in the body also");
- }
- var b1 = 1;
- foo6();
-
- var a1 = 10;
- function foo7(b = function () { return a1; }) {
- assert.areEqual(undefined, a1, "Inside the function body the assignment hasn't happened yet");
- var a1 = 20;
- assert.areEqual(20, a1, "Assignment to the symbol inside the function changes the value");
- return b;
- }
- assert.areEqual(10, foo7()(), "Function in the param scope correctly binds to the outer variable");
-
- function foo8(a = x1, b = function g() {
- return function h() {
- assert.areEqual(10, x1, "x1 is captured from the outer scope");
- };
- }) {
- var x1 = 100;
- b()();
- };
- var x1 = 10;
- foo8();
-
- var x2 = 1;
- function foo9(a = x2, b = function() { return x2; }) {
- {
- function x2() {
- }
- }
- var x2 = 2;
- return b;
- }
- assert.areEqual(1, foo9()(), "Symbol capture at the param scope is unaffected by the inner definitions");
-
- var x3 = 1;
- function foo10(a = x3, b = function(_x) { return x3; }) {
- var x3 = 2;
- return b;
- }
- assert.areEqual(1, foo10()(), "Symbol capture at the param scope is unaffected by other references in the body and param");
- }
- },
- {
- name: "Parameter scope shadowing tests",
- body: function () {
- // These tests exercise logic in FindOrAddPidRef for when we need to look at parameter scope
- // Original sym in parameter scope
- var test0 = function(arg1 = _strvar0) {
- for (var _strvar0 in f32) {
- for (var _strvar0 in obj1) {
- }
- }
- }
- // False positive PidRef (no decl) at parameter scope
- function test1() {
- for (var _strvar0 in a) {
- var f = function(b = _strvar0) {
- for (var _strvar0 in c) {}
- };
- }
- }
- function test2() {
- let l = (z) => {
- let w = { z };
- assert.areEqual(10, w.z, "Identifier reference in object literal should get the correct reference from the arguments");
- var z;
- }
- l(10);
- };
- test2();
- }
- },
- {
- name: "Arrow function bodies in parameter scope",
- body: function () {
- // Nested parameter scopes
- function arrow(a = ((x = 1) => x)()) { return a; }
- assert.areEqual(1, arrow(), "Arrow function with default value works at parameter scope");
- function nestedArrow(a = (b = (x = () => 1)) => 1) { return a; }
- assert.areEqual(1, nestedArrow()(), "Nested arrow function with default value works at parameter scope");
- }
- },
- {
- name: "OS 1583694: Arguments sym is not set correctly on undo defer",
- body: function () {
- eval();
- var arguments;
- }
- },
- {
- name: "Unmapped arguments - Non simple parameter list",
- body: function () {
- function f1 (x = 10, y = 20, z) {
- x += 2;
- y += 2;
- z += 2;
- assert.areEqual(arguments[0], undefined, "arguments[0] is not mapped with first formal and did not change when the first formal changed");
- assert.areEqual(arguments[1], undefined, "arguments[1] is not mapped with second formal and did not change when the second formal changed");
- assert.areEqual(arguments[2], 30, "arguments[2] is not mapped with third formal and did not change when the third formal changed");
- arguments[0] = 1;
- arguments[1] = 2;
- arguments[2] = 3;
- assert.areEqual(x, 12, "Changing arguments[0], did not change the first formal");
- assert.areEqual(y, 22, "Changing arguments[1], did not change the second formal");
- assert.areEqual(z, 32, "Changing arguments[2], did not change the third formal");
- }
- f1(undefined, undefined, 30);
- function f2 (x = 10, y = 20, z) {
- eval('');
- x += 2;
- y += 2;
- z += 2;
- assert.areEqual(arguments[0], undefined, "Function has eval - arguments[0] is not mapped with first formal and did not change when the first formal changed");
- assert.areEqual(arguments[1], undefined, "Function has eval - arguments[1] is not mapped with second formal and did not change when the second formal changed");
- assert.areEqual(arguments[2], 30, "Function has eval - arguments[2] is not mapped with third formal and did not change when the third formal changed");
- arguments[0] = 1;
- arguments[1] = 2;
- arguments[2] = 3;
- assert.areEqual(x, 12, "Function has eval - Changing arguments[0], did not change the first formal");
- assert.areEqual(y, 22, "Function has eval - Changing arguments[1], did not change the second formal");
- assert.areEqual(z, 32, "Function has eval - Changing arguments[2], did not change the third formal");
- }
- f2(undefined, undefined, 30);
- function f3 (x = 10, y = 20, z) {
- (function () {
- eval('');
- })();
- x += 2;
- y += 2;
- z += 2;
- assert.areEqual(arguments[0], undefined, "Function's inner function has eval - arguments[0] is not mapped with first formal and did not change when the first formal changed");
- assert.areEqual(arguments[1], undefined, "Function's inner function has eval - arguments[1] is not mapped with second formal and did not change when the second formal changed");
- assert.areEqual(arguments[2], 30, "Function's inner function has eval - arguments[2] is not mapped with third formal and did not change when the third formal changed");
- arguments[0] = 1;
- arguments[1] = 2;
- arguments[2] = 3;
- assert.areEqual(x, 12, "Function's inner function has eval - Changing arguments[0], did not change the first formal");
- assert.areEqual(y, 22, "Function's inner function has eval - Changing arguments[1], did not change the second formal");
- assert.areEqual(z, 32, "Function's inner function has eval - Changing arguments[2], did not change the third formal");
- }
- f3(undefined, undefined, 30);
- function f4 (a, b, c, d = 1) {
- var e = 10;
- assert.areEqual(2, arguments[0], "Unmapped arguments value has the expected value in the body");
- (function () {
- eval('');
- }());
- };
- f4.call(1, 2);
-
- function f5 (a, b, c, d = 1) {
- var e = 10;
- var d = 11;
- assert.areEqual(2, arguments[0], "Unmapped arguments value has the expected value, even with duplicate symbol in the body");
- (function () {
- eval('');
- }());
- };
- f5.call(1, 2);
- function f6() {
- return ((a, b = (c = arguments) => c) => b)(2);
- };
- assert.areEqual(1, f6(1)()[0], "Nested lambda should capture the arguments from the outer function");
- }
- },
- {
- name: "Param of lambda has default as function",
- body: function () {
- assert.doesNotThrow(function () { eval("[ (a = function () { }) => {} ];"); }, "Lambda defined, inside an array literal, has a default as a function should not assert");
- }
- },
- {
- name: "Shadowing arguments symbol",
- body: function () {
- function f1(a, b = arguments[0]) {
- assert.areEqual(1, arguments[0], "Initial value of arguments symbol in the body should be same as the arguments from the param scope");
- var arguments = [10, 20];
- assert.areEqual(1, b, "Arguments value is the initial value in the param scope too");
- assert.areEqual(10, arguments[0], "Arguments value is updated in the body");
- }
- f1(1);
- function f2(a = 1, arguments) {
- assert.areEqual(2, arguments, "Initial value of arguments symbol in the body should be same as the arguments from the param scope");
- var arguments = [10, 20];
- assert.areEqual(10, arguments[0], "Arguments value is updated in the body");
- }
- f2(undefined, 2);
- function f3(a, b = arguments[0]) {
- assert.areEqual(10, arguments(), "Arguments symbol is overwritten by the the function definition");
- function arguments() {
- return 10;
- }
- assert.areEqual(1, b, "Arguments value is the initial value in the param scope too");
- }
- f3(1);
- function f4(a = 1, arguments, c = arguments) {
- assert.areEqual(10, arguments(), "In the body function definition shadows the formal");
- assert.areEqual(2, c, "Value of the formal is assigned properly");
- function arguments() {
- return 10;
- }
- }
- f4(undefined, 2);
- function f5(a, b = arguments) {
- function arguments(c) {
- return arguments;
- }
- assert.areEqual(30, arguments(10, 20, 30)[2], "Inside the arguments function the arguments symbol should points to the passed in values");
- assert.areEqual(4, b[3], "In the param scope arguments symbol referes to the passed in values");
- }
- f5(1, undefined, 3, 4, 5);
- function f6(a, b = arguments) {
- function arguments(c) {
- if (!arguments.length) {
- return arguments.callee(10, 20, 30);
- }
- return arguments;
- }
- assert.areEqual(20, arguments()[1], "In the function body arguments refers to the inner function");
- assert.areEqual(3, b[2], "In the param scope arguments symbol referes to the passed in values");
- }
- f6(1, undefined, 3, 4);
- function f7(a, b = function arguments(c) {
- if (!c) {
- return arguments.callee(10, 20);
- }
- return c + arguments[1];
- }) {
- assert.areEqual(30, b(), "Function defined in the param scope can be called recursively");
- assert.areEqual(1, arguments[0], "Arguments symbol is unaffected by the function expression");
- }
- f7(1);
- function f8(a, b = arguments) {
- var c = function arguments(c) {
- if (!arguments.length) {
- return arguments.callee(10, 20, 30);
- }
- return arguments;
- }
- assert.areEqual(30, c()[2], "In the function body the arguments function expression with name is not visible");
- assert.areEqual(1, b[0], "In the param scope arguments symbol referes to the passed in values");
- }
- f8(1);
- assert.throws(function () { eval("function f(a, b = arguments) { class arguments { } }"); }, SyntaxError, "Class cannot be named arguments", "Invalid usage of 'arguments' in strict mode");
- assert.throws(function () { eval("function f(a, arguments) { class arguments { } }"); }, SyntaxError, "Class cannot be named arguments even when one of the formal is named arguments", "Let/Const redeclaration");
- function f9( a = 0, b = {
- arguments() {
- return 10;
- }
- }, c = arguments) {
- with (b) {
- assert.areEqual(10, arguments(), "Inside with the right the arguments function inside the object is used in the body also");
- }
- assert.areEqual(1, arguments[0], "Arguments symbol should be unaffected after with construct");
- assert.areEqual(1, c[0], "Arguments symbol from param scope should be unaffected after with construct");
- }
- f9(1);
- function f10(a = 1, b = () => {
- assert.areEqual(undefined, arguments, "Due to the decalration in the body arguments symbol is shadowed inside the lambda");
- var arguments = 100;
- assert.areEqual(100, arguments, "After the assignment value of arguments is updated inside the lambda");
- }, c = arguments) {
- assert.areEqual(10, arguments[0], "In the body the value of arguments is retained");
- assert.areEqual(10, c[0], "Arguments symbol is not affected in the param scope");
- b();
- }
- f10(10);
- function f11(a = 1, b = () => {
- assert.areEqual(100, arguments(), "Inside the lambda the function definition shadows the parent's arguments symbol");
- function arguments() {
- return 100;
- }
- }, c = arguments) {
- assert.areEqual(10, arguments[0], "In the body the value of arguments is retained");
- b();
- assert.areEqual(10, c[0], "Arguments symbol is not affected in the param scope");
- }
- f11(10);
- function f12({a = 1, arguments}) {
- assert.areEqual(2, arguments, "Initial value of arguments symbol in the body should be same as the arguments from the param scope's destructured pattern");
- var arguments = [10, 20];
- assert.areEqual(10, arguments[0], "Arguments value is updated in the body");
- }
- f12({arguments : 2});
- function f13(a = 1, {arguments, c = arguments}) {
- assert.areEqual(10, arguments(), "In the body function definition shadows the destructured formal");
- assert.areEqual(2, c, "Value of the formal is assigned properly");
- function arguments() {
- return 10;
- }
- }
- f13(undefined, { arguments: 2 });
- function f14(a, b = arguments[0]) {
- assert.areEqual(1, arguments[0], "Function in block causes a var declaration to be hoisted and the initial value should be same as the arguments symbol");
- {
- {
- function arguments() {
- return 10;
- }
- }
- }
- assert.areEqual(1, b, "Arguments value is the initial value in the param scope too");
- assert.areEqual(10, arguments(), "Hoisted var binding is updated after the block is exected");
- }
- f14(1);
- function f15() {
- function f16() {
- eval("");
- this.arguments = 1;
- }
- var obj = new f16();
-
- function arguments() {
- return 10;
- }
- assert.areEqual(1, obj.arguments, "Child function having eval should work fine with a duplicate arguments definition in the parent body");
- };
- f15();
- }
- },
- {
- name: "Shadowing arguments symbol - Eval",
- body: function () {
- function f1(a, b = arguments[0]) {
- assert.areEqual(1, arguments[0], "Initial value of arguments symbol in the body should be same as the arguments from the param scope");
- var arguments = [10, 20];
- assert.areEqual(1, b, "Arguments value is the initial value in the param scope too");
- assert.areEqual(10, eval("arguments[0]"), "Arguments value is updated in the body");
- }
- f1(1);
- function f2(a = 1, arguments) {
- assert.areEqual(2, eval("arguments"), "Initial value of arguments symbol in the body should be same as the arguments from the param scope");
- var arguments = [10, 20];
- assert.areEqual(10, arguments[0], "Arguments value is updated in the body");
- }
- f2(undefined, 2);
- function f3(a, b = arguments[0]) {
- assert.areEqual(10, eval("arguments()"), "Arguments symbol is overwritten by the the function definition");
- function arguments() {
- return 10;
- }
- assert.areEqual(1, b, "Arguments value is the initial value in the param scope too");
- }
- f3(1);
- function f4(a = 1, arguments, c = arguments) {
- assert.areEqual(10, arguments(), "In the body function definition shadows the formal");
- assert.areEqual(2, eval("c"), "Value of the formal is assigned properly");
- function arguments() {
- return 10;
- }
- }
- f4(undefined, 2);
- function f5(a, b, c = arguments) {
- function arguments(c) {
- return eval("arguments");
- }
- assert.areEqual(30, arguments(10, 20, 30)[2], "In the function body arguments refers to the inner function");
- assert.areEqual(2, c[1], "In the param scope arguments symbol referes to the passed in values");
- }
- f5(1, 2, undefined, 4);
- function f6(a, b = function arguments(c) {
- if (!c) {
- return arguments.callee(10, 20);
- }
- return c + arguments[1];
- }) {
- assert.areEqual(30, eval("b()"), "Function defined in the param scope can be called recursively");
- assert.areEqual(1, arguments[0], "Arguments symbol is unaffected by the function expression");
- }
- f6(1);
- function f7(a, b = arguments) {
- var c = function arguments(c) {
- if (!arguments.length) {
- return arguments.callee(10, 20, 30);
- }
- return arguments;
- }
- assert.areEqual(10, eval("c()[0]"), "In the function body the arguments function expression with name is not visible");
- assert.areEqual(4, b[3], "In the param scope arguments symbol referes to the passed in values");
- }
- f7(1, undefined, 3, 4);
- }
- }
- ];
- testRunner.runTests(tests, { verbose: WScript.Arguments[0] != "summary" });
|