summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIdan Horowitz <idan.horowitz@gmail.com>2021-06-06 00:43:03 +0300
committerLinus Groh <mail@linusgroh.de>2021-06-05 23:54:08 +0100
commit2a8f4f097c30fd15f8d421aa062dcab3f63cc13f (patch)
tree8bd56b53b4ad6e05e867bd017fb270331c323185
parent31534055e4e2ab17d3db6cb761e528568cd6e5f9 (diff)
downloadserenity-2a8f4f097c30fd15f8d421aa062dcab3f63cc13f.zip
LibJS: Throw TypeError on write to non-writable property in strict mode
-rw-r--r--Userland/Libraries/LibJS/Runtime/ErrorTypes.h1
-rw-r--r--Userland/Libraries/LibJS/Runtime/Object.cpp2
-rw-r--r--Userland/Libraries/LibJS/Tests/non-writable-assignment.js12
3 files changed, 15 insertions, 0 deletions
diff --git a/Userland/Libraries/LibJS/Runtime/ErrorTypes.h b/Userland/Libraries/LibJS/Runtime/ErrorTypes.h
index f4fcf82b2a..8b11a0faa8 100644
--- a/Userland/Libraries/LibJS/Runtime/ErrorTypes.h
+++ b/Userland/Libraries/LibJS/Runtime/ErrorTypes.h
@@ -23,6 +23,7 @@
M(Convert, "Cannot convert {} to {}") \
M(ConvertUndefinedToObject, "Cannot convert undefined to object") \
M(DescChangeNonConfigurable, "Cannot change attributes of non-configurable property '{}'") \
+ M(DescWriteNonWritable, "Cannot write to non-writable property '{}'") \
M(DivisionByZero, "Division by zero") \
M(FunctionArgsNotObject, "Argument array must be an object") \
M(GetCapabilitiesExecutorCalledMultipleTimes, "GetCapabilitiesExecutor was called multiple times") \
diff --git a/Userland/Libraries/LibJS/Runtime/Object.cpp b/Userland/Libraries/LibJS/Runtime/Object.cpp
index c7fa05b863..4667eed1d2 100644
--- a/Userland/Libraries/LibJS/Runtime/Object.cpp
+++ b/Userland/Libraries/LibJS/Runtime/Object.cpp
@@ -637,6 +637,8 @@ bool Object::put_own_property(const StringOrSymbol& property_name, Value value,
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())
+ vm().throw_exception<TypeError>(global_object(), ErrorType::DescWriteNonWritable, property_name.to_display_string());
return false;
}
diff --git a/Userland/Libraries/LibJS/Tests/non-writable-assignment.js b/Userland/Libraries/LibJS/Tests/non-writable-assignment.js
new file mode 100644
index 0000000000..2674cf6221
--- /dev/null
+++ b/Userland/Libraries/LibJS/Tests/non-writable-assignment.js
@@ -0,0 +1,12 @@
+test("normal mode", () => {
+ expect(() => {
+ NaN = 5; // NaN is a non-writable global variable
+ }).not.toThrow();
+});
+
+test("strict mode", () => {
+ expect(() => {
+ "use strict";
+ NaN = 5; // NaN is a non-writable global variable
+ }).toThrowWithMessage(TypeError, "Cannot write to non-writable property 'NaN'");
+});