1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-25 09:42:35 +00:00
serenity/Userland/Libraries/LibJS/Tests/functions/function-evaluation-order.js
Linus Groh 87068896d0 LibJS: Evaluate NewExpression arguments before checking constructor type
Exactly like in 99f9609, which fixed the same issue in CallExpression,
the spec tells us to *first* evaluate the arguments, if any, and *then*
check if the provided value is a constructor function.
2021-12-30 01:02:30 +01:00

65 lines
1.8 KiB
JavaScript

describe("CallExpression", () => {
test("callee is evaluated before arguments", () => {
function foo() {}
const values = [];
[foo][(values.push("callee"), 0)](values.push("args"));
expect(values).toEqual(["callee", "args"]);
});
test("arguments are evaluated in order", () => {
function foo() {}
const values = [];
foo(values.push("arg1"), values.push("arg2"), values.push("arg3"));
expect(values).toEqual(["arg1", "arg2", "arg3"]);
});
test("arguments are evaluated before callee is checked for its type", () => {
const values = [];
expect(() => {
"foo"(values.push("args"));
}).toThrowWithMessage(TypeError, "foo is not a function");
expect(values).toEqual(["args"]);
expect(() => {
"foo"(bar);
}).toThrowWithMessage(ReferenceError, "'bar' is not defined");
});
});
describe("NewExpression", () => {
test("callee is evaluated before arguments", () => {
function Foo() {}
const values = [];
new [Foo][(values.push("callee"), 0)](values.push("args"));
expect(values).toEqual(["callee", "args"]);
});
test("arguments are evaluated in order", () => {
function Foo() {}
const values = [];
new Foo(values.push("arg1"), values.push("arg2"), values.push("arg3"));
expect(values).toEqual(["arg1", "arg2", "arg3"]);
});
test("arguments are evaluated before callee is checked for its type", () => {
const values = [];
expect(() => {
new "Foo"(values.push("args"));
}).toThrowWithMessage(TypeError, "Foo is not a constructor");
expect(values).toEqual(["args"]);
expect(() => {
new "Foo"(bar);
}).toThrowWithMessage(ReferenceError, "'bar' is not defined");
});
});