src: add napi_define_class() null checks
napi_define_class is tested by passing NULL to all parameters that are pointers, one at a time. Moreover, two bugs were corrected. One was utf8name and the second was the property descriptor pointer. These pointers were assumed to be non-NULL and now we have NULL checks. PR-URL: https://github.com/nodejs/node/pull/27945 Reviewed-By: Gabriel Schulhof <gabriel.schulhof@intel.com>
This commit is contained in:
parent
9611d75943
commit
a18e27df9e
@ -24,6 +24,9 @@
|
|||||||
RETURN_STATUS_IF_FALSE((env), \
|
RETURN_STATUS_IF_FALSE((env), \
|
||||||
(len == NAPI_AUTO_LENGTH) || len <= INT_MAX, \
|
(len == NAPI_AUTO_LENGTH) || len <= INT_MAX, \
|
||||||
napi_invalid_arg); \
|
napi_invalid_arg); \
|
||||||
|
RETURN_STATUS_IF_FALSE((env), \
|
||||||
|
(str) != nullptr, \
|
||||||
|
napi_invalid_arg); \
|
||||||
auto str_maybe = v8::String::NewFromUtf8( \
|
auto str_maybe = v8::String::NewFromUtf8( \
|
||||||
(env)->isolate, (str), v8::NewStringType::kInternalized, \
|
(env)->isolate, (str), v8::NewStringType::kInternalized, \
|
||||||
static_cast<int>(len)); \
|
static_cast<int>(len)); \
|
||||||
@ -768,6 +771,10 @@ napi_status napi_define_class(napi_env env,
|
|||||||
CHECK_ARG(env, result);
|
CHECK_ARG(env, result);
|
||||||
CHECK_ARG(env, constructor);
|
CHECK_ARG(env, constructor);
|
||||||
|
|
||||||
|
if (property_count > 0) {
|
||||||
|
CHECK_ARG(env, properties);
|
||||||
|
}
|
||||||
|
|
||||||
v8::Isolate* isolate = env->isolate;
|
v8::Isolate* isolate = env->isolate;
|
||||||
|
|
||||||
v8::EscapableHandleScope scope(isolate);
|
v8::EscapableHandleScope scope(isolate);
|
||||||
|
@ -48,3 +48,14 @@ assert.throws(() => { test_object.readonlyAccessor2 = 3; },
|
|||||||
// to the instance
|
// to the instance
|
||||||
assert.strictEqual(TestConstructor.staticReadonlyAccessor1, 10);
|
assert.strictEqual(TestConstructor.staticReadonlyAccessor1, 10);
|
||||||
assert.strictEqual(test_object.staticReadonlyAccessor1, undefined);
|
assert.strictEqual(test_object.staticReadonlyAccessor1, undefined);
|
||||||
|
|
||||||
|
// Verify that passing NULL to napi_define_class() results in the correct
|
||||||
|
// error.
|
||||||
|
assert.deepStrictEqual(TestConstructor.TestDefineClass(), {
|
||||||
|
envIsNull: 'pass',
|
||||||
|
nameIsNull: 'pass',
|
||||||
|
cbIsNull: 'pass',
|
||||||
|
cbDataIsNull: 'pass',
|
||||||
|
propertiesIsNull: 'pass',
|
||||||
|
resultIsNull: 'pass'
|
||||||
|
});
|
||||||
|
@ -4,6 +4,140 @@
|
|||||||
static double value_ = 1;
|
static double value_ = 1;
|
||||||
static double static_value_ = 10;
|
static double static_value_ = 10;
|
||||||
|
|
||||||
|
static napi_value TestDefineClass(napi_env env,
|
||||||
|
napi_callback_info info) {
|
||||||
|
napi_status ret[7];
|
||||||
|
napi_value result, return_value, prop_value;
|
||||||
|
|
||||||
|
napi_property_descriptor property_descriptor = {
|
||||||
|
"TestDefineClass",
|
||||||
|
NULL,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
napi_enumerable | napi_static,
|
||||||
|
NULL};
|
||||||
|
|
||||||
|
ret[0] = napi_define_class(NULL,
|
||||||
|
"TrackedFunction",
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
&property_descriptor,
|
||||||
|
&result);
|
||||||
|
|
||||||
|
ret[1] = napi_define_class(env,
|
||||||
|
NULL,
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
&property_descriptor,
|
||||||
|
&result);
|
||||||
|
|
||||||
|
ret[2] = napi_define_class(env,
|
||||||
|
"TrackedFunction",
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
&property_descriptor,
|
||||||
|
&result);
|
||||||
|
|
||||||
|
ret[3] = napi_define_class(env,
|
||||||
|
"TrackedFunction",
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
&property_descriptor,
|
||||||
|
&result);
|
||||||
|
|
||||||
|
ret[4] = napi_define_class(env,
|
||||||
|
"TrackedFunction",
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
NULL,
|
||||||
|
&result);
|
||||||
|
|
||||||
|
ret[5] = napi_define_class(env,
|
||||||
|
"TrackedFunction",
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
TestDefineClass,
|
||||||
|
NULL,
|
||||||
|
1,
|
||||||
|
&property_descriptor,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_object(env, &return_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[0] == napi_invalid_arg ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"envIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[1] == napi_invalid_arg ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"nameIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[2] == napi_invalid_arg ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"cbIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[3] == napi_ok ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"cbDataIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[4] == napi_invalid_arg ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"propertiesIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
NAPI_CALL(env, napi_create_string_utf8(env,
|
||||||
|
(ret[5] == napi_invalid_arg ?
|
||||||
|
"pass" : "fail"),
|
||||||
|
NAPI_AUTO_LENGTH,
|
||||||
|
&prop_value));
|
||||||
|
NAPI_CALL(env, napi_set_named_property(env,
|
||||||
|
return_value,
|
||||||
|
"resultIsNull",
|
||||||
|
prop_value));
|
||||||
|
|
||||||
|
return return_value;
|
||||||
|
}
|
||||||
|
|
||||||
static napi_value GetValue(napi_env env, napi_callback_info info) {
|
static napi_value GetValue(napi_env env, napi_callback_info info) {
|
||||||
size_t argc = 0;
|
size_t argc = 0;
|
||||||
NAPI_CALL(env, napi_get_cb_info(env, info, &argc, NULL, NULL, NULL));
|
NAPI_CALL(env, napi_get_cb_info(env, info, &argc, NULL, NULL, NULL));
|
||||||
@ -74,17 +208,26 @@ napi_value Init(napi_env env, napi_value exports) {
|
|||||||
env, "MyObject_Extra", 8, NewExtra, NULL, 0, NULL, &cons));
|
env, "MyObject_Extra", 8, NewExtra, NULL, 0, NULL, &cons));
|
||||||
|
|
||||||
napi_property_descriptor properties[] = {
|
napi_property_descriptor properties[] = {
|
||||||
{ "echo", 0, Echo, 0, 0, 0, napi_enumerable, 0 },
|
{ "echo", NULL, Echo, NULL, NULL, NULL, napi_enumerable, NULL },
|
||||||
{ "readwriteValue", 0, 0, 0, 0, number, napi_enumerable | napi_writable, 0 },
|
{ "readwriteValue", NULL, NULL, NULL, NULL, number,
|
||||||
{ "readonlyValue", 0, 0, 0, 0, number, napi_enumerable, 0},
|
napi_enumerable | napi_writable, NULL },
|
||||||
{ "hiddenValue", 0, 0, 0, 0, number, napi_default, 0},
|
{ "readonlyValue", NULL, NULL, NULL, NULL, number, napi_enumerable,
|
||||||
{ "readwriteAccessor1", 0, 0, GetValue, SetValue, 0, napi_default, 0},
|
NULL },
|
||||||
{ "readwriteAccessor2", 0, 0, GetValue, SetValue, 0, napi_writable, 0},
|
{ "hiddenValue", NULL, NULL, NULL, NULL, number, napi_default, NULL },
|
||||||
{ "readonlyAccessor1", 0, 0, GetValue, NULL, 0, napi_default, 0},
|
{ "readwriteAccessor1", NULL, NULL, GetValue, SetValue, NULL, napi_default,
|
||||||
{ "readonlyAccessor2", 0, 0, GetValue, NULL, 0, napi_writable, 0},
|
NULL },
|
||||||
{ "staticReadonlyAccessor1", 0, 0, GetStaticValue, NULL, 0,
|
{ "readwriteAccessor2", NULL, NULL, GetValue, SetValue, NULL,
|
||||||
napi_default | napi_static, 0},
|
napi_writable, NULL },
|
||||||
{ "constructorName", 0, 0, 0, 0, cons, napi_enumerable | napi_static, 0 },
|
{ "readonlyAccessor1", NULL, NULL, GetValue, NULL, NULL, napi_default,
|
||||||
|
NULL },
|
||||||
|
{ "readonlyAccessor2", NULL, NULL, GetValue, NULL, NULL, napi_writable,
|
||||||
|
NULL },
|
||||||
|
{ "staticReadonlyAccessor1", NULL, NULL, GetStaticValue, NULL, NULL,
|
||||||
|
napi_default | napi_static, NULL},
|
||||||
|
{ "constructorName", NULL, NULL, NULL, NULL, cons,
|
||||||
|
napi_enumerable | napi_static, NULL },
|
||||||
|
{ "TestDefineClass", NULL, TestDefineClass, NULL, NULL, NULL,
|
||||||
|
napi_enumerable | napi_static, NULL },
|
||||||
};
|
};
|
||||||
|
|
||||||
NAPI_CALL(env, napi_define_class(env, "MyObject", NAPI_AUTO_LENGTH, New,
|
NAPI_CALL(env, napi_define_class(env, "MyObject", NAPI_AUTO_LENGTH, New,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user