2018-04-28 17:26:38 +09:00
|
|
|
pub extern crate futures;
|
|
|
|
use std::any::TypeId;
|
|
|
|
use std::ffi::CString;
|
|
|
|
use std::marker::PhantomData;
|
|
|
|
use std::mem;
|
2019-05-16 01:41:22 +09:00
|
|
|
use std::ops::{Deref, DerefMut};
|
2018-04-28 17:26:38 +09:00
|
|
|
use std::os::raw::{c_char, c_void};
|
|
|
|
use std::ptr;
|
2019-05-16 01:41:22 +09:00
|
|
|
use std::slice;
|
2018-04-28 17:26:38 +09:00
|
|
|
use std::string::String as RustString;
|
|
|
|
|
|
|
|
mod executor;
|
|
|
|
pub mod sys;
|
|
|
|
|
2020-01-16 23:10:06 +09:00
|
|
|
pub use sys::{napi_valuetype, Status};
|
2018-04-28 17:26:38 +09:00
|
|
|
|
|
|
|
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)]
|
|
|
|
pub struct Error {
|
|
|
|
status: Status,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Env(sys::napi_env);
|
|
|
|
|
|
|
|
// Value types
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Any;
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Undefined;
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
2019-05-23 20:28:51 +09:00
|
|
|
pub struct Boolean {
|
|
|
|
value: bool,
|
|
|
|
}
|
2018-04-28 17:26:38 +09:00
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
2019-05-23 20:28:51 +09:00
|
|
|
pub struct Number {
|
|
|
|
int: i64,
|
|
|
|
}
|
2018-04-28 17:26:38 +09:00
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct String;
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Object;
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Function;
|
|
|
|
|
2019-05-16 01:41:22 +09:00
|
|
|
#[derive(Clone, Copy, Debug)]
|
2019-05-23 20:28:51 +09:00
|
|
|
pub struct Buffer {
|
|
|
|
data: *const u8,
|
2020-02-18 22:09:17 +09:00
|
|
|
size: u64,
|
2019-05-23 20:28:51 +09:00
|
|
|
}
|
2019-05-16 01:41:22 +09:00
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub struct Value<'env, T> {
|
|
|
|
env: &'env Env,
|
|
|
|
raw_value: sys::napi_value,
|
2019-05-23 20:28:51 +09:00
|
|
|
value: T,
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Ref<T> {
|
|
|
|
raw_env: sys::napi_env,
|
|
|
|
raw_ref: sys::napi_ref,
|
|
|
|
_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,
|
|
|
|
raw_descriptor: sys::napi_property_descriptor,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct TaggedObject<T> {
|
|
|
|
type_id: TypeId,
|
|
|
|
object: Option<T>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! register_module {
|
|
|
|
($module_name:ident, $init:ident) => {
|
|
|
|
#[no_mangle]
|
|
|
|
#[cfg_attr(target_os = "linux", link_section = ".ctors")]
|
|
|
|
#[cfg_attr(target_os = "macos", link_section = "__DATA,__mod_init_func")]
|
|
|
|
#[cfg_attr(target_os = "windows", link_section = ".CRT$XCU")]
|
|
|
|
pub static __REGISTER_MODULE: extern "C" fn() = {
|
|
|
|
use std::io::Write;
|
|
|
|
use std::os::raw::c_char;
|
|
|
|
use std::ptr;
|
|
|
|
use $crate::sys;
|
|
|
|
|
|
|
|
extern "C" fn register_module() {
|
|
|
|
static mut MODULE_DESCRIPTOR: Option<sys::napi_module> = None;
|
|
|
|
unsafe {
|
|
|
|
MODULE_DESCRIPTOR = Some(sys::napi_module {
|
|
|
|
nm_version: 1,
|
|
|
|
nm_flags: 0,
|
|
|
|
nm_filename: concat!(file!(), "\0").as_ptr() as *const c_char,
|
|
|
|
nm_register_func: Some(init_module),
|
|
|
|
nm_modname: concat!(stringify!($module_name), "\0").as_ptr() as *const c_char,
|
|
|
|
nm_priv: 0 as *mut _,
|
|
|
|
reserved: [0 as *mut _; 4],
|
|
|
|
});
|
|
|
|
|
|
|
|
sys::napi_module_register(MODULE_DESCRIPTOR.as_mut().unwrap() as *mut sys::napi_module);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" fn init_module(
|
|
|
|
raw_env: sys::napi_env,
|
|
|
|
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);
|
|
|
|
|
|
|
|
let result = $init(&env, &mut exports);
|
|
|
|
|
|
|
|
match result {
|
|
|
|
Ok(Some(exports)) => exports.into_raw(),
|
|
|
|
Ok(None) => ptr::null_mut(),
|
|
|
|
Err(e) => {
|
|
|
|
let _ = writeln!(::std::io::stderr(), "Error initializing module: {:?}", e);
|
|
|
|
ptr::null_mut()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
register_module
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! callback {
|
|
|
|
($callback_expr:expr) => {{
|
|
|
|
use std::io::Write;
|
|
|
|
use std::mem;
|
|
|
|
use std::os::raw::c_char;
|
|
|
|
use std::ptr;
|
|
|
|
use $crate::sys;
|
|
|
|
use $crate::{Any, Env, Status, Value};
|
|
|
|
|
|
|
|
extern "C" fn raw_callback(
|
|
|
|
raw_env: sys::napi_env,
|
|
|
|
cb_info: sys::napi_callback_info,
|
|
|
|
) -> sys::napi_value {
|
|
|
|
const MAX_ARGC: usize = 8;
|
|
|
|
let mut argc = MAX_ARGC;
|
2020-02-18 22:09:17 +09:00
|
|
|
let mut raw_args =
|
|
|
|
unsafe { mem::MaybeUninit::<[$crate::sys::napi_value; MAX_ARGC]>::uninit().assume_init() };
|
2018-04-28 17:26:38 +09:00
|
|
|
let mut raw_this = ptr::null_mut();
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_get_cb_info(
|
|
|
|
raw_env,
|
|
|
|
cb_info,
|
2020-02-18 22:09:17 +09:00
|
|
|
&mut argc as *mut usize as *mut u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
&mut raw_args[0],
|
|
|
|
&mut raw_this,
|
|
|
|
ptr::null_mut(),
|
|
|
|
);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
let env = Env::from_raw(raw_env);
|
|
|
|
let this = Value::from_raw(&env, raw_this);
|
2020-02-18 22:09:17 +09:00
|
|
|
let mut args = unsafe { mem::MaybeUninit::<[Value<Any>; 8]>::uninit().assume_init() };
|
|
|
|
for (i, raw_arg) in raw_args.iter().enumerate() {
|
2018-04-28 17:26:38 +09:00
|
|
|
args[i] = Value::from_raw(&env, *raw_arg)
|
|
|
|
}
|
|
|
|
|
|
|
|
let callback = $callback_expr;
|
|
|
|
let result = callback(&env, this, &args[0..argc]);
|
|
|
|
|
|
|
|
match result {
|
|
|
|
Ok(Some(result)) => result.into_raw(),
|
|
|
|
Ok(None) => env.get_undefined().into_raw(),
|
|
|
|
Err(e) => {
|
2020-02-19 00:05:13 +09:00
|
|
|
if !cfg!(windows) {
|
|
|
|
let _ = writeln!(::std::io::stderr(), "Error calling function: {:?}", e);
|
|
|
|
}
|
2018-04-28 17:26:38 +09:00
|
|
|
let message = format!("{:?}", e);
|
|
|
|
unsafe {
|
|
|
|
$crate::sys::napi_throw_error(raw_env, ptr::null(), message.as_ptr() as *const c_char);
|
|
|
|
}
|
|
|
|
env.get_undefined().into_raw()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
raw_callback
|
|
|
|
}};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Error {
|
|
|
|
pub fn new(status: Status) -> Self {
|
|
|
|
Error { status: status }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<std::ffi::NulError> for Error {
|
|
|
|
fn from(_error: std::ffi::NulError) -> Self {
|
|
|
|
Error {
|
|
|
|
status: Status::StringContainsNull,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Env {
|
|
|
|
pub fn from_raw(env: sys::napi_env) -> Self {
|
|
|
|
Env(env)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_undefined<'a>(&'a self) -> Value<'a, Undefined> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe { sys::napi_get_undefined(self.0, &mut raw_value) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, Undefined)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_boolean(&self, value: bool) -> Value<Boolean> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe { sys::napi_get_boolean(self.0, value, &mut raw_value) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, Boolean { value })
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_int64<'a>(&'a self, int: i64) -> Value<'a, 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) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, Number { int })
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_string<'a, 'b>(&'a self, s: &'b str) -> Value<'a, String> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe {
|
2020-02-18 22:09:17 +09:00
|
|
|
sys::napi_create_string_utf8(
|
|
|
|
self.0,
|
|
|
|
s.as_ptr() as *const c_char,
|
|
|
|
s.len() as u64,
|
|
|
|
&mut raw_value,
|
|
|
|
)
|
2018-04-28 17:26:38 +09:00
|
|
|
};
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, String)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_string_utf16<'a, 'b>(&'a self, chars: &[u16]) -> Value<'a, String> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
2020-02-18 22:09:17 +09:00
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_create_string_utf16(self.0, chars.as_ptr(), chars.len() as u64, &mut raw_value)
|
|
|
|
};
|
2018-04-28 17:26:38 +09:00
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, String)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_object<'a>(&'a self) -> Value<'a, Object> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe { sys::napi_create_object(self.0, &mut raw_value) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, Object)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_array_with_length(&self, length: usize) -> Value<Object> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
2020-02-18 22:09:17 +09:00
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_create_array_with_length(self.0, length as u64, &mut raw_value) };
|
2018-04-28 17:26:38 +09:00
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_value, Object)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
2020-02-18 22:09:17 +09:00
|
|
|
pub fn create_buffer(&self, length: u64) -> Value<Buffer> {
|
2019-05-16 01:41:22 +09:00
|
|
|
let mut raw_value = ptr::null_mut();
|
2019-05-23 20:28:51 +09:00
|
|
|
let mut data = ptr::null_mut();
|
|
|
|
let status = unsafe { sys::napi_create_buffer(self.0, length, &mut data, &mut raw_value) };
|
2019-05-16 01:41:22 +09:00
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(
|
|
|
|
self,
|
|
|
|
raw_value,
|
|
|
|
Buffer {
|
|
|
|
data: data as *const u8,
|
|
|
|
size: length,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-02-18 22:09:17 +09:00
|
|
|
pub fn create_buffer_with_data(&self, data_ptr: *const u8, length: u64) -> Value<Buffer> {
|
2019-05-23 20:28:51 +09:00
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let mut data_raw_ptr = data_ptr as *mut c_void;
|
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_create_buffer(self.0, length, &mut data_raw_ptr, &mut raw_value) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
Value::from_raw_value(
|
|
|
|
self,
|
|
|
|
raw_value,
|
|
|
|
Buffer {
|
|
|
|
data: data_ptr,
|
|
|
|
size: length,
|
|
|
|
},
|
|
|
|
)
|
2019-05-16 01:41:22 +09:00
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
pub fn create_function<'a, 'b>(
|
|
|
|
&'a self,
|
|
|
|
name: &'b str,
|
|
|
|
callback: Callback,
|
|
|
|
) -> Value<'a, Function> {
|
|
|
|
let mut raw_result = ptr::null_mut();
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_create_function(
|
|
|
|
self.0,
|
|
|
|
name.as_ptr() as *const c_char,
|
2020-02-18 22:09:17 +09:00
|
|
|
name.len() as u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
Some(callback),
|
|
|
|
callback as *mut c_void,
|
|
|
|
&mut raw_result,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_result, Function)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_reference<T>(&self, value: &Value<T>) -> Ref<T> {
|
|
|
|
let mut raw_ref = ptr::null_mut();
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_create_reference(self.0, value.raw_value, 1, &mut raw_ref);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
};
|
|
|
|
|
|
|
|
Ref {
|
|
|
|
raw_env: self.0,
|
|
|
|
raw_ref,
|
|
|
|
_marker: PhantomData,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_reference_value<T: ValueType>(&self, reference: &Ref<T>) -> Value<T> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_get_reference_value(self.0, reference.raw_ref, &mut raw_value);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
};
|
|
|
|
|
|
|
|
Value::from_raw(self, raw_value)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn define_class<'a, 'b>(
|
|
|
|
&'a self,
|
|
|
|
name: &'b str,
|
|
|
|
constructor_cb: Callback,
|
|
|
|
properties: Vec<Property>,
|
|
|
|
) -> Value<'a, Function> {
|
|
|
|
let mut raw_result = ptr::null_mut();
|
|
|
|
let raw_properties = properties
|
|
|
|
.into_iter()
|
|
|
|
.map(|prop| prop.into_raw(self))
|
|
|
|
.collect::<Vec<sys::napi_property_descriptor>>();
|
|
|
|
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_define_class(
|
|
|
|
self.0,
|
|
|
|
name.as_ptr() as *const c_char,
|
2020-02-18 22:09:17 +09:00
|
|
|
name.len() as u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
Some(constructor_cb),
|
|
|
|
ptr::null_mut(),
|
2020-02-18 22:09:17 +09:00
|
|
|
raw_properties.len() as u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
raw_properties.as_ptr(),
|
|
|
|
&mut raw_result,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
|
2019-05-23 20:28:51 +09:00
|
|
|
Value::from_raw_value(self, raw_result, Function)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn wrap<T: 'static>(&self, js_object: &mut Value<Object>, native_object: T) -> Result<()> {
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_wrap(
|
|
|
|
self.0,
|
|
|
|
js_object.raw_value,
|
|
|
|
Box::into_raw(Box::new(TaggedObject::new(native_object))) as *mut c_void,
|
|
|
|
Some(raw_finalize::<T>),
|
|
|
|
ptr::null_mut(),
|
|
|
|
ptr::null_mut(),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
check_status(status).or(Ok(()))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn unwrap<T: 'static>(&self, js_object: &Value<Object>) -> Result<&mut T> {
|
|
|
|
unsafe {
|
|
|
|
let mut unknown_tagged_object: *mut c_void = ptr::null_mut();
|
|
|
|
let status = sys::napi_unwrap(self.0, js_object.raw_value, &mut unknown_tagged_object);
|
|
|
|
check_status(status)?;
|
|
|
|
|
|
|
|
let type_id: *const TypeId = mem::transmute(unknown_tagged_object);
|
|
|
|
if *type_id == TypeId::of::<T>() {
|
|
|
|
let tagged_object: *mut TaggedObject<T> = mem::transmute(unknown_tagged_object);
|
|
|
|
(*tagged_object).object.as_mut().ok_or(Error {
|
|
|
|
status: Status::InvalidArg,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Err(Error {
|
|
|
|
status: Status::InvalidArg,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn drop_wrapped<T: 'static>(&self, js_object: Value<Object>) -> Result<()> {
|
|
|
|
unsafe {
|
|
|
|
let mut unknown_tagged_object: *mut c_void = ptr::null_mut();
|
|
|
|
let status = sys::napi_unwrap(self.0, js_object.raw_value, &mut unknown_tagged_object);
|
|
|
|
check_status(status)?;
|
|
|
|
|
|
|
|
let type_id: *const TypeId = mem::transmute(unknown_tagged_object);
|
|
|
|
if *type_id == TypeId::of::<T>() {
|
|
|
|
let tagged_object: *mut TaggedObject<T> = mem::transmute(unknown_tagged_object);
|
|
|
|
(*tagged_object).object = None;
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(Error {
|
|
|
|
status: Status::InvalidArg,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn async_init(&self, resource: Option<Value<Object>>, name: &str) -> AsyncContext {
|
|
|
|
let raw_resource = resource
|
|
|
|
.map(|r| r.into_raw())
|
|
|
|
.unwrap_or_else(|| self.create_object().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);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
|
|
|
|
let status = sys::napi_create_reference(self.0, raw_resource, 1, &mut raw_resource_ref);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
AsyncContext {
|
|
|
|
raw_env: self.0,
|
|
|
|
raw_resource: raw_resource_ref,
|
|
|
|
raw_context,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_promise(&self) -> (Value<Object>, Deferred) {
|
|
|
|
let mut raw_promise = ptr::null_mut();
|
|
|
|
let mut raw_deferred = ptr::null_mut();
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
sys::napi_create_promise(self.0, &mut raw_deferred, &mut raw_promise);
|
|
|
|
}
|
|
|
|
|
2019-05-23 20:28:51 +09:00
|
|
|
(
|
|
|
|
Value::from_raw_value(self, raw_promise, Object),
|
|
|
|
Deferred(raw_deferred),
|
|
|
|
)
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn resolve_deferred<T: ValueType>(&self, deferred: Deferred, value: Value<T>) {
|
|
|
|
unsafe {
|
|
|
|
sys::napi_resolve_deferred(self.0, deferred.0, value.into_raw());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_executor(&self) -> executor::LibuvExecutor {
|
|
|
|
let event_loop = unsafe { sys::uv_default_loop() };
|
|
|
|
executor::LibuvExecutor::new(event_loop)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait ValueType: Copy {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(env: sys::napi_env, raw: sys::napi_value) -> Self;
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool;
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for Any {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(_env: sys::napi_env, _raw: sys::napi_value) -> Self {
|
|
|
|
Any
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(_env: sys::napi_env, _raw: sys::napi_value) -> bool {
|
2018-04-28 17:26:38 +09:00
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for Undefined {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(_env: sys::napi_env, _raw: sys::napi_value) -> Self {
|
|
|
|
Undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
get_raw_type(env, raw) == sys::napi_valuetype::napi_undefined
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for Boolean {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(env: sys::napi_env, raw: sys::napi_value) -> Self {
|
|
|
|
let mut value = true;
|
|
|
|
let status = unsafe { sys::napi_get_value_bool(env, raw, &mut value) };
|
|
|
|
debug_assert!(Status::from(status) == Status::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
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for Number {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(env: sys::napi_env, raw: sys::napi_value) -> Self {
|
|
|
|
let mut int: i64 = 0;
|
|
|
|
let status = unsafe { sys::napi_get_value_int64(env, raw, &mut int) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
Number { int }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
get_raw_type(env, raw) == sys::napi_valuetype::napi_number
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for String {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(_env: sys::napi_env, _raw: sys::napi_value) -> Self {
|
|
|
|
String {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
get_raw_type(env, raw) == sys::napi_valuetype::napi_string
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ValueType for Object {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(_env: sys::napi_env, _raw: sys::napi_value) -> Self {
|
|
|
|
Object {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
get_raw_type(env, raw) == sys::napi_valuetype::napi_object
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-16 01:41:22 +09:00
|
|
|
impl ValueType for Buffer {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(env: sys::napi_env, raw: sys::napi_value) -> Self {
|
|
|
|
let mut data = ptr::null_mut();
|
2020-02-18 22:09:17 +09:00
|
|
|
let mut size: u64 = 0;
|
2019-05-23 20:28:51 +09:00
|
|
|
let status = unsafe { sys::napi_get_buffer_info(env, raw, &mut data, &mut size) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
Buffer {
|
|
|
|
data: data as *const u8,
|
|
|
|
size,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
let mut result = false;
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_is_buffer(env, raw, &mut result);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
}
|
|
|
|
result
|
2019-05-16 01:41:22 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
impl ValueType for Function {
|
2019-05-23 20:28:51 +09:00
|
|
|
fn from_raw(_env: sys::napi_env, _raw: sys::napi_value) -> Self {
|
|
|
|
Function {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn matches_raw_type(env: sys::napi_env, raw: sys::napi_value) -> bool {
|
|
|
|
get_raw_type(env, raw) == sys::napi_valuetype::napi_function
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'env, T: ValueType> Value<'env, T> {
|
2019-05-23 20:28:51 +09:00
|
|
|
pub fn from_raw_value(env: &'env Env, raw_value: sys::napi_value, value: T) -> Self {
|
|
|
|
Self {
|
|
|
|
env,
|
|
|
|
raw_value,
|
|
|
|
value,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
pub fn from_raw(env: &'env Env, raw_value: sys::napi_value) -> Self {
|
|
|
|
Self {
|
|
|
|
env,
|
|
|
|
raw_value,
|
2019-05-23 20:28:51 +09:00
|
|
|
value: T::from_raw(env.0, raw_value),
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn into_raw(self) -> sys::napi_value {
|
|
|
|
self.raw_value
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn try_into<S: ValueType>(self) -> Result<Value<'env, S>> {
|
2019-05-23 20:28:51 +09:00
|
|
|
if S::matches_raw_type(self.env.0, self.raw_value) {
|
|
|
|
Ok(Value::from_raw(self.env, self.raw_value))
|
|
|
|
} else {
|
|
|
|
Err(Error {
|
|
|
|
status: Status::GenericFailure,
|
|
|
|
})
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn coerce_to_number(self) -> Result<Value<'env, 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) };
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(Value {
|
|
|
|
env: self.env,
|
|
|
|
raw_value: self.raw_value,
|
2019-05-23 20:28:51 +09:00
|
|
|
value: Number::from_raw(self.env.0, self.raw_value),
|
2018-04-28 17:26:38 +09:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn coerce_to_string(self) -> Result<Value<'env, String>> {
|
|
|
|
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) };
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(Value {
|
|
|
|
env: self.env,
|
|
|
|
raw_value: self.raw_value,
|
2019-05-23 20:28:51 +09:00
|
|
|
value: String,
|
2018-04-28 17:26:38 +09:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn coerce_to_object(self) -> Result<Value<'env, Object>> {
|
|
|
|
let mut new_raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_coerce_to_object(
|
|
|
|
self.env.0,
|
|
|
|
self.raw_value,
|
|
|
|
(&mut new_raw_value) as *mut sys::napi_value,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(Value {
|
|
|
|
env: self.env,
|
|
|
|
raw_value: self.raw_value,
|
2019-05-23 20:28:51 +09:00
|
|
|
value: Object,
|
2018-04-28 17:26:38 +09:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-23 20:28:51 +09:00
|
|
|
#[inline]
|
|
|
|
fn get_raw_type(env: sys::napi_env, raw_value: sys::napi_value) -> sys::napi_valuetype {
|
|
|
|
unsafe {
|
2020-02-18 22:09:17 +09:00
|
|
|
let value_type = ptr::null_mut();
|
|
|
|
let status = sys::napi_typeof(env, raw_value, value_type);
|
2019-05-23 20:28:51 +09:00
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2020-02-18 22:09:17 +09:00
|
|
|
*value_type
|
2019-05-23 20:28:51 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
impl<'env> Value<'env, String> {
|
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
let mut raw_length = ptr::null_mut();
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_get_named_property(
|
|
|
|
self.env.0,
|
|
|
|
self.raw_value,
|
|
|
|
"length\0".as_ptr() as *const c_char,
|
|
|
|
&mut raw_length,
|
|
|
|
);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
}
|
|
|
|
let length: Value<Number> = Value::from_raw(self.env, raw_length);
|
|
|
|
length.into()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-23 20:28:51 +09:00
|
|
|
impl<'env> Deref for Value<'env, String> {
|
|
|
|
type Target = [u8];
|
|
|
|
|
|
|
|
fn deref(&self) -> &[u8] {
|
2020-02-18 22:09:17 +09:00
|
|
|
let mut written_char_count: u64 = 0;
|
2019-05-23 20:28:51 +09:00
|
|
|
let len = self.len() + 1;
|
|
|
|
let mut result = Vec::with_capacity(len);
|
|
|
|
unsafe {
|
|
|
|
let status = sys::napi_get_value_string_utf8(
|
|
|
|
self.env.0,
|
|
|
|
self.raw_value,
|
|
|
|
result.as_mut_ptr(),
|
2020-02-18 22:09:17 +09:00
|
|
|
len as u64,
|
|
|
|
&mut written_char_count,
|
2019-05-23 20:28:51 +09:00
|
|
|
);
|
|
|
|
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
let ptr = result.as_ptr();
|
|
|
|
mem::forget(result);
|
2020-02-18 22:09:17 +09:00
|
|
|
slice::from_raw_parts(ptr as *const u8, written_char_count as usize)
|
2019-05-23 20:28:51 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
impl<'env> Into<Vec<u16>> for Value<'env, String> {
|
|
|
|
fn into(self) -> Vec<u16> {
|
|
|
|
let mut result = Vec::with_capacity(self.len() + 1); // Leave room for trailing null byte
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let mut written_char_count = 0;
|
|
|
|
let status = sys::napi_get_value_string_utf16(
|
|
|
|
self.env.0,
|
|
|
|
self.raw_value,
|
|
|
|
result.as_mut_ptr(),
|
2020-02-18 22:09:17 +09:00
|
|
|
result.capacity() as u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
&mut written_char_count,
|
|
|
|
);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2020-02-18 22:09:17 +09:00
|
|
|
result.set_len(written_char_count as usize);
|
2018-04-28 17:26:38 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'env> Into<usize> for Value<'env, Number> {
|
|
|
|
fn into(self) -> usize {
|
|
|
|
let mut result = 0;
|
|
|
|
let status = unsafe { sys::napi_get_value_int64(self.env.0, self.raw_value, &mut result) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
result as usize
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'env> Into<i64> for Value<'env, Number> {
|
|
|
|
fn into(self) -> i64 {
|
|
|
|
let mut result = 0;
|
|
|
|
let status = unsafe { sys::napi_get_value_int64(self.env.0, self.raw_value, &mut result) };
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'env> Into<f64> for Value<'env, Number> {
|
|
|
|
fn into(self) -> f64 {
|
|
|
|
let mut result = 0_f64;
|
|
|
|
let status = unsafe { sys::napi_get_value_double(self.env.0, self.raw_value, &mut result) };
|
|
|
|
debug_assert!(Status::from(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<()> {
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_set_property(
|
|
|
|
self.raw_env(),
|
|
|
|
self.raw_value(),
|
|
|
|
key.raw_value,
|
|
|
|
value.raw_value,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_named_property<'a, T, V: Into<Value<'a, T>>>(
|
|
|
|
&mut self,
|
|
|
|
name: &'a str,
|
|
|
|
value: V,
|
|
|
|
) -> Result<()> {
|
|
|
|
let key = CString::new(name)?;
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_set_named_property(
|
|
|
|
self.raw_env(),
|
|
|
|
self.raw_value(),
|
|
|
|
key.as_ptr(),
|
|
|
|
value.into().raw_value,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_named_property<T: ValueType>(&self, name: &str) -> Result<Value<T>> {
|
|
|
|
let key = CString::new(name)?;
|
|
|
|
let mut raw_value = ptr::null_mut();
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_get_named_property(
|
|
|
|
self.raw_env(),
|
|
|
|
self.raw_value(),
|
|
|
|
key.as_ptr(),
|
|
|
|
&mut raw_value,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
check_status(status)?;
|
|
|
|
Value::<Any>::from_raw(self.env, raw_value).try_into()
|
|
|
|
}
|
|
|
|
|
2019-01-27 07:33:47 +09:00
|
|
|
pub fn get_property_names<T: ValueType>(&self) -> Result<Value<T>> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
2019-05-16 01:41:22 +09:00
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_get_property_names(self.raw_env(), self.raw_value(), &mut raw_value) };
|
2019-01-27 07:33:47 +09:00
|
|
|
check_status(status)?;
|
|
|
|
Value::<Any>::from_raw(self.env, raw_value).try_into()
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2019-01-27 07:33:47 +09:00
|
|
|
pub fn get_index<T: ValueType>(&self, index: u32) -> Result<Value<T>> {
|
|
|
|
let mut raw_value = ptr::null_mut();
|
2019-05-16 01:41:22 +09:00
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_get_element(self.raw_env(), self.raw_value(), index, &mut raw_value) };
|
2019-01-27 07:33:47 +09:00
|
|
|
check_status(status)?;
|
|
|
|
Value::<Any>::from_raw(self.env, raw_value).try_into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_array(&self) -> Result<bool> {
|
|
|
|
let mut is_array = false;
|
2019-05-16 01:41:22 +09:00
|
|
|
let status = unsafe { sys::napi_is_array(self.raw_env(), self.raw_value(), &mut is_array) };
|
2019-01-27 07:33:47 +09:00
|
|
|
check_status(status)?;
|
|
|
|
Ok(is_array)
|
|
|
|
}
|
|
|
|
|
2020-01-16 23:10:06 +09:00
|
|
|
pub fn is_buffer(&self) -> Result<bool> {
|
|
|
|
let mut is_buffer = false;
|
|
|
|
let status = unsafe { sys::napi_is_buffer(self.raw_env(), self.raw_value(), &mut is_buffer) };
|
|
|
|
check_status(status)?;
|
|
|
|
Ok(is_buffer)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_buffer(&self) -> Value<'env, Buffer> {
|
|
|
|
Value::from_raw(self.env, self.raw_value)
|
|
|
|
}
|
|
|
|
|
2019-01-27 07:33:47 +09:00
|
|
|
pub fn get_array_length(&self) -> Result<u32> {
|
|
|
|
if self.is_array()? != true {
|
|
|
|
return Err(Error {
|
2019-05-16 01:41:22 +09:00
|
|
|
status: Status::ArrayExpected,
|
|
|
|
});
|
2019-01-27 07:33:47 +09:00
|
|
|
}
|
|
|
|
let mut length: u32 = 0;
|
2019-05-16 01:41:22 +09:00
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_get_array_length(self.raw_env(), self.raw_value(), &mut length) };
|
2019-01-27 07:33:47 +09:00
|
|
|
check_status(status)?;
|
|
|
|
Ok(length)
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
fn raw_value(&self) -> sys::napi_value {
|
|
|
|
self.raw_value
|
|
|
|
}
|
|
|
|
|
|
|
|
fn raw_env(&self) -> sys::napi_env {
|
|
|
|
self.env.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-16 01:41:22 +09:00
|
|
|
impl<'env> Deref for Value<'env, Buffer> {
|
|
|
|
type Target = [u8];
|
|
|
|
|
|
|
|
fn deref(&self) -> &[u8] {
|
2020-02-18 22:09:17 +09:00
|
|
|
unsafe { slice::from_raw_parts(self.value.data, self.value.size as usize) }
|
2019-05-16 01:41:22 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'env> DerefMut for Value<'env, Buffer> {
|
|
|
|
fn deref_mut(&mut self) -> &mut [u8] {
|
|
|
|
let mut data = ptr::null_mut();
|
2019-05-23 20:28:51 +09:00
|
|
|
let mut size = 0;
|
|
|
|
let status =
|
|
|
|
unsafe { sys::napi_get_buffer_info(self.env.0, self.raw_value, &mut data, &mut size) };
|
2019-05-16 01:41:22 +09:00
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
2020-02-18 22:09:17 +09:00
|
|
|
unsafe { slice::from_raw_parts_mut(data as *mut _, size as usize) }
|
2019-05-16 01:41:22 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
impl<'env> Value<'env, Function> {
|
|
|
|
pub fn call(
|
|
|
|
&self,
|
|
|
|
this: Option<&Value<'env, Object>>,
|
|
|
|
args: &[Value<'env, Any>],
|
|
|
|
) -> Result<Value<'env, Any>> {
|
|
|
|
let raw_this = this
|
|
|
|
.map(|v| v.into_raw())
|
|
|
|
.unwrap_or_else(|| self.env.get_undefined().into_raw());
|
2020-02-18 22:09:17 +09:00
|
|
|
let mut raw_args = unsafe { mem::MaybeUninit::<[sys::napi_value; 8]>::uninit().assume_init() };
|
2018-04-28 17:26:38 +09:00
|
|
|
for (i, arg) in args.into_iter().enumerate() {
|
|
|
|
raw_args[i] = arg.raw_value;
|
|
|
|
}
|
|
|
|
let mut return_value = ptr::null_mut();
|
|
|
|
let status = unsafe {
|
|
|
|
sys::napi_call_function(
|
|
|
|
self.env.0,
|
|
|
|
raw_this,
|
|
|
|
self.raw_value,
|
2020-02-18 22:09:17 +09:00
|
|
|
args.len() as u64,
|
2018-04-28 17:26:38 +09:00
|
|
|
&raw_args[0],
|
|
|
|
&mut return_value,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
check_status(status)?;
|
|
|
|
|
|
|
|
Ok(Value::from_raw(self.env, return_value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-16 23:10:06 +09:00
|
|
|
impl<'env> Value<'env, Any> {
|
|
|
|
pub fn get_type(&self) -> sys::napi_valuetype {
|
|
|
|
get_raw_type(self.env.0, self.raw_value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:26:38 +09:00
|
|
|
impl<T> Drop for Ref<T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
|
|
|
let mut ref_count = 0;
|
|
|
|
let status = sys::napi_reference_unref(self.raw_env, self.raw_ref, &mut ref_count);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
|
|
|
|
if ref_count == 0 {
|
|
|
|
let status = sys::napi_delete_reference(self.raw_env, self.raw_ref);
|
|
|
|
debug_assert!(Status::from(status) == Status::Ok);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
|
|
|
name: RustString::from(name),
|
|
|
|
raw_descriptor: sys::napi_property_descriptor {
|
|
|
|
utf8name: ptr::null_mut(),
|
|
|
|
name: ptr::null_mut(),
|
|
|
|
method: None,
|
|
|
|
getter: None,
|
|
|
|
setter: None,
|
|
|
|
value: ptr::null_mut(),
|
|
|
|
attributes: sys::napi_property_attributes::napi_default,
|
|
|
|
data: ptr::null_mut(),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn with_value<T>(mut self, value: Value<T>) -> Self {
|
|
|
|
self.raw_descriptor.value = value.raw_value;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn with_method(mut self, callback: Callback) -> Self {
|
|
|
|
self.raw_descriptor.method = Some(callback);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn with_getter(mut self, callback: Callback) -> Self {
|
|
|
|
self.raw_descriptor.getter = Some(callback);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn into_raw(mut self, env: &Env) -> sys::napi_property_descriptor {
|
|
|
|
self.raw_descriptor.name = env.create_string(&self.name).into_raw();
|
|
|
|
self.raw_descriptor
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: 'static> TaggedObject<T> {
|
|
|
|
fn new(object: T) -> Self {
|
|
|
|
TaggedObject {
|
|
|
|
type_id: TypeId::of::<T>(),
|
|
|
|
object: Some(object),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_status(code: sys::napi_status) -> Result<()> {
|
|
|
|
let status = Status::from(code);
|
|
|
|
match status {
|
|
|
|
Status::Ok => Ok(()),
|
|
|
|
_ => Err(Error { status }),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" fn raw_finalize<T>(
|
|
|
|
_raw_env: sys::napi_env,
|
|
|
|
finalize_data: *mut c_void,
|
|
|
|
_finalize_hint: *mut c_void,
|
|
|
|
) {
|
|
|
|
unsafe {
|
|
|
|
let tagged_object: *mut TaggedObject<T> = mem::transmute(finalize_data);
|
|
|
|
Box::from_raw(tagged_object);
|
|
|
|
}
|
|
|
|
}
|