refactor(napi): thread safe function redesign
This commit is contained in:
parent
64bf2e81be
commit
c5f2b6699d
5 changed files with 184 additions and 181 deletions
|
@ -16,6 +16,8 @@ use crate::{sys, Error, NodeVersion, Result, Status};
|
||||||
use crate::js_values::{De, Ser};
|
use crate::js_values::{De, Ser};
|
||||||
#[cfg(all(any(feature = "libuv", feature = "tokio_rt"), napi4))]
|
#[cfg(all(any(feature = "libuv", feature = "tokio_rt"), napi4))]
|
||||||
use crate::promise;
|
use crate::promise;
|
||||||
|
#[cfg(napi4)]
|
||||||
|
use crate::threadsafe_function::{ThreadSafeCallContext, ThreadsafeFunction};
|
||||||
#[cfg(all(feature = "tokio_rt", napi4))]
|
#[cfg(all(feature = "tokio_rt", napi4))]
|
||||||
use crate::tokio_rt::{get_tokio_sender, Message};
|
use crate::tokio_rt::{get_tokio_sender, Message};
|
||||||
#[cfg(all(feature = "libuv", napi4))]
|
#[cfg(all(feature = "libuv", napi4))]
|
||||||
|
@ -554,6 +556,20 @@ impl Env {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(napi4)]
|
||||||
|
pub fn create_threadsafe_function<
|
||||||
|
T: Send,
|
||||||
|
V: NapiValue,
|
||||||
|
R: 'static + Send + FnMut(ThreadSafeCallContext<T>) -> Result<Vec<V>>,
|
||||||
|
>(
|
||||||
|
&self,
|
||||||
|
func: JsFunction,
|
||||||
|
max_queue_size: u64,
|
||||||
|
callback: R,
|
||||||
|
) -> Result<ThreadsafeFunction<T>> {
|
||||||
|
ThreadsafeFunction::create(self.0, func, max_queue_size, callback)
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(all(feature = "libuv", napi4))]
|
#[cfg(all(feature = "libuv", napi4))]
|
||||||
pub fn execute<
|
pub fn execute<
|
||||||
T: 'static + Send,
|
T: 'static + Send,
|
||||||
|
|
|
@ -71,8 +71,6 @@ pub(crate) struct TSFNValue(sys::napi_threadsafe_function);
|
||||||
|
|
||||||
unsafe impl Send for TSFNValue {}
|
unsafe impl Send for TSFNValue {}
|
||||||
|
|
||||||
unsafe impl Sync for TSFNValue {}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub(crate) async fn resolve_from_future<T: Send, F: Future<Output = Result<T>>>(
|
pub(crate) async fn resolve_from_future<T: Send, F: Future<Output = Result<T>>>(
|
||||||
tsfn_value: TSFNValue,
|
tsfn_value: TSFNValue,
|
||||||
|
|
|
@ -1,13 +1,19 @@
|
||||||
use std::convert::Into;
|
use std::convert::Into;
|
||||||
|
use std::marker::PhantomData;
|
||||||
use std::os::raw::{c_char, c_void};
|
use std::os::raw::{c_char, c_void};
|
||||||
use std::ptr;
|
use std::ptr;
|
||||||
|
|
||||||
use crate::error::check_status;
|
use crate::error::check_status;
|
||||||
use crate::{sys, Env, JsFunction, JsUnknown, Result};
|
use crate::{sys, Env, JsFunction, NapiValue, Result};
|
||||||
|
|
||||||
use sys::napi_threadsafe_function_call_mode;
|
use sys::napi_threadsafe_function_call_mode;
|
||||||
use sys::napi_threadsafe_function_release_mode;
|
use sys::napi_threadsafe_function_release_mode;
|
||||||
|
|
||||||
|
pub struct ThreadSafeCallContext<T: 'static> {
|
||||||
|
pub env: Env,
|
||||||
|
pub value: T,
|
||||||
|
}
|
||||||
|
|
||||||
#[repr(u8)]
|
#[repr(u8)]
|
||||||
pub enum ThreadsafeFunctionCallMode {
|
pub enum ThreadsafeFunctionCallMode {
|
||||||
NonBlocking,
|
NonBlocking,
|
||||||
|
@ -46,12 +52,6 @@ impl Into<napi_threadsafe_function_release_mode> for ThreadsafeFunctionReleaseMo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait ToJs: Copy + Clone {
|
|
||||||
type Output;
|
|
||||||
|
|
||||||
fn resolve(&self, env: &mut Env, output: Self::Output) -> Result<Vec<JsUnknown>>;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Communicate with the addon's main thread by invoking a JavaScript function from other threads.
|
/// Communicate with the addon's main thread by invoking a JavaScript function from other threads.
|
||||||
///
|
///
|
||||||
/// ## Example
|
/// ## Example
|
||||||
|
@ -62,71 +62,71 @@ pub trait ToJs: Copy + Clone {
|
||||||
/// extern crate napi_derive;
|
/// extern crate napi_derive;
|
||||||
///
|
///
|
||||||
/// use std::thread;
|
/// use std::thread;
|
||||||
|
///
|
||||||
/// use napi::{
|
/// use napi::{
|
||||||
/// Number, Result, Env, CallContext, JsUndefined, JsFunction,
|
/// threadsafe_function::{
|
||||||
|
/// ThreadSafeCallContext, ThreadsafeFunctionCallMode, ThreadsafeFunctionReleaseMode,
|
||||||
|
/// },
|
||||||
|
/// CallContext, Error, JsFunction, JsNumber, JsUndefined, Result, Status,
|
||||||
/// };
|
/// };
|
||||||
/// use napi::threadsafe_function::{
|
|
||||||
/// ToJs, ThreadsafeFunction, ThreadsafeFunctionCallMode, ThreadsafeFunctionReleaseMode,
|
|
||||||
/// };
|
|
||||||
///
|
|
||||||
/// // Define a struct for handling the data passed from `ThreadsafeFunction::call`
|
|
||||||
/// // and return the data to be used for the js callback.
|
|
||||||
/// #[derive(Clone, Copy)]
|
|
||||||
/// struct HandleNumber;
|
|
||||||
///
|
|
||||||
/// impl ToJs for HandleNumber {
|
|
||||||
/// type Output = u8;
|
|
||||||
///
|
|
||||||
/// fn resolve(&self, env: &mut Env, output: Self::Output) -> Result<Vec<JsUnknown>> {
|
|
||||||
/// let value = env.create_uint32(output as u32)?.into_unknown()?;
|
|
||||||
/// // The first argument in the NodeJS callback will be either a null or an error
|
|
||||||
/// // depending on the result returned by this function.
|
|
||||||
/// // If this Result is Ok, the first argument will be null.
|
|
||||||
/// // If this Result is Err, the first argument will be the error.
|
|
||||||
/// Ok(vec![value])
|
|
||||||
/// }
|
|
||||||
/// }
|
|
||||||
///
|
|
||||||
/// #[js_function(1)]
|
/// #[js_function(1)]
|
||||||
/// fn test_threadsafe_function(ctx: CallContext) -> Result<JsUndefined> {
|
/// pub fn test_threadsafe_function(ctx: CallContext) -> Result<JsUndefined> {
|
||||||
/// // The callback function from js which will be called in `ThreadsafeFunction::call`.
|
|
||||||
/// let func = ctx.get::<JsFunction>(0)?;
|
/// let func = ctx.get::<JsFunction>(0)?;
|
||||||
///
|
|
||||||
/// let to_js = HandleNumber;
|
/// let tsfn =
|
||||||
/// let tsfn = ThreadsafeFunction::create(ctx.env, func, to_js, 0)?;
|
/// ctx
|
||||||
///
|
/// .env
|
||||||
|
/// .create_threadsafe_function(func, 0, |ctx: ThreadSafeCallContext<Vec<u32>>| {
|
||||||
|
/// ctx
|
||||||
|
/// .value
|
||||||
|
/// .iter()
|
||||||
|
/// .map(|v| ctx.env.create_uint32(*v))
|
||||||
|
/// .collect::<Result<Vec<JsNumber>>>()
|
||||||
|
/// })?;
|
||||||
|
|
||||||
/// thread::spawn(move || {
|
/// thread::spawn(move || {
|
||||||
/// let output: u8 = 42;
|
/// let output: Vec<u32> = vec![42, 1, 2, 3];
|
||||||
/// // It's okay to call a threadsafe function multiple times.
|
/// /// It's okay to call a threadsafe function multiple times.
|
||||||
/// tsfn.call(Ok(output), ThreadsafeFunctionCallMode::Blocking).unwrap();
|
/// tsfn.call(Ok(output.clone()), ThreadsafeFunctionCallMode::Blocking);
|
||||||
/// tsfn.call(Ok(output), ThreadsafeFunctionCallMode::Blocking).unwrap();
|
/// tsfn.call(Ok(output.clone()), ThreadsafeFunctionCallMode::NonBlocking);
|
||||||
/// // We should call `ThreadsafeFunction::release` manually when we don't
|
/// tsfn.release(ThreadsafeFunctionReleaseMode::Release);
|
||||||
/// // need the instance anymore, or it will prevent Node.js from exiting
|
|
||||||
/// // automatically and possibly cause memory leaks.
|
|
||||||
/// tsfn.release(ThreadsafeFunctionReleaseMode::Release).unwrap();
|
|
||||||
/// });
|
/// });
|
||||||
///
|
|
||||||
/// ctx.env.get_undefined()
|
/// ctx.env.get_undefined()
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
#[derive(Debug, Clone, Copy)]
|
pub struct ThreadsafeFunction<T: 'static> {
|
||||||
pub struct ThreadsafeFunction<T: ToJs> {
|
raw_tsfn: sys::napi_threadsafe_function,
|
||||||
raw_value: sys::napi_threadsafe_function,
|
_phantom: PhantomData<T>,
|
||||||
to_js: T,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe impl<T: ToJs> Send for ThreadsafeFunction<T> {}
|
unsafe impl<T> Send for ThreadsafeFunction<T> {}
|
||||||
unsafe impl<T: ToJs> Sync for ThreadsafeFunction<T> {}
|
unsafe impl<T> Sync for ThreadsafeFunction<T> {}
|
||||||
|
|
||||||
impl<T: ToJs> ThreadsafeFunction<T> {
|
#[repr(transparent)]
|
||||||
|
struct ThreadSafeContext<T: 'static, V: NapiValue>(
|
||||||
|
Box<dyn FnMut(ThreadSafeCallContext<T>) -> Result<Vec<V>>>,
|
||||||
|
);
|
||||||
|
|
||||||
|
impl<T: 'static> ThreadsafeFunction<T> {
|
||||||
/// See [napi_create_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_create_threadsafe_function)
|
/// See [napi_create_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_create_threadsafe_function)
|
||||||
/// for more information.
|
/// for more information.
|
||||||
pub fn create(env: &Env, func: JsFunction, to_js: T, max_queue_size: u64) -> Result<Self> {
|
#[inline(always)]
|
||||||
|
pub fn create<
|
||||||
|
V: NapiValue,
|
||||||
|
R: 'static + Send + FnMut(ThreadSafeCallContext<T>) -> Result<Vec<V>>,
|
||||||
|
>(
|
||||||
|
env: sys::napi_env,
|
||||||
|
func: JsFunction,
|
||||||
|
max_queue_size: u64,
|
||||||
|
callback: R,
|
||||||
|
) -> Result<Self> {
|
||||||
let mut async_resource_name = ptr::null_mut();
|
let mut async_resource_name = ptr::null_mut();
|
||||||
let s = "napi_rs_threadsafe_function";
|
let s = "napi_rs_threadsafe_function";
|
||||||
check_status(unsafe {
|
check_status(unsafe {
|
||||||
sys::napi_create_string_utf8(
|
sys::napi_create_string_utf8(
|
||||||
env.0,
|
env,
|
||||||
s.as_ptr() as *const c_char,
|
s.as_ptr() as *const c_char,
|
||||||
s.len() as u64,
|
s.len() as u64,
|
||||||
&mut async_resource_name,
|
&mut async_resource_name,
|
||||||
|
@ -134,59 +134,65 @@ impl<T: ToJs> ThreadsafeFunction<T> {
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
let initial_thread_count: u64 = 1;
|
let initial_thread_count: u64 = 1;
|
||||||
let mut result = ptr::null_mut();
|
let mut raw_tsfn = ptr::null_mut();
|
||||||
let tsfn = ThreadsafeFunction {
|
let context = ThreadSafeContext(Box::from(callback));
|
||||||
to_js,
|
let ptr = Box::into_raw(Box::new(context)) as *mut _;
|
||||||
raw_value: result,
|
check_status(unsafe {
|
||||||
};
|
|
||||||
|
|
||||||
let ptr = Box::into_raw(Box::from(tsfn)) as *mut _ as *mut c_void;
|
|
||||||
|
|
||||||
let status = unsafe {
|
|
||||||
sys::napi_create_threadsafe_function(
|
sys::napi_create_threadsafe_function(
|
||||||
env.0,
|
env,
|
||||||
func.0.value,
|
func.0.value,
|
||||||
ptr::null_mut(),
|
ptr::null_mut(),
|
||||||
async_resource_name,
|
async_resource_name,
|
||||||
max_queue_size,
|
max_queue_size,
|
||||||
initial_thread_count,
|
initial_thread_count,
|
||||||
ptr,
|
ptr,
|
||||||
Some(thread_finalize_cb::<T>),
|
Some(thread_finalize_cb::<T, V>),
|
||||||
ptr,
|
ptr,
|
||||||
Some(call_js_cb::<T>),
|
Some(call_js_cb::<T, V>),
|
||||||
&mut result,
|
&mut raw_tsfn,
|
||||||
)
|
)
|
||||||
};
|
})?;
|
||||||
check_status(status)?;
|
|
||||||
|
|
||||||
Ok(ThreadsafeFunction {
|
Ok(ThreadsafeFunction {
|
||||||
to_js,
|
raw_tsfn,
|
||||||
raw_value: result,
|
_phantom: PhantomData,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [napi_call_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_call_threadsafe_function)
|
/// See [napi_call_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_call_threadsafe_function)
|
||||||
/// for more information.
|
/// for more information.
|
||||||
pub fn call(&self, value: Result<T::Output>, mode: ThreadsafeFunctionCallMode) -> Result<()> {
|
pub fn call(&self, value: Result<T>, mode: ThreadsafeFunctionCallMode) {
|
||||||
check_status(unsafe {
|
let status = unsafe {
|
||||||
sys::napi_call_threadsafe_function(
|
sys::napi_call_threadsafe_function(
|
||||||
self.raw_value,
|
self.raw_tsfn,
|
||||||
Box::into_raw(Box::from(value)) as *mut _ as *mut c_void,
|
Box::into_raw(Box::new(value)) as *mut _,
|
||||||
mode.into(),
|
mode.into(),
|
||||||
)
|
)
|
||||||
})
|
};
|
||||||
|
debug_assert!(
|
||||||
|
status == sys::napi_status::napi_ok,
|
||||||
|
"Threadsafe Function call failed"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [napi_acquire_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_acquire_threadsafe_function)
|
/// See [napi_acquire_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_acquire_threadsafe_function)
|
||||||
/// for more information.
|
/// for more information.
|
||||||
pub fn acquire(&self) -> Result<()> {
|
pub fn acquire(&self) {
|
||||||
check_status(unsafe { sys::napi_acquire_threadsafe_function(self.raw_value) })
|
let status = unsafe { sys::napi_acquire_threadsafe_function(self.raw_tsfn) };
|
||||||
|
debug_assert!(
|
||||||
|
status == sys::napi_status::napi_ok,
|
||||||
|
"Threadsafe Function acquire failed"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [napi_release_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_release_threadsafe_function)
|
/// See [napi_release_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_release_threadsafe_function)
|
||||||
/// for more information.
|
/// for more information.
|
||||||
pub fn release(&self, mode: ThreadsafeFunctionReleaseMode) -> Result<()> {
|
pub fn release(self, mode: ThreadsafeFunctionReleaseMode) {
|
||||||
check_status(unsafe { sys::napi_release_threadsafe_function(self.raw_value, mode.into()) })
|
let status = unsafe { sys::napi_release_threadsafe_function(self.raw_tsfn, mode.into()) };
|
||||||
|
debug_assert!(
|
||||||
|
status == sys::napi_status::napi_ok,
|
||||||
|
"Threadsafe Function call failed"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [napi_ref_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_ref_threadsafe_function)
|
/// See [napi_ref_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_ref_threadsafe_function)
|
||||||
|
@ -194,73 +200,76 @@ impl<T: ToJs> ThreadsafeFunction<T> {
|
||||||
///
|
///
|
||||||
/// "ref" is a keyword so that we use "refer" here.
|
/// "ref" is a keyword so that we use "refer" here.
|
||||||
pub fn refer(&self, env: &Env) -> Result<()> {
|
pub fn refer(&self, env: &Env) -> Result<()> {
|
||||||
check_status(unsafe { sys::napi_ref_threadsafe_function(env.0, self.raw_value) })
|
check_status(unsafe { sys::napi_ref_threadsafe_function(env.0, self.raw_tsfn) })
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [napi_unref_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_unref_threadsafe_function)
|
/// See [napi_unref_threadsafe_function](https://nodejs.org/api/n-api.html#n_api_napi_unref_threadsafe_function)
|
||||||
/// for more information.
|
/// for more information.
|
||||||
pub fn unref(&self, env: &Env) -> Result<()> {
|
pub fn unref(&self, env: &Env) -> Result<()> {
|
||||||
check_status(unsafe { sys::napi_unref_threadsafe_function(env.0, self.raw_value) })
|
check_status(unsafe { sys::napi_unref_threadsafe_function(env.0, self.raw_tsfn) })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe extern "C" fn thread_finalize_cb<T: ToJs>(
|
unsafe extern "C" fn thread_finalize_cb<T: 'static, V: NapiValue>(
|
||||||
_raw_env: sys::napi_env,
|
_raw_env: sys::napi_env,
|
||||||
finalize_data: *mut c_void,
|
finalize_data: *mut c_void,
|
||||||
_finalize_hint: *mut c_void,
|
_finalize_hint: *mut c_void,
|
||||||
) {
|
) {
|
||||||
// cleanup
|
// cleanup
|
||||||
Box::from_raw(finalize_data as *mut ThreadsafeFunction<T>);
|
Box::from_raw(finalize_data as *mut ThreadSafeContext<T, V>);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe extern "C" fn call_js_cb<T: ToJs>(
|
unsafe extern "C" fn call_js_cb<T: 'static, V: NapiValue>(
|
||||||
raw_env: sys::napi_env,
|
raw_env: sys::napi_env,
|
||||||
js_callback: sys::napi_value,
|
js_callback: sys::napi_value,
|
||||||
context: *mut c_void,
|
context: *mut c_void,
|
||||||
data: *mut c_void,
|
data: *mut c_void,
|
||||||
) {
|
) {
|
||||||
let mut env = Env::from_raw(raw_env);
|
|
||||||
let mut recv = ptr::null_mut();
|
let mut recv = ptr::null_mut();
|
||||||
sys::napi_get_undefined(raw_env, &mut recv);
|
sys::napi_get_undefined(raw_env, &mut recv);
|
||||||
|
|
||||||
let tsfn = Box::leak(Box::from_raw(context as *mut ThreadsafeFunction<T>));
|
let ctx = Box::leak(Box::from_raw(context as *mut ThreadSafeContext<T, V>));
|
||||||
let val = Box::from_raw(data as *mut Result<T::Output>);
|
let val = Box::from_raw(data as *mut Result<T>);
|
||||||
|
|
||||||
let ret = val.and_then(|v| tsfn.to_js.resolve(&mut env, v));
|
let ret = val.and_then(|v| {
|
||||||
|
(ctx.0)(ThreadSafeCallContext {
|
||||||
|
env: Env::from_raw(raw_env),
|
||||||
|
value: v,
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
let status;
|
let status;
|
||||||
|
|
||||||
// Follow async callback conventions: https://nodejs.org/en/knowledge/errors/what-are-the-error-conventions/
|
// Follow async callback conventions: https://nodejs.org/en/knowledge/errors/what-are-the-error-conventions/
|
||||||
// Check if the Result is okay, if so, pass a null as the first (error) argument automatically.
|
// Check if the Result is okay, if so, pass a null as the first (error) argument automatically.
|
||||||
// If the Result is an error, pass that as the first argument.
|
// If the Result is an error, pass that as the first argument.
|
||||||
if ret.is_ok() {
|
match ret {
|
||||||
let values = ret.unwrap();
|
Ok(values) => {
|
||||||
let js_null = env.get_null().unwrap();
|
let mut js_null = ptr::null_mut();
|
||||||
let mut raw_values: Vec<sys::napi_value> = vec![];
|
sys::napi_get_null(raw_env, &mut js_null);
|
||||||
raw_values.push(js_null.0.value);
|
let args_length = values.len() + 1;
|
||||||
for item in values.iter() {
|
let mut args: Vec<sys::napi_value> = Vec::with_capacity(args_length);
|
||||||
raw_values.push(item.0.value)
|
args.push(js_null);
|
||||||
}
|
args.extend(values.iter().map(|v| v.raw()));
|
||||||
|
|
||||||
status = sys::napi_call_function(
|
status = sys::napi_call_function(
|
||||||
raw_env,
|
raw_env,
|
||||||
recv,
|
recv,
|
||||||
js_callback,
|
js_callback,
|
||||||
(values.len() + 1) as u64,
|
args_length as _,
|
||||||
raw_values.as_ptr(),
|
args.as_ptr(),
|
||||||
ptr::null_mut(),
|
ptr::null_mut(),
|
||||||
);
|
);
|
||||||
} else {
|
}
|
||||||
let mut err = env.create_error(ret.err().unwrap()).unwrap();
|
Err(e) => {
|
||||||
status = sys::napi_call_function(
|
status = sys::napi_call_function(
|
||||||
raw_env,
|
raw_env,
|
||||||
recv,
|
recv,
|
||||||
js_callback,
|
js_callback,
|
||||||
1,
|
1,
|
||||||
&mut err.0.value,
|
[e.into_raw(raw_env)].as_mut_ptr(),
|
||||||
ptr::null_mut(),
|
ptr::null_mut(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
debug_assert!(status == sys::napi_status::napi_ok, "CallJsCB failed");
|
debug_assert!(status == sys::napi_status::napi_ok, "CallJsCB failed");
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,8 +5,8 @@ mod tsfn;
|
||||||
use tsfn::*;
|
use tsfn::*;
|
||||||
|
|
||||||
pub fn register_js(module: &mut Module) -> Result<()> {
|
pub fn register_js(module: &mut Module) -> Result<()> {
|
||||||
module.create_named_method("testTsfnError", test_tsfn_error)?;
|
|
||||||
module.create_named_method("testThreadsafeFunction", test_threadsafe_function)?;
|
module.create_named_method("testThreadsafeFunction", test_threadsafe_function)?;
|
||||||
|
module.create_named_method("testTsfnError", test_tsfn_error)?;
|
||||||
module.create_named_method("testTokioReadfile", test_tokio_readfile)?;
|
module.create_named_method("testTokioReadfile", test_tokio_readfile)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,47 +1,35 @@
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
|
||||||
use napi::threadsafe_function::{
|
use napi::{
|
||||||
ThreadsafeFunction, ThreadsafeFunctionCallMode, ThreadsafeFunctionReleaseMode, ToJs,
|
threadsafe_function::{
|
||||||
|
ThreadSafeCallContext, ThreadsafeFunctionCallMode, ThreadsafeFunctionReleaseMode,
|
||||||
|
},
|
||||||
|
CallContext, Error, JsFunction, JsNumber, JsString, JsUndefined, Result, Status,
|
||||||
};
|
};
|
||||||
use napi::{CallContext, Env, Error, JsFunction, JsString, JsUndefined, JsUnknown, Result, Status};
|
|
||||||
use tokio;
|
use tokio;
|
||||||
|
|
||||||
#[derive(Clone, Copy)]
|
|
||||||
struct HandleNumber;
|
|
||||||
|
|
||||||
impl ToJs for HandleNumber {
|
|
||||||
type Output = Vec<u8>;
|
|
||||||
|
|
||||||
fn resolve(&self, env: &mut Env, output: Self::Output) -> Result<Vec<JsUnknown>> {
|
|
||||||
let mut items: Vec<JsUnknown> = vec![];
|
|
||||||
for item in output.iter() {
|
|
||||||
let value = env.create_uint32((*item) as u32)?.into_unknown();
|
|
||||||
items.push(value);
|
|
||||||
}
|
|
||||||
Ok(items)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[js_function(1)]
|
#[js_function(1)]
|
||||||
pub fn test_threadsafe_function(ctx: CallContext) -> Result<JsUndefined> {
|
pub fn test_threadsafe_function(ctx: CallContext) -> Result<JsUndefined> {
|
||||||
let func = ctx.get::<JsFunction>(0)?;
|
let func = ctx.get::<JsFunction>(0)?;
|
||||||
|
|
||||||
let to_js = HandleNumber;
|
let tsfn =
|
||||||
let tsfn = ThreadsafeFunction::create(ctx.env, func, to_js, 0)?;
|
ctx
|
||||||
|
.env
|
||||||
|
.create_threadsafe_function(func, 0, |ctx: ThreadSafeCallContext<Vec<u32>>| {
|
||||||
|
ctx
|
||||||
|
.value
|
||||||
|
.iter()
|
||||||
|
.map(|v| ctx.env.create_uint32(*v))
|
||||||
|
.collect::<Result<Vec<JsNumber>>>()
|
||||||
|
})?;
|
||||||
|
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
let output: Vec<u8> = vec![42, 1, 2, 3];
|
let output: Vec<u32> = vec![42, 1, 2, 3];
|
||||||
// It's okay to call a threadsafe function multiple times.
|
// It's okay to call a threadsafe function multiple times.
|
||||||
tsfn
|
tsfn.call(Ok(output.clone()), ThreadsafeFunctionCallMode::Blocking);
|
||||||
.call(Ok(output.clone()), ThreadsafeFunctionCallMode::Blocking)
|
tsfn.call(Ok(output.clone()), ThreadsafeFunctionCallMode::NonBlocking);
|
||||||
.unwrap();
|
tsfn.release(ThreadsafeFunctionReleaseMode::Release);
|
||||||
tsfn
|
|
||||||
.call(Ok(output.clone()), ThreadsafeFunctionCallMode::Blocking)
|
|
||||||
.unwrap();
|
|
||||||
tsfn
|
|
||||||
.release(ThreadsafeFunctionReleaseMode::Release)
|
|
||||||
.unwrap();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
ctx.env.get_undefined()
|
ctx.env.get_undefined()
|
||||||
|
@ -50,36 +38,22 @@ pub fn test_threadsafe_function(ctx: CallContext) -> Result<JsUndefined> {
|
||||||
#[js_function(1)]
|
#[js_function(1)]
|
||||||
pub fn test_tsfn_error(ctx: CallContext) -> Result<JsUndefined> {
|
pub fn test_tsfn_error(ctx: CallContext) -> Result<JsUndefined> {
|
||||||
let func = ctx.get::<JsFunction>(0)?;
|
let func = ctx.get::<JsFunction>(0)?;
|
||||||
let to_js = HandleNumber;
|
let tsfn = ctx
|
||||||
let tsfn = ThreadsafeFunction::create(ctx.env, func, to_js, 0)?;
|
.env
|
||||||
|
.create_threadsafe_function(func, 0, |ctx: ThreadSafeCallContext<()>| {
|
||||||
|
ctx.env.get_undefined().map(|v| vec![v])
|
||||||
|
})?;
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
tsfn
|
tsfn.call(
|
||||||
.call(
|
|
||||||
Err(Error::new(Status::Unknown, "invalid".to_owned())),
|
Err(Error::new(Status::Unknown, "invalid".to_owned())),
|
||||||
ThreadsafeFunctionCallMode::Blocking,
|
ThreadsafeFunctionCallMode::Blocking,
|
||||||
)
|
);
|
||||||
.unwrap();
|
tsfn.release(ThreadsafeFunctionReleaseMode::Release);
|
||||||
tsfn
|
|
||||||
.release(ThreadsafeFunctionReleaseMode::Release)
|
|
||||||
.unwrap();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
ctx.env.get_undefined()
|
ctx.env.get_undefined()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
|
||||||
struct HandleBuffer;
|
|
||||||
|
|
||||||
impl ToJs for HandleBuffer {
|
|
||||||
type Output = Vec<u8>;
|
|
||||||
|
|
||||||
fn resolve(&self, env: &mut Env, output: Self::Output) -> Result<Vec<JsUnknown>> {
|
|
||||||
let value = env.create_buffer_with_data(output.to_vec())?.into_unknown();
|
|
||||||
Ok(vec![value])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn read_file_content(filepath: &Path) -> Result<Vec<u8>> {
|
async fn read_file_content(filepath: &Path) -> Result<Vec<u8>> {
|
||||||
tokio::fs::read(filepath)
|
tokio::fs::read(filepath)
|
||||||
.await
|
.await
|
||||||
|
@ -92,16 +66,22 @@ pub fn test_tokio_readfile(ctx: CallContext) -> Result<JsUndefined> {
|
||||||
let js_func = ctx.get::<JsFunction>(1)?;
|
let js_func = ctx.get::<JsFunction>(1)?;
|
||||||
let path_str = js_filepath.into_utf8()?.to_owned()?;
|
let path_str = js_filepath.into_utf8()?.to_owned()?;
|
||||||
|
|
||||||
let to_js = HandleBuffer;
|
let tsfn =
|
||||||
let tsfn = ThreadsafeFunction::create(ctx.env, js_func, to_js, 0)?;
|
ctx
|
||||||
let mut rt = tokio::runtime::Runtime::new().unwrap();
|
.env
|
||||||
|
.create_threadsafe_function(js_func, 0, |ctx: ThreadSafeCallContext<Vec<u8>>| {
|
||||||
|
ctx
|
||||||
|
.env
|
||||||
|
.create_buffer_with_data(ctx.value)
|
||||||
|
.map(|v| vec![v.into_raw()])
|
||||||
|
})?;
|
||||||
|
let mut rt = tokio::runtime::Runtime::new()
|
||||||
|
.map_err(|e| Error::from_reason(format!("Create tokio runtime failed {}", e)))?;
|
||||||
|
|
||||||
rt.block_on(async move {
|
rt.block_on(async move {
|
||||||
let ret = read_file_content(&Path::new(&path_str)).await;
|
let ret = read_file_content(&Path::new(&path_str)).await;
|
||||||
let _ = tsfn.call(ret, ThreadsafeFunctionCallMode::Blocking);
|
tsfn.call(ret, ThreadsafeFunctionCallMode::Blocking);
|
||||||
tsfn
|
tsfn.release(ThreadsafeFunctionReleaseMode::Release);
|
||||||
.release(ThreadsafeFunctionReleaseMode::Release)
|
|
||||||
.unwrap();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
ctx.env.get_undefined()
|
ctx.env.get_undefined()
|
||||||
|
|
Loading…
Reference in a new issue