2021-09-23 02:29:09 +09:00
|
|
|
use std::convert::{From, TryFrom};
|
|
|
|
use std::error;
|
2021-11-22 17:51:21 +09:00
|
|
|
use std::ffi::{CStr, CString};
|
2021-09-23 02:29:09 +09:00
|
|
|
use std::fmt;
|
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
use std::fmt::Display;
|
|
|
|
use std::os::raw::{c_char, c_void};
|
|
|
|
use std::ptr;
|
|
|
|
|
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
use serde::{de, ser};
|
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
use serde_json::Error as SerdeJSONError;
|
|
|
|
|
|
|
|
use crate::{check_status, sys, Status};
|
|
|
|
|
|
|
|
pub type Result<T> = std::result::Result<T, Error>;
|
|
|
|
|
|
|
|
/// Represent `JsError`.
|
|
|
|
/// Return this Error in `js_function`, **napi-rs** will throw it as `JsError` for you.
|
|
|
|
/// If you want throw it as `TypeError` or `RangeError`, you can use `JsTypeError/JsRangeError::from(Error).throw_into(env)`
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct Error {
|
|
|
|
pub status: Status,
|
|
|
|
pub reason: String,
|
2021-11-13 21:51:14 +09:00
|
|
|
// Convert raw `JsError` into Error
|
|
|
|
// Only be used in `async fn(p: Promise<T>)` scenario
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
pub(crate) maybe_raw: sys::napi_ref,
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
2021-11-13 21:51:14 +09:00
|
|
|
unsafe impl Send for Error {}
|
|
|
|
unsafe impl Sync for Error {}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
impl error::Error for Error {}
|
|
|
|
|
2022-05-30 19:43:35 +09:00
|
|
|
impl From<std::convert::Infallible> for Error {
|
|
|
|
fn from(_: std::convert::Infallible) -> Self {
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
impl ser::Error for Error {
|
|
|
|
fn custom<T: Display>(msg: T) -> Self {
|
|
|
|
Error::new(Status::InvalidArg, msg.to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
impl de::Error for Error {
|
|
|
|
fn custom<T: Display>(msg: T) -> Self {
|
|
|
|
Error::new(Status::InvalidArg, msg.to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(feature = "serde-json")]
|
|
|
|
impl From<SerdeJSONError> for Error {
|
|
|
|
fn from(value: SerdeJSONError) -> Self {
|
|
|
|
Error::new(Status::InvalidArg, format!("{}", value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
impl From<sys::napi_ref> for Error {
|
|
|
|
fn from(value: sys::napi_ref) -> Self {
|
|
|
|
Self {
|
|
|
|
status: Status::InvalidArg,
|
|
|
|
reason: "".to_string(),
|
|
|
|
maybe_raw: value,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-14 09:03:31 +09:00
|
|
|
#[cfg(feature = "anyhow")]
|
|
|
|
impl From<anyhow::Error> for Error {
|
|
|
|
fn from(value: anyhow::Error) -> Self {
|
|
|
|
Error::new(Status::GenericFailure, format!("{}", value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
impl fmt::Display for Error {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
if !self.reason.is_empty() {
|
|
|
|
write!(f, "{:?}, {}", self.status, self.reason)
|
|
|
|
} else {
|
|
|
|
write!(f, "{:?}", self.status)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Error {
|
|
|
|
pub fn new(status: Status, reason: String) -> Self {
|
2021-11-13 21:51:14 +09:00
|
|
|
Error {
|
|
|
|
status,
|
|
|
|
reason,
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
maybe_raw: ptr::null_mut(),
|
|
|
|
}
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_status(status: Status) -> Self {
|
|
|
|
Error {
|
|
|
|
status,
|
|
|
|
reason: "".to_owned(),
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
maybe_raw: ptr::null_mut(),
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-01 23:56:32 +09:00
|
|
|
pub fn from_reason<T: Into<String>>(reason: T) -> Self {
|
2021-09-23 02:29:09 +09:00
|
|
|
Error {
|
|
|
|
status: Status::GenericFailure,
|
2022-05-01 23:56:32 +09:00
|
|
|
reason: reason.into(),
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
maybe_raw: ptr::null_mut(),
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<std::ffi::NulError> for Error {
|
|
|
|
fn from(error: std::ffi::NulError) -> Self {
|
|
|
|
Error {
|
|
|
|
status: Status::GenericFailure,
|
|
|
|
reason: format!("{}", error),
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
maybe_raw: ptr::null_mut(),
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<std::io::Error> for Error {
|
|
|
|
fn from(error: std::io::Error) -> Self {
|
|
|
|
Error {
|
|
|
|
status: Status::GenericFailure,
|
|
|
|
reason: format!("{}", error),
|
2021-12-08 13:52:33 +09:00
|
|
|
#[cfg(all(feature = "tokio_rt", feature = "napi4"))]
|
2021-11-13 21:51:14 +09:00
|
|
|
maybe_raw: ptr::null_mut(),
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct ExtendedErrorInfo {
|
|
|
|
pub message: String,
|
|
|
|
pub engine_reserved: *mut c_void,
|
|
|
|
pub engine_error_code: u32,
|
|
|
|
pub error_code: Status,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<sys::napi_extended_error_info> for ExtendedErrorInfo {
|
|
|
|
type Error = Error;
|
|
|
|
|
|
|
|
fn try_from(value: sys::napi_extended_error_info) -> Result<Self> {
|
|
|
|
Ok(Self {
|
|
|
|
message: unsafe {
|
|
|
|
CString::from_raw(value.error_message as *mut c_char)
|
|
|
|
.into_string()
|
|
|
|
.map_err(|e| Error::new(Status::GenericFailure, format!("{}", e)))?
|
|
|
|
},
|
|
|
|
engine_error_code: value.engine_error_code,
|
|
|
|
engine_reserved: value.engine_reserved,
|
|
|
|
error_code: Status::from(value.error_code),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct JsError(Error);
|
|
|
|
|
2022-08-14 09:03:31 +09:00
|
|
|
#[cfg(feature = "anyhow")]
|
|
|
|
impl From<anyhow::Error> for JsError {
|
|
|
|
fn from(value: anyhow::Error) -> Self {
|
|
|
|
JsError(Error::new(Status::GenericFailure, value.to_string()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
pub struct JsTypeError(Error);
|
|
|
|
|
|
|
|
pub struct JsRangeError(Error);
|
|
|
|
|
2021-12-03 17:30:31 +09:00
|
|
|
#[cfg(feature = "experimental")]
|
|
|
|
pub struct JsSyntaxError(Error);
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
macro_rules! impl_object_methods {
|
|
|
|
($js_value:ident, $kind:expr) => {
|
|
|
|
impl $js_value {
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// This function is safety if env is not null ptr.
|
|
|
|
pub unsafe fn into_value(self, env: sys::napi_env) -> sys::napi_value {
|
|
|
|
let error_status = format!("{:?}", self.0.status);
|
|
|
|
let status_len = error_status.len();
|
|
|
|
let error_code_string = CString::new(error_status).unwrap();
|
|
|
|
let reason_len = self.0.reason.len();
|
|
|
|
let reason = CString::new(self.0.reason).unwrap();
|
|
|
|
let mut error_code = ptr::null_mut();
|
|
|
|
let mut reason_string = ptr::null_mut();
|
|
|
|
let mut js_error = ptr::null_mut();
|
2022-01-13 12:15:02 +09:00
|
|
|
let create_code_status = unsafe {
|
|
|
|
sys::napi_create_string_utf8(env, error_code_string.as_ptr(), status_len, &mut error_code)
|
|
|
|
};
|
2021-09-23 02:29:09 +09:00
|
|
|
debug_assert!(create_code_status == sys::Status::napi_ok);
|
2022-01-13 12:15:02 +09:00
|
|
|
let create_reason_status = unsafe {
|
|
|
|
sys::napi_create_string_utf8(env, reason.as_ptr(), reason_len, &mut reason_string)
|
|
|
|
};
|
2021-09-23 02:29:09 +09:00
|
|
|
debug_assert!(create_reason_status == sys::Status::napi_ok);
|
2022-01-13 12:15:02 +09:00
|
|
|
let create_error_status = unsafe { $kind(env, error_code, reason_string, &mut js_error) };
|
2021-09-23 02:29:09 +09:00
|
|
|
debug_assert!(create_error_status == sys::Status::napi_ok);
|
|
|
|
js_error
|
|
|
|
}
|
|
|
|
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// This function is safety if env is not null ptr.
|
|
|
|
pub unsafe fn throw_into(self, env: sys::napi_env) {
|
2021-11-05 19:31:36 +09:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
let reason = self.0.reason.clone();
|
|
|
|
let status = self.0.status;
|
2021-11-13 21:51:14 +09:00
|
|
|
if status == Status::PendingException {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-13 12:15:02 +09:00
|
|
|
let js_error = unsafe { self.into_value(env) };
|
2021-11-05 19:31:36 +09:00
|
|
|
#[cfg(debug_assertions)]
|
2022-01-13 12:15:02 +09:00
|
|
|
let throw_status = unsafe { sys::napi_throw(env, js_error) };
|
|
|
|
unsafe { sys::napi_throw(env, js_error) };
|
2021-11-05 19:31:36 +09:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
assert!(
|
|
|
|
throw_status == sys::Status::napi_ok,
|
|
|
|
"Throw error failed, status: [{}], raw message: \"{}\", raw status: [{}]",
|
|
|
|
Status::from(throw_status),
|
|
|
|
reason,
|
|
|
|
Status::from(status)
|
|
|
|
);
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
2022-03-05 22:29:57 +09:00
|
|
|
#[allow(clippy::not_unsafe_ptr_arg_deref)]
|
2021-09-23 02:29:09 +09:00
|
|
|
pub fn throw(&self, env: sys::napi_env) -> Result<()> {
|
2021-11-22 17:51:21 +09:00
|
|
|
let error_status = format!("{:?}\0", self.0.status);
|
2021-09-23 02:29:09 +09:00
|
|
|
let status_len = error_status.len();
|
2021-11-22 17:51:21 +09:00
|
|
|
let error_code_string =
|
|
|
|
unsafe { CStr::from_bytes_with_nul_unchecked(error_status.as_bytes()) };
|
2021-09-23 02:29:09 +09:00
|
|
|
let reason_len = self.0.reason.len();
|
2021-11-22 17:51:21 +09:00
|
|
|
let reason_c_string = format!("{}\0", self.0.reason.clone());
|
|
|
|
let reason = unsafe { CStr::from_bytes_with_nul_unchecked(reason_c_string.as_bytes()) };
|
2021-09-23 02:29:09 +09:00
|
|
|
let mut error_code = ptr::null_mut();
|
|
|
|
let mut reason_string = ptr::null_mut();
|
|
|
|
let mut js_error = ptr::null_mut();
|
|
|
|
check_status!(unsafe {
|
|
|
|
sys::napi_create_string_utf8(env, error_code_string.as_ptr(), status_len, &mut error_code)
|
|
|
|
})?;
|
|
|
|
check_status!(unsafe {
|
|
|
|
sys::napi_create_string_utf8(env, reason.as_ptr(), reason_len, &mut reason_string)
|
|
|
|
})?;
|
|
|
|
check_status!(unsafe { $kind(env, error_code, reason_string, &mut js_error) })?;
|
|
|
|
check_status!(unsafe { sys::napi_throw(env, js_error) })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Error> for $js_value {
|
|
|
|
fn from(err: Error) -> Self {
|
|
|
|
Self(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl_object_methods!(JsError, sys::napi_create_error);
|
|
|
|
impl_object_methods!(JsTypeError, sys::napi_create_type_error);
|
|
|
|
impl_object_methods!(JsRangeError, sys::napi_create_range_error);
|
2021-12-02 01:18:38 +09:00
|
|
|
#[cfg(feature = "experimental")]
|
|
|
|
impl_object_methods!(JsSyntaxError, sys::node_api_create_syntax_error);
|
2021-09-23 02:29:09 +09:00
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! error {
|
|
|
|
($status:expr, $($msg:tt)*) => {
|
|
|
|
$crate::Error::new($status, format!($($msg)*))
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! check_status {
|
|
|
|
($code:expr) => {{
|
|
|
|
let c = $code;
|
|
|
|
match c {
|
|
|
|
$crate::sys::Status::napi_ok => Ok(()),
|
|
|
|
_ => Err($crate::Error::new($crate::Status::from(c), "".to_owned())),
|
|
|
|
}
|
|
|
|
}};
|
|
|
|
|
|
|
|
($code:expr, $($msg:tt)*) => {{
|
|
|
|
let c = $code;
|
|
|
|
match c {
|
|
|
|
$crate::sys::Status::napi_ok => Ok(()),
|
|
|
|
_ => Err($crate::Error::new($crate::Status::from(c), format!($($msg)*))),
|
|
|
|
}
|
|
|
|
}};
|
|
|
|
}
|