LibJS: Throw TypeError when calling class constructor without 'new'

This commit is contained in:
Linus Groh 2020-11-11 21:37:40 +00:00 committed by Andreas Kling
parent b07c7f589f
commit 1b0c862f3a
Notes: sideshowbarker 2024-07-19 01:26:14 +09:00
5 changed files with 21 additions and 5 deletions

View file

@ -715,6 +715,7 @@ Value ClassExpression::execute(Interpreter& interpreter, GlobalObject& global_ob
ASSERT(class_constructor_value.is_function() && class_constructor_value.as_function().is_script_function());
ScriptFunction* class_constructor = static_cast<ScriptFunction*>(&class_constructor_value.as_function());
class_constructor->set_is_class_constructor();
Value super_constructor = js_undefined();
if (!m_super_class.is_null()) {
super_constructor = m_super_class->execute(interpreter, global_object);

View file

@ -36,6 +36,7 @@
M(BigIntBadOperatorOtherType, "Cannot use {} operator with BigInt and other type") \
M(BigIntIntArgument, "BigInt argument must be an integer") \
M(BigIntInvalidValue, "Invalid value for BigInt: {}") \
M(ClassConstructorWithoutNew, "Class constructor {} must be called with 'new'") \
M(ClassDoesNotExtendAConstructorOrNull, "Class extends value {} is not a constructor or null") \
M(Convert, "Cannot convert {} to {}") \
M(ConvertUndefinedToObject, "Cannot convert undefined to object") \

View file

@ -110,7 +110,7 @@ LexicalEnvironment* ScriptFunction::create_environment()
return environment;
}
Value ScriptFunction::call()
Value ScriptFunction::execute_function_body()
{
auto& vm = this->vm();
@ -150,13 +150,22 @@ Value ScriptFunction::call()
return interpreter->execute_statement(global_object(), m_body, move(arguments), ScopeType::Function);
}
Value ScriptFunction::call()
{
if (m_is_class_constructor) {
vm().throw_exception<TypeError>(global_object(), ErrorType::ClassConstructorWithoutNew, m_name);
return {};
}
return execute_function_body();
}
Value ScriptFunction::construct(Function&)
{
if (m_is_arrow_function) {
vm().throw_exception<TypeError>(global_object(), ErrorType::NotAConstructor, m_name);
return {};
}
return call();
return execute_function_body();
}
JS_DEFINE_NATIVE_GETTER(ScriptFunction::length_getter)

View file

@ -50,6 +50,8 @@ public:
virtual const FlyString& name() const override { return m_name; };
void set_name(const FlyString& name) { m_name = name; };
void set_is_class_constructor() { m_is_class_constructor = true; };
protected:
virtual bool is_strict_mode() const final { return m_is_strict; }
@ -58,6 +60,8 @@ private:
virtual LexicalEnvironment* create_environment() override;
virtual void visit_children(Visitor&) override;
Value execute_function_body();
JS_DECLARE_NATIVE_GETTER(length_getter);
JS_DECLARE_NATIVE_GETTER(name_getter);
@ -68,6 +72,7 @@ private:
i32 m_function_length { 0 };
bool m_is_strict { false };
bool m_is_arrow_function { false };
bool m_is_class_constructor { false };
};
}

View file

@ -46,18 +46,18 @@ test("constructor length affects class length", () => {
expect(B).toHaveLength(2);
});
test.skip("must be invoked with 'new'", () => {
test("must be invoked with 'new'", () => {
class A {
constructor() {}
}
expect(() => {
A();
}).toThrow(TypeError); // FIXME: Add message when this test works
}).toThrowWithMessage(TypeError, "Class constructor A must be called with 'new'");
expect(() => {
A.prototype.constructor();
}).toThrow(TypeError); // FIXME: Add message when this test works
}).toThrowWithMessage(TypeError, "Class constructor A must be called with 'new'");
});
test("implicit constructor", () => {