123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 |
- /*
- * Copyright (c) 2021, Linus Groh <linusg@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <LibJS/Runtime/Error.h>
- #include <LibJS/Runtime/FunctionObject.h>
- #include <LibJS/Runtime/GlobalObject.h>
- #include <LibJS/Runtime/Object.h>
- #include <LibJS/Runtime/PropertyDescriptor.h>
- #include <LibJS/Runtime/Value.h>
- namespace JS {
- // 6.2.5.1 IsAccessorDescriptor ( Desc ), https://tc39.es/ecma262/#sec-isaccessordescriptor
- bool PropertyDescriptor::is_accessor_descriptor() const
- {
- // 1. If Desc is undefined, return false.
- // 2. If both Desc.[[Get]] and Desc.[[Set]] are absent, return false.
- if (!get.has_value() && !set.has_value())
- return false;
- // 3. Return true.
- return true;
- }
- // 6.2.5.2 IsDataDescriptor ( Desc ), https://tc39.es/ecma262/#sec-isdatadescriptor
- bool PropertyDescriptor::is_data_descriptor() const
- {
- // 1. If Desc is undefined, return false.
- // 2. If both Desc.[[Value]] and Desc.[[Writable]] are absent, return false.
- if (!value.has_value() && !writable.has_value())
- return false;
- // 3. Return true.
- return true;
- }
- // 6.2.5.3 IsGenericDescriptor ( Desc ), https://tc39.es/ecma262/#sec-isgenericdescriptor
- bool PropertyDescriptor::is_generic_descriptor() const
- {
- // 1. If Desc is undefined, return false.
- // 2. If IsAccessorDescriptor(Desc) and IsDataDescriptor(Desc) are both false, return true.
- if (!is_accessor_descriptor() && !is_data_descriptor())
- return true;
- // 3. Return false.
- return false;
- }
- // 6.2.5.4 FromPropertyDescriptor ( Desc ), https://tc39.es/ecma262/#sec-frompropertydescriptor
- Value from_property_descriptor(GlobalObject& global_object, Optional<PropertyDescriptor> const& property_descriptor)
- {
- if (!property_descriptor.has_value())
- return js_undefined();
- auto& vm = global_object.vm();
- auto* object = Object::create(global_object, global_object.object_prototype());
- if (property_descriptor->value.has_value())
- MUST(object->create_data_property_or_throw(vm.names.value, *property_descriptor->value));
- if (property_descriptor->writable.has_value())
- MUST(object->create_data_property_or_throw(vm.names.writable, Value(*property_descriptor->writable)));
- if (property_descriptor->get.has_value())
- MUST(object->create_data_property_or_throw(vm.names.get, *property_descriptor->get ? Value(*property_descriptor->get) : js_undefined()));
- if (property_descriptor->set.has_value())
- MUST(object->create_data_property_or_throw(vm.names.set, *property_descriptor->set ? Value(*property_descriptor->set) : js_undefined()));
- if (property_descriptor->enumerable.has_value())
- MUST(object->create_data_property_or_throw(vm.names.enumerable, Value(*property_descriptor->enumerable)));
- if (property_descriptor->configurable.has_value())
- MUST(object->create_data_property_or_throw(vm.names.configurable, Value(*property_descriptor->configurable)));
- return object;
- }
- // 6.2.5.5 ToPropertyDescriptor ( Obj ), https://tc39.es/ecma262/#sec-topropertydescriptor
- ThrowCompletionOr<PropertyDescriptor> to_property_descriptor(GlobalObject& global_object, Value argument)
- {
- auto& vm = global_object.vm();
- // 1. If Type(Obj) is not Object, throw a TypeError exception.
- if (!argument.is_object())
- return vm.throw_completion<TypeError>(global_object, ErrorType::NotAnObject, argument.to_string_without_side_effects());
- auto& object = argument.as_object();
- // 2. Let desc be a new Property Descriptor that initially has no fields.
- PropertyDescriptor descriptor;
- // 3. Let hasEnumerable be ? HasProperty(Obj, "enumerable").
- auto has_enumerable = TRY(object.has_property(vm.names.enumerable));
- // 4. If hasEnumerable is true, then
- if (has_enumerable) {
- // a. Let enumerable be ! ToBoolean(? Get(Obj, "enumerable")).
- auto enumerable = TRY(object.get(vm.names.enumerable)).to_boolean();
- // b. Set desc.[[Enumerable]] to enumerable.
- descriptor.enumerable = enumerable;
- }
- // 5. Let hasConfigurable be ? HasProperty(Obj, "configurable").
- auto has_configurable = TRY(object.has_property(vm.names.configurable));
- // 6. If hasConfigurable is true, then
- if (has_configurable) {
- // a. Let configurable be ! ToBoolean(? Get(Obj, "configurable")).
- auto configurable = TRY(object.get(vm.names.configurable)).to_boolean();
- // b. Set desc.[[Configurable]] to configurable.
- descriptor.configurable = configurable;
- }
- // 7. Let hasValue be ? HasProperty(Obj, "value").
- auto has_value = TRY(object.has_property(vm.names.value));
- // 8. If hasValue is true, then
- if (has_value) {
- // a. Let value be ? Get(Obj, "value").
- auto value = TRY(object.get(vm.names.value));
- // b. Set desc.[[Value]] to value.
- descriptor.value = value;
- }
- // 9. Let hasWritable be ? HasProperty(Obj, "writable").
- auto has_writable = TRY(object.has_property(vm.names.writable));
- // 10. If hasWritable is true, then
- if (has_writable) {
- // a. Let writable be ! ToBoolean(? Get(Obj, "writable")).
- auto writable = TRY(object.get(vm.names.writable)).to_boolean();
- // b. Set desc.[[Writable]] to writable.
- descriptor.writable = writable;
- }
- // 11. Let hasGet be ? HasProperty(Obj, "get").
- auto has_get = TRY(object.has_property(vm.names.get));
- // 12. If hasGet is true, then
- if (has_get) {
- // a. Let getter be ? Get(Obj, "get").
- auto getter = TRY(object.get(vm.names.get));
- // b. If IsCallable(getter) is false and getter is not undefined, throw a TypeError exception.
- if (!getter.is_function() && !getter.is_undefined())
- return vm.throw_completion<TypeError>(global_object, ErrorType::AccessorBadField, "get");
- // c. Set desc.[[Get]] to getter.
- descriptor.get = getter.is_function() ? &getter.as_function() : nullptr;
- }
- // 13. Let hasSet be ? HasProperty(Obj, "set").
- auto has_set = TRY(object.has_property(vm.names.set));
- // 14. If hasSet is true, then
- if (has_set) {
- // a. Let setter be ? Get(Obj, "set").
- auto setter = TRY(object.get(vm.names.set));
- // b. If IsCallable(setter) is false and setter is not undefined, throw a TypeError exception.
- if (!setter.is_function() && !setter.is_undefined())
- return vm.throw_completion<TypeError>(global_object, ErrorType::AccessorBadField, "set");
- // c. Set desc.[[Set]] to setter.
- descriptor.set = setter.is_function() ? &setter.as_function() : nullptr;
- }
- // 15. If desc.[[Get]] is present or desc.[[Set]] is present, then
- if (descriptor.get.has_value() || descriptor.set.has_value()) {
- // a. If desc.[[Value]] is present or desc.[[Writable]] is present, throw a TypeError exception.
- if (descriptor.value.has_value() || descriptor.writable.has_value())
- return vm.throw_completion<TypeError>(global_object, ErrorType::AccessorValueOrWritable);
- }
- // 16. Return desc.
- return descriptor;
- }
- // 6.2.5.6 CompletePropertyDescriptor ( Desc ), https://tc39.es/ecma262/#sec-completepropertydescriptor
- void PropertyDescriptor::complete()
- {
- if (is_generic_descriptor() || is_data_descriptor()) {
- if (!value.has_value())
- value = Value {};
- if (!writable.has_value())
- writable = false;
- } else {
- if (!get.has_value())
- get = nullptr;
- if (!set.has_value())
- set = nullptr;
- }
- if (!enumerable.has_value())
- enumerable = false;
- if (!configurable.has_value())
- configurable = false;
- }
- // Non-standard, just a convenient way to get from three Optional<bool> to PropertyAttributes.
- PropertyAttributes PropertyDescriptor::attributes() const
- {
- u8 attributes = 0;
- if (writable.value_or(false))
- attributes |= Attribute::Writable;
- if (enumerable.value_or(false))
- attributes |= Attribute::Enumerable;
- if (configurable.value_or(false))
- attributes |= Attribute::Configurable;
- return { attributes };
- }
- }
|