diff --git a/napi-inl.h b/napi-inl.h index 98c578e..f72e1da 100644 --- a/napi-inl.h +++ b/napi-inl.h @@ -4118,7 +4118,8 @@ inline napi_status ThreadSafeFunction::Abort() { inline ThreadSafeFunction::ConvertibleContext ThreadSafeFunction::GetContext() const { void* context; - napi_get_threadsafe_function_context(_tsfn, &context); + napi_status status = napi_get_threadsafe_function_context(_tsfn, &context); + NAPI_FATAL_IF_FAILED(status, "ThreadSafeFunction::GetContext", "napi_get_threadsafe_function_context"); return ConvertibleContext({ context }); } diff --git a/test/binding.cc b/test/binding.cc index 0999ec7..5c3cd6b 100644 --- a/test/binding.cc +++ b/test/binding.cc @@ -40,6 +40,7 @@ Object InitObjectDeprecated(Env env); #endif // !NODE_ADDON_API_DISABLE_DEPRECATED Object InitPromise(Env env); #if (NAPI_VERSION > 3) +Object InitThreadSafeFunctionCtx(Env env); Object InitThreadSafeFunctionExistingTsfn(Env env); Object InitThreadSafeFunctionPtr(Env env); Object InitThreadSafeFunctionSum(Env env); @@ -92,6 +93,7 @@ Object Init(Env env, Object exports) { #endif // !NODE_ADDON_API_DISABLE_DEPRECATED exports.Set("promise", InitPromise(env)); #if (NAPI_VERSION > 3) + exports.Set("threadsafe_function_ctx", InitThreadSafeFunctionCtx(env)); exports.Set("threadsafe_function_existing_tsfn", InitThreadSafeFunctionExistingTsfn(env)); exports.Set("threadsafe_function_ptr", InitThreadSafeFunctionPtr(env)); exports.Set("threadsafe_function_sum", InitThreadSafeFunctionSum(env)); diff --git a/test/binding.gyp b/test/binding.gyp index a21ef8e..ced1a68 100644 --- a/test/binding.gyp +++ b/test/binding.gyp @@ -36,6 +36,7 @@ 'object/object.cc', 'object/set_property.cc', 'promise.cc', + 'threadsafe_function/threadsafe_function_ctx.cc', 'threadsafe_function/threadsafe_function_existing_tsfn.cc', 'threadsafe_function/threadsafe_function_ptr.cc', 'threadsafe_function/threadsafe_function_sum.cc', diff --git a/test/index.js b/test/index.js index 2269aed..e05c25a 100644 --- a/test/index.js +++ b/test/index.js @@ -39,6 +39,7 @@ let testModules = [ 'object/object_deprecated', 'object/set_property', 'promise', + 'threadsafe_function/threadsafe_function_ctx', 'threadsafe_function/threadsafe_function_existing_tsfn', 'threadsafe_function/threadsafe_function_ptr', 'threadsafe_function/threadsafe_function_sum', @@ -69,6 +70,7 @@ if (napiVersion < 3) { if (napiVersion < 4) { testModules.splice(testModules.indexOf('asyncprogressworker'), 1); + testModules.splice(testModules.indexOf('threadsafe_function/threadsafe_function_ctx'), 1); testModules.splice(testModules.indexOf('threadsafe_function/threadsafe_function_existing_tsfn'), 1); testModules.splice(testModules.indexOf('threadsafe_function/threadsafe_function_ptr'), 1); testModules.splice(testModules.indexOf('threadsafe_function/threadsafe_function_sum'), 1); diff --git a/test/threadsafe_function/threadsafe_function_ctx.cc b/test/threadsafe_function/threadsafe_function_ctx.cc new file mode 100644 index 0000000..bae83ba --- /dev/null +++ b/test/threadsafe_function/threadsafe_function_ctx.cc @@ -0,0 +1,63 @@ +#include "napi.h" + +#if (NAPI_VERSION > 3) + +using namespace Napi; + +namespace { + +class TSFNWrap : public ObjectWrap { +public: + static Object Init(Napi::Env env, Object exports); + TSFNWrap(const CallbackInfo &info); + + Napi::Value GetContext(const CallbackInfo & /*info*/) { + Reference *ctx = _tsfn.GetContext(); + return ctx->Value(); + }; + + Napi::Value Release(const CallbackInfo &info) { + Napi::Env env = info.Env(); + _deferred = std::unique_ptr(new Promise::Deferred(env)); + _tsfn.Release(); + return _deferred->Promise(); + }; + +private: + ThreadSafeFunction _tsfn; + std::unique_ptr _deferred; +}; + +Object TSFNWrap::Init(Napi::Env env, Object exports) { + Function func = + DefineClass(env, "TSFNWrap", + {InstanceMethod("getContext", &TSFNWrap::GetContext), + InstanceMethod("release", &TSFNWrap::Release)}); + + exports.Set("TSFNWrap", func); + return exports; +} + +TSFNWrap::TSFNWrap(const CallbackInfo &info) : ObjectWrap(info) { + Napi::Env env = info.Env(); + + Reference *_ctx = new Reference; + *_ctx = Persistent(info[0]); + + _tsfn = ThreadSafeFunction::New( + info.Env(), Function::New(env, [](const CallbackInfo & /*info*/) {}), + Object::New(env), "Test", 1, 1, _ctx, + [this](Napi::Env env, Reference *ctx) { + _deferred->Resolve(env.Undefined()); + ctx->Reset(); + delete ctx; + }); +} + +} // namespace + +Object InitThreadSafeFunctionCtx(Env env) { + return TSFNWrap::Init(env, Object::New(env)); +} + +#endif diff --git a/test/threadsafe_function/threadsafe_function_ctx.js b/test/threadsafe_function/threadsafe_function_ctx.js new file mode 100644 index 0000000..d091bbb --- /dev/null +++ b/test/threadsafe_function/threadsafe_function_ctx.js @@ -0,0 +1,16 @@ +'use strict'; + +const assert = require('assert'); +const buildType = process.config.target_defaults.default_configuration; + +module.exports = Promise.all[ + test(require(`../build/${buildType}/binding.node`)), + test(require(`../build/${buildType}/binding_noexcept.node`)) +]; + +async function test(binding) { + const ctx = { }; + const tsfn = new binding.threadsafe_function_ctx.TSFNWrap(ctx); + assert(tsfn.getContext() === ctx); + await tsfn.release(); +}