mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 15:02:06 +00:00 
			
		
		
		
	 c114be95f5
			
		
	
	
		c114be95f5
		
	
	
	
	
		
			
			Before this was a mix of different strategies but copy_data_properties does all of that in a spec way. This fixes numeric properties in object spreading. And ensures that any new properties added during spreading are not taken into account.
		
			
				
	
	
		
			174 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const testObjSpread = obj => {
 | |
|     expect(obj).toEqual({
 | |
|         foo: 0,
 | |
|         bar: 1,
 | |
|         baz: 2,
 | |
|         qux: 3,
 | |
|     });
 | |
| };
 | |
| 
 | |
| const testObjStrSpread = obj => {
 | |
|     expect(obj).toEqual(["a", "b", "c", "d"]);
 | |
| };
 | |
| 
 | |
| test("spread object literal inside object literal", () => {
 | |
|     const obj = {
 | |
|         foo: 0,
 | |
|         ...{ bar: 1, baz: 2 },
 | |
|         qux: 3,
 | |
|     };
 | |
|     testObjSpread(obj);
 | |
| });
 | |
| 
 | |
| test("spread object with assigned property inside object literal", () => {
 | |
|     const obj = { foo: 0, bar: 1, baz: 2 };
 | |
|     obj.qux = 3;
 | |
|     testObjSpread({ ...obj });
 | |
| });
 | |
| 
 | |
| test("spread object inside object literal", () => {
 | |
|     let a = { bar: 1, baz: 2 };
 | |
|     const obj = { foo: 0, ...a, qux: 3 };
 | |
|     testObjSpread(obj);
 | |
| });
 | |
| 
 | |
| test("complex nested object spreading", () => {
 | |
|     const obj = {
 | |
|         ...{},
 | |
|         ...{
 | |
|             ...{ foo: 0, bar: 1, baz: 2 },
 | |
|         },
 | |
|         qux: 3,
 | |
|     };
 | |
|     testObjSpread(obj);
 | |
| });
 | |
| 
 | |
| test("spread string in object literal", () => {
 | |
|     const obj = { ..."abcd" };
 | |
|     testObjStrSpread(obj);
 | |
| });
 | |
| 
 | |
| test("spread array in object literal", () => {
 | |
|     const obj = { ...["a", "b", "c", "d"] };
 | |
|     testObjStrSpread(obj);
 | |
| });
 | |
| 
 | |
| test("spread array with holes in object literal", () => {
 | |
|     const obj = { ...[, , "a", , , , "b", "c", , "d", , ,] };
 | |
|     expect(obj).toEqual({ 2: "a", 6: "b", 7: "c", 9: "d" });
 | |
| });
 | |
| 
 | |
| test("spread string object in object literal", () => {
 | |
|     const obj = { ...String("abcd") };
 | |
|     testObjStrSpread(obj);
 | |
| });
 | |
| 
 | |
| test("spread object with non-enumerable property", () => {
 | |
|     const a = { foo: 0 };
 | |
|     Object.defineProperty(a, "bar", {
 | |
|         value: 1,
 | |
|         enumerable: false,
 | |
|     });
 | |
|     const obj = { ...a };
 | |
|     expect(obj.foo).toBe(0);
 | |
|     expect(obj).not.toHaveProperty("bar");
 | |
| });
 | |
| 
 | |
| test("spread object with symbol keys", () => {
 | |
|     const s = Symbol("baz");
 | |
|     const a = {
 | |
|         foo: "bar",
 | |
|         [s]: "qux",
 | |
|     };
 | |
|     const obj = { ...a };
 | |
|     expect(obj.foo).toBe("bar");
 | |
|     expect(obj[s]).toBe("qux");
 | |
| });
 | |
| 
 | |
| test("spreading non-spreadable values", () => {
 | |
|     let empty = {
 | |
|         ...undefined,
 | |
|         ...null,
 | |
|         ...1,
 | |
|         ...true,
 | |
|         ...function () {},
 | |
|         ...Date,
 | |
|     };
 | |
|     expect(Object.getOwnPropertyNames(empty)).toHaveLength(0);
 | |
| });
 | |
| 
 | |
| test("respects custom Symbol.iterator method", () => {
 | |
|     let o = {
 | |
|         [Symbol.iterator]() {
 | |
|             return {
 | |
|                 i: 0,
 | |
|                 next() {
 | |
|                     if (this.i++ == 3) {
 | |
|                         return { done: true };
 | |
|                     }
 | |
|                     return { value: this.i, done: false };
 | |
|                 },
 | |
|             };
 | |
|         },
 | |
|     };
 | |
| 
 | |
|     let a = [...o];
 | |
|     expect(a).toEqual([1, 2, 3]);
 | |
| });
 | |
| 
 | |
| test("object with numeric indices", () => {
 | |
|     const obj = { 0: 0, 1: 1, foo: "bar" };
 | |
|     const result = { ...obj };
 | |
|     expect(result).toHaveProperty("0", 0);
 | |
|     expect(result).toHaveProperty("1", 1);
 | |
|     expect(result).toHaveProperty("foo", "bar");
 | |
| });
 | |
| 
 | |
| describe("modification of spreadable objects during spread", () => {
 | |
|     test("spreading object", () => {
 | |
|         const object = {
 | |
|             0: 0,
 | |
|             2: 2,
 | |
|             9999: 9999,
 | |
|             bar: 44,
 | |
|             get 3() {
 | |
|                 object[4] = 4;
 | |
|                 object[5000] = 5000;
 | |
|                 return 3;
 | |
|             },
 | |
|         };
 | |
| 
 | |
|         const result = { ...object };
 | |
|         expect(Object.getOwnPropertyNames(result)).toHaveLength(5);
 | |
|         expect(Object.getOwnPropertyNames(result)).not.toContain("4");
 | |
|         expect(Object.getOwnPropertyNames(result)).toContain("bar");
 | |
|     });
 | |
| 
 | |
|     test("spreading array", () => {
 | |
|         const array = [0];
 | |
|         array[2] = 2;
 | |
|         array[999] = 999;
 | |
|         Object.defineProperty(array, 3, {
 | |
|             get() {
 | |
|                 array[4] = 4;
 | |
|                 array[1000] = 1000;
 | |
|                 return 3;
 | |
|             },
 | |
|             enumerable: true,
 | |
|         });
 | |
| 
 | |
|         const objectResult = { ...array };
 | |
|         expect(Object.getOwnPropertyNames(objectResult)).toHaveLength(4);
 | |
|         expect(Object.getOwnPropertyNames(objectResult)).not.toContain("4");
 | |
| 
 | |
|         const arrayResult = [...array];
 | |
|         expect(arrayResult).toHaveLength(1001);
 | |
|         expect(arrayResult).toHaveProperty("0", 0);
 | |
|         expect(arrayResult).toHaveProperty("2", 2);
 | |
|         expect(arrayResult).toHaveProperty("3", 3);
 | |
|         // Yes the in flight added items need to be here in this case! (since it uses an iterator)
 | |
|         expect(arrayResult).toHaveProperty("4", 4);
 | |
|         expect(arrayResult).toHaveProperty("999", 999);
 | |
|         expect(arrayResult).toHaveProperty("1000", 1000);
 | |
|     });
 | |
| });
 |