mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-24 23:12:32 +00:00 
			
		
		
		
	 8fa6861f66
			
		
	
	
		8fa6861f66
		
	
	
	
	
		
			
			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>
		
			
				
	
	
		
			148 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			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);
 |