blob: 1fbac5a0721b5b3a89e05b049877d2fc822c8818 [file] [log] [blame]
// Copyright 2016 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 assertThrowsAsync(run, errorType, message) {
var actual;
var hadValue = false;
var hadError = false;
var promise = run();
if (typeof promise !== "object" || typeof promise.then !== "function") {
throw new MjsUnitAssertionError(
"Expected " + run.toString() +
" to return a Promise, but it returned " + PrettyPrint(promise));
}
promise.then(function(value) { hadValue = true; actual = value; },
function(error) { hadError = true; actual = error; });
assertFalse(hadValue || hadError);
%RunMicrotasks();
if (!hadError) {
throw new MjsUnitAssertionError(
"Expected " + run + "() to throw " + errorType.name +
", but did not throw.");
}
if (!(actual instanceof errorType))
throw new MjsUnitAssertionError(
"Expected " + run + "() to throw " + errorType.name +
", but threw '" + actual + "'");
if (message !== void 0 && actual.message !== message)
throw new MjsUnitAssertionError(
"Expected " + run + "() to throw '" + message + "', but threw '" +
actual.message + "'");
};
function assertEqualsAsync(expected, run, msg) {
var actual;
var hadValue = false;
var hadError = false;
var promise = run();
if (typeof promise !== "object" || typeof promise.then !== "function") {
throw new MjsUnitAssertionError(
"Expected " + run.toString() +
" to return a Promise, but it returned " + PrettyPrint(promise));
}
promise.then(function(value) { hadValue = true; actual = value; },
function(error) { hadError = true; actual = error; });
assertFalse(hadValue || hadError);
%RunMicrotasks();
if (hadError) throw actual;
assertTrue(
hadValue, "Expected '" + run.toString() + "' to produce a value");
assertEquals(expected, actual, msg);
};
(function TestDefaultEvaluationOrder() {
var y = 0;
var z = 0;
var w = 0;
async function f1(x = (y = 1)) { z = 1; await undefined; w = 1; };
assertEquals(0, y);
assertEquals(0, z);
assertEquals(0, w);
f1();
assertEquals(1, y);
assertEquals(1, z);
assertEquals(0, w);
%RunMicrotasks();
assertEquals(1, y);
assertEquals(1, z);
assertEquals(1, w);
})();
(function TestShadowingOfParameters() {
async function f1({x}) { var x = 2; return x }
assertEqualsAsync(2, () => f1({x: 1}));
async function f2({x}) { { var x = 2; } return x; }
assertEqualsAsync(2, () => f2({x: 1}));
async function f3({x}) { var y = x; var x = 2; return y; }
assertEqualsAsync(1, () => f3({x: 1}));
async function f4({x}) { { var y = x; var x = 2; } return y; }
assertEqualsAsync(1, () => f4({x: 1}));
async function f5({x}, g = () => x) { var x = 2; return g(); }
assertEqualsAsync(1, () => f5({x: 1}));
async function f6({x}, g = () => x) { { var x = 2; } return g(); }
assertEqualsAsync(1, () => f6({x: 1}));
async function f7({x}) { var g = () => x; var x = 2; return g(); }
assertEqualsAsync(2, () => f7({x: 1}));
async function f8({x}) { { var g = () => x; var x = 2; } return g(); }
assertEqualsAsync(2, () => f8({x: 1}));
async function f9({x}, g = () => eval("x")) { var x = 2; return g(); }
assertEqualsAsync(1, () => f9({x: 1}));
async function f10({x}, y) { var y; return y }
assertEqualsAsync(2, () => f10({x: 6}, 2));
async function f11({x}, y) { var z = y; var y = 2; return z; }
assertEqualsAsync(1, () => f11({x: 6}, 1));
async function f12(y, g = () => y) { var y = 2; return g(); }
assertEqualsAsync(1, () => f12(1));
async function f13({x}, y, [z], v) { var x, y, z; return x*y*z*v }
assertEqualsAsync(210, () => f13({x: 2}, 3, [5], 7));
async function f20({x}) { function x() { return 2 }; return x(); }
assertEqualsAsync(2, () => f20({x: 1}));
// Annex B 3.3 function hoisting is blocked by the conflicting x declaration
async function f21({x}) { { function x() { return 2 } } return x; }
assertEqualsAsync(1, () => f21({x: 1}));
var g1 = async ({x}) => { var x = 2; return x };
assertEqualsAsync(2, () => g1({x: 1}));
var g2 = async ({x}) => { { var x = 2; } return x; };
assertEqualsAsync(2, () => g2({x: 1}));
var g3 = async ({x}) => { var y = x; var x = 2; return y; };
assertEqualsAsync(1, () => g3({x: 1}));
var g4 = async ({x}) => { { var y = x; var x = 2; } return y; };
assertEqualsAsync(1, () => g4({x: 1}));
var g5 = async ({x}, g = () => x) => { var x = 2; return g(); };
assertEqualsAsync(1, () => g5({x: 1}));
var g6 = async ({x}, g = () => x) => { { var x = 2; } return g(); };
assertEqualsAsync(1, () => g6({x: 1}));
var g7 = async ({x}) => { var g = () => x; var x = 2; return g(); };
assertEqualsAsync(2, () => g7({x: 1}));
var g8 = async ({x}) => { { var g = () => x; var x = 2; } return g(); };
assertEqualsAsync(2, () => g8({x: 1}));
var g9 = async ({x}, g = () => eval("x")) => { var x = 2; return g(); };
assertEqualsAsync(1, () => g9({x: 1}));
var g10 = async ({x}, y) => { var y; return y };
assertEqualsAsync(2, () => g10({x: 6}, 2));
var g11 = async ({x}, y) => { var z = y; var y = 2; return z; };
assertEqualsAsync(1, () => g11({x: 6}, 1));
var g12 = async (y, g = () => y) => { var y = 2; return g(); };
assertEqualsAsync(1, () => g12(1));
var g13 = async ({x}, y, [z], v) => { var x, y, z; return x*y*z*v };
assertEqualsAsync(210, () => g13({x: 2}, 3, [5], 7));
var g20 = async ({x}) => { function x() { return 2 }; return x(); }
assertEqualsAsync(2, () => g20({x: 1}));
var g21 = async ({x}) => { { function x() { return 2 } } return x(); }
assertThrowsAsync(() => g21({x: 1}), TypeError);
// These errors are not recognized in lazy parsing; see mjsunit/bugs/bug-2728.js
assertThrows("'use strict'; (async function f(x) { let x = 0; })()", SyntaxError);
assertThrows("'use strict'; (async function f({x}) { let x = 0; })()", SyntaxError);
assertThrows("'use strict'; (async function f(x) { const x = 0; })()", SyntaxError);
assertThrows("'use strict'; (async function f({x}) { const x = 0; })()", SyntaxError);
assertThrows("'use strict'; let g = async (x) => { let x = 0; }", SyntaxError);
assertThrows("'use strict'; let g = async ({x}) => { let x = 0; }", SyntaxError);
assertThrows("'use strict'; let g = async (x) => { const x = 0; }", SyntaxError);
assertThrows("'use strict'; let g = async ({x}) => { const x = 0; }", SyntaxError);
}());
(function TestDefaults() {
async function f1(x = 1) { return x }
assertEqualsAsync(1, () => f1());
assertEqualsAsync(1, () => f1(undefined));
assertEqualsAsync(2, () => f1(2));
assertEqualsAsync(null, () => f1(null));
async function f2(x, y = x) { return x + y; }
assertEqualsAsync(8, () => f2(4));
assertEqualsAsync(8, () => f2(4, undefined));
assertEqualsAsync(6, () => f2(4, 2));
async function f3(x = 1, y) { return x + y; }
assertEqualsAsync(8, () => f3(5, 3));
assertEqualsAsync(3, () => f3(undefined, 2));
assertEqualsAsync(6, () => f3(4, 2));
async function f4(x = () => 1) { return x() }
assertEqualsAsync(1, () => f4());
assertEqualsAsync(1, () => f4(undefined));
assertEqualsAsync(2, () => f4(() => 2));
assertThrowsAsync(() => f4(null), TypeError);
async function f5(x, y = () => x) { return x + y(); }
assertEqualsAsync(8, () => f5(4));
assertEqualsAsync(8, () => f5(4, undefined));
assertEqualsAsync(6, () => f5(4, () => 2));
async function f6(x = {a: 1, m() { return 2 }}) { return x.a + x.m(); }
assertEqualsAsync(3, () => f6());
assertEqualsAsync(3, () => f6(undefined));
assertEqualsAsync(5, () => f6({a: 2, m() { return 3 }}));
var g1 = async (x = 1) => { return x };
assertEqualsAsync(1, () => g1());
assertEqualsAsync(1, () => g1(undefined));
assertEqualsAsync(2, () => g1(2));
assertEqualsAsync(null, () => g1(null));
var g2 = async (x, y = x) => { return x + y; };
assertEqualsAsync(8, () => g2(4));
assertEqualsAsync(8, () => g2(4, undefined));
assertEqualsAsync(6, () => g2(4, 2));
var g3 = async (x = 1, y) => { return x + y; };
assertEqualsAsync(8, () => g3(5, 3));
assertEqualsAsync(3, () => g3(undefined, 2));
assertEqualsAsync(6, () => g3(4, 2));
var g4 = async (x = () => 1) => { return x() };
assertEqualsAsync(1, () => g4());
assertEqualsAsync(1, () => g4(undefined));
assertEqualsAsync(2, () => g4(() => 2));
assertThrowsAsync(() => g4(null), TypeError);
var g5 = async (x, y = () => x) => { return x + y(); };
assertEqualsAsync(8, () => g5(4));
assertEqualsAsync(8, () => g5(4, undefined));
assertEqualsAsync(6, () => g5(4, () => 2));
var g6 = async (x = {a: 1, m() { return 2 }}) => { return x.a + x.m(); };
assertEqualsAsync(3, () => g6());
assertEqualsAsync(3, () => g6(undefined));
assertEqualsAsync(5, () => g6({a: 2, m() { return 3 }}));
}());
(function TestEvalInParameters() {
async function f1(x = eval(0)) { return x }
assertEqualsAsync(0, f1);
async function f2(x = () => eval(1)) { return x() }
assertEqualsAsync(1, f2);
})();
(function TestParameterScopingSloppy() {
var x = 1;
async function f1(a = x) { var x = 2; return a; }
assertEqualsAsync(1, f1);
async function f2(a = x) { function x() {}; return a; }
assertEqualsAsync(1, () => f2());
async function f3(a = eval("x")) { var x; return a; }
assertEqualsAsync(1, () => f3());
async function f31(a = eval("'use strict'; x")) { var x; return a; }
assertEqualsAsync(1, () => f31());
async function f4(a = function() { return x }) { var x; return a(); }
assertEqualsAsync(1, () => f4());
async function f5(a = () => x) { var x; return a(); }
assertEqualsAsync(1, () => f5());
async function f6(a = () => eval("x")) { var x; return a(); }
assertEqualsAsync(1, () => f6());
async function f61(a = () => { 'use strict'; return eval("x") }) { var x; return a(); }
assertEqualsAsync(1, () => f61());
async function f62(a = () => eval("'use strict'; x")) { var x; return a(); }
assertEqualsAsync(1, () => f62());
var g1 = async (a = x) => { var x = 2; return a; };
assertEqualsAsync(1, () => g1());
var g2 = async (a = x) => { function x() {}; return a; };
assertEqualsAsync(1, () => g2());
var g3 = async (a = eval("x")) => { var x; return a; };
assertEqualsAsync(1, g3);
var g31 = async (a = eval("'use strict'; x")) => { var x; return a; };
assertEqualsAsync(1, () => g31());
var g4 = async (a = function() { return x }) => { var x; return a(); };
assertEqualsAsync(1, () => g4());
var g5 = async (a = () => x) => { var x; return a(); };
assertEqualsAsync(1, () => g5());
var g6 = async (a = () => eval("x")) => { var x; return a(); };
assertEqualsAsync(1, () => g6());
var g61 = async (a = () => { 'use strict'; return eval("x") }) => { var x; return a(); };
assertEqualsAsync(1, () => g61());
var g62 = async (a = () => eval("'use strict'; x")) => { var x; return a(); };
assertEqualsAsync(1, () => g62());
var f11 = async function f(x = f) { var f; return x; }
assertEqualsAsync(f11, f11);
var f12 = async function f(x = f) { function f() {}; return x; }
assertEqualsAsync(f12, f12);
var f13 = async function f(f = 7, x = f) { return x; }
assertEqualsAsync(7, f13);
var o1 = {f: async function(x = this) { return x; }};
assertEqualsAsync(o1, () => o1.f());
assertEqualsAsync(1, () => o1.f(1));
})();
(function TestParameterScopingStrict() {
"use strict";
var x = 1;
async function f1(a = x) { let x = 2; return a; }
assertEqualsAsync(1, () => f1());
async function f2(a = x) { const x = 2; return a; }
assertEqualsAsync(1, () => f2());
async function f3(a = x) { function x() {}; return a; }
assertEqualsAsync(1, () => f3());
async function f4(a = eval("x")) { var x; return a; }
assertEqualsAsync(1, () => f4());
async function f5(a = () => eval("x")) { var x; return a(); }
assertEqualsAsync(1, () => f5());
var g1 = async (a = x) => { let x = 2; return a; };
assertEqualsAsync(1, () => g1());
var g2 = async (a = x) => { const x = 2; return a; };
assertEqualsAsync(1, () => g2());
var g3 = async (a = x) => { function x() {}; return a; };
assertEqualsAsync(1, () => g3());
var g4 = async (a = eval("x")) => { var x; return a; };
assertEqualsAsync(1, () => g4());
var g5 = async (a = () => eval("x")) => { var x; return a(); };
assertEqualsAsync(1, () => g5());
var f11 = async function f(x = f) { let f; return x; }
assertEqualsAsync(f11, f11);
var f12 = async function f(x = f) { const f = 0; return x; }
assertEqualsAsync(f12, f12);
var f13 = async function f(x = f) { function f() {}; return x; }
assertEqualsAsync(f13, f13);
})();
(function TestSloppyEvalScoping() {
var x = 1;
async function f1(y = eval("var x = 2")) { with ({}) { return x; } }
assertEqualsAsync(1, () => f1());
async function f2(y = eval("var x = 2"), z = x) { return z; }
assertEqualsAsync(1, () => f2());
assertEqualsAsync(1, () => f2(0));
async function f3(y = eval("var x = 2"), z = eval("x")) { return z; }
assertEqualsAsync(1, () => f3());
assertEqualsAsync(1, () => f3(0));
async function f8(y = (eval("var x = 2"), x)) { return y; }
assertEqualsAsync(2, () => f8());
assertEqualsAsync(0, () => f8(0));
async function f11(z = eval("var y = 2")) { return y; }
assertThrowsAsync(f11, ReferenceError);
async function f12(z = eval("var y = 2"), b = y) {}
assertThrowsAsync(f12, ReferenceError);
async function f13(z = eval("var y = 2"), b = eval("y")) {}
assertThrowsAsync(f13, ReferenceError);
async function f21(f = () => x) { eval("var x = 2"); return f() }
assertEqualsAsync(1, () => f21());
assertEqualsAsync(3, () => f21(() => 3));
async function f22(f = () => eval("x")) { eval("var x = 2"); return f() }
assertEqualsAsync(1, () => f22());
assertEqualsAsync(3, () => f22(() => 3));
var g1 = async (y = eval("var x = 2")) => { with ({}) { return x; } };
assertEqualsAsync(1, () => g1());
var g2 = async (y = eval("var x = 2"), z = x) => { return z; };
assertEqualsAsync(1, () => g2());
assertEqualsAsync(1, () => g2(0));
var g3 = async (y = eval("var x = 2"), z = eval("x")) => { return z; };
assertEqualsAsync(1, () => g3());
assertEqualsAsync(1, () => g3(0));
var g8 = async (y = (eval("var x = 2"), x)) => { return y; };
assertEqualsAsync(2, () => g8());
assertEqualsAsync(0, () => g8(0));
var g11 = async (z = eval("var y = 2")) => { return y; };
assertThrowsAsync(g11, ReferenceError);
var g12 = async (z = eval("var y = 2"), b = y) => {};
assertThrowsAsync(g12, ReferenceError);
var g13 = async (z = eval("var y = 2"), b = eval("y")) => {};
assertThrowsAsync(g13, ReferenceError);
var g21 = async (f = () => x) => { eval("var x = 2"); return f() };
assertEqualsAsync(1, () => g21());
assertEqualsAsync(3, () => g21(() => 3));
var g22 = async (f = () => eval("x")) => { eval("var x = 2"); return f() };
assertEqualsAsync(1, () => g22());
assertEqualsAsync(3, () => g22(() => 3));
})();
(function TestStrictEvalScoping() {
'use strict';
var x = 1;
async function f1(y = eval("var x = 2")) { return x; }
assertEqualsAsync(1, () => f1());
async function f2(y = eval("var x = 2"), z = x) { return z; }
assertEqualsAsync(1, () => f2());
assertEqualsAsync(1, () => f2(0));
async function f3(y = eval("var x = 2"), z = eval("x")) { return z; }
assertEqualsAsync(1, () => f3());
assertEqualsAsync(1, () => f3(0));
async function f8(y = (eval("var x = 2"), x)) { return y; }
assertEqualsAsync(1, () => f8());
assertEqualsAsync(0, () => f8(0));
async function f11(z = eval("var y = 2")) { return y; }
assertThrowsAsync(f11, ReferenceError);
async function f12(z = eval("var y = 2"), b = y) {}
assertThrowsAsync(f12, ReferenceError);
async function f13(z = eval("var y = 2"), b = eval("y")) {}
assertThrowsAsync(f13, ReferenceError);
async function f21(f = () => x) { eval("var x = 2"); return f() }
assertEqualsAsync(1, () => f21());
assertEqualsAsync(3, () => f21(() => 3));
async function f22(f = () => eval("x")) { eval("var x = 2"); return f() }
assertEqualsAsync(1, () => f22());
assertEqualsAsync(3, () => f22(() => 3));
})();
(function TestParameterTDZSloppy() {
async function f1(a = x, x) { return a }
assertThrowsAsync(() => f1(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f1(4, 5));
async function f2(a = eval("x"), x) { return a }
assertThrowsAsync(() => f2(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f2(4, 5));
async function f3(a = eval("'use strict'; x"), x) { return a }
assertThrowsAsync(() => f3(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f3(4, 5));
async function f4(a = () => x, x) { return a() }
assertEqualsAsync(4, () => f4(() => 4, 5));
async function f5(a = () => eval("x"), x) { return a() }
assertEqualsAsync(4, () => f5(() => 4, 5));
async function f6(a = () => eval("'use strict'; x"), x) { return a() }
assertEqualsAsync(4, () => f6(() => 4, 5));
async function f11(a = x, x = 2) { return a }
assertThrowsAsync(() => f11(), ReferenceError);
assertThrowsAsync(() => f11(undefined), ReferenceError);
assertThrowsAsync(() => f11(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f1(4, 5));
async function f12(a = eval("x"), x = 2) { return a }
assertThrowsAsync(() => f12(), ReferenceError);
assertThrowsAsync(() => f12(undefined), ReferenceError);
assertThrowsAsync(() => f12(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f12(4, 5));
async function f13(a = eval("'use strict'; x"), x = 2) { return a }
assertThrowsAsync(() => f13(), ReferenceError);
assertThrowsAsync(() => f13(undefined), ReferenceError);
assertThrowsAsync(() => f13(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f13(4, 5));
async function f21(x = function() { return a }, ...a) { return x()[0] }
assertEqualsAsync(4, () => f21(undefined, 4));
async function f22(x = () => a, ...a) { return x()[0] }
assertEqualsAsync(4, () => f22(undefined, 4));
async function f23(x = () => eval("a"), ...a) { return x()[0] }
assertEqualsAsync(4, () => f23(undefined, 4));
async function f24(x = () => {'use strict'; return eval("a") }, ...a) {
return x()[0]
}
assertEqualsAsync(4, () => f24(undefined, 4));
async function f25(x = () => eval("'use strict'; a"), ...a) { return x()[0] }
assertEqualsAsync(4, () => f25(undefined, 4));
var g1 = async (x = function() { return a }, ...a) => { return x()[0] };
assertEqualsAsync(4, () => g1(undefined, 4));
var g2 = async (x = () => a, ...a) => { return x()[0] };
assertEqualsAsync(4, () => g2(undefined, 4));
})();
(function TestParameterTDZStrict() {
"use strict";
async function f1(a = eval("x"), x) { return a }
assertThrowsAsync(() => f1(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f1(4, 5));
async function f2(a = () => eval("x"), x) { return a() }
assertEqualsAsync(4, () => f2(() => 4, 5));
async function f11(a = eval("x"), x = 2) { return a }
assertThrowsAsync(() => f11(), ReferenceError);
assertThrowsAsync(() => f11(undefined), ReferenceError);
assertThrowsAsync(() => f11(undefined, 4), ReferenceError);
assertEqualsAsync(4, () => f11(4, 5));
async function f21(x = () => eval("a"), ...a) { return x()[0] }
assertEqualsAsync(4, () => f21(undefined, 4));
})();
(function TestArgumentsForNonSimpleParameters() {
async function f1(x = 900) { arguments[0] = 1; return x }
assertEqualsAsync(9, () => f1(9));
assertEqualsAsync(900, () => f1());
async function f2(x = 1001) { x = 2; return arguments[0] }
assertEqualsAsync(10, () => f2(10));
assertEqualsAsync(undefined, () => f2());
}());
(function TestFunctionLength() {
assertEquals(0, (async function(x = 1) {}).length);
assertEquals(0, (async function(x = 1, ...a) {}).length);
assertEquals(1, (async function(x, y = 1) {}).length);
assertEquals(1, (async function(x, y = 1, ...a) {}).length);
assertEquals(2, (async function(x, y, z = 1) {}).length);
assertEquals(2, (async function(x, y, z = 1, ...a) {}).length);
assertEquals(1, (async function(x, y = 1, z) {}).length);
assertEquals(1, (async function(x, y = 1, z, ...a) {}).length);
assertEquals(1, (async function(x, y = 1, z, v = 2) {}).length);
assertEquals(1, (async function(x, y = 1, z, v = 2, ...a) {}).length);
})();
(function TestDirectiveThrows() {
"use strict";
assertThrows("(async function(x=1){'use strict';})", SyntaxError);
assertThrows("(async function(a, x=1){'use strict';})", SyntaxError);
assertThrows("(async function({x}){'use strict';})", SyntaxError);
})();