use libquickjs_ng_sys::{JSContext, JSValue};
use serde::{ser, Serialize};
use crate::utils::{
create_bigint, create_bool, create_empty_array, create_empty_object, create_float, create_int,
create_null, create_string, create_undefined, own_raw_value,
};
use crate::value::{OwnedJsArray, OwnedJsObject, OwnedJsValue};
use super::error::{Error, Result};
const MAX_SAFE_INTEGER: u64 = 9007199254740991;
pub struct Serializer {
context: *mut JSContext,
root: OwnedJsValue,
paths: Vec<OwnedJsValue>,
current: Option<OwnedJsValue>,
current_is_key: bool,
current_key: Option<String>,
}
pub fn to_js<T>(context: *mut JSContext, value: &T) -> Result<OwnedJsValue>
where
T: Serialize,
{
let root = own_raw_value(context, create_undefined());
let mut serializer = Serializer {
context,
root,
paths: Vec::new(),
current: None,
current_is_key: false,
current_key: None,
};
value.serialize(&mut serializer)?;
Ok(serializer.root)
}
impl Serializer {
#[inline]
pub(self) fn set_node_value(&mut self, value: JSValue) -> Result<()> {
if let Some(current) = self.current.as_mut() {
if current.is_undefined() {
current.replace(value);
} else if current.is_array() {
let current = OwnedJsArray::try_from_value(current.clone())?;
current.push(OwnedJsValue::new(self.context, value))?;
} else if current.is_object() {
if let Some(key) = self.current_key.take() {
let current = OwnedJsObject::try_from_value(current.clone())?;
current.set_property(&key, OwnedJsValue::new(self.context, value))?;
} else {
#[cfg(debug_assertions)]
{
let v = OwnedJsValue::new(self.context, value);
println!("tag: {:?}", v.tag());
println!("tag: {}", v.to_json_string(0)?);
}
return Err(Error::Message("current key is None".to_string()));
}
} else {
#[cfg(debug_assertions)]
{
println!("current tag: {:?}", current.tag());
println!("current value: {}", current.to_json_string(0)?);
}
return Err(Error::Message(
"current is not undefined, object or array.".to_string(),
));
}
} else {
self.root.replace(value);
let current = self.root.clone();
self.current = Some(current);
}
Ok(())
}
#[inline]
pub(self) fn push_object(&mut self) -> Result<()> {
let value = create_empty_object(self.context).unwrap();
self.set_node_value(value)?;
if let Some(current) = self.current.take() {
self.paths.push(current);
}
self.current = Some(OwnedJsValue::own(self.context, &value));
Ok(())
}
#[inline]
pub(self) fn push_array(&mut self) -> Result<()> {
let value = create_empty_array(self.context).unwrap();
self.set_node_value(value)?;
if let Some(current) = self.current.take() {
self.paths.push(current);
}
self.current = Some(OwnedJsValue::own(self.context, &value));
Ok(())
}
#[inline]
pub(self) fn pop(&mut self) -> Result<()> {
self.current = self.paths.pop();
Ok(())
}
}
impl<'a> ser::Serializer for &'a mut Serializer {
type Ok = ();
type Error = Error;
type SerializeSeq = Self;
type SerializeTuple = Self;
type SerializeTupleStruct = Self;
type SerializeTupleVariant = Self;
type SerializeMap = Self;
type SerializeStruct = Self;
type SerializeStructVariant = Self;
fn serialize_bool(self, v: bool) -> Result<()> {
let value = create_bool(self.context, v);
self.set_node_value(value)
}
fn serialize_i8(self, v: i8) -> Result<()> {
self.serialize_i32(i32::from(v))
}
fn serialize_i16(self, v: i16) -> Result<()> {
self.serialize_i32(i32::from(v))
}
fn serialize_i32(self, v: i32) -> Result<()> {
let value = create_int(self.context, v as i32);
self.set_node_value(value)
}
fn serialize_i64(self, v: i64) -> Result<()> {
if v > MAX_SAFE_INTEGER as i64 {
let value = create_bigint(self.context, (v as i64).into())?;
self.set_node_value(value)
} else if v > i32::MAX as i64 {
let value = create_float(self.context, v as f64);
self.set_node_value(value)
} else {
let value = create_int(self.context, v as i32);
self.set_node_value(value)
}
}
fn serialize_u8(self, v: u8) -> Result<()> {
self.serialize_i32(i32::from(v))
}
fn serialize_u16(self, v: u16) -> Result<()> {
self.serialize_i32(i32::from(v))
}
fn serialize_u32(self, v: u32) -> Result<()> {
if v > i32::MAX as u32 {
let value = create_float(self.context, v as f64);
self.set_node_value(value)
} else {
let value = create_int(self.context, v as i32);
self.set_node_value(value)
}
}
fn serialize_u64(self, v: u64) -> Result<()> {
if v > i64::MAX as u64 {
let value = create_bigint(self.context, num_bigint::BigInt::from(v).into())?;
self.set_node_value(value)
} else if v > MAX_SAFE_INTEGER {
let value = create_bigint(self.context, (v as i64).into())?;
self.set_node_value(value)
} else if v > i32::MAX as u64 {
let value = create_float(self.context, v as f64);
self.set_node_value(value)
} else {
let value = create_int(self.context, v as i32);
self.set_node_value(value)
}
}
fn serialize_f32(self, v: f32) -> Result<()> {
self.serialize_f64(f64::from(v))
}
fn serialize_f64(self, v: f64) -> Result<()> {
let value = create_float(self.context, v);
self.set_node_value(value)
}
fn serialize_char(self, v: char) -> Result<()> {
self.serialize_str(&v.to_string())
}
fn serialize_str(self, v: &str) -> Result<()> {
if self.current_is_key {
self.current_key = Some(v.to_string());
self.current_is_key = false;
Ok(())
} else {
let value = create_string(self.context, v).unwrap();
self.set_node_value(value)
}
}
fn serialize_bytes(self, v: &[u8]) -> Result<()> {
use serde::ser::SerializeSeq;
let mut seq = self.serialize_seq(Some(v.len()))?;
for byte in v {
seq.serialize_element(byte)?;
}
seq.end()
}
fn serialize_none(self) -> Result<()> {
self.serialize_unit()
}
fn serialize_some<T>(self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(self)
}
fn serialize_unit(self) -> Result<()> {
let value = create_null();
self.set_node_value(value)
}
fn serialize_unit_struct(self, _name: &'static str) -> Result<()> {
self.serialize_unit()
}
fn serialize_unit_variant(
self,
_name: &'static str,
_variant_index: u32,
variant: &'static str,
) -> Result<()> {
self.serialize_str(variant)
}
fn serialize_newtype_struct<T>(self, _name: &'static str, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(self)
}
fn serialize_newtype_variant<T>(
self,
_name: &'static str,
_variant_index: u32,
variant: &'static str,
value: &T,
) -> Result<()>
where
T: ?Sized + Serialize,
{
self.push_object()?;
self.current_is_key = true;
variant.serialize(&mut *self)?;
value.serialize(&mut *self)?;
self.pop()?;
Ok(())
}
fn serialize_seq(self, _len: Option<usize>) -> Result<Self::SerializeSeq> {
self.push_array()?;
Ok(self)
}
fn serialize_tuple(self, len: usize) -> Result<Self::SerializeTuple> {
self.serialize_seq(Some(len))
}
fn serialize_tuple_struct(
self,
_name: &'static str,
len: usize,
) -> Result<Self::SerializeTupleStruct> {
self.serialize_seq(Some(len))
}
fn serialize_tuple_variant(
self,
_name: &'static str,
_variant_index: u32,
variant: &'static str,
_len: usize,
) -> Result<Self::SerializeTupleVariant> {
self.push_object()?;
self.current_is_key = true;
variant.serialize(&mut *self)?;
self.push_array()?;
Ok(self)
}
fn serialize_map(self, _len: Option<usize>) -> Result<Self::SerializeMap> {
self.push_object()?;
Ok(self)
}
fn serialize_struct(self, _name: &'static str, len: usize) -> Result<Self::SerializeStruct> {
self.serialize_map(Some(len))
}
fn serialize_struct_variant(
self,
_name: &'static str,
_variant_index: u32,
variant: &'static str,
_len: usize,
) -> Result<Self::SerializeStructVariant> {
self.push_object()?;
self.current_is_key = true;
variant.serialize(&mut *self)?;
self.push_object()?;
Ok(self)
}
}
impl<'a> ser::SerializeSeq for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_element<T>(&mut self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeTuple for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_element<T>(&mut self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeTupleStruct for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_field<T>(&mut self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeTupleVariant for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_field<T>(&mut self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeMap for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_key<T>(&mut self, key: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
if self.current_key.is_none() {
self.current_is_key = true;
} else {
return Err(Error::Message(
"Cannot serialize map with more than one key".to_string(),
));
}
key.serialize(&mut **self)
}
fn serialize_value<T>(&mut self, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeStruct for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_field<T>(&mut self, key: &'static str, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
if self.current_key.is_none() {
self.current_is_key = true;
} else {
return Err(Error::Message(
"Cannot serialize map with more than one key".to_string(),
));
}
key.serialize(&mut **self)?;
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
Ok(())
}
}
impl<'a> ser::SerializeStructVariant for &'a mut Serializer {
type Ok = ();
type Error = Error;
fn serialize_field<T>(&mut self, key: &'static str, value: &T) -> Result<()>
where
T: ?Sized + Serialize,
{
if self.current_key.is_none() {
self.current_is_key = true;
} else {
return Err(Error::Message(
"Cannot serialize map with more than one key".to_string(),
));
}
key.serialize(&mut **self)?;
value.serialize(&mut **self)
}
fn end(self) -> Result<()> {
self.pop()?;
self.pop()?;
Ok(())
}
}