1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-25 18:22:32 +00:00
serenity/Userland/Libraries/LibJS/Tests/builtins/Object/Object.freeze.js
Linus Groh ac3e7ef791 LibJS: Fix crash in Object.{freeze,seal}() with indexed properties
This was failing to take two things into account:

- When constructing a PropertyName from a value, it won't automatically
  convert to Type::Number for something like string "0", even though
  that's how things work internally, since indexed properties are stored
  separately. This will be improved in a future patch, it's a footgun
  and should happen automatically.
- Those can't be looked up on the shape, we have to go through the
  indexed properties instead.

Additionally it now operates on the shape or indexed properties directly
as define_property() was overly strict and would throw if a property was
already non-configurable.

Fixes #6469.
2021-04-20 09:38:22 +02:00

73 lines
2.1 KiB
JavaScript

test("length is 1", () => {
expect(Object.freeze).toHaveLength(1);
});
describe("normal behavior", () => {
test("returns given argument", () => {
const o = {};
expect(Object.freeze(42)).toBe(42);
expect(Object.freeze("foobar")).toBe("foobar");
expect(Object.freeze(o)).toBe(o);
});
test("prevents addition of new properties", () => {
const o = {};
expect(o.foo).toBeUndefined();
Object.freeze(o);
o.foo = "bar";
expect(o.foo).toBeUndefined();
});
test("prevents deletion of existing properties", () => {
const o = { foo: "bar" };
expect(o.foo).toBe("bar");
Object.freeze(o);
delete o.foo;
expect(o.foo).toBe("bar");
});
test("prevents changing attributes of existing properties", () => {
const o = { foo: "bar" };
Object.freeze(o);
// FIXME: These don't change anything and should not throw!
// expect(Object.defineProperty(o, "foo", {})).toBe(o);
// expect(Object.defineProperty(o, "foo", { configurable: false })).toBe(o);
expect(() => {
Object.defineProperty(o, "foo", { configurable: true });
}).toThrowWithMessage(
TypeError,
"Cannot change attributes of non-configurable property 'foo'"
);
});
test("prevents changing value of existing properties", () => {
const o = { foo: "bar" };
expect(o.foo).toBe("bar");
Object.freeze(o);
o.foo = "baz";
expect(o.foo).toBe("bar");
});
// #6469
test("works with indexed properties", () => {
const a = ["foo"];
expect(a[0]).toBe("foo");
Object.freeze(a);
a[0] = "bar";
expect(a[0]).toBe("foo");
});
test("works with properties that are already non-configurable", () => {
const o = {};
Object.defineProperty(o, "foo", {
value: "bar",
configurable: false,
writable: true,
enumerable: true,
});
expect(o.foo).toBe("bar");
Object.freeze(o);
o.foo = "baz";
expect(o.foo).toBe("bar");
});
});