mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 10:08:12 +00:00
LibJS: Convert has_binding() to ThrowCompletionOr
Also add spec step comments to it while we're here.
This commit is contained in:
parent
617d3cd3d3
commit
fbb176c926
11 changed files with 50 additions and 29 deletions
|
@ -3190,7 +3190,7 @@ void ScopeNode::block_declaration_instantiation(GlobalObject& global_object, Env
|
||||||
if (is_constant_declaration) {
|
if (is_constant_declaration) {
|
||||||
environment->create_immutable_binding(global_object, name, true);
|
environment->create_immutable_binding(global_object, name, true);
|
||||||
} else {
|
} else {
|
||||||
if (!environment->has_binding(name))
|
if (!MUST(environment->has_binding(name)))
|
||||||
environment->create_mutable_binding(global_object, name, false);
|
environment->create_mutable_binding(global_object, name, false);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -558,7 +558,7 @@ ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& glo
|
||||||
while (this_environment != variable_environment) {
|
while (this_environment != variable_environment) {
|
||||||
if (!is<ObjectEnvironment>(*this_environment)) {
|
if (!is<ObjectEnvironment>(*this_environment)) {
|
||||||
program.for_each_var_declared_name([&](auto const& name) {
|
program.for_each_var_declared_name([&](auto const& name) {
|
||||||
if (this_environment->has_binding(name)) {
|
if (MUST(this_environment->has_binding(name))) {
|
||||||
vm.throw_exception<SyntaxError>(global_object, ErrorType::FixmeAddAnErrorStringWithMessage, "Var already declared lexically");
|
vm.throw_exception<SyntaxError>(global_object, ErrorType::FixmeAddAnErrorStringWithMessage, "Var already declared lexically");
|
||||||
return IterationDecision::Break;
|
return IterationDecision::Break;
|
||||||
}
|
}
|
||||||
|
@ -605,7 +605,7 @@ ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& glo
|
||||||
auto* this_environment = lexical_environment;
|
auto* this_environment = lexical_environment;
|
||||||
|
|
||||||
while (this_environment != variable_environment) {
|
while (this_environment != variable_environment) {
|
||||||
if (!is<ObjectEnvironment>(*this_environment) && this_environment->has_binding(function_name))
|
if (!is<ObjectEnvironment>(*this_environment) && MUST(this_environment->has_binding(function_name)))
|
||||||
return IterationDecision::Continue;
|
return IterationDecision::Continue;
|
||||||
|
|
||||||
this_environment = this_environment->outer_environment();
|
this_environment = this_environment->outer_environment();
|
||||||
|
@ -628,7 +628,7 @@ ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& glo
|
||||||
if (vm.exception())
|
if (vm.exception())
|
||||||
return IterationDecision::Break;
|
return IterationDecision::Break;
|
||||||
} else {
|
} else {
|
||||||
if (!variable_environment->has_binding(function_name)) {
|
if (!MUST(variable_environment->has_binding(function_name))) {
|
||||||
variable_environment->create_mutable_binding(global_object, function_name, true);
|
variable_environment->create_mutable_binding(global_object, function_name, true);
|
||||||
variable_environment->initialize_binding(global_object, function_name, js_undefined());
|
variable_environment->initialize_binding(global_object, function_name, js_undefined());
|
||||||
VERIFY(!vm.exception());
|
VERIFY(!vm.exception());
|
||||||
|
@ -700,7 +700,7 @@ ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& glo
|
||||||
if (auto* exception = vm.exception())
|
if (auto* exception = vm.exception())
|
||||||
return throw_completion(exception->value());
|
return throw_completion(exception->value());
|
||||||
} else {
|
} else {
|
||||||
auto binding_exists = variable_environment->has_binding(declaration.name());
|
auto binding_exists = MUST(variable_environment->has_binding(declaration.name()));
|
||||||
|
|
||||||
if (!binding_exists) {
|
if (!binding_exists) {
|
||||||
variable_environment->create_mutable_binding(global_object, declaration.name(), true);
|
variable_environment->create_mutable_binding(global_object, declaration.name(), true);
|
||||||
|
@ -721,7 +721,7 @@ ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& glo
|
||||||
if (auto* exception = vm.exception())
|
if (auto* exception = vm.exception())
|
||||||
return throw_completion(exception->value());
|
return throw_completion(exception->value());
|
||||||
} else {
|
} else {
|
||||||
auto binding_exists = variable_environment->has_binding(var_name);
|
auto binding_exists = MUST(variable_environment->has_binding(var_name));
|
||||||
|
|
||||||
if (!binding_exists) {
|
if (!binding_exists) {
|
||||||
variable_environment->create_mutable_binding(global_object, var_name, true);
|
variable_environment->create_mutable_binding(global_object, var_name, true);
|
||||||
|
|
|
@ -35,7 +35,7 @@ void DeclarativeEnvironment::visit_edges(Visitor& visitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 9.1.1.1.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-declarative-environment-records-hasbinding-n
|
// 9.1.1.1.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-declarative-environment-records-hasbinding-n
|
||||||
bool DeclarativeEnvironment::has_binding(FlyString const& name, Optional<size_t>* out_index) const
|
ThrowCompletionOr<bool> DeclarativeEnvironment::has_binding(FlyString const& name, Optional<size_t>* out_index) const
|
||||||
{
|
{
|
||||||
auto it = m_names.find(name);
|
auto it = m_names.find(name);
|
||||||
if (it == m_names.end())
|
if (it == m_names.end())
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
#include <AK/FlyString.h>
|
#include <AK/FlyString.h>
|
||||||
#include <AK/HashMap.h>
|
#include <AK/HashMap.h>
|
||||||
|
#include <LibJS/Runtime/Completion.h>
|
||||||
#include <LibJS/Runtime/Environment.h>
|
#include <LibJS/Runtime/Environment.h>
|
||||||
#include <LibJS/Runtime/Value.h>
|
#include <LibJS/Runtime/Value.h>
|
||||||
|
|
||||||
|
@ -21,7 +22,7 @@ public:
|
||||||
explicit DeclarativeEnvironment(Environment* parent_scope);
|
explicit DeclarativeEnvironment(Environment* parent_scope);
|
||||||
virtual ~DeclarativeEnvironment() override;
|
virtual ~DeclarativeEnvironment() override;
|
||||||
|
|
||||||
virtual bool has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
virtual ThrowCompletionOr<bool> has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
||||||
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
||||||
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
||||||
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
||||||
|
|
|
@ -357,7 +357,7 @@ ThrowCompletionOr<void> ECMAScriptFunctionObject::function_declaration_instantia
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto const& parameter_name : parameter_names) {
|
for (auto const& parameter_name : parameter_names) {
|
||||||
if (environment->has_binding(parameter_name))
|
if (MUST(environment->has_binding(parameter_name)))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
environment->create_mutable_binding(global_object(), parameter_name, false);
|
environment->create_mutable_binding(global_object(), parameter_name, false);
|
||||||
|
|
|
@ -33,7 +33,7 @@ public:
|
||||||
|
|
||||||
virtual Object* with_base_object() const { return nullptr; }
|
virtual Object* with_base_object() const { return nullptr; }
|
||||||
|
|
||||||
virtual bool has_binding([[maybe_unused]] FlyString const& name, [[maybe_unused]] Optional<size_t>* out_index = nullptr) const { return false; }
|
virtual ThrowCompletionOr<bool> has_binding([[maybe_unused]] FlyString const& name, [[maybe_unused]] Optional<size_t>* out_index = nullptr) const { return false; }
|
||||||
virtual void create_mutable_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, [[maybe_unused]] bool can_be_deleted) { }
|
virtual void create_mutable_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, [[maybe_unused]] bool can_be_deleted) { }
|
||||||
virtual void create_immutable_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, [[maybe_unused]] bool strict) { }
|
virtual void create_immutable_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, [[maybe_unused]] bool strict) { }
|
||||||
virtual void initialize_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, Value) { }
|
virtual void initialize_binding(GlobalObject&, [[maybe_unused]] FlyString const& name, Value) { }
|
||||||
|
|
|
@ -39,17 +39,22 @@ ThrowCompletionOr<Value> GlobalEnvironment::get_this_binding(GlobalObject&) cons
|
||||||
}
|
}
|
||||||
|
|
||||||
// 9.1.1.4.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-global-environment-records-hasbinding-n
|
// 9.1.1.4.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-global-environment-records-hasbinding-n
|
||||||
bool GlobalEnvironment::has_binding(FlyString const& name, Optional<size_t>*) const
|
ThrowCompletionOr<bool> GlobalEnvironment::has_binding(FlyString const& name, Optional<size_t>*) const
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name))
|
// 1. Let DclRec be envRec.[[DeclarativeRecord]].
|
||||||
|
// 2. If DclRec.HasBinding(N) is true, return true.
|
||||||
|
if (MUST(m_declarative_record->has_binding(name)))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
// 3. Let ObjRec be envRec.[[ObjectRecord]].
|
||||||
|
// 4. Return ? ObjRec.HasBinding(N).
|
||||||
return m_object_record->has_binding(name);
|
return m_object_record->has_binding(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 9.1.1.4.2 CreateMutableBinding ( N, D ), https://tc39.es/ecma262/#sec-global-environment-records-createmutablebinding-n-d
|
// 9.1.1.4.2 CreateMutableBinding ( N, D ), https://tc39.es/ecma262/#sec-global-environment-records-createmutablebinding-n-d
|
||||||
void GlobalEnvironment::create_mutable_binding(GlobalObject& global_object, FlyString const& name, bool can_be_deleted)
|
void GlobalEnvironment::create_mutable_binding(GlobalObject& global_object, FlyString const& name, bool can_be_deleted)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name)) {
|
if (MUST(m_declarative_record->has_binding(name))) {
|
||||||
global_object.vm().throw_exception<TypeError>(global_object, ErrorType::FixmeAddAnErrorString);
|
global_object.vm().throw_exception<TypeError>(global_object, ErrorType::FixmeAddAnErrorString);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -59,7 +64,7 @@ void GlobalEnvironment::create_mutable_binding(GlobalObject& global_object, FlyS
|
||||||
// 9.1.1.4.3 CreateImmutableBinding ( N, S ), https://tc39.es/ecma262/#sec-global-environment-records-createimmutablebinding-n-s
|
// 9.1.1.4.3 CreateImmutableBinding ( N, S ), https://tc39.es/ecma262/#sec-global-environment-records-createimmutablebinding-n-s
|
||||||
void GlobalEnvironment::create_immutable_binding(GlobalObject& global_object, FlyString const& name, bool strict)
|
void GlobalEnvironment::create_immutable_binding(GlobalObject& global_object, FlyString const& name, bool strict)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name)) {
|
if (MUST(m_declarative_record->has_binding(name))) {
|
||||||
global_object.vm().throw_exception<TypeError>(global_object, ErrorType::FixmeAddAnErrorString);
|
global_object.vm().throw_exception<TypeError>(global_object, ErrorType::FixmeAddAnErrorString);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -69,7 +74,7 @@ void GlobalEnvironment::create_immutable_binding(GlobalObject& global_object, Fl
|
||||||
// 9.1.1.4.4 InitializeBinding ( N, V ), https://tc39.es/ecma262/#sec-global-environment-records-initializebinding-n-v
|
// 9.1.1.4.4 InitializeBinding ( N, V ), https://tc39.es/ecma262/#sec-global-environment-records-initializebinding-n-v
|
||||||
void GlobalEnvironment::initialize_binding(GlobalObject& global_object, FlyString const& name, Value value)
|
void GlobalEnvironment::initialize_binding(GlobalObject& global_object, FlyString const& name, Value value)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name)) {
|
if (MUST(m_declarative_record->has_binding(name))) {
|
||||||
m_declarative_record->initialize_binding(global_object, name, value);
|
m_declarative_record->initialize_binding(global_object, name, value);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -79,7 +84,7 @@ void GlobalEnvironment::initialize_binding(GlobalObject& global_object, FlyStrin
|
||||||
// 9.1.1.4.5 SetMutableBinding ( N, V, S ), https://tc39.es/ecma262/#sec-global-environment-records-setmutablebinding-n-v-s
|
// 9.1.1.4.5 SetMutableBinding ( N, V, S ), https://tc39.es/ecma262/#sec-global-environment-records-setmutablebinding-n-v-s
|
||||||
void GlobalEnvironment::set_mutable_binding(GlobalObject& global_object, FlyString const& name, Value value, bool strict)
|
void GlobalEnvironment::set_mutable_binding(GlobalObject& global_object, FlyString const& name, Value value, bool strict)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name)) {
|
if (MUST(m_declarative_record->has_binding(name))) {
|
||||||
m_declarative_record->set_mutable_binding(global_object, name, value, strict);
|
m_declarative_record->set_mutable_binding(global_object, name, value, strict);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -90,7 +95,7 @@ void GlobalEnvironment::set_mutable_binding(GlobalObject& global_object, FlyStri
|
||||||
// 9.1.1.4.6 GetBindingValue ( N, S ), https://tc39.es/ecma262/#sec-global-environment-records-getbindingvalue-n-s
|
// 9.1.1.4.6 GetBindingValue ( N, S ), https://tc39.es/ecma262/#sec-global-environment-records-getbindingvalue-n-s
|
||||||
Value GlobalEnvironment::get_binding_value(GlobalObject& global_object, FlyString const& name, bool strict)
|
Value GlobalEnvironment::get_binding_value(GlobalObject& global_object, FlyString const& name, bool strict)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name))
|
if (MUST(m_declarative_record->has_binding(name)))
|
||||||
return m_declarative_record->get_binding_value(global_object, name, strict);
|
return m_declarative_record->get_binding_value(global_object, name, strict);
|
||||||
return m_object_record->get_binding_value(global_object, name, strict);
|
return m_object_record->get_binding_value(global_object, name, strict);
|
||||||
}
|
}
|
||||||
|
@ -98,7 +103,7 @@ Value GlobalEnvironment::get_binding_value(GlobalObject& global_object, FlyStrin
|
||||||
// 9.1.1.4.7 DeleteBinding ( N ), https://tc39.es/ecma262/#sec-global-environment-records-deletebinding-n
|
// 9.1.1.4.7 DeleteBinding ( N ), https://tc39.es/ecma262/#sec-global-environment-records-deletebinding-n
|
||||||
bool GlobalEnvironment::delete_binding(GlobalObject& global_object, FlyString const& name)
|
bool GlobalEnvironment::delete_binding(GlobalObject& global_object, FlyString const& name)
|
||||||
{
|
{
|
||||||
if (m_declarative_record->has_binding(name))
|
if (MUST(m_declarative_record->has_binding(name)))
|
||||||
return m_declarative_record->delete_binding(global_object, name);
|
return m_declarative_record->delete_binding(global_object, name);
|
||||||
|
|
||||||
bool existing_prop = TRY_OR_DISCARD(m_object_record->binding_object().has_own_property(name));
|
bool existing_prop = TRY_OR_DISCARD(m_object_record->binding_object().has_own_property(name));
|
||||||
|
@ -121,7 +126,7 @@ bool GlobalEnvironment::has_var_declaration(FlyString const& name) const
|
||||||
// 9.1.1.4.13 HasLexicalDeclaration ( N ), https://tc39.es/ecma262/#sec-haslexicaldeclaration
|
// 9.1.1.4.13 HasLexicalDeclaration ( N ), https://tc39.es/ecma262/#sec-haslexicaldeclaration
|
||||||
bool GlobalEnvironment::has_lexical_declaration(FlyString const& name) const
|
bool GlobalEnvironment::has_lexical_declaration(FlyString const& name) const
|
||||||
{
|
{
|
||||||
return m_declarative_record->has_binding(name);
|
return MUST(m_declarative_record->has_binding(name));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 9.1.1.4.14 HasRestrictedGlobalProperty ( N ), https://tc39.es/ecma262/#sec-hasrestrictedglobalproperty
|
// 9.1.1.4.14 HasRestrictedGlobalProperty ( N ), https://tc39.es/ecma262/#sec-hasrestrictedglobalproperty
|
||||||
|
|
|
@ -19,7 +19,7 @@ public:
|
||||||
virtual bool has_this_binding() const final { return true; }
|
virtual bool has_this_binding() const final { return true; }
|
||||||
virtual ThrowCompletionOr<Value> get_this_binding(GlobalObject&) const final;
|
virtual ThrowCompletionOr<Value> get_this_binding(GlobalObject&) const final;
|
||||||
|
|
||||||
virtual bool has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
virtual ThrowCompletionOr<bool> has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
||||||
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
||||||
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
||||||
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
||||||
|
|
|
@ -25,20 +25,37 @@ void ObjectEnvironment::visit_edges(Cell::Visitor& visitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 9.1.1.2.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-object-environment-records-hasbinding-n
|
// 9.1.1.2.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-object-environment-records-hasbinding-n
|
||||||
bool ObjectEnvironment::has_binding(FlyString const& name, Optional<size_t>*) const
|
ThrowCompletionOr<bool> ObjectEnvironment::has_binding(FlyString const& name, Optional<size_t>*) const
|
||||||
{
|
{
|
||||||
auto& vm = this->vm();
|
auto& vm = this->vm();
|
||||||
bool found_binding = TRY_OR_DISCARD(m_binding_object.has_property(name));
|
|
||||||
|
// 1. Let bindingObject be envRec.[[BindingObject]].
|
||||||
|
|
||||||
|
// 2. Let foundBinding be ? HasProperty(bindingObject, N).
|
||||||
|
bool found_binding = TRY(m_binding_object.has_property(name));
|
||||||
|
|
||||||
|
// 3. If foundBinding is false, return false.
|
||||||
if (!found_binding)
|
if (!found_binding)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
// 4. If envRec.[[IsWithEnvironment]] is false, return true.
|
||||||
if (!m_with_environment)
|
if (!m_with_environment)
|
||||||
return true;
|
return true;
|
||||||
auto unscopables = TRY_OR_DISCARD(m_binding_object.get(*vm.well_known_symbol_unscopables()));
|
|
||||||
|
// 5. Let unscopables be ? Get(bindingObject, @@unscopables).
|
||||||
|
auto unscopables = TRY(m_binding_object.get(*vm.well_known_symbol_unscopables()));
|
||||||
|
|
||||||
|
// 6. If Type(unscopables) is Object, then
|
||||||
if (unscopables.is_object()) {
|
if (unscopables.is_object()) {
|
||||||
auto blocked = TRY_OR_DISCARD(unscopables.as_object().get(name));
|
// a. Let blocked be ! ToBoolean(? Get(unscopables, N)).
|
||||||
if (blocked.to_boolean())
|
auto blocked = TRY(unscopables.as_object().get(name)).to_boolean();
|
||||||
|
|
||||||
|
// b. If blocked is true, return false.
|
||||||
|
if (blocked)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 7. Return true.
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,7 @@ public:
|
||||||
};
|
};
|
||||||
ObjectEnvironment(Object& binding_object, IsWithEnvironment, Environment* outer_environment);
|
ObjectEnvironment(Object& binding_object, IsWithEnvironment, Environment* outer_environment);
|
||||||
|
|
||||||
virtual bool has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
virtual ThrowCompletionOr<bool> has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
|
||||||
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
virtual void create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
|
||||||
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
virtual void create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
|
||||||
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
virtual void initialize_binding(GlobalObject&, FlyString const& name, Value) override;
|
||||||
|
|
|
@ -445,9 +445,7 @@ Reference VM::get_identifier_reference(Environment* environment, FlyString name,
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<size_t> index;
|
Optional<size_t> index;
|
||||||
auto exists = environment->has_binding(name, &index);
|
auto exists = TRY_OR_DISCARD(environment->has_binding(name, &index));
|
||||||
if (exception())
|
|
||||||
return {};
|
|
||||||
|
|
||||||
Optional<EnvironmentCoordinate> environment_coordinate;
|
Optional<EnvironmentCoordinate> environment_coordinate;
|
||||||
if (index.has_value())
|
if (index.has_value())
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue