2021-09-28 01:01:19 +09:00
|
|
|
use serde_json::{Map, Value};
|
|
|
|
|
2021-11-15 19:56:06 +09:00
|
|
|
use crate::{
|
|
|
|
bindgen_runtime::Null, check_status, sys, type_of, Error, JsObject, Result, Status, ValueType,
|
|
|
|
};
|
2021-09-28 01:01:19 +09:00
|
|
|
|
|
|
|
use super::{FromNapiValue, Object, ToNapiValue};
|
|
|
|
|
|
|
|
impl ToNapiValue for Value {
|
|
|
|
unsafe fn to_napi_value(env: sys::napi_env, val: Self) -> Result<sys::napi_value> {
|
|
|
|
match val {
|
|
|
|
Value::Null => Null::to_napi_value(env, Null),
|
|
|
|
Value::Bool(b) => bool::to_napi_value(env, b),
|
|
|
|
Value::Number(n) => {
|
|
|
|
if n.is_i64() {
|
|
|
|
i64::to_napi_value(env, n.as_i64().unwrap())
|
|
|
|
} else if n.is_f64() {
|
|
|
|
f64::to_napi_value(env, n.as_f64().unwrap())
|
|
|
|
} else {
|
|
|
|
let n = n.as_u64().unwrap();
|
|
|
|
if n > u32::MAX as u64 {
|
|
|
|
todo!("impl BigInt")
|
|
|
|
} else {
|
|
|
|
u32::to_napi_value(env, n as u32)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Value::String(s) => String::to_napi_value(env, s),
|
|
|
|
Value::Array(arr) => Vec::<Value>::to_napi_value(env, arr),
|
|
|
|
Value::Object(obj) => Map::to_napi_value(env, obj),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromNapiValue for Value {
|
|
|
|
unsafe fn from_napi_value(env: sys::napi_env, napi_val: sys::napi_value) -> Result<Self> {
|
|
|
|
let ty = type_of!(env, napi_val)?;
|
|
|
|
let val = match ty {
|
|
|
|
ValueType::Boolean => Value::Bool(bool::from_napi_value(env, napi_val)?),
|
|
|
|
ValueType::Number => {
|
|
|
|
return Err(Error::new(
|
|
|
|
Status::InvalidArg,
|
|
|
|
"Js Number is not be able to convert to rust.".to_owned(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
ValueType::String => Value::String(String::from_napi_value(env, napi_val)?),
|
|
|
|
ValueType::Object => {
|
|
|
|
let mut is_arr = false;
|
|
|
|
check_status!(
|
|
|
|
sys::napi_is_array(env, napi_val, &mut is_arr),
|
|
|
|
"Failed to detect whether given js is an array"
|
|
|
|
)?;
|
|
|
|
|
|
|
|
if is_arr {
|
|
|
|
Value::Array(Vec::<Value>::from_napi_value(env, napi_val)?)
|
|
|
|
} else {
|
|
|
|
Value::Object(Map::<String, Value>::from_napi_value(env, napi_val)?)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[cfg(feature = "napi6")]
|
2021-11-11 01:33:51 +09:00
|
|
|
ValueType::BigInt => todo!(),
|
2021-09-28 01:01:19 +09:00
|
|
|
_ => Value::Null,
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(val)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ToNapiValue for Map<String, Value> {
|
|
|
|
unsafe fn to_napi_value(env: sys::napi_env, val: Self) -> Result<sys::napi_value> {
|
|
|
|
let mut obj = Object::new(env)?;
|
|
|
|
|
|
|
|
for (k, v) in val.into_iter() {
|
|
|
|
obj.set(k, v)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
Object::to_napi_value(env, obj)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromNapiValue for Map<String, Value> {
|
|
|
|
unsafe fn from_napi_value(env: sys::napi_env, napi_val: sys::napi_value) -> Result<Self> {
|
2021-11-15 19:56:06 +09:00
|
|
|
let obj = JsObject(crate::Value {
|
2021-09-28 01:01:19 +09:00
|
|
|
env,
|
2021-11-15 19:56:06 +09:00
|
|
|
value: napi_val,
|
|
|
|
value_type: ValueType::Object,
|
|
|
|
});
|
2021-09-28 01:01:19 +09:00
|
|
|
|
|
|
|
let mut map = Map::new();
|
|
|
|
for key in Object::keys(&obj)?.into_iter() {
|
|
|
|
if let Some(val) = obj.get(&key)? {
|
|
|
|
map.insert(key, val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(map)
|
|
|
|
}
|
|
|
|
}
|