use std::os::raw::c_void;
use std::ptr;
use std::{
marker::PhantomData,
sync::{Arc, RwLock},
};
#[cfg(feature = "deferred_trace")]
use crate::{bindgen_runtime::JsObjectValue, JsValue};
use crate::{
bindgen_runtime::{Object, ToNapiValue},
check_status, sys, Env, Error, Result,
};
#[cfg(feature = "deferred_trace")]
#[repr(transparent)]
#[derive(Clone)]
struct DeferredTrace(sys::napi_ref);
#[cfg(feature = "deferred_trace")]
impl DeferredTrace {
fn new(raw_env: sys::napi_env) -> Result<Self> {
let env = Env::from_raw(raw_env);
let reason = env.create_string("none")?;
let mut js_error = ptr::null_mut();
check_status!(
unsafe { sys::napi_create_error(raw_env, ptr::null_mut(), reason.raw(), &mut js_error) },
"Create error in DeferredTrace failed"
)?;
let mut result = ptr::null_mut();
check_status!(
unsafe { sys::napi_create_reference(raw_env, js_error, 1, &mut result) },
"Create reference in DeferredTrace failed"
)?;
Ok(Self(result))
}
fn into_rejected(self, raw_env: sys::napi_env, mut err: Error) -> Result<sys::napi_value> {
let env = Env::from_raw(raw_env);
let mut raw = ptr::null_mut();
check_status!(
unsafe { sys::napi_get_reference_value(raw_env, self.0, &mut raw) },
"Failed to get referenced value in DeferredTrace"
)?;
let mut obj = Object::from_raw(raw_env, raw);
let err_value = if !err.maybe_raw.is_null() {
let mut err_raw_value = std::ptr::null_mut();
check_status!(
unsafe { sys::napi_get_reference_value(raw_env, err.maybe_raw, &mut err_raw_value) },
"Get error reference in `to_napi_value` failed"
)?;
let err_obj = Object::from_raw(raw_env, err_raw_value);
let err_value = if err_obj.has_named_property("message")? {
Ok(err_obj.raw())
} else {
obj.set_named_property("message", "")?;
obj.set_named_property("code", "")?;
Ok(raw)
};
let mut ref_count = 0;
check_status!(
unsafe { sys::napi_reference_unref(raw_env, err.maybe_raw, &mut ref_count) },
"Unref error reference in `to_napi_value` failed"
)?;
if ref_count == 0 {
check_status!(
unsafe { sys::napi_delete_reference(raw_env, err.maybe_raw) },
"Delete error reference in `to_napi_value` failed"
)?;
}
err.maybe_env = ptr::null_mut();
err.maybe_raw = ptr::null_mut();
err_value
} else {
obj.set_named_property("message", &err.reason)?;
obj.set_named_property(
"code",
env.create_string_from_std(format!("{}", err.status))?,
)?;
Ok(raw)
};
check_status!(
unsafe { sys::napi_delete_reference(raw_env, self.0) },
"Failed to get referenced value in DeferredTrace"
)?;
err_value
}
}
type FinalizeCallback = Arc<RwLock<Option<Box<dyn FnOnce(sys::napi_env)>>>>;
struct DeferredData<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>> {
resolver: Result<Resolver>,
#[cfg(feature = "deferred_trace")]
trace: DeferredTrace,
tsfn: sys::napi_threadsafe_function,
finalize_callback: FinalizeCallback,
}
pub struct JsDeferred<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>> {
pub(crate) tsfn: sys::napi_threadsafe_function,
#[cfg(feature = "deferred_trace")]
trace: DeferredTrace,
finalize_callback: FinalizeCallback,
_data: PhantomData<Data>,
_resolver: PhantomData<Resolver>,
}
impl<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>> Clone
for JsDeferred<Data, Resolver>
{
fn clone(&self) -> Self {
Self {
tsfn: self.tsfn,
#[cfg(feature = "deferred_trace")]
trace: self.trace.clone(),
finalize_callback: self.finalize_callback.clone(),
_data: PhantomData,
_resolver: PhantomData,
}
}
}
unsafe impl<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>> Send
for JsDeferred<Data, Resolver>
{
}
impl<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>> JsDeferred<Data, Resolver> {
pub(crate) fn new(env: &Env) -> Result<(Self, Object<'_>)> {
let (tsfn, promise) = js_deferred_new_raw(env, Some(napi_resolve_deferred::<Data, Resolver>))?;
let deferred = Self {
tsfn,
#[cfg(feature = "deferred_trace")]
trace: DeferredTrace::new(env.0)?,
finalize_callback: Default::default(),
_data: PhantomData,
_resolver: PhantomData,
};
Ok((deferred, promise))
}
pub fn resolve(self, resolver: Resolver) {
self.call_tsfn(Ok(resolver))
}
pub fn reject(self, error: Error) {
self.call_tsfn(Err(error))
}
#[allow(clippy::arc_with_non_send_sync)]
pub fn set_finalize_callback(
&mut self,
finalize_callback: Option<Box<dyn FnOnce(sys::napi_env)>>,
) {
self.finalize_callback = Arc::new(RwLock::new(finalize_callback));
}
fn call_tsfn(self, result: Result<Resolver>) {
let data = DeferredData {
resolver: result,
#[cfg(feature = "deferred_trace")]
trace: self.trace,
tsfn: self.tsfn,
finalize_callback: self.finalize_callback.clone(),
};
let status = unsafe {
sys::napi_call_threadsafe_function(
self.tsfn,
Box::into_raw(Box::from(data)).cast(),
sys::ThreadsafeFunctionCallMode::blocking,
)
};
debug_assert!(
status == sys::Status::napi_ok,
"Call threadsafe function in JsDeferred failed"
);
}
}
fn js_deferred_new_raw(
env: &Env,
resolve_deferred: sys::napi_threadsafe_function_call_js,
) -> Result<(sys::napi_threadsafe_function, Object<'_>)> {
let mut raw_promise = ptr::null_mut();
let mut raw_deferred = ptr::null_mut();
check_status!(
unsafe { sys::napi_create_promise(env.0, &mut raw_deferred, &mut raw_promise) },
"Create promise in JsDeferred failed"
)?;
let mut async_resource_name = ptr::null_mut();
check_status!(
unsafe {
sys::napi_create_string_utf8(
env.0,
c"napi_resolve_deferred".as_ptr().cast(),
22,
&mut async_resource_name,
)
},
"Create async resource name in JsDeferred failed"
)?;
let mut tsfn = ptr::null_mut();
check_status!(
unsafe {
sys::napi_create_threadsafe_function(
env.0,
ptr::null_mut(),
ptr::null_mut(),
async_resource_name,
0,
1,
ptr::null_mut(),
None,
raw_deferred.cast(),
resolve_deferred,
&mut tsfn,
)
},
"Create threadsafe function in JsDeferred failed"
)?;
let promise = Object::from_raw(env.0, raw_promise);
Ok((tsfn, promise))
}
extern "C" fn napi_resolve_deferred<Data: ToNapiValue, Resolver: FnOnce(Env) -> Result<Data>>(
env: sys::napi_env,
_js_callback: sys::napi_value,
context: *mut c_void,
data: *mut c_void,
) {
let deferred = context.cast();
let deferred_data: Box<DeferredData<Data, Resolver>> = unsafe { Box::from_raw(data.cast()) };
let tsfn: *mut napi_sys::napi_threadsafe_function__ = deferred_data.tsfn;
let finalize_callback = RwLock::write(&deferred_data.finalize_callback)
.expect("RwLock Poison")
.take();
let result = deferred_data
.resolver
.and_then(|resolver| resolver(Env::from_raw(env)))
.and_then(|res| unsafe { ToNapiValue::to_napi_value(env, res) });
let release_tsfn_result = check_status!(
unsafe {
sys::napi_release_threadsafe_function(tsfn, sys::ThreadsafeFunctionReleaseMode::release)
},
"Release threadsafe function in JsDeferred failed"
);
if let Err(e) = release_tsfn_result.and(result).and_then(|res| {
check_status!(
unsafe { sys::napi_resolve_deferred(env, deferred, res) },
"Resolve deferred value failed"
)
.map(|_| {
#[cfg(feature = "deferred_trace")]
{
let _status = unsafe { sys::napi_delete_reference(env, deferred_data.trace.0) };
if _status != sys::Status::napi_ok && cfg!(debug_assertions) {
eprintln!(
"Failed to delete reference in deferred {}",
crate::Status::from(_status)
);
}
}
})
}) {
#[cfg(feature = "deferred_trace")]
let error = deferred_data.trace.into_rejected(env, e);
#[cfg(not(feature = "deferred_trace"))]
let error = Ok::<sys::napi_value, Error>(unsafe { crate::JsError::from(e).into_value(env) });
match error {
Ok(error) => {
unsafe { sys::napi_reject_deferred(env, deferred, error) };
if let Some(finalize_callback) = finalize_callback {
finalize_callback(env);
}
}
Err(err) => {
if let Some(finalize_callback) = finalize_callback {
finalize_callback(env);
}
if cfg!(debug_assertions) {
eprintln!("Failed to reject deferred: {err:?}");
let mut err = ptr::null_mut();
let mut err_msg = ptr::null_mut();
unsafe {
sys::napi_create_string_utf8(env, c"Rejection failed".as_ptr().cast(), 0, &mut err_msg);
sys::napi_create_error(env, ptr::null_mut(), err_msg, &mut err);
sys::napi_reject_deferred(env, deferred, err);
}
}
}
}
} else if let Some(finalize_callback) = finalize_callback {
finalize_callback(env);
}
}