2020-07-14 23:57:11 +09:00
|
|
|
//! High level NodeJS [N-API](https://nodejs.org/api/n-api.html) binding
|
|
|
|
//!
|
|
|
|
//! **napi-rs** provides minimal overhead to write N-API modules in `Rust`.
|
2020-09-03 21:38:28 +09:00
|
|
|
//!
|
2020-07-14 23:57:11 +09:00
|
|
|
//! ## Feature flags
|
2020-09-03 21:38:28 +09:00
|
|
|
//!
|
2020-07-14 23:57:11 +09:00
|
|
|
//! ### libuv
|
|
|
|
//! With `libuv` feature, you can execute a rust future in `libuv` in NodeJS, and return a `promise` object.
|
|
|
|
//! ```
|
|
|
|
//! use std::thread;
|
|
|
|
//! use std::fs;
|
|
|
|
//!
|
|
|
|
//! use futures::prelude::*;
|
|
|
|
//! use futures::channel::oneshot;
|
|
|
|
//! use napi::{CallContext, Result, JsString, JsObject, Status, Error};
|
|
|
|
//!
|
|
|
|
//! #[js_function(1)]
|
|
|
|
//! pub fn uv_read_file(ctx: CallContext) -> Result<JsObject> {
|
2020-10-14 12:29:51 +09:00
|
|
|
//! let path = ctx.get::<JsString>(0)?;
|
|
|
|
//! let (sender, receiver) = oneshot::channel();
|
|
|
|
//! let p = path.as_str()?.to_owned();
|
|
|
|
//! thread::spawn(|| {
|
|
|
|
//! let res = fs::read(p).map_err(|e| Error::new(Status::Unknown, format!("{}", e)));
|
|
|
|
//! sender.send(res).expect("Send data failed");
|
|
|
|
//! });
|
|
|
|
//! ctx.env.execute(receiver.map_err(|e| Error::new(Status::Unknown, format!("{}", e))).map(|x| x.and_then(|x| x)), |&mut env, data| {
|
|
|
|
//! env.create_buffer_with_data(data)
|
|
|
|
//! })
|
2020-07-14 23:57:11 +09:00
|
|
|
//! }
|
|
|
|
//! ```
|
|
|
|
//! ### tokio_rt
|
|
|
|
//! With `tokio_rt` feature, `napi-rs` provides a ***tokio runtime*** in an additional thread.
|
|
|
|
//! And you can easily run tokio `future` in it and return `promise`.
|
|
|
|
//!
|
|
|
|
//! ```
|
|
|
|
//! use futures::prelude::*;
|
|
|
|
//! use napi::{CallContext, Error, JsObject, JsString, Result, Status};
|
|
|
|
//! use tokio;
|
|
|
|
//!
|
|
|
|
//! #[js_function(1)]
|
|
|
|
//! pub fn tokio_readfile(ctx: CallContext) -> Result<JsObject> {
|
2020-10-14 12:29:51 +09:00
|
|
|
//! let js_filepath = ctx.get::<JsString>(0)?;
|
|
|
|
//! let path_str = js_filepath.as_str()?;
|
|
|
|
//! ctx.env.execute_tokio_future(
|
|
|
|
//! tokio::fs::read(path_str.to_owned())
|
|
|
|
//! .map(|v| v.map_err(|e| Error::new(Status::Unknown, format!("failed to read file, {}", e)))),
|
|
|
|
//! |&mut env, data| env.create_buffer_with_data(data),
|
|
|
|
//! )
|
2020-07-14 23:57:11 +09:00
|
|
|
//! }
|
|
|
|
//! ```
|
|
|
|
//!
|
|
|
|
//! ***Tokio channel in `napi-rs` buffer size is default `100`.***
|
|
|
|
//!
|
|
|
|
//! ***You can adjust it via `NAPI_RS_TOKIO_CHANNEL_BUFFER_SIZE` environment variable***
|
|
|
|
//!
|
|
|
|
//! ```
|
|
|
|
//! NAPI_RS_TOKIO_CHANNEL_BUFFER_SIZE=1000 node ./app.js
|
|
|
|
//! ```
|
|
|
|
//!
|
2020-09-03 21:38:28 +09:00
|
|
|
//! ### latin1
|
|
|
|
//!
|
|
|
|
//! Decode latin1 string from JavaScript using [encoding_rs](https://docs.rs/encoding_rs).
|
|
|
|
//!
|
|
|
|
//! With this feature, you can use `JsString.as_latin1_string` function
|
|
|
|
//!
|
|
|
|
//! ### serde-json
|
|
|
|
//!
|
|
|
|
//! Enable Serialize/Deserialize data cross `JavaScript Object` and `Rust struct`.
|
|
|
|
//!
|
|
|
|
//! ```
|
|
|
|
//! #[derive(Serialize, Debug, Deserialize)]
|
|
|
|
//! struct AnObject {
|
2020-10-14 12:29:51 +09:00
|
|
|
//! a: u32,
|
|
|
|
//! b: Vec<f64>,
|
|
|
|
//! c: String,
|
2020-09-03 21:38:28 +09:00
|
|
|
//! }
|
|
|
|
//!
|
|
|
|
//! #[js_function(1)]
|
|
|
|
//! fn deserialize_from_js(ctx: CallContext) -> Result<JsUndefined> {
|
2020-10-14 12:29:51 +09:00
|
|
|
//! let arg0 = ctx.get::<JsUnknown>(0)?;
|
|
|
|
//! let de_serialized: AnObject = ctx.env.from_js_value(arg0)?;
|
|
|
|
//! ...
|
2020-09-03 21:38:28 +09:00
|
|
|
//! }
|
|
|
|
//!
|
|
|
|
//! #[js_function]
|
|
|
|
//! fn serialize(ctx: CallContext) -> Result<JsUnknown> {
|
2020-10-14 12:29:51 +09:00
|
|
|
//! let value = AnyObject { a: 1, b: vec![0.1, 2.22], c: "hello" };
|
|
|
|
//! ctx.env.to_js_value(&value)
|
2020-09-03 21:38:28 +09:00
|
|
|
//! }
|
|
|
|
//! ```
|
|
|
|
//!
|
2020-07-14 23:57:11 +09:00
|
|
|
|
2020-05-12 14:59:20 +09:00
|
|
|
mod async_work;
|
2020-04-26 19:46:56 +09:00
|
|
|
mod call_context;
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(feature = "napi3")]
|
2020-10-04 17:02:04 +09:00
|
|
|
mod cleanup_env;
|
2020-06-21 20:10:06 +09:00
|
|
|
mod env;
|
|
|
|
mod error;
|
|
|
|
mod js_values;
|
|
|
|
mod module;
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2020-07-03 01:31:50 +09:00
|
|
|
mod promise;
|
2020-07-01 21:44:29 +09:00
|
|
|
mod status;
|
2020-05-11 01:28:06 +09:00
|
|
|
mod task;
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(feature = "napi3")]
|
2020-10-04 17:02:04 +09:00
|
|
|
pub use cleanup_env::CleanupEnvHook;
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(feature = "napi4")]
|
2020-06-19 21:42:18 +09:00
|
|
|
pub mod threadsafe_function;
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2020-07-08 01:59:09 +09:00
|
|
|
mod tokio_rt;
|
2020-05-06 00:13:23 +09:00
|
|
|
mod version;
|
2020-10-31 10:24:19 +09:00
|
|
|
#[cfg(target_os = "windows")]
|
|
|
|
mod win_delay_load_hook;
|
2018-04-28 17:26:38 +09:00
|
|
|
|
2020-07-15 01:43:20 +09:00
|
|
|
pub use napi_sys as sys;
|
|
|
|
|
2020-04-21 01:20:35 +09:00
|
|
|
pub use call_context::CallContext;
|
2020-06-21 20:10:06 +09:00
|
|
|
pub use env::*;
|
|
|
|
pub use error::{Error, Result};
|
|
|
|
pub use js_values::*;
|
|
|
|
pub use module::Module;
|
2020-07-01 21:44:29 +09:00
|
|
|
pub use status::Status;
|
2020-05-11 01:28:06 +09:00
|
|
|
pub use task::Task;
|
2020-05-06 00:13:23 +09:00
|
|
|
pub use version::NodeVersion;
|
2018-04-28 17:26:38 +09:00
|
|
|
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2020-07-08 01:59:09 +09:00
|
|
|
pub use tokio_rt::shutdown as shutdown_tokio_rt;
|
|
|
|
|
2020-08-26 01:07:27 +09:00
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
#[macro_use]
|
|
|
|
extern crate serde;
|
|
|
|
|
2020-09-30 15:34:26 +09:00
|
|
|
pub type ContextlessResult<T> = Result<Option<T>>;
|
|
|
|
|
2020-07-14 23:57:11 +09:00
|
|
|
/// register nodejs module
|
|
|
|
///
|
|
|
|
/// ## Example
|
|
|
|
/// ```
|
|
|
|
/// register_module!(test_module, init);
|
|
|
|
///
|
|
|
|
/// fn init(module: &mut Module) -> Result<()> {
|
2020-10-14 12:29:51 +09:00
|
|
|
/// module.create_named_method("nativeFunction", native_function)?;
|
2020-07-14 23:57:11 +09:00
|
|
|
/// }
|
|
|
|
/// ```
|
2018-04-28 17:26:38 +09:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! register_module {
|
|
|
|
($module_name:ident, $init:ident) => {
|
2020-07-08 01:59:09 +09:00
|
|
|
#[inline]
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2020-07-08 01:59:09 +09:00
|
|
|
fn check_status(code: $crate::sys::napi_status) -> Result<()> {
|
2020-07-15 00:52:42 +09:00
|
|
|
use $crate::{Error, Status};
|
2020-07-08 01:59:09 +09:00
|
|
|
let status = Status::from(code);
|
|
|
|
match status {
|
|
|
|
Status::Ok => Ok(()),
|
|
|
|
_ => Err(Error::from_status(status)),
|
|
|
|
}
|
|
|
|
}
|
2020-10-14 12:29:51 +09:00
|
|
|
|
2020-11-04 16:54:50 +09:00
|
|
|
#[no_mangle]
|
|
|
|
unsafe extern "C" fn napi_register_module_v1(
|
2020-11-12 12:41:41 +09:00
|
|
|
raw_env: $crate::sys::napi_env,
|
|
|
|
raw_exports: $crate::sys::napi_value,
|
|
|
|
) -> $crate::sys::napi_value {
|
|
|
|
use std::ffi::CString;
|
|
|
|
use std::io::Write;
|
|
|
|
use std::os::raw::c_char;
|
|
|
|
use std::ptr;
|
|
|
|
use $crate::{Env, JsObject, NapiValue};
|
|
|
|
|
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
|
|
|
use $crate::shutdown_tokio_rt;
|
|
|
|
|
2020-11-04 16:54:50 +09:00
|
|
|
let env = Env::from_raw(raw_env);
|
|
|
|
let mut exports: JsObject = JsObject::from_raw_unchecked(raw_env, raw_exports);
|
|
|
|
let mut cjs_module = Module { env, exports };
|
|
|
|
let result = $init(&mut cjs_module);
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2020-11-04 16:54:50 +09:00
|
|
|
let hook_result = check_status(unsafe {
|
2020-11-12 12:41:41 +09:00
|
|
|
$crate::sys::napi_add_env_cleanup_hook(raw_env, Some(shutdown_tokio_rt), ptr::null_mut())
|
2020-11-04 16:54:50 +09:00
|
|
|
});
|
2020-11-10 12:09:25 +09:00
|
|
|
#[cfg(not(all(feature = "tokio_rt", feature = "napi4")))]
|
2020-11-04 16:54:50 +09:00
|
|
|
let hook_result = Ok(());
|
|
|
|
match hook_result.and_then(move |_| result) {
|
|
|
|
Ok(_) => cjs_module.exports.raw(),
|
|
|
|
Err(e) => {
|
|
|
|
unsafe {
|
2020-11-12 12:41:41 +09:00
|
|
|
$crate::sys::napi_throw_error(
|
2020-11-04 16:54:50 +09:00
|
|
|
raw_env,
|
|
|
|
ptr::null(),
|
|
|
|
CString::from_vec_unchecked(format!("Error initializing module: {}", e).into())
|
|
|
|
.as_ptr() as *const _,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
ptr::null_mut()
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
2020-10-14 12:29:51 +09:00
|
|
|
}
|
2018-04-28 17:26:38 +09:00
|
|
|
};
|
|
|
|
}
|