From f6b610ba486cc470872804f08e4cd65c003d06e6 Mon Sep 17 00:00:00 2001 From: legendecas Date: Fri, 7 May 2021 11:35:43 +0800 Subject: [PATCH] src: return bool on object freeze and seal (#991) These operations can call into JavaScript by using Proxy handlers, hence JavaScript exceptions might pending at the end of the operations. --- napi-inl.h | 10 ++++++---- napi.h | 4 ++-- test/common/index.js | 4 ++-- test/object/object_freeze_seal.cc | 8 ++++---- test/object/object_freeze_seal.js | 28 ++++++++++++++++++++++++++-- 5 files changed, 40 insertions(+), 14 deletions(-) diff --git a/napi-inl.h b/napi-inl.h index be20dc085..7e9dd726a 100644 --- a/napi-inl.h +++ b/napi-inl.h @@ -1389,14 +1389,16 @@ inline void Object::AddFinalizer(Finalizer finalizeCallback, } #if NAPI_VERSION >= 8 -inline void Object::Freeze() { +inline bool Object::Freeze() { napi_status status = napi_object_freeze(_env, _value); - NAPI_THROW_IF_FAILED_VOID(_env, status); + NAPI_THROW_IF_FAILED(_env, status, false); + return true; } -inline void Object::Seal() { +inline bool Object::Seal() { napi_status status = napi_object_seal(_env, _value); - NAPI_THROW_IF_FAILED_VOID(_env, status); + NAPI_THROW_IF_FAILED(_env, status, false); + return true; } #endif // NAPI_VERSION >= 8 diff --git a/napi.h b/napi.h index 8be2ca3f7..7dc9b17df 100644 --- a/napi.h +++ b/napi.h @@ -773,8 +773,8 @@ namespace Napi { T* data, Hint* finalizeHint); #if NAPI_VERSION >= 8 - void Freeze(); - void Seal(); + bool Freeze(); + bool Seal(); #endif // NAPI_VERSION >= 8 }; diff --git a/test/common/index.js b/test/common/index.js index 65bdc10b7..ab6d12bc8 100644 --- a/test/common/index.js +++ b/test/common/index.js @@ -76,7 +76,7 @@ exports.mustNotCall = function(msg) { }; exports.runTest = async function(test, buildType) { - buildType = buildType || process.config.target_defaults.default_configuration; + buildType = buildType || process.config.target_defaults.default_configuration || 'Release'; const bindings = [ `../build/${buildType}/binding.node`, @@ -90,7 +90,7 @@ exports.runTest = async function(test, buildType) { } exports.runTestWithBindingPath = async function(test, buildType) { - buildType = buildType || process.config.target_defaults.default_configuration; + buildType = buildType || process.config.target_defaults.default_configuration || 'Release'; const bindings = [ `../build/${buildType}/binding.node`, diff --git a/test/object/object_freeze_seal.cc b/test/object/object_freeze_seal.cc index 2d6f4b616..51071a1c6 100644 --- a/test/object/object_freeze_seal.cc +++ b/test/object/object_freeze_seal.cc @@ -4,14 +4,14 @@ using namespace Napi; -void Freeze(const CallbackInfo& info) { +Value Freeze(const CallbackInfo& info) { Object obj = info[0].As(); - obj.Freeze(); + return Boolean::New(info.Env(), obj.Freeze()); } -void Seal(const CallbackInfo& info) { +Value Seal(const CallbackInfo& info) { Object obj = info[0].As(); - obj.Seal(); + return Boolean::New(info.Env(), obj.Seal()); } Object InitObjectFreezeSeal(Env env) { diff --git a/test/object/object_freeze_seal.js b/test/object/object_freeze_seal.js index b6434b6e3..62c1c48f3 100644 --- a/test/object/object_freeze_seal.js +++ b/test/object/object_freeze_seal.js @@ -7,7 +7,7 @@ module.exports = require('../common').runTest(test); function test(binding) { { const obj = { x: 'a', y: 'b', z: 'c' }; - binding.object_freeze_seal.freeze(obj); + assert.strictEqual(binding.object_freeze_seal.freeze(obj), true); assert.strictEqual(Object.isFrozen(obj), true); assert.throws(() => { obj.x = 10; @@ -20,9 +20,21 @@ function test(binding) { }, /Cannot delete property 'x' of #/); } + { + const obj = new Proxy({ x: 'a', y: 'b', z: 'c' }, { + preventExtensions() { + throw new Error('foo'); + }, + }); + + assert.throws(() => { + binding.object_freeze_seal.freeze(obj); + }, /foo/); + } + { const obj = { x: 'a', y: 'b', z: 'c' }; - binding.object_freeze_seal.seal(obj); + assert.strictEqual(binding.object_freeze_seal.seal(obj), true); assert.strictEqual(Object.isSealed(obj), true); assert.throws(() => { obj.w = 'd'; @@ -34,4 +46,16 @@ function test(binding) { // so this should not throw. obj.x = 'd'; } + + { + const obj = new Proxy({ x: 'a', y: 'b', z: 'c' }, { + preventExtensions() { + throw new Error('foo'); + }, + }); + + assert.throws(() => { + binding.object_freeze_seal.seal(obj); + }, /foo/); + } }