edon 0.0.27

Library embedding Nodejs (libnode) within Rust
Documentation
use std::collections::HashMap;
use std::ffi::c_char;
use std::ffi::c_int;
use std::ffi::c_uint;
use std::ffi::CString;
use std::os::raw::c_void;
use std::ptr;
use std::sync::LazyLock;
use std::sync::RwLock;

use crate::napi::JsObject;
use crate::napi::NapiValue;
use crate::Env;

type InitFn =
  unsafe extern "C" fn(libnode_sys::napi_env, libnode_sys::napi_value) -> libnode_sys::napi_value;

static NAPI_MODULE_NAMES: LazyLock<RwLock<HashMap<String, CString>>> =
  LazyLock::new(Default::default);

fn set_napi_module_register_name<S: AsRef<str>>(name: S) -> bool {
  let mut napi_module_names = NAPI_MODULE_NAMES.write().unwrap();
  let name = name.as_ref().to_string();
  let cname = CString::new(name.clone()).unwrap();
  if napi_module_names.contains_key(&name) {
    return false;
  }
  napi_module_names.insert(name, cname);
  true
}

fn get_napi_module_register_name<S: AsRef<str>>(name: S) -> Option<*const c_char> {
  let napi_module_names = NAPI_MODULE_NAMES.read().unwrap();
  let cname = napi_module_names.get(name.as_ref())?;
  Some(cname.as_ptr())
}

pub fn napi_module_register<
  S: AsRef<str>,
  F: 'static + Fn(Env, JsObject) -> crate::Result<JsObject>,
>(
  module_name: S,
  register_function: F,
) -> crate::Result<()> {
  if !set_napi_module_register_name(&module_name) {
    return Err(crate::Error::NapiModuleAlreadyRegistered);
  }

  let wrapped_fn = move |napi_env: libnode_sys::napi_env,
                         napi_value: libnode_sys::napi_value|
        -> libnode_sys::napi_value {
    let env = unsafe { Env::from_raw(napi_env) };
    let exports = unsafe { JsObject::from_raw_unchecked(napi_env, napi_value) };
    register_function(env, exports).unwrap();
    napi_value
  };

  let target_fn_leaked: &'static _ = Box::leak(Box::new(wrapped_fn));
  let target_fn_closure = libffi::high::Closure2::new(target_fn_leaked);
  let &target_fn_ptr = target_fn_closure.code_ptr();
  let target_fn: InitFn = unsafe { std::mem::transmute(target_fn_ptr) };
  std::mem::forget(target_fn_closure);

  let nm = Box::into_raw(Box::new(libnode_sys::napi_module {
    nm_version: 131 as c_int,
    nm_flags: 0 as c_uint,
    nm_filename: get_napi_module_register_name(&module_name).unwrap(),
    nm_register_func: Some(target_fn),
    nm_modname: get_napi_module_register_name(&module_name).unwrap(),
    nm_priv: get_napi_module_register_name(&module_name).unwrap() as *mut c_void,
    reserved: [
      std::ptr::null_mut::<c_void>(),
      ptr::null_mut(),
      ptr::null_mut(),
      ptr::null_mut(),
    ],
  }));

  unsafe {
    libnode_sys::napi_module_register(nm);
  }

  Ok(())
}