mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-23 01:52:32 +00:00 
			
		
		
		
	 02e97b3313
			
		
	
	
		02e97b3313
		
	
	
	
	
		
			
			This fixes 2 bugs in our current implementation: * Properties deleted during iteration were still being iterated * Properties with the same name in both the object and it's prototype were iterated twice
		
			
				
	
	
		
			122 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| test("iterate through empty string", () => {
 | |
|     const a = [];
 | |
|     for (const property in "") {
 | |
|         a.push(property);
 | |
|     }
 | |
|     expect(a).toEqual([]);
 | |
| });
 | |
| 
 | |
| test("iterate through number", () => {
 | |
|     const a = [];
 | |
|     for (const property in 123) {
 | |
|         a.push(property);
 | |
|     }
 | |
|     expect(a).toEqual([]);
 | |
| });
 | |
| 
 | |
| test("iterate through empty object", () => {
 | |
|     const a = [];
 | |
|     for (const property in {}) {
 | |
|         a.push(property);
 | |
|     }
 | |
|     expect(a).toEqual([]);
 | |
| });
 | |
| 
 | |
| test("iterate through string", () => {
 | |
|     const a = [];
 | |
|     for (const property in "hello") {
 | |
|         a.push(property);
 | |
|     }
 | |
|     expect(a).toEqual(["0", "1", "2", "3", "4"]);
 | |
| });
 | |
| 
 | |
| test("iterate through object", () => {
 | |
|     const a = [];
 | |
|     for (const property in { a: 1, b: 2, c: 2 }) {
 | |
|         a.push(property);
 | |
|     }
 | |
|     expect(a).toEqual(["a", "b", "c"]);
 | |
| });
 | |
| 
 | |
| test("iterate through undefined", () => {
 | |
|     for (const property in undefined) {
 | |
|         expect.fail();
 | |
|     }
 | |
| });
 | |
| 
 | |
| test("use already-declared variable", () => {
 | |
|     var property;
 | |
|     for (property in "abc");
 | |
|     expect(property).toBe("2");
 | |
| });
 | |
| 
 | |
| test("allow binding patterns", () => {
 | |
|     const expected = [
 | |
|         ["1", "3", []],
 | |
|         ["s", undefined, []],
 | |
|         ["l", "n", ["g", "N", "a", "m", "e"]],
 | |
|     ];
 | |
|     let counter = 0;
 | |
| 
 | |
|     for (let [a, , b, ...c] in { 123: 1, sm: 2, longName: 3 }) {
 | |
|         expect(a).toBe(expected[counter][0]);
 | |
|         expect(b).toBe(expected[counter][1]);
 | |
|         expect(c).toEqual(expected[counter][2]);
 | |
|         counter++;
 | |
|     }
 | |
|     expect(counter).toBe(3);
 | |
| });
 | |
| 
 | |
| describe("special left hand sides", () => {
 | |
|     test("allow member expression as variable", () => {
 | |
|         const f = {};
 | |
|         for (f.a in "abc");
 | |
|         expect(f.a).toBe("2");
 | |
|     });
 | |
| 
 | |
|     test("allow member expression of function call", () => {
 | |
|         const b = {};
 | |
|         function f() {
 | |
|             return b;
 | |
|         }
 | |
| 
 | |
|         for (f().a in "abc");
 | |
| 
 | |
|         expect(f().a).toBe("2");
 | |
|         expect(b.a).toBe("2");
 | |
|     });
 | |
| 
 | |
|     test("call function is allowed in parsing but fails in runtime", () => {
 | |
|         function f() {
 | |
|             expect().fail();
 | |
|         }
 | |
| 
 | |
|         // Does not fail since it does not iterate
 | |
|         expect("for (f() in []);").toEvalTo(undefined);
 | |
| 
 | |
|         expect(() => {
 | |
|             eval("for (f() in [0]) { expect().fail() }");
 | |
|         }).toThrowWithMessage(ReferenceError, "Invalid left-hand side in assignment");
 | |
|     });
 | |
| });
 | |
| 
 | |
| test("remove properties while iterating", () => {
 | |
|     const from = [1, 2, 3];
 | |
|     const to = [];
 | |
|     for (const prop in from) {
 | |
|         to.push(prop);
 | |
|         from.pop();
 | |
|     }
 | |
|     expect(to).toEqual(["0", "1"]);
 | |
| });
 | |
| 
 | |
| test("duplicated properties in prototype", () => {
 | |
|     const object = { a: 1 };
 | |
|     const proto = { a: 2 };
 | |
|     Object.setPrototypeOf(object, proto);
 | |
|     const a = [];
 | |
|     for (const prop in object) {
 | |
|         a.push(prop);
 | |
|     }
 | |
|     expect(a).toEqual(["a"]);
 | |
| });
 |