mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 15:52:07 +00:00 
			
		
		
		
	 16ac43c9d4
			
		
	
	
		16ac43c9d4
		
	
	
	
	
		
			
			If we don't check that a private identifier is valid this can break the assumption that we have a private environment when evaluation the private identifier. Also an unknown private identifier this should be a SyntaxError.
		
			
				
	
	
		
			185 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			185 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| test("basic functionality", () => {
 | |
|     class A {
 | |
|         #number = 3;
 | |
| 
 | |
|         getNumber() {
 | |
|             return this.#number;
 | |
|         }
 | |
| 
 | |
|         #string = "foo";
 | |
| 
 | |
|         getString() {
 | |
|             return this.#string;
 | |
|         }
 | |
| 
 | |
|         #uninitialized;
 | |
| 
 | |
|         getUninitialized() {
 | |
|             return this.#uninitialized;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const a = new A();
 | |
|     expect(a.getNumber()).toBe(3);
 | |
|     expect(a.getString()).toBe("foo");
 | |
|     expect(a.getUninitialized()).toBeUndefined();
 | |
| 
 | |
|     expect("a.#number").not.toEval();
 | |
|     expect("a.#string").not.toEval();
 | |
|     expect("a.#uninitialized").not.toEval();
 | |
| });
 | |
| 
 | |
| test("initializer has correct this value", () => {
 | |
|     class A {
 | |
|         #thisVal = this;
 | |
| 
 | |
|         getThisVal() {
 | |
|             return this.#thisVal;
 | |
|         }
 | |
| 
 | |
|         #thisName = this.#thisVal;
 | |
| 
 | |
|         getThisName() {
 | |
|             return this.#thisName;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const a = new A();
 | |
|     expect(a.getThisVal()).toBe(a);
 | |
|     expect(a.getThisName()).toBe(a);
 | |
| });
 | |
| 
 | |
| test("static fields", () => {
 | |
|     class A {
 | |
|         static #simple = 1;
 | |
| 
 | |
|         static getStaticSimple() {
 | |
|             return this.#simple;
 | |
|         }
 | |
| 
 | |
|         static #thisVal = this;
 | |
|         static #thisName = this.name;
 | |
|         static #thisVal2 = this.#thisVal;
 | |
| 
 | |
|         static getThisVal() {
 | |
|             return this.#thisVal;
 | |
|         }
 | |
| 
 | |
|         static getThisName() {
 | |
|             return this.#thisName;
 | |
|         }
 | |
| 
 | |
|         static getThisVal2() {
 | |
|             return this.#thisVal2;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect(A.getStaticSimple()).toBe(1);
 | |
| 
 | |
|     expect(A.getThisVal()).toBe(A);
 | |
|     expect(A.getThisName()).toBe("A");
 | |
|     expect(A.getThisVal2()).toBe(A);
 | |
| 
 | |
|     expect("A.#simple").not.toEval();
 | |
| });
 | |
| 
 | |
| test("slash after private identifier is treated as division", () => {
 | |
|     class A {
 | |
|         static #field = 4;
 | |
|         static #divided = this.#field / 2;
 | |
| 
 | |
|         static getDivided() {
 | |
|             return this.#divided;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     expect(A.getDivided()).toBe(2);
 | |
| });
 | |
| 
 | |
| test("private identifier not followed by 'in' throws", () => {
 | |
|     expect(`class A { #field = 2; method() { return #field instanceof 1; }}`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return #field < 1; }}`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return #field + 1; }}`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return #field ** 1; }}`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return !#field; } }`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return ~#field; } }`).not.toEval();
 | |
|     expect(`class A { #field = 2; method() { return ++#field; } }`).not.toEval();
 | |
| 
 | |
|     expect(`class A { #field = 2; method() { return #field in 1; }}`).toEval();
 | |
| });
 | |
| 
 | |
| test("cannot have static and non static field with the same description", () => {
 | |
|     expect("class A { static #simple; #simple; }").not.toEval();
 | |
| });
 | |
| 
 | |
| test("'arguments' is not allowed in class field initializer", () => {
 | |
|     expect("class A { #a = arguments; }").not.toEval();
 | |
|     expect("class B { static #b = arguments; }").not.toEval();
 | |
| 
 | |
|     class C {
 | |
|         #c = eval("arguments");
 | |
|     }
 | |
| 
 | |
|     expect(() => {
 | |
|         new C();
 | |
|     }).toThrowWithMessage(SyntaxError, "'arguments' is not allowed in class field initializer");
 | |
| 
 | |
|     expect(() => {
 | |
|         class D {
 | |
|             static #d = eval("arguments");
 | |
|         }
 | |
|     }).toThrowWithMessage(SyntaxError, "'arguments' is not allowed in class field initializer");
 | |
| });
 | |
| 
 | |
| test("using 'arguments' via indirect eval throws at runtime instead of parse time", () => {
 | |
|     const indirect = eval;
 | |
| 
 | |
|     class A {
 | |
|         #a = indirect("arguments");
 | |
|     }
 | |
| 
 | |
|     expect(() => {
 | |
|         new A();
 | |
|     }).toThrowWithMessage(ReferenceError, "'arguments' is not defined");
 | |
| 
 | |
|     expect(() => {
 | |
|         class B {
 | |
|             static #b = indirect("arguments");
 | |
|         }
 | |
|     }).toThrowWithMessage(ReferenceError, "'arguments' is not defined");
 | |
| });
 | |
| 
 | |
| test("unknown private name gives SyntaxError", () => {
 | |
|     expect(`#n`).not.toEval();
 | |
|     expect(`obj.#n`).not.toEval();
 | |
|     expect(`this.#n`).not.toEval();
 | |
|     expect(`if (#n) 1;`).not.toEval();
 | |
|     expect(`1?.#n`).not.toEval();
 | |
|     expect(`1?.n.#n`).not.toEval();
 | |
| });
 | |
| 
 | |
| // OSS-FUZZ Issue 53363: top level unknown private names seg faults
 | |
| expect(() => eval(`#n`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 | |
| expect(() => eval(`obj.#n`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 | |
| expect(() => eval(`this.#n`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 | |
| expect(() => eval(`if (#n) 1;`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 | |
| expect(() => eval(`1?.#n`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 | |
| expect(() => eval(`1?.n.#n`)).toThrowWithMessage(
 | |
|     SyntaxError,
 | |
|     "Reference to undeclared private field or method '#n'"
 | |
| );
 |