serenity/Userland/Libraries/LibJS/Tests/syntax/functions-in-tree-order-strict.js
davidot 8fa6861f66 LibJS: Initialize functions in spec order
This is only visible with something like `Object.getOwnPropertyNames` on
the global object. All other declaration instantiations put the
functions on an environment making the order invisible.
Note that spec order is not quite tree order as in non-strict mode
functions which get hoisted out of blocks appear before top level
functions.

Co-authored-by: Hendiadyoin1 <leon.a@serenityos.org>
2022-11-17 16:05:20 +00:00

148 lines
3.3 KiB
JavaScript

"use strict";
// Note: This must be checked at script level because that is the only place
// where function order is visible. We introduce some test(...) with
// names to make sure the file does have some tests and a suite.
// Note: In strict mode we expect almost the same result except that the
// functions in blocks do not get hoisted up.
// Only the indirect eval copy gets the global var environment.
const evalViaArrow = x => eval(x);
const evalRef = eval;
const expectedBeforeEval = "3478CDHIOP";
const expectedAfterEval = "3478CDHIOP";
const expectedAfterEvalRef = "3478CDHIOPKJL";
const expectOrderToBe = expectedOrder => {
const currentOrder = Object.getOwnPropertyNames(this)
.filter(s => s.length === 2 && s[0] === "f")
.map(s => s[1])
.join("");
expect(currentOrder).toBe(expectedOrder);
};
test("in strict mode: function order should be in tree order and nothing in eval should be included, in strict mode functions should not be hoisted", () => {
expectOrderToBe(expectedBeforeEval);
});
{
function f1() {}
expectOrderToBe(expectedBeforeEval);
function f2() {}
}
expectOrderToBe(expectedBeforeEval);
function f3() {}
expectOrderToBe(expectedBeforeEval);
function f4() {}
expectOrderToBe(expectedBeforeEval);
{
function f5() {}
function f6() {}
}
function f7() {}
function f8() {}
expectOrderToBe(expectedBeforeEval);
eval(`
expectOrderToBe(expectedAfterEval);
function f9() {}
{
function fA() {}
}
function fB() {}
expectOrderToBe(expectedAfterEval);
`);
expectOrderToBe(expectedAfterEval);
function fC() {}
function fD() {}
expectOrderToBe(expectedAfterEval);
// This eval does not do anything because it goes via a function, this means
// its parent environment is not the global environment so it does not have
// a global var environment and does not put the functions on `this`.
evalViaArrow(`
expectOrderToBe(expectedAfterEval);
function fE() {}
{
expectOrderToBe(expectedAfterEval);
function fF() {}
}
function fG() {}
expectOrderToBe(expectedAfterEval);
`);
test("in strict mode: function order should be in tree order, functions in eval should be in order but at the back, in strict mode functions should not be hoisted", () => {
expectOrderToBe(expectedAfterEval);
});
function fH() {}
function fI() {}
expectOrderToBe(expectedAfterEval);
// This is an indirect eval, but still has the global scope as immediate
// parent so it does influence the global `this`.
evalRef(`
expectOrderToBe(expectedAfterEvalRef);
console.log(2, JSON.stringify(Object.getOwnPropertyNames(this).filter(s => s.length === 2)));
function fJ() {}
{
expectOrderToBe(expectedAfterEvalRef);
function fK() {}
}
function fL() {}
expectOrderToBe(expectedAfterEvalRef);
`);
{
function fM() {}
function fN() {}
}
test("in strict mode: function order should be in tree order, functions in evalRef should be in order but at the back, in strict mode functions should not be hoisted", () => {
expectOrderToBe(expectedAfterEvalRef);
});
function fO() {}
function fP() {}
{
function fQ() {}
{
expectOrderToBe(expectedAfterEvalRef);
}
function fR() {}
}
expectOrderToBe(expectedAfterEvalRef);