@@ -712,6 +712,45 @@ bool FindWrapper(v8::Local<v8::Object> obj,
712
712
return true ;
713
713
}
714
714
715
+ static void DeleteEnv (napi_env env, void * data, void * hint) {
716
+ delete env;
717
+ }
718
+
719
+ napi_env GetEnv (v8::Local<v8::Context> context) {
720
+ napi_env result;
721
+
722
+ auto isolate = context->GetIsolate ();
723
+ auto global = context->Global ();
724
+
725
+ // In the case of the string for which we grab the private and the value of
726
+ // the private on the global object we can call .ToLocalChecked() directly
727
+ // because we need to stop hard if either of them is empty.
728
+ //
729
+ // Re https://github.com/nodejs/node/pull/14217#discussion_r128775149
730
+ auto key = v8::Private::ForApi (isolate,
731
+ v8::String::NewFromOneByte (isolate,
732
+ reinterpret_cast <const uint8_t *>(" N-API Environment" ),
733
+ v8::NewStringType::kInternalized ).ToLocalChecked ());
734
+ auto value = global->GetPrivate (context, key).ToLocalChecked ();
735
+
736
+ if (value->IsExternal ()) {
737
+ result = static_cast <napi_env>(value.As <v8::External>()->Value ());
738
+ } else {
739
+ result = new napi_env__ (isolate);
740
+ auto external = v8::External::New (isolate, result);
741
+
742
+ // We must also stop hard if the result of assigning the env to the global
743
+ // is either nothing or false.
744
+ CHECK (global->SetPrivate (context, key, external).FromJust ());
745
+
746
+ // Create a self-destructing reference to external that will get rid of the
747
+ // napi_env when external goes out of scope.
748
+ Reference::New (result, external, 0 , true , DeleteEnv, nullptr , nullptr );
749
+ }
750
+
751
+ return result;
752
+ }
753
+
715
754
} // end of namespace v8impl
716
755
717
756
// Intercepts the Node-V8 module registration callback. Converts parameters
@@ -723,9 +762,9 @@ void napi_module_register_cb(v8::Local<v8::Object> exports,
723
762
void * priv) {
724
763
napi_module* mod = static_cast <napi_module*>(priv);
725
764
726
- // Create a new napi_env for this module. Once module unloading is supported
727
- // we shall have to call delete on this object from there .
728
- napi_env env = new napi_env__ (context-> GetIsolate () );
765
+ // Create a new napi_env for this module or reference one if a pre-existing
766
+ // one is found .
767
+ napi_env env = v8impl::GetEnv (context);
729
768
730
769
mod->nm_register_func (
731
770
env,
0 commit comments