ModuleNamespaceObject.cpp 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. /*
  2. * Copyright (c) 2022, David Tuin <davidot@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/QuickSort.h>
  7. #include <LibJS/Runtime/GlobalObject.h>
  8. #include <LibJS/Runtime/ModuleNamespaceObject.h>
  9. namespace JS {
  10. ModuleNamespaceObject::ModuleNamespaceObject(Realm& realm, Module* module, Vector<DeprecatedFlyString> exports)
  11. : Object(ConstructWithPrototypeTag::Tag, realm.intrinsics().object_prototype())
  12. , m_module(module)
  13. , m_exports(move(exports))
  14. {
  15. // Note: We just perform step 6 of 10.4.6.12 ModuleNamespaceCreate ( module, exports ), https://tc39.es/ecma262/#sec-modulenamespacecreate
  16. // 6. Let sortedExports be a List whose elements are the elements of exports ordered as if an Array of the same values had been sorted using %Array.prototype.sort% using undefined as comparefn.
  17. quick_sort(m_exports, [&](DeprecatedFlyString const& lhs, DeprecatedFlyString const& rhs) {
  18. return lhs.view() < rhs.view();
  19. });
  20. }
  21. void ModuleNamespaceObject::initialize(Realm& realm)
  22. {
  23. auto& vm = this->vm();
  24. Base::initialize(realm);
  25. // 28.3.1 @@toStringTag, https://tc39.es/ecma262/#sec-@@tostringtag
  26. define_direct_property(vm.well_known_symbol_to_string_tag(), PrimitiveString::create(vm, "Module"_string), 0);
  27. }
  28. // 10.4.6.1 [[GetPrototypeOf]] ( ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-getprototypeof
  29. ThrowCompletionOr<Object*> ModuleNamespaceObject::internal_get_prototype_of() const
  30. {
  31. // 1. Return null.
  32. return nullptr;
  33. }
  34. // 10.4.6.2 [[SetPrototypeOf]] ( V ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-setprototypeof-v
  35. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_set_prototype_of(Object* prototype)
  36. {
  37. // 1. Return ! SetImmutablePrototype(O, V).
  38. return MUST(set_immutable_prototype(prototype));
  39. }
  40. // 10.4.6.3 [[IsExtensible]] ( ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-isextensible
  41. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_is_extensible() const
  42. {
  43. // 1. Return false.
  44. return false;
  45. }
  46. // 10.4.6.4 [[PreventExtensions]] ( ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-preventextensions
  47. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_prevent_extensions()
  48. {
  49. // 1. Return true.
  50. return true;
  51. }
  52. // 10.4.6.5 [[GetOwnProperty]] ( P ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-getownproperty-p
  53. ThrowCompletionOr<Optional<PropertyDescriptor>> ModuleNamespaceObject::internal_get_own_property(PropertyKey const& property_key) const
  54. {
  55. // 1. If Type(P) is Symbol, return OrdinaryGetOwnProperty(O, P).
  56. if (property_key.is_symbol())
  57. return Object::internal_get_own_property(property_key);
  58. // 2. Let exports be O.[[Exports]].
  59. // 3. If P is not an element of exports, return undefined.
  60. auto export_element = m_exports.find(property_key.to_string());
  61. if (export_element.is_end())
  62. return Optional<PropertyDescriptor> {};
  63. // 4. Let value be ? O.[[Get]](P, O).
  64. auto value = TRY(internal_get(property_key, this));
  65. // 5. Return PropertyDescriptor { [[Value]]: value, [[Writable]]: true, [[Enumerable]]: true, [[Configurable]]: false }.
  66. return PropertyDescriptor { .value = value, .writable = true, .enumerable = true, .configurable = false };
  67. }
  68. // 10.4.6.6 [[DefineOwnProperty]] ( P, Desc ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-defineownproperty-p-desc
  69. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_define_own_property(PropertyKey const& property_key, PropertyDescriptor const& descriptor)
  70. {
  71. // 1. If Type(P) is Symbol, return ! OrdinaryDefineOwnProperty(O, P, Desc).
  72. if (property_key.is_symbol())
  73. return MUST(Object::internal_define_own_property(property_key, descriptor));
  74. // 2. Let current be ? O.[[GetOwnProperty]](P).
  75. auto current = TRY(internal_get_own_property(property_key));
  76. // 3. If current is undefined, return false.
  77. if (!current.has_value())
  78. return false;
  79. // 4. If Desc has a [[Configurable]] field and Desc.[[Configurable]] is true, return false.
  80. if (descriptor.configurable.has_value() && descriptor.configurable.value())
  81. return false;
  82. // 5. If Desc has an [[Enumerable]] field and Desc.[[Enumerable]] is false, return false.
  83. if (descriptor.enumerable.has_value() && !descriptor.enumerable.value())
  84. return false;
  85. // 6. If IsAccessorDescriptor(Desc) is true, return false.
  86. if (descriptor.is_accessor_descriptor())
  87. return false;
  88. // 7. If Desc has a [[Writable]] field and Desc.[[Writable]] is false, return false.
  89. if (descriptor.writable.has_value() && !descriptor.writable.value())
  90. return false;
  91. // 8. If Desc has a [[Value]] field, return SameValue(Desc.[[Value]], current.[[Value]]).
  92. if (descriptor.value.has_value())
  93. return same_value(descriptor.value.value(), current->value.value());
  94. // 9. Return true.
  95. return true;
  96. }
  97. // 10.4.6.7 [[HasProperty]] ( P ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-hasproperty-p
  98. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_has_property(PropertyKey const& property_key) const
  99. {
  100. // 1. If Type(P) is Symbol, return ! OrdinaryHasProperty(O, P).
  101. if (property_key.is_symbol())
  102. return MUST(Object::internal_has_property(property_key));
  103. // 2. Let exports be O.[[Exports]].
  104. // 3. If P is an element of exports, return true.
  105. auto export_element = m_exports.find(property_key.to_string());
  106. if (!export_element.is_end())
  107. return true;
  108. // 4. Return false.
  109. return false;
  110. }
  111. // 10.4.6.8 [[Get]] ( P, Receiver ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-get-p-receiver
  112. ThrowCompletionOr<Value> ModuleNamespaceObject::internal_get(PropertyKey const& property_key, Value receiver, CacheablePropertyMetadata* cacheable_metadata) const
  113. {
  114. auto& vm = this->vm();
  115. // 1. If Type(P) is Symbol, then
  116. if (property_key.is_symbol()) {
  117. // a. Return ! OrdinaryGet(O, P, Receiver).
  118. return MUST(Object::internal_get(property_key, receiver, cacheable_metadata));
  119. }
  120. // 2. Let exports be O.[[Exports]].
  121. // 3. If P is not an element of exports, return undefined.
  122. auto export_element = m_exports.find(property_key.to_string());
  123. if (export_element.is_end())
  124. return js_undefined();
  125. // 4. Let m be O.[[Module]].
  126. // 5. Let binding be ! m.ResolveExport(P).
  127. auto binding = MUST(m_module->resolve_export(vm, property_key.to_string()));
  128. // 6. Assert: binding is a ResolvedBinding Record.
  129. VERIFY(binding.is_valid());
  130. // 7. Let targetModule be binding.[[Module]].
  131. auto target_module = binding.module;
  132. // 8. Assert: targetModule is not undefined.
  133. VERIFY(target_module);
  134. // 9. If binding.[[BindingName]] is namespace, then
  135. if (binding.is_namespace()) {
  136. // a. Return ? GetModuleNamespace(targetModule).
  137. return TRY(target_module->get_module_namespace(vm));
  138. }
  139. // 10. Let targetEnv be targetModule.[[Environment]].
  140. auto* target_environment = target_module->environment();
  141. // 11. If targetEnv is empty, throw a ReferenceError exception.
  142. if (!target_environment)
  143. return vm.throw_completion<ReferenceError>(ErrorType::ModuleNoEnvironment);
  144. // 12. Return ? targetEnv.GetBindingValue(binding.[[BindingName]], true).
  145. return target_environment->get_binding_value(vm, binding.export_name, true);
  146. }
  147. // 10.4.6.9 [[Set]] ( P, V, Receiver ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-set-p-v-receiver
  148. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_set(PropertyKey const&, Value, Value)
  149. {
  150. // 1. Return false.
  151. return false;
  152. }
  153. // 10.4.6.10 [[Delete]] ( P ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-delete-p
  154. ThrowCompletionOr<bool> ModuleNamespaceObject::internal_delete(PropertyKey const& property_key)
  155. {
  156. // 1. If Type(P) is Symbol, then
  157. if (property_key.is_symbol()) {
  158. // a. Return ! OrdinaryDelete(O, P).
  159. return MUST(Object::internal_delete(property_key));
  160. }
  161. // 2. Let exports be O.[[Exports]].
  162. // 3. If P is an element of exports, return false.
  163. auto export_element = m_exports.find(property_key.to_string());
  164. if (!export_element.is_end())
  165. return false;
  166. // 4. Return true.
  167. return true;
  168. }
  169. // 10.4.6.11 [[OwnPropertyKeys]] ( ), https://tc39.es/ecma262/#sec-module-namespace-exotic-objects-ownpropertykeys
  170. ThrowCompletionOr<MarkedVector<Value>> ModuleNamespaceObject::internal_own_property_keys() const
  171. {
  172. // 1. Let exports be O.[[Exports]].
  173. // NOTE: We only add the exports after we know the size of symbolKeys
  174. MarkedVector<Value> exports { vm().heap() };
  175. // 2. Let symbolKeys be OrdinaryOwnPropertyKeys(O).
  176. auto symbol_keys = MUST(Object::internal_own_property_keys());
  177. // 3. Return the list-concatenation of exports and symbolKeys.
  178. exports.ensure_capacity(m_exports.size() + symbol_keys.size());
  179. for (auto const& export_name : m_exports)
  180. exports.unchecked_append(PrimitiveString::create(vm(), export_name));
  181. exports.extend(symbol_keys);
  182. return exports;
  183. }
  184. }