Merge pull request #42 from Brooooooklyn/refactor-async

refactor(napi): async and lifetime span
This commit is contained in:
LongYinan 2020-05-09 14:30:13 +08:00 committed by GitHub
commit 617e9c78b0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
24 changed files with 1645 additions and 462 deletions

2
.prettierignore Normal file
View file

@ -0,0 +1,2 @@
target
node_modules

View file

@ -27,6 +27,7 @@ This library depends on N-API and requires Node 8.9 or later. It is still pretty
One nice feature is that this crate allows you to build add-ons purely with the Rust toolchain and without involving `node-gyp`.
## Taste
```rust
#[js_function(1)] // ------> arguments length, omit for zero
fn fibonacci(ctx: CallContext) -> Result<Value<Number>> {
@ -62,7 +63,7 @@ napi-build = "0.1"
And create `build.rs` in your own project:
```rs
```rust
// build.rs
extern crate napi_build;
@ -78,7 +79,6 @@ See the included [test_module](./test_module) for an example add-on.
Run `cargo build` to produce the `Dynamic lib` file. And install the `napi-rs` to help you copy `Dynamic lib` file to `.node` file in case you can `require` it in your program.
```json
{
"package": "your pkg",
"dependencies": {
@ -128,7 +128,7 @@ npm test
### Create JavaScript values
| NAPI | NAPI Version | Minimal Node version | Status |
| ------------------------------------------------------------ | ------------ | -------------------- | ------ |
| ------------------------------------------------------------------------------------------------------------ | ------------ | -------------------- | ------ |
| [Enum types](https://nodejs.org/api/n-api.html#n_api_enum_types) | 6 | v13.7.0 | ⛔️ |
| [napi_create_array](https://nodejs.org/api/n-api.html#n_api_napi_create_array) | 1 | v8.0.0 | ✅ |
| [napi_create_array_with_length](https://nodejs.org/api/n-api.html#n_api_napi_create_array_with_length) | 1 | v8.0.0 | ✅ |
@ -157,7 +157,7 @@ npm test
### [Functions to convert from N-API to C types](https://nodejs.org/api/n-api.html#n_api_functions_to_convert_from_n_api_to_c_types)
| NAPI | NAPI Version | Minimal Node Version | Status |
| ---- | ------------ | -------------------- | ------ |
| ---------------------------------------------------------------------------------------------------- | ------------ | -------------------- | ------ |
| [napi_get_array_length](https://nodejs.org/api/n-api.html#n_api_napi_get_array_length) | 1 | v8.0.0 | ✅ |
| [napi_get_arraybuffer_info](https://nodejs.org/api/n-api.html#n_api_napi_get_arraybuffer_info) | 1 | v8.0.0 | ✅ |
| [napi_get_buffer_info](https://nodejs.org/api/n-api.html#n_api_napi_get_buffer_info) | 1 | v8.0.0 | ✅ |
@ -185,7 +185,7 @@ npm test
### [Working with JavaScript Values and Abstract Operations](https://nodejs.org/api/n-api.html#n_api_working_with_javascript_values_and_abstract_operations)
| NAPI | NAPI Version | Minimal Node Version | Status |
| ---- | ------------ | -------------------- | ------ |
| ---------------------------------------------------------------------------------------------------- | ------------ | -------------------- | ------ |
| [napi_coerce_to_bool](https://nodejs.org/api/n-api.html#n_api_napi_coerce_to_bool) | 1 | v8.0.0 | ✅ |
| [napi_coerce_to_number](https://nodejs.org/api/n-api.html#n_api_napi_coerce_to_number) | 1 | v8.0.0 | ✅ |
| [napi_coerce_to_object](https://nodejs.org/api/n-api.html#n_api_napi_coerce_to_object) | 1 | v8.0.0 | ✅ |

View file

@ -1,4 +1,5 @@
# napi-build
> Build support for napi-rs
Setup `N-API` build in your `build.rs`:

View file

@ -8,10 +8,7 @@ use std::convert::TryInto;
register_module!(test_module, init);
fn init<'env>(
env: &'env Env,
exports: &'env mut Value<'env, Object>,
) -> Result<Option<Value<'env, Object>>> {
fn init(env: &Env, exports: &mut Value<Object>) -> Result<Option<Value<Object>>> {
exports.set_named_property("testThrow", env.create_function("testThrow", test_throw)?)?;
exports.set_named_property("fibonacci", env.create_function("fibonacci", fibonacci)?)?;
@ -20,7 +17,7 @@ fn init<'env>(
#[js_function]
fn test_throw(_ctx: CallContext) -> Result<Value<Any>> {
Err(Error::new(Status::GenericFailure))
Err(Error::from_status(Status::GenericFailure))
}
#[js_function(1)]

View file

@ -102,16 +102,13 @@ pub fn js_function(attr: TokenStream, input: TokenStream) -> TokenStream {
}
let env = Env::from_raw(raw_env);
let call_ctx = CallContext::new(&env, raw_this, raw_args, #arg_len_span);
let call_ctx = CallContext::new(env, raw_this, raw_args, #arg_len_span);
let result = call_ctx.and_then(|ctx| #new_fn_name(ctx));
has_error = has_error && result.is_err();
match result {
Ok(result) => result.into_raw(),
Err(e) => {
if !cfg!(windows) {
let _ = writeln!(::std::io::stderr(), "Error calling function: {:?}", e);
}
let message = format!("{:?}", e);
unsafe {
napi_rs::sys::napi_throw_error(raw_env, ptr::null(), message.as_ptr() as *const c_char);

View file

@ -7,6 +7,7 @@ description = "N-API bindings"
readme = "README.md"
repository = "https://github.com/Brooooooklyn/napi-rs"
keywords = ["NodeJS", "FFI", "NAPI", "n-api"]
edition = "2018"
[dependencies]
futures = { version = "0.3", features = ["default", "thread-pool"] }
@ -25,7 +26,6 @@ tar = "0.4"
[build-dependencies]
bindgen = "0.53"
cc = "1.0"
glob = "0.3"
napi-build = { version = "0.1", path = "../build" }
regex = "1.3"

View file

@ -1,5 +1,4 @@
extern crate bindgen;
extern crate cc;
#[cfg(windows)]
extern crate flate2;
extern crate glob;
@ -36,8 +35,6 @@ fn main() {
);
}
env::set_var("CARGO_RUSTC_FLAGS", "-Clink-args=-export_dynamic");
if node_major_version < 10 {
panic!("node version is too low")
}
@ -60,31 +57,6 @@ fn main() {
.expect("Unable to generate napi bindings")
.write_to_file(out_path.join("bindings.rs"))
.expect("Unable to write napi bindings");
let mut bindings_path = PathBuf::from("src");
bindings_path.push("sys");
bindings_path.push("bindings.cc");
let mut cc_builder = cc::Build::new();
cc_builder
.cpp(true)
.include(&node_include_path)
.file(bindings_path);
if !cfg!(windows) {
cc_builder.flag("-Wno-unused-parameter");
};
if cfg!(target_os = "macos") {
cc_builder.flag("-std=c++0x");
} else if cfg!(linux) || cfg!(target_env = "gnu") {
cc_builder.flag("-std=c++14");
}
cc_builder
.cargo_metadata(true)
.out_dir(&out_path)
.compile("napi-bindings");
}
#[cfg(target_os = "windows")]

View file

@ -1,32 +1,35 @@
use crate::{sys, Any, Env, Error, Result, Status, Value, ValueType};
pub struct CallContext<'env, T: ValueType = Any> {
pub env: &'env Env,
pub this: Value<'env, T>,
pub struct CallContext<T: ValueType = Any> {
pub env: Env,
pub this: Value<T>,
args: [sys::napi_value; 8],
arg_len: usize,
}
impl<'env, T: ValueType> CallContext<'env, T> {
impl<T: ValueType> CallContext<T> {
pub fn new(
env: &'env Env,
env: Env,
this: sys::napi_value,
args: [sys::napi_value; 8],
arg_len: usize,
) -> Result<Self> {
Ok(Self {
env,
this: Value::<'env, T>::from_raw(env, this)?,
this: Value::<T>::from_raw(env.0, this)?,
args,
arg_len,
})
}
pub fn get<ArgType: ValueType>(&'env self, index: usize) -> Result<Value<'env, ArgType>> {
pub fn get<ArgType: ValueType>(&self, index: usize) -> Result<Value<ArgType>> {
if index + 1 > self.arg_len {
Err(Error::new(Status::GenericFailure))
Err(Error {
status: Status::GenericFailure,
reason: Some("Arguments index out of range".to_owned()),
})
} else {
Value::<'env, ArgType>::from_raw(&self.env, self.args[index])
Value::<ArgType>::from_raw(self.env.0, self.args[index])
}
}
}

View file

@ -1,4 +1,3 @@
use super::sys;
use futures::task::Poll;
use std::future::Future;
use std::mem;
@ -7,16 +6,7 @@ use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, RawWaker, RawWakerVTable, Waker};
pub struct LibuvExecutor {
event_loop: *mut sys::uv_loop_s,
}
#[derive(Clone, Debug)]
struct LibuvWaker(*mut sys::uv_async_t);
unsafe impl Send for LibuvWaker {}
unsafe impl Sync for LibuvWaker {}
use crate::{sys, Error, Result, Status};
const UV_ASYNC_V_TABLE: RawWakerVTable = RawWakerVTable::new(
clone_executor,
@ -39,27 +29,28 @@ unsafe fn wake_uv_async_by_ref(uv_async_t: *const ()) {
assert!(status == 0, "wake_uv_async_by_ref failed");
}
unsafe fn drop_uv_async(uv_async_t_ptr: *const ()) {
sys::uv_unref(uv_async_t_ptr as *mut sys::uv_handle_t);
}
unsafe fn drop_uv_async(_uv_async_t_ptr: *const ()) {}
struct Task<'a> {
future: Pin<Box<dyn Future<Output = ()>>>,
context: Context<'a>,
}
impl LibuvExecutor {
pub fn new(event_loop: *mut sys::uv_loop_s) -> Self {
Self { event_loop }
}
pub fn execute<F: 'static + Future<Output = ()>>(&self, future: F) {
pub fn execute<F: 'static + Future<Output = ()>>(
event_loop: *mut sys::uv_loop_s,
future: F,
) -> Result<()> {
let uninit = mem::MaybeUninit::<sys::uv_async_t>::uninit();
let uv_async_t: Box<sys::uv_async_t> = unsafe { Box::new(uninit.assume_init()) };
let uv_async_t_ref = Box::leak(uv_async_t);
unsafe {
let status = sys::uv_async_init(self.event_loop, uv_async_t_ref, Some(poll_future));
assert!(status == 0, "Non-zero status returned from uv_async_init");
let status = sys::uv_async_init(event_loop, uv_async_t_ref, Some(poll_future));
if status != 0 {
return Err(Error {
status: Status::Unknown,
reason: Some("Non-zero status returned from uv_async_init".to_owned()),
});
}
};
unsafe {
let waker = Waker::from_raw(RawWaker::new(
@ -77,8 +68,8 @@ impl LibuvExecutor {
uv_async_t_ref as *mut _ as *mut sys::uv_handle_t,
Arc::into_raw(arc_task) as *mut c_void,
);
}
}
};
Ok(())
}
}
@ -96,10 +87,7 @@ unsafe extern "C" fn poll_future(handle: *mut sys::uv_async_t) {
let mut task = Arc::from_raw(data_ptr);
if let Some(mut_task) = Arc::get_mut(&mut task) {
if mut_task.poll_future() {
sys::uv_close(
handle as *mut sys::uv_handle_t,
Some(drop_handle_after_close),
);
sys::uv_close(handle as *mut sys::uv_handle_t, None);
} else {
Arc::into_raw(task);
};
@ -107,7 +95,3 @@ unsafe extern "C" fn poll_future(handle: *mut sys::uv_async_t) {
Arc::into_raw(task);
}
}
unsafe extern "C" fn drop_handle_after_close(handle: *mut sys::uv_handle_t) {
Box::from_raw(handle);
}

View file

@ -1,5 +1,7 @@
extern crate futures;
use core::fmt::Debug;
use futures::prelude::*;
use std::any::TypeId;
use std::convert::{TryFrom, TryInto};
use std::ffi::CString;
@ -14,17 +16,21 @@ use std::string::String as RustString;
mod call_context;
mod executor;
mod promise;
pub mod sys;
mod version;
pub use call_context::CallContext;
pub use sys::{napi_valuetype, Status};
pub use version::NodeVersion;
pub type Result<T> = std::result::Result<T, Error>;
pub type Callback = extern "C" fn(sys::napi_env, sys::napi_callback_info) -> sys::napi_value;
#[derive(Debug)]
#[derive(Debug, Clone)]
pub struct Error {
status: Status,
pub status: Status,
pub reason: Option<String>,
}
#[derive(Clone, Copy, Debug)]
@ -75,8 +81,8 @@ pub struct ArrayBuffer {
}
#[derive(Clone, Copy, Debug)]
pub struct Value<'env, T> {
env: &'env Env,
pub struct Value<T> {
env: sys::napi_env,
raw_value: sys::napi_value,
value: T,
}
@ -87,14 +93,6 @@ pub struct Ref<T> {
_marker: PhantomData<T>,
}
pub struct AsyncContext {
raw_env: sys::napi_env,
raw_context: sys::napi_async_context,
raw_resource: sys::napi_ref,
}
pub struct Deferred(sys::napi_deferred);
#[derive(Clone, Debug)]
pub struct Property {
name: RustString,
@ -141,7 +139,7 @@ macro_rules! register_module {
raw_exports: sys::napi_value,
) -> sys::napi_value {
let env = Env::from_raw(raw_env);
let mut exports: Value<Object> = Value::from_raw(&env, raw_exports).unwrap();
let mut exports: Value<Object> = Value::from_raw(raw_env, raw_exports).unwrap();
let result = $init(&env, &mut exports);
@ -162,15 +160,19 @@ macro_rules! register_module {
}
impl Error {
pub fn new(status: Status) -> Self {
Error { status: status }
pub fn from_status(status: Status) -> Self {
Error {
status: status,
reason: None,
}
}
}
impl From<std::ffi::NulError> for Error {
fn from(_error: std::ffi::NulError) -> Self {
fn from(error: std::ffi::NulError) -> Self {
Error {
status: Status::StringExpected,
reason: Some(format!("{:?}", error)),
}
}
}
@ -180,14 +182,14 @@ impl Env {
Env(env)
}
pub fn get_undefined<'a>(&'a self) -> Result<Value<'a, Undefined>> {
pub fn get_undefined(&self) -> Result<Value<Undefined>> {
let mut raw_value = ptr::null_mut();
let status = unsafe { sys::napi_get_undefined(self.0, &mut raw_value) };
check_status(status)?;
Ok(Value::from_raw_value(self, raw_value, Undefined))
}
pub fn get_null<'a>(&'a self) -> Result<Value<'a, Null>> {
pub fn get_null(&self) -> Result<Value<Null>> {
let mut raw_value = ptr::null_mut();
let status = unsafe { sys::napi_get_null(self.0, &mut raw_value) };
check_status(status)?;
@ -201,7 +203,7 @@ impl Env {
Ok(Value::from_raw_value(self, raw_value, Boolean { value }))
}
pub fn create_int32<'a>(&'a self, int: i32) -> Result<Value<'a, Number>> {
pub fn create_int32(&self, int: i32) -> Result<Value<Number>> {
let mut raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_create_int32(self.0, int, (&mut raw_value) as *mut sys::napi_value) };
@ -209,7 +211,7 @@ impl Env {
Ok(Value::from_raw_value(self, raw_value, Number::Int32(int)))
}
pub fn create_int64<'a>(&'a self, int: i64) -> Result<Value<'a, Number>> {
pub fn create_int64(&self, int: i64) -> Result<Value<Number>> {
let mut raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_create_int64(self.0, int, (&mut raw_value) as *mut sys::napi_value) };
@ -217,7 +219,7 @@ impl Env {
Ok(Value::from_raw_value(self, raw_value, Number::Int(int)))
}
pub fn create_uint32<'a>(&'a self, number: u32) -> Result<Value<'a, Number>> {
pub fn create_uint32(&self, number: u32) -> Result<Value<Number>> {
let mut raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_create_uint32(self.0, number, (&mut raw_value) as *mut sys::napi_value) };
@ -225,7 +227,7 @@ impl Env {
Ok(Value::from_raw_value(self, raw_value, Number::U32(number)))
}
pub fn create_double<'a>(&'a self, double: f64) -> Result<Value<'a, Number>> {
pub fn create_double(&self, double: f64) -> Result<Value<Number>> {
let mut raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_create_double(self.0, double, (&mut raw_value) as *mut sys::napi_value) };
@ -237,7 +239,7 @@ impl Env {
))
}
pub fn create_string<'a, 'b>(&'a self, s: &'b str) -> Result<Value<'a, JsString>> {
pub fn create_string<'a, 'b>(&'a self, s: &'b str) -> Result<Value<JsString>> {
let mut raw_value = ptr::null_mut();
let status = unsafe {
sys::napi_create_string_utf8(
@ -260,7 +262,7 @@ impl Env {
Ok(Value::from_raw_value(self, raw_value, JsString))
}
pub fn create_object<'a>(&'a self) -> Result<Value<'a, Object>> {
pub fn create_object(&self) -> Result<Value<Object>> {
let mut raw_value = ptr::null_mut();
let status = unsafe { sys::napi_create_object(self.0, &mut raw_value) };
check_status(status)?;
@ -365,11 +367,7 @@ impl Env {
))
}
pub fn create_function<'a, 'b>(
&'a self,
name: &'b str,
callback: Callback,
) -> Result<Value<'a, Function>> {
pub fn create_function(&self, name: &str, callback: Callback) -> Result<Value<Function>> {
let mut raw_result = ptr::null_mut();
let status = unsafe {
sys::napi_create_function(
@ -414,7 +412,7 @@ impl Env {
check_status(status)?;
};
Value::from_raw(self, raw_value)
Value::from_raw(self.0, raw_value)
}
pub fn define_class<'a, 'b>(
@ -422,7 +420,7 @@ impl Env {
name: &'b str,
constructor_cb: Callback,
properties: Vec<Property>,
) -> Result<Value<'a, Function>> {
) -> Result<Value<Function>> {
let mut raw_result = ptr::null_mut();
let raw_properties = properties
.into_iter()
@ -473,10 +471,14 @@ impl Env {
let tagged_object: *mut TaggedObject<T> = mem::transmute(unknown_tagged_object);
(*tagged_object).object.as_mut().ok_or(Error {
status: Status::InvalidArg,
reason: Some("Invalid argument, nothing attach to js_object".to_owned()),
})
} else {
Err(Error {
status: Status::InvalidArg,
reason: Some(
"Invalid argument, T on unrwap is not the type of wrapped object".to_owned(),
),
})
}
}
@ -496,35 +498,40 @@ impl Env {
} else {
Err(Error {
status: Status::InvalidArg,
reason: Some(
"Invalid argument, T on drop_wrapped is not the type of wrapped object".to_owned(),
),
})
}
}
}
pub fn async_init(&self, resource: Option<Value<Object>>, name: &str) -> Result<AsyncContext> {
let raw_resource = resource
.map(|r| Ok(r.into_raw()))
.unwrap_or_else(|| self.create_object().map(|o| o.into_raw()))?;
let raw_name = self.create_string(name)?.into_raw();
let mut raw_context = ptr::null_mut();
let mut raw_resource_ref = ptr::null_mut();
unsafe {
let status = sys::napi_async_init(self.0, raw_resource, raw_name, &mut raw_context);
check_status(status)?;
let status = sys::napi_create_reference(self.0, raw_resource, 1, &mut raw_resource_ref);
pub fn create_error(&self, e: Error) -> Result<Value<Object>> {
let reason = e.reason.unwrap_or("".to_owned());
let reason_string = self.create_string(reason.as_str())?;
let mut result = ptr::null_mut();
let status = unsafe {
sys::napi_create_error(
self.0,
ptr::null_mut(),
reason_string.into_raw(),
&mut result,
)
};
check_status(status)?;
Ok(Value::from_raw_value(self, result, Object))
}
Ok(AsyncContext {
raw_env: self.0,
raw_resource: raw_resource_ref,
raw_context,
})
}
pub fn create_promise(&self) -> Result<(Value<Object>, Deferred)> {
pub fn perform_async_operation<
T: 'static,
V: 'static + ValueType,
F: 'static + Future<Output = Result<T>>,
R: 'static + FnOnce(&mut Env, T) -> Result<Value<V>>,
>(
&self,
deferred: F,
resolver: R,
) -> Result<Value<Object>> {
let mut raw_promise = ptr::null_mut();
let mut raw_deferred = ptr::null_mut();
@ -533,28 +540,41 @@ impl Env {
check_status(status)?;
}
Ok((
Value::from_raw_value(self, raw_promise, Object),
Deferred(raw_deferred),
))
}
pub fn resolve_deferred<T: ValueType>(&self, deferred: Deferred, value: Value<T>) -> Result<()> {
unsafe {
let status = sys::napi_resolve_deferred(self.0, deferred.0, value.into_raw());
check_status(status)
}
}
pub fn create_executor(&self) -> executor::LibuvExecutor {
let event_loop = unsafe { sys::uv_default_loop() };
executor::LibuvExecutor::new(event_loop)
let raw_env = self.0;
executor::execute(
event_loop,
promise::resolve(self.0, deferred, resolver, raw_deferred).map(move |v| match v {
Ok(value) => value,
Err(e) => {
let cloned_error = e.clone();
unsafe {
sys::napi_throw_error(
raw_env,
ptr::null(),
e.reason.unwrap_or(format!("{:?}", e.status)).as_ptr() as *const _,
);
};
eprintln!("{:?}", &cloned_error);
panic!(cloned_error);
}
}),
)?;
Ok(Value::from_raw_value(self, raw_promise, Object))
}
pub fn get_node_version(&self) -> Result<NodeVersion> {
let mut result = ptr::null();
check_status(unsafe { sys::napi_get_node_version(self.0, &mut result) })?;
let version = unsafe { *result };
version.try_into()
}
}
pub trait ValueType: Copy {
pub trait ValueType: Copy + Debug {
fn from_raw(env: sys::napi_env, raw: sys::napi_value) -> Result<Self>;
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool;
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool>;
}
impl ValueType for Any {
@ -562,8 +582,8 @@ impl ValueType for Any {
Ok(Any)
}
fn matches_raw_type(_env: sys::napi_env, _raw: sys::napi_value) -> bool {
true
fn matches_raw_type(_env: sys::napi_env, _raw: sys::napi_value) -> Result<bool> {
Ok(true)
}
}
@ -572,8 +592,8 @@ impl ValueType for Undefined {
Ok(Undefined)
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_undefined
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_undefined)
}
}
@ -582,8 +602,8 @@ impl ValueType for Null {
Ok(Null)
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_null
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_null)
}
}
@ -595,8 +615,8 @@ impl ValueType for Boolean {
Ok(Boolean { value })
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_boolean
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_boolean)
}
}
@ -608,8 +628,8 @@ impl ValueType for Number {
Ok(Number::Double(double))
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_number
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_number)
}
}
@ -618,8 +638,8 @@ impl ValueType for JsString {
Ok(JsString {})
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_string
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_string)
}
}
@ -628,8 +648,8 @@ impl ValueType for Object {
Ok(Object {})
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_object
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_object)
}
}
@ -645,13 +665,13 @@ impl ValueType for Buffer {
})
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
let mut result = false;
unsafe {
let status = sys::napi_is_buffer(env, raw, &mut result);
debug_assert!(Status::from(status) == Status::Ok);
check_status(status)?;
}
result
Ok(result)
}
}
@ -667,21 +687,21 @@ impl ValueType for ArrayBuffer {
})
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
let mut result = false;
unsafe {
let status = sys::napi_is_arraybuffer(env, raw, &mut result);
debug_assert!(Status::from(status) == Status::Ok);
check_status(status)?;
}
result
Ok(result)
}
}
impl<'env> Value<'env, Buffer> {
impl Value<Buffer> {
#[inline]
pub fn from_value(env: &'env Env, value: &Value<'env, Any>) -> Result<Value<'env, Buffer>> {
pub fn from_value(env: &Env, value: &Value<Any>) -> Result<Value<Buffer>> {
Ok(Value {
env,
env: env.0,
raw_value: value.raw_value,
value: Buffer::from_raw(env.0, value.into_raw())?,
})
@ -693,25 +713,25 @@ impl ValueType for Function {
Ok(Function {})
}
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
get_raw_type(env, raw) == sys::napi_valuetype::napi_function
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> Result<bool> {
Ok(get_raw_type(env, raw)? == sys::napi_valuetype::napi_function)
}
}
impl<'env, T: ValueType> Value<'env, T> {
pub fn from_raw_value(env: &'env Env, raw_value: sys::napi_value, value: T) -> Self {
impl<T: ValueType> Value<T> {
pub fn from_raw_value(env: &Env, raw_value: sys::napi_value, value: T) -> Self {
Self {
env,
env: env.0,
raw_value,
value,
}
}
pub fn from_raw(env: &'env Env, raw_value: sys::napi_value) -> Result<Self> {
pub fn from_raw(env: sys::napi_env, raw_value: sys::napi_value) -> Result<Self> {
Ok(Self {
env,
raw_value,
value: T::from_raw(env.0, raw_value)?,
value: T::from_raw(env, raw_value)?,
})
}
@ -719,22 +739,22 @@ impl<'env, T: ValueType> Value<'env, T> {
self.raw_value
}
pub fn coerce_to_number(self) -> Result<Value<'env, Number>> {
pub fn coerce_to_number(self) -> Result<Value<Number>> {
let mut new_raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_coerce_to_number(self.env.0, self.raw_value, &mut new_raw_value) };
unsafe { sys::napi_coerce_to_number(self.env, self.raw_value, &mut new_raw_value) };
check_status(status)?;
Ok(Value {
env: self.env,
raw_value: self.raw_value,
value: Number::from_raw(self.env.0, self.raw_value)?,
value: Number::from_raw(self.env, self.raw_value)?,
})
}
pub fn coerce_to_string(self) -> Result<Value<'env, JsString>> {
pub fn coerce_to_string(self) -> Result<Value<JsString>> {
let mut new_raw_value = ptr::null_mut();
let status =
unsafe { sys::napi_coerce_to_string(self.env.0, self.raw_value, &mut new_raw_value) };
unsafe { sys::napi_coerce_to_string(self.env, self.raw_value, &mut new_raw_value) };
check_status(status)?;
Ok(Value {
env: self.env,
@ -743,11 +763,11 @@ impl<'env, T: ValueType> Value<'env, T> {
})
}
pub fn coerce_to_object(self) -> Result<Value<'env, Object>> {
pub fn coerce_to_object(self) -> Result<Value<Object>> {
let mut new_raw_value = ptr::null_mut();
let status = unsafe {
sys::napi_coerce_to_object(
self.env.0,
self.env,
self.raw_value,
(&mut new_raw_value) as *mut sys::napi_value,
)
@ -761,7 +781,7 @@ impl<'env, T: ValueType> Value<'env, T> {
}
#[inline]
pub fn into_any(self) -> Value<'env, Any> {
pub fn into_any(self) -> Value<Any> {
Value {
env: self.env,
raw_value: self.raw_value,
@ -771,27 +791,26 @@ impl<'env, T: ValueType> Value<'env, T> {
}
#[inline]
fn get_raw_type(env: sys::napi_env, raw_value: sys::napi_value) -> sys::napi_valuetype {
fn get_raw_type(env: sys::napi_env, raw_value: sys::napi_value) -> Result<sys::napi_valuetype> {
unsafe {
let value_type = ptr::null_mut();
let status = sys::napi_typeof(env, raw_value, value_type);
debug_assert!(Status::from(status) == Status::Ok);
*value_type
check_status(sys::napi_typeof(env, raw_value, value_type))?;
Ok(*value_type)
}
}
impl<'env> Value<'env, Boolean> {
impl Value<Boolean> {
pub fn get_value(&self) -> bool {
self.value.value
}
}
impl<'env> Value<'env, JsString> {
impl Value<JsString> {
pub fn len(&self) -> Result<usize> {
let mut raw_length = ptr::null_mut();
unsafe {
let status = sys::napi_get_named_property(
self.env.0,
self.env,
self.raw_value,
"length\0".as_ptr() as *const c_char,
&mut raw_length,
@ -803,7 +822,7 @@ impl<'env> Value<'env, JsString> {
}
}
impl<'env> Value<'env, JsString> {
impl Value<JsString> {
#[inline]
pub fn get_ref(&self) -> Result<&[u8]> {
let mut written_char_count: u64 = 0;
@ -811,7 +830,7 @@ impl<'env> Value<'env, JsString> {
let mut result = Vec::with_capacity(len);
unsafe {
let status = sys::napi_get_value_string_utf8(
self.env.0,
self.env,
self.raw_value,
result.as_mut_ptr(),
len as u64,
@ -829,7 +848,10 @@ impl<'env> Value<'env, JsString> {
}
pub fn as_str(&self) -> Result<&str> {
str::from_utf8(self.get_ref()?).map_err(|_| Error::new(Status::GenericFailure))
str::from_utf8(self.get_ref()?).map_err(|e| Error {
status: Status::GenericFailure,
reason: Some(format!("{:?}", e)),
})
}
pub fn get_ref_mut(&mut self) -> Result<&mut [u8]> {
@ -838,7 +860,7 @@ impl<'env> Value<'env, JsString> {
let mut result = Vec::with_capacity(len);
unsafe {
let status = sys::napi_get_value_string_utf8(
self.env.0,
self.env,
self.raw_value,
result.as_mut_ptr(),
len as u64,
@ -856,16 +878,16 @@ impl<'env> Value<'env, JsString> {
}
}
impl<'env> TryFrom<Value<'env, JsString>> for Vec<u16> {
impl TryFrom<Value<JsString>> for Vec<u16> {
type Error = Error;
fn try_from(value: Value<'env, JsString>) -> Result<Vec<u16>> {
fn try_from(value: Value<JsString>) -> Result<Vec<u16>> {
let mut result = Vec::with_capacity(value.len()? + 1); // Leave room for trailing null byte
unsafe {
let mut written_char_count = 0;
let status = sys::napi_get_value_string_utf16(
value.env.0,
value.env,
value.raw_value,
result.as_mut_ptr(),
result.capacity() as u64,
@ -879,63 +901,63 @@ impl<'env> TryFrom<Value<'env, JsString>> for Vec<u16> {
}
}
impl<'env> TryFrom<Value<'env, Number>> for usize {
impl TryFrom<Value<Number>> for usize {
type Error = Error;
fn try_from(value: Value<'env, Number>) -> Result<usize> {
fn try_from(value: Value<Number>) -> Result<usize> {
let mut result = 0;
let status = unsafe { sys::napi_get_value_int64(value.env.0, value.raw_value, &mut result) };
let status = unsafe { sys::napi_get_value_int64(value.env, value.raw_value, &mut result) };
check_status(status)?;
Ok(result as usize)
}
}
impl<'env> TryFrom<Value<'env, Number>> for u32 {
impl TryFrom<Value<Number>> for u32 {
type Error = Error;
fn try_from(value: Value<'env, Number>) -> Result<u32> {
fn try_from(value: Value<Number>) -> Result<u32> {
let mut result = 0;
let status = unsafe { sys::napi_get_value_uint32(value.env.0, value.raw_value, &mut result) };
let status = unsafe { sys::napi_get_value_uint32(value.env, value.raw_value, &mut result) };
check_status(status)?;
Ok(result)
}
}
impl<'env> TryFrom<Value<'env, Number>> for i32 {
impl TryFrom<Value<Number>> for i32 {
type Error = Error;
fn try_from(value: Value<'env, Number>) -> Result<i32> {
fn try_from(value: Value<Number>) -> Result<i32> {
let mut result = 0;
let status = unsafe { sys::napi_get_value_int32(value.env.0, value.raw_value, &mut result) };
let status = unsafe { sys::napi_get_value_int32(value.env, value.raw_value, &mut result) };
check_status(status)?;
Ok(result)
}
}
impl<'env> TryFrom<Value<'env, Number>> for i64 {
impl TryFrom<Value<Number>> for i64 {
type Error = Error;
fn try_from(value: Value<'env, Number>) -> Result<i64> {
fn try_from(value: Value<Number>) -> Result<i64> {
let mut result = 0;
let status = unsafe { sys::napi_get_value_int64(value.env.0, value.raw_value, &mut result) };
let status = unsafe { sys::napi_get_value_int64(value.env, value.raw_value, &mut result) };
check_status(status)?;
Ok(result)
}
}
impl<'env> TryFrom<Value<'env, Number>> for f64 {
impl TryFrom<Value<Number>> for f64 {
type Error = Error;
fn try_from(value: Value<'env, Number>) -> Result<f64> {
fn try_from(value: Value<Number>) -> Result<f64> {
let mut result = 0_f64;
let status = unsafe { sys::napi_get_value_double(value.env.0, value.raw_value, &mut result) };
let status = unsafe { sys::napi_get_value_double(value.env, value.raw_value, &mut result) };
check_status(status)?;
Ok(result)
}
}
impl<'env> Value<'env, Object> {
pub fn set_property<'a, K, V>(&mut self, key: Value<K>, value: Value<V>) -> Result<()> {
impl Value<Object> {
pub fn set_property<K, V>(&mut self, key: Value<K>, value: Value<V>) -> Result<()> {
let status = unsafe {
sys::napi_set_property(
self.raw_env(),
@ -948,11 +970,7 @@ impl<'env> Value<'env, Object> {
Ok(())
}
pub fn set_named_property<'a, T, V: Into<Value<'a, T>>>(
&mut self,
name: &'a str,
value: V,
) -> Result<()> {
pub fn set_named_property<T, V: Into<Value<T>>>(&mut self, name: &str, value: V) -> Result<()> {
let key = CString::new(name)?;
let status = unsafe {
sys::napi_set_named_property(
@ -990,7 +1008,7 @@ impl<'env> Value<'env, Object> {
}
pub fn set_index<'a, T>(&mut self, index: usize, value: Value<T>) -> Result<()> {
self.set_property(self.env.create_int64(index as i64)?, value)
self.set_property(Env::from_raw(self.env).create_int64(index as i64)?, value)
}
pub fn get_index<T: ValueType>(&self, index: u32) -> Result<Value<T>> {
@ -1015,7 +1033,7 @@ impl<'env> Value<'env, Object> {
Ok(is_buffer)
}
pub fn to_buffer(&self) -> Result<Value<'env, Buffer>> {
pub fn to_buffer(&self) -> Result<Value<Buffer>> {
Value::from_raw(self.env, self.raw_value)
}
@ -1023,6 +1041,7 @@ impl<'env> Value<'env, Object> {
if self.is_array()? != true {
return Err(Error {
status: Status::ArrayExpected,
reason: Some("Object is not array".to_owned()),
});
}
let mut length: u32 = 0;
@ -1037,17 +1056,17 @@ impl<'env> Value<'env, Object> {
}
fn raw_env(&self) -> sys::napi_env {
self.env.0
self.env
}
}
impl<'env> AsRef<[u8]> for Value<'env, Buffer> {
impl AsRef<[u8]> for Value<Buffer> {
fn as_ref(&self) -> &[u8] {
self.deref()
}
}
impl<'env> Deref for Value<'env, Buffer> {
impl Deref for Value<Buffer> {
type Target = [u8];
fn deref(&self) -> &[u8] {
@ -1055,13 +1074,13 @@ impl<'env> Deref for Value<'env, Buffer> {
}
}
impl<'env> DerefMut for Value<'env, Buffer> {
impl DerefMut for Value<Buffer> {
fn deref_mut(&mut self) -> &mut [u8] {
unsafe { slice::from_raw_parts_mut(self.value.data as *mut _, self.value.size as usize) }
}
}
impl<'env> Deref for Value<'env, ArrayBuffer> {
impl Deref for Value<ArrayBuffer> {
type Target = [u8];
fn deref(&self) -> &[u8] {
@ -1069,22 +1088,26 @@ impl<'env> Deref for Value<'env, ArrayBuffer> {
}
}
impl<'env> DerefMut for Value<'env, ArrayBuffer> {
impl DerefMut for Value<ArrayBuffer> {
fn deref_mut(&mut self) -> &mut [u8] {
unsafe { slice::from_raw_parts_mut(self.value.data as *mut _, self.value.size as usize) }
}
}
impl<'env> Value<'env, Function> {
pub fn call(
&self,
this: Option<&Value<'env, Object>>,
args: &[Value<'env, Any>],
) -> Result<Value<'env, Any>> {
impl Value<Function> {
pub fn call(&self, this: Option<&Value<Object>>, args: &[Value<Any>]) -> Result<Value<Any>> {
let raw_this = this
.map(|v| v.into_raw())
.or_else(|| self.env.get_undefined().ok().map(|u| u.into_raw()))
.ok_or(Error::new(Status::Unknown))?;
.or_else(|| {
Env::from_raw(self.env)
.get_undefined()
.ok()
.map(|u| u.into_raw())
})
.ok_or(Error {
status: Status::Unknown,
reason: Some("Get raw this failed".to_owned()),
})?;
let mut raw_args = unsafe { mem::MaybeUninit::<[sys::napi_value; 8]>::uninit().assume_init() };
for (i, arg) in args.into_iter().enumerate() {
raw_args[i] = arg.raw_value;
@ -1092,7 +1115,7 @@ impl<'env> Value<'env, Function> {
let mut return_value = ptr::null_mut();
let status = unsafe {
sys::napi_call_function(
self.env.0,
self.env,
raw_this,
self.raw_value,
args.len() as u64,
@ -1106,9 +1129,9 @@ impl<'env> Value<'env, Function> {
}
}
impl<'env> Value<'env, Any> {
pub fn get_type(&self) -> sys::napi_valuetype {
get_raw_type(self.env.0, self.raw_value)
impl Value<Any> {
pub fn get_type(&self) -> Result<sys::napi_valuetype> {
get_raw_type(self.env, self.raw_value)
}
}
@ -1127,34 +1150,6 @@ impl<T> Drop for Ref<T> {
}
}
impl AsyncContext {
pub fn enter<'a, F: 'a + FnOnce(&mut Env)>(&'a self, run_in_context: F) {
let mut env = Env::from_raw(self.raw_env);
let mut handle_scope = ptr::null_mut();
let mut callback_scope = ptr::null_mut();
let mut raw_resource = ptr::null_mut();
unsafe {
sys::napi_open_handle_scope(env.0, &mut handle_scope);
sys::napi_get_reference_value(env.0, self.raw_resource, &mut raw_resource);
sys::extras_open_callback_scope(self.raw_context, raw_resource, &mut callback_scope);
}
run_in_context(&mut env);
unsafe {
sys::extras_close_callback_scope(callback_scope);
sys::napi_close_handle_scope(env.0, handle_scope);
}
}
}
impl Drop for AsyncContext {
fn drop(&mut self) {
unsafe {
sys::napi_delete_reference(self.raw_env, self.raw_resource);
}
}
}
impl Property {
pub fn new(name: &str) -> Self {
Property {
@ -1202,11 +1197,12 @@ impl<T: 'static> TaggedObject<T> {
}
}
#[inline]
fn check_status(code: sys::napi_status) -> Result<()> {
let status = Status::from(code);
match status {
Status::Ok => Ok(()),
_ => Err(Error { status }),
_ => Err(Error::from_status(status)),
}
}

65
napi/src/promise.rs Normal file
View file

@ -0,0 +1,65 @@
use futures::prelude::*;
use std::os::raw::c_char;
use std::ptr;
use crate::{check_status, sys, Env, Result, Value, ValueType};
#[inline]
pub async fn resolve<
T,
V: ValueType,
R: FnOnce(&mut Env, T) -> Result<Value<V>>,
F: Future<Output = Result<T>>,
>(
env: sys::napi_env,
fut: F,
resolver: R,
raw_deferred: sys::napi_deferred,
) -> Result<()> {
let mut raw_resource = ptr::null_mut();
let status = unsafe { sys::napi_create_object(env, &mut raw_resource) };
check_status(status)?;
let mut raw_name = ptr::null_mut();
let s = "napi_async_context";
let status = unsafe {
sys::napi_create_string_utf8(
env,
s.as_ptr() as *const c_char,
s.len() as u64,
&mut raw_name,
)
};
check_status(status)?;
let mut raw_context = ptr::null_mut();
unsafe {
let status = sys::napi_async_init(env, raw_resource, raw_name, &mut raw_context);
check_status(status)?;
}
let mut handle_scope = ptr::null_mut();
match fut.await {
Ok(v) => unsafe {
check_status(sys::napi_open_handle_scope(env, &mut handle_scope))?;
let mut tmp_env = Env::from_raw(env);
let js_value = resolver(&mut tmp_env, v)?;
check_status(sys::napi_resolve_deferred(
env,
raw_deferred,
js_value.raw_value,
))?;
check_status(sys::napi_close_handle_scope(env, handle_scope))?;
},
Err(e) => unsafe {
check_status(sys::napi_open_handle_scope(env, &mut handle_scope))?;
check_status(sys::napi_reject_deferred(
env,
raw_deferred,
Env::from_raw(env)
.create_error(e)
.map(|e| e.into_raw())
.unwrap_or(ptr::null_mut()),
))?;
check_status(sys::napi_close_handle_scope(env, handle_scope))?;
},
};
Ok(())
}

View file

@ -1,31 +0,0 @@
#include "bindings.h"
#include <stdio.h>
#include <node.h>
#include <v8.h>
static v8::Local<v8::Value> V8LocalValueFromJsValue(napi_value v)
{
v8::Local<v8::Value> local;
memcpy(&local, &v, sizeof(v));
return local;
}
EXTERN_C_START
NAPI_EXTERN void extras_open_callback_scope(napi_async_context napi_async_context,
napi_value napi_resource_object,
extras_callback_scope *result)
{
v8::Isolate *isolate = v8::Isolate::GetCurrent();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
v8::Local<v8::Object> resource_object = V8LocalValueFromJsValue(napi_resource_object)->ToObject(context).ToLocalChecked();
node::async_context *node_async_context = reinterpret_cast<node::async_context *>(napi_async_context);
*result = reinterpret_cast<extras_callback_scope>(new node::CallbackScope(isolate, resource_object, *node_async_context));
}
NAPI_EXTERN void extras_close_callback_scope(extras_callback_scope callback_scope)
{
delete reinterpret_cast<node::CallbackScope *>(callback_scope);
}
EXTERN_C_END

View file

@ -1,15 +1,2 @@
#include <node_api.h>
#include <uv.h>
#include <string.h>
typedef struct extras_callback_scope__ *extras_callback_scope;
EXTERN_C_START
NAPI_EXTERN void extras_open_callback_scope(napi_async_context napi_async_context,
napi_value napi_resource_object,
extras_callback_scope *result);
NAPI_EXTERN void extras_close_callback_scope(extras_callback_scope callback_scope);
EXTERN_C_END

View file

@ -1,6 +1,6 @@
use super::napi_status;
#[derive(Eq, PartialEq, Debug)]
#[derive(Eq, PartialEq, Debug, Clone, Copy)]
pub enum Status {
Ok,
InvalidArg,

29
napi/src/version.rs Normal file
View file

@ -0,0 +1,29 @@
use crate::{sys, Error, Status};
use std::convert::TryFrom;
use std::ffi::CStr;
#[derive(Debug, Clone, Copy)]
pub struct NodeVersion {
pub major: u32,
pub minor: u32,
pub patch: u32,
pub release: &'static str,
}
impl TryFrom<sys::napi_node_version> for NodeVersion {
type Error = Error;
fn try_from(value: sys::napi_node_version) -> Result<NodeVersion, Error> {
Ok(NodeVersion {
major: value.major,
minor: value.minor,
patch: value.patch,
release: unsafe {
CStr::from_ptr(value.release).to_str().map_err(|_| Error {
status: Status::StringExpected,
reason: Some("Invalid release name".to_owned()),
})?
},
})
}
}

View file

@ -17,6 +17,13 @@
}
],
"license": "MIT",
"scripts": {
"format": "run-p format:md format:json format:yaml format:source",
"format:md": "prettier --parser markdown --write './**/*.md'",
"format:json": "prettier --parser json --write './**/*.json'",
"format:source": "prettier --config ./package.json --write './**/*.js'",
"format:yaml": "prettier --parser yaml --write './**/*.{yml,yaml}'"
},
"bugs": {
"url": "https://github.com/Brooooooklyn/napi-rs/issues"
},
@ -32,12 +39,23 @@
"trailingComma": "all",
"arrowParens": "always"
},
"files": [
"scripts/napi.js",
"LICENSE"
],
"files": ["scripts/napi.js", "LICENSE"],
"lint-staged": {
"*.js": ["prettier --write"],
"*.@(yml|yaml)": ["prettier --parser yaml --write"],
"*.json": ["prettier --parser json --write"],
"*.md": ["prettier --parser markdown --write"]
},
"husky": {
"hooks": {
"pre-commit": "lint-staged && cargo fmt --all"
}
},
"devDependencies": {
"@types/node": "^13.13.5",
"husky": "^4.2.5",
"lint-staged": "^10.2.2",
"npm-run-all": "^4.1.5",
"prettier": "^2.0.5"
}
}

View file

@ -11,7 +11,10 @@ let tomlContent
let moduleName
try {
tomlContentString = fs.readFileSync(path.join(process.cwd(), 'Cargo.toml'), 'utf-8')
tomlContentString = fs.readFileSync(
path.join(process.cwd(), 'Cargo.toml'),
'utf-8',
)
} catch {
throw new TypeError('Can not find Cargo.toml in process.cwd')
}
@ -60,7 +63,9 @@ const targetDir = argv.release ? 'release' : 'debug'
const platformName = argv.platform ? `.${platform}` : ''
let subcommand = argv._[0] || path.join('target', targetDir, `${moduleName}${platformName}.node`)
let subcommand =
argv._[0] ||
path.join('target', targetDir, `${moduleName}${platformName}.node`)
const parsedDist = path.parse(subcommand)
if (!parsedDist.name || parsedDist.name === '.') {
@ -73,11 +78,13 @@ if (!parsedDist.ext) {
const pos = __dirname.indexOf('node_modules')
const dylibContent = fs.readFileSync(path.join(
const dylibContent = fs.readFileSync(
path.join(
__dirname.substring(0, pos),
'target',
targetDir,
`${dylibName}${libExt}`,
))
),
)
fs.writeFileSync(subcommand, dylibContent)

View file

@ -1,4 +1,4 @@
const testModule = require(`./target/debug/test_module.node`)
const testModule = require('./index.node')
function testSpawn() {
console.log('=== Test spawning a future on libuv event loop')
@ -9,19 +9,21 @@ function testThrow() {
console.log('=== Test throwing from Rust')
try {
testModule.testThrow()
} catch (e) {
return
}
console.error('Expected function to throw an error')
process.exit(1)
} catch (e) {
console.error(e)
}
}
const future = testSpawn()
// https://github.com/nodejs/node/issues/29355
setTimeout(() => {
future.then(testThrow).catch((e) => {
future
.then((value) => {
console.info(`${value} from napi`)
testThrow()
})
.catch((e) => {
console.error(e)
process.exit(1)
})
}, 1000)

View file

@ -2,8 +2,8 @@
"name": "test-module",
"version": "1.0.0",
"scripts": {
"build": "cargo build && node ../scripts/napi.js",
"build-release": "cargo build --release && node ../scripts/napi.js --release",
"build": "cargo build && node ../scripts/napi.js ./index",
"build-release": "cargo build --release && node ../scripts/napi.js --release ./index",
"test": "node ./index.js"
}
}

View file

@ -9,49 +9,36 @@ use napi::{Any, CallContext, Env, Error, Object, Result, Status, Value};
register_module!(test_module, init);
fn init<'env>(
env: &'env Env,
exports: &'env mut Value<'env, Object>,
) -> Result<Option<Value<'env, Object>>> {
fn init(env: &Env, exports: &mut Value<Object>) -> Result<Option<Value<Object>>> {
exports.set_named_property("testSpawn", env.create_function("testSpawn", test_spawn)?)?;
exports.set_named_property("testThrow", env.create_function("testThrow", test_throw)?)?;
Ok(None)
}
#[js_function]
fn test_spawn<'a>(ctx: CallContext<'a>) -> Result<Value<'a, Object>> {
fn test_spawn(ctx: CallContext) -> Result<Value<Object>> {
use futures::executor::ThreadPool;
use futures::StreamExt;
let env = ctx.env;
let async_context = env.async_init(None, "test_spawn")?;
let pool = ThreadPool::new().expect("Failed to build pool");
let (promise, deferred) = env.create_promise()?;
let (tx, rx) = futures::channel::mpsc::unbounded::<i32>();
let fut_values = async move {
let fut_tx_result = async move {
(0..20).for_each(|v| {
(0..200).for_each(|v| {
tx.unbounded_send(v).expect("Failed to send");
})
};
pool.spawn_ok(fut_tx_result);
let fut_values = rx.map(|v| v * 2).collect::<Vec<i32>>();
let results = fut_values.await;
if !cfg!(windows) {
let fut = rx.map(|v| v * 2).collect::<Vec<i32>>();
let results = fut.await;
println!("Collected result lenght {}", results.len());
};
async_context.enter(|env| {
env
.resolve_deferred(deferred, env.get_undefined().unwrap())
.unwrap();
});
Ok(results.len() as u32)
};
env.create_executor().execute(fut_values);
Ok(promise)
env.perform_async_operation(fut_values, |&mut env, len| env.create_uint32(len))
}
#[js_function]
fn test_throw<'a>(_ctx: CallContext<'a>) -> Result<Value<'a, Any>> {
Err(Error::new(Status::GenericFailure))
fn test_throw(_ctx: CallContext) -> Result<Value<Any>> {
Err(Error::from_status(Status::GenericFailure))
}

1167
yarn.lock

File diff suppressed because it is too large Load diff