summaryrefslogtreecommitdiff
path: root/Userland/Libraries
diff options
context:
space:
mode:
authorIdan Horowitz <idan.horowitz@gmail.com>2021-06-16 13:07:16 +0300
committerLinus Groh <mail@linusgroh.de>2021-06-16 12:57:48 +0100
commitc619ad4fecc7ba95462539786ff65b6325724cbd (patch)
tree1651a4fd56b3cc840433787bc7a4b155db5f84d9 /Userland/Libraries
parent0c8dce60a2481380a31f4ae527e033e63d2ed295 (diff)
downloadserenity-c619ad4fecc7ba95462539786ff65b6325724cbd.zip
LibJS: Allow no-op define property calls on non-configurable objects
This brings us slightly closer to the specification's 10.1.6.3 ValidateAndApplyPropertyDescriptor.
Diffstat (limited to 'Userland/Libraries')
-rw-r--r--Userland/Libraries/LibJS/Runtime/Object.cpp35
-rw-r--r--Userland/Libraries/LibJS/Tests/builtins/Object/Object.defineProperty.js4
2 files changed, 31 insertions, 8 deletions
diff --git a/Userland/Libraries/LibJS/Runtime/Object.cpp b/Userland/Libraries/LibJS/Runtime/Object.cpp
index 930519b35e..50f13d7776 100644
--- a/Userland/Libraries/LibJS/Runtime/Object.cpp
+++ b/Userland/Libraries/LibJS/Runtime/Object.cpp
@@ -657,11 +657,35 @@ bool Object::put_own_property(const StringOrSymbol& property_name, Value value,
VERIFY(metadata.has_value());
}
- if (!new_property && mode == PutOwnPropertyMode::DefineProperty && !metadata.value().attributes.is_configurable() && attributes != metadata.value().attributes) {
- dbgln_if(OBJECT_DEBUG, "Disallow reconfig of non-configurable property");
- if (throw_exceptions)
- vm().throw_exception<TypeError>(global_object(), ErrorType::DescChangeNonConfigurable, property_name.to_display_string());
- return false;
+ auto value_here = m_storage[metadata.value().offset];
+ if (!new_property && mode == PutOwnPropertyMode::DefineProperty && !metadata.value().attributes.is_configurable()) {
+ if ((attributes.has_configurable() && attributes.is_configurable()) || (attributes.has_enumerable() && attributes.is_enumerable() != metadata.value().attributes.is_enumerable())) {
+ dbgln_if(OBJECT_DEBUG, "Disallow reconfig of non-configurable property");
+ if (throw_exceptions)
+ vm().throw_exception<TypeError>(global_object(), ErrorType::DescChangeNonConfigurable, property_name.to_display_string());
+ return false;
+ }
+
+ if (value_here.is_accessor() != value.is_accessor()) {
+ dbgln_if(OBJECT_DEBUG, "Disallow reconfig of non-configurable property");
+ if (throw_exceptions)
+ vm().throw_exception<TypeError>(global_object(), ErrorType::DescChangeNonConfigurable, property_name.to_display_string());
+ return false;
+ }
+
+ if (!value_here.is_accessor() && !metadata.value().attributes.is_writable() && ((attributes.has_writable() && attributes.is_writable()) || (!value.is_empty() && !same_value(value, value_here)))) {
+ dbgln_if(OBJECT_DEBUG, "Disallow reconfig of non-configurable property");
+ if (throw_exceptions)
+ vm().throw_exception<TypeError>(global_object(), ErrorType::DescChangeNonConfigurable, property_name.to_display_string());
+ return false;
+ }
+
+ if (value_here.is_accessor() && ((attributes.has_setter() && value.as_accessor().setter() != value_here.as_accessor().setter()) || (attributes.has_getter() && value.as_accessor().getter() != value_here.as_accessor().getter()))) {
+ dbgln_if(OBJECT_DEBUG, "Disallow reconfig of non-configurable property");
+ if (throw_exceptions)
+ vm().throw_exception<TypeError>(global_object(), ErrorType::DescChangeNonConfigurable, property_name.to_display_string());
+ return false;
+ }
}
if (mode == PutOwnPropertyMode::DefineProperty && attributes != metadata.value().attributes) {
@@ -675,7 +699,6 @@ bool Object::put_own_property(const StringOrSymbol& property_name, Value value,
dbgln_if(OBJECT_DEBUG, "Reconfigured property {}, new shape says offset is {} and my storage capacity is {}", property_name.to_display_string(), metadata.value().offset, m_storage.size());
}
- auto value_here = m_storage[metadata.value().offset];
if (!new_property && mode == PutOwnPropertyMode::Put && !value_here.is_accessor() && !metadata.value().attributes.is_writable()) {
dbgln_if(OBJECT_DEBUG, "Disallow write to non-writable property");
if (throw_exceptions && vm().in_strict_mode())
diff --git a/Userland/Libraries/LibJS/Tests/builtins/Object/Object.defineProperty.js b/Userland/Libraries/LibJS/Tests/builtins/Object/Object.defineProperty.js
index bebe32997d..644091c90c 100644
--- a/Userland/Libraries/LibJS/Tests/builtins/Object/Object.defineProperty.js
+++ b/Userland/Libraries/LibJS/Tests/builtins/Object/Object.defineProperty.js
@@ -166,7 +166,7 @@ describe("errors", () => {
Object.defineProperty(o, "foo", { value: 1, writable: true, enumerable: true });
expect(() => {
- Object.defineProperty(o, "foo", { value: 2, writable: false, enumerable: true });
+ Object.defineProperty(o, "foo", { value: 2, writable: true, enumerable: false });
}).toThrowWithMessage(
TypeError,
"Cannot change attributes of non-configurable property 'foo'"
@@ -179,7 +179,7 @@ describe("errors", () => {
Object.defineProperty(o, s, { value: 1, writable: true, enumerable: true });
expect(() => {
- Object.defineProperty(o, s, { value: 2, writable: false, enumerable: true });
+ Object.defineProperty(o, s, { value: 2, writable: true, enumerable: false });
}).toThrowWithMessage(
TypeError,
"Cannot change attributes of non-configurable property 'Symbol(foo)'"