mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 09:22:34 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			152 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| describe("[[DefineProperty]] trap normal behavior", () => {
 | |
|     test("forwarding when not defined in handler", () => {
 | |
|         let p = new Proxy({}, { defineProperty: null });
 | |
|         expect(Object.defineProperty(p, "foo", {})).toBe(p);
 | |
|         p = new Proxy({}, { defineProperty: undefined });
 | |
|         expect(Object.defineProperty(p, "foo", {})).toBe(p);
 | |
|         p = new Proxy({}, {});
 | |
|         expect(Object.defineProperty(p, "foo", {})).toBe(p);
 | |
|     });
 | |
| 
 | |
|     test("correct arguments passed to trap", () => {
 | |
|         let o = {};
 | |
|         p = new Proxy(o, {
 | |
|             defineProperty(target, name, descriptor) {
 | |
|                 expect(target).toBe(o);
 | |
|                 expect(name).toBe("foo");
 | |
|                 expect(descriptor.configurable).toBeTrue();
 | |
|                 expect(descriptor.enumerable).toBeUndefined();
 | |
|                 expect(descriptor.writable).toBeTrue();
 | |
|                 expect(descriptor.value).toBe(10);
 | |
|                 expect(descriptor.get).toBeUndefined();
 | |
|                 expect(descriptor.set).toBeUndefined();
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         Object.defineProperty(p, "foo", { configurable: true, writable: true, value: 10 });
 | |
|     });
 | |
| 
 | |
|     test("correct arguments passed to trap even for number", () => {
 | |
|         let o = {};
 | |
|         p = new Proxy(o, {
 | |
|             defineProperty(target, name, descriptor) {
 | |
|                 expect(target).toBe(o);
 | |
|                 expect(name).toBe("1");
 | |
|                 expect(descriptor.configurable).toBeTrue();
 | |
|                 expect(descriptor.enumerable).toBeUndefined();
 | |
|                 expect(descriptor.writable).toBeTrue();
 | |
|                 expect(descriptor.value).toBe(10);
 | |
|                 expect(descriptor.get).toBeUndefined();
 | |
|                 expect(descriptor.set).toBeUndefined();
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         Object.defineProperty(p, 1, { configurable: true, writable: true, value: 10 });
 | |
|     });
 | |
| 
 | |
|     test("optionally ignoring the define call", () => {
 | |
|         let o = {};
 | |
|         let p = new Proxy(o, {
 | |
|             defineProperty(target, name, descriptor) {
 | |
|                 if (target[name] === undefined) Object.defineProperty(target, name, descriptor);
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         Object.defineProperty(p, "foo", {
 | |
|             value: 10,
 | |
|             enumerable: true,
 | |
|             configurable: false,
 | |
|             writable: true,
 | |
|         });
 | |
|         expect(p).toHaveEnumerableProperty("foo");
 | |
|         expect(p).not.toHaveConfigurableProperty("foo");
 | |
|         expect(p).toHaveWritableProperty("foo");
 | |
|         expect(p).toHaveValueProperty("foo", 10);
 | |
|         expect(p).not.toHaveGetterProperty("foo");
 | |
|         expect(p).not.toHaveSetterProperty("foo");
 | |
| 
 | |
|         Object.defineProperty(p, "foo", {
 | |
|             value: 20,
 | |
|             enumerable: true,
 | |
|             configurable: false,
 | |
|             writable: true,
 | |
|         });
 | |
|         expect(p).toHaveEnumerableProperty("foo");
 | |
|         expect(p).not.toHaveConfigurableProperty("foo");
 | |
|         expect(p).toHaveWritableProperty("foo");
 | |
|         expect(p).toHaveValueProperty("foo", 10);
 | |
|         expect(p).not.toHaveGetterProperty("foo");
 | |
|         expect(p).not.toHaveSetterProperty("foo");
 | |
|     });
 | |
| });
 | |
| 
 | |
| describe("[[DefineProperty]] invariants", () => {
 | |
|     test("trap can't return false", () => {
 | |
|         let p = new Proxy(
 | |
|             {},
 | |
|             {
 | |
|                 defineProperty() {
 | |
|                     return false;
 | |
|                 },
 | |
|             }
 | |
|         );
 | |
| 
 | |
|         expect(() => {
 | |
|             Object.defineProperty(p, "foo", {});
 | |
|         }).toThrowWithMessage(TypeError, "Object's [[DefineOwnProperty]] method returned false");
 | |
|     });
 | |
| 
 | |
|     test("trap cannot return true for a non-extensible target if the property does not exist", () => {
 | |
|         let o = {};
 | |
|         Object.preventExtensions(o);
 | |
|         let p = new Proxy(o, {
 | |
|             defineProperty() {
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         expect(() => {
 | |
|             Object.defineProperty(p, "foo", {});
 | |
|         }).toThrowWithMessage(
 | |
|             TypeError,
 | |
|             "Proxy handler's defineProperty trap violates invariant: a property cannot be reported as being defined if the property does not exist on the target and the target is non-extensible"
 | |
|         );
 | |
|     });
 | |
| 
 | |
|     test("trap cannot return true for a non-configurable property if it doesn't already exist on the target", () => {
 | |
|         let o = {};
 | |
|         Object.defineProperty(o, "foo", { value: 10, configurable: true });
 | |
|         let p = new Proxy(o, {
 | |
|             defineProperty() {
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         expect(() => {
 | |
|             Object.defineProperty(p, "bar", { value: 6, configurable: false });
 | |
|         }).toThrowWithMessage(
 | |
|             TypeError,
 | |
|             "Proxy handler's defineProperty trap violates invariant: a property cannot be defined as non-configurable if it does not already exist on the target object"
 | |
|         );
 | |
|     });
 | |
| 
 | |
|     test("trap cannot return true for a non-configurable property if it already exists as a configurable property", () => {
 | |
|         let o = {};
 | |
|         Object.defineProperty(o, "foo", { value: 10, configurable: true });
 | |
|         let p = new Proxy(o, {
 | |
|             defineProperty() {
 | |
|                 return true;
 | |
|             },
 | |
|         });
 | |
| 
 | |
|         expect(() => {
 | |
|             Object.defineProperty(p, "foo", { value: 6, configurable: false });
 | |
|         }).toThrowWithMessage(
 | |
|             TypeError,
 | |
|             "Proxy handler's defineProperty trap violates invariant: a property cannot be defined as non-configurable if it already exists on the target object as a configurable property"
 | |
|         );
 | |
|     });
 | |
| });
 | 
