mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 00:12:33 +00:00 
			
		
		
		
	 bfedec6a98
			
		
	
	
		bfedec6a98
		
	
	
	
	
		
			
			Previously the variable and lexical environments were already kept in a NativeFunction call. However when we (try to) call a private method from within an async function we go through async_block_start which sets up a NativeFunction to call. This is technically not exactly as the spec describes it, as that requires you to actually "continue" the context. Since we don't have that concept (yet) we use this as an implementation detail to access the private environment from within a native function. Note that this not allow general private environment access since most things get blocked by the parser already.
		
			
				
	
	
		
			144 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| test("basic functionality", () => {
 | |
|     class A {
 | |
|         static method() {
 | |
|             return 10;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect(A.method()).toBe(10);
 | |
|     expect(new A().method).toBeUndefined();
 | |
| });
 | |
| 
 | |
| test("extended name syntax", () => {
 | |
|     class A {
 | |
|         static method() {
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         static 12() {
 | |
|             return 2;
 | |
|         }
 | |
| 
 | |
|         static [`he${"llo"}`]() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect(A.method()).toBe(1);
 | |
|     expect(A[12]()).toBe(2);
 | |
|     expect(A.hello()).toBe(3);
 | |
| });
 | |
| 
 | |
| test("bound |this|", () => {
 | |
|     class A {
 | |
|         static method() {
 | |
|             expect(this).toBe(A);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     A.method();
 | |
| });
 | |
| 
 | |
| test("inherited static methods", () => {
 | |
|     class Parent {
 | |
|         static method() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {}
 | |
| 
 | |
|     expect(Parent.method()).toBe(3);
 | |
|     expect(Child.method()).toBe(3);
 | |
|     expect(new Parent()).not.toHaveProperty("method");
 | |
|     expect(new Child()).not.toHaveProperty("method");
 | |
| });
 | |
| 
 | |
| test("static method overriding", () => {
 | |
|     class Parent {
 | |
|         static method() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         static method() {
 | |
|             return 10;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect(Parent.method()).toBe(3);
 | |
|     expect(Child.method()).toBe(10);
 | |
| });
 | |
| 
 | |
| test("static function named 'async'", () => {
 | |
|     class A {
 | |
|         static async() {
 | |
|             return "static function named async";
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect("async" in A).toBeTrue();
 | |
|     expect(A.async()).toBe("static function named async");
 | |
| });
 | |
| 
 | |
| test("can call other private methods from static method", () => {
 | |
|     class A {
 | |
|         static #a() {
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         static async #b() {
 | |
|             return 2;
 | |
|         }
 | |
| 
 | |
|         static syncA() {
 | |
|             return this.#a();
 | |
|         }
 | |
| 
 | |
|         static async asyncA() {
 | |
|             return this.#a();
 | |
|         }
 | |
| 
 | |
|         static syncB() {
 | |
|             return this.#b();
 | |
|         }
 | |
| 
 | |
|         static async asyncB() {
 | |
|             return this.#b();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     var called = false;
 | |
| 
 | |
|     async function check() {
 | |
|         called = true;
 | |
|         expect(A.syncA()).toBe(1);
 | |
|         expect(await A.asyncA()).toBe(1);
 | |
|         expect(await A.syncB()).toBe(2);
 | |
|         expect(await A.asyncB()).toBe(2);
 | |
|         return 3;
 | |
|     }
 | |
| 
 | |
|     var error = null;
 | |
|     var failed = false;
 | |
| 
 | |
|     check().then(
 | |
|         value => {
 | |
|             expect(called).toBeTrue();
 | |
|             expect(value).toBe(3);
 | |
|         },
 | |
|         thrownError => {
 | |
|             failed = true;
 | |
|             error = thrownError;
 | |
|         }
 | |
|     );
 | |
| 
 | |
|     runQueuedPromiseJobs();
 | |
| 
 | |
|     expect(called).toBeTrue();
 | |
| 
 | |
|     if (failed) throw error;
 | |
| 
 | |
|     expect(failed).toBeFalse();
 | |
| });
 |