use rbs::{value::map::ValueMap, Value};
use serde::{ser, Serialize};
pub fn fast_serialize<T: Serialize>(mut value: T) -> Result<Value, rbs::Error> {
let type_name = std::any::type_name::<T>();
if type_name == std::any::type_name::<Value>() {
let addr = std::ptr::addr_of_mut!(value);
let v = unsafe { &mut *addr.cast() };
return Ok(std::mem::take(v));
}
if type_name == std::any::type_name::<&Value>() {
let addr = std::ptr::addr_of!(value);
return Ok(unsafe { *addr.cast::<&Value>() }.clone());
}
if type_name == std::any::type_name::<&&Value>() {
let addr = std::ptr::addr_of!(value);
return Ok(unsafe { **addr.cast::<&&Value>() }.clone());
}
value.serialize(Serializer)
}
struct Serializer;
impl serde::Serializer for Serializer {
type Ok = rbs::Value;
type Error = rbs::Error;
type SerializeSeq = SerializeVec;
type SerializeTuple = SerializeVec;
type SerializeMap = DefaultSerializeMap;
type SerializeTupleStruct = SerializeVec;
type SerializeStruct = DefaultSerializeMap;
type SerializeStructVariant = DefaultSerializeMap;
type SerializeTupleVariant = SerializeTupleVariant;
#[inline]
fn serialize_bool(self, val: bool) -> Result<Self::Ok, Self::Error> {
Ok(Value::Bool(val))
}
#[inline]
fn serialize_i8(self, val: i8) -> Result<Self::Ok, Self::Error> {
Ok(Value::I32(i32::from(val)))
}
#[inline]
fn serialize_i16(self, val: i16) -> Result<Self::Ok, Self::Error> {
Ok(Value::I32(i32::from(val)))
}
#[inline]
fn serialize_i32(self, val: i32) -> Result<Self::Ok, Self::Error> {
Ok(Value::I32(val))
}
#[inline]
fn serialize_i64(self, val: i64) -> Result<Self::Ok, Self::Error> {
Ok(Value::I64(val))
}
#[inline]
fn serialize_u8(self, val: u8) -> Result<Self::Ok, Self::Error> {
Ok(Value::U32(u32::from(val)))
}
#[inline]
fn serialize_u16(self, val: u16) -> Result<Self::Ok, Self::Error> {
Ok(Value::U32(u32::from(val)))
}
#[inline]
fn serialize_u32(self, val: u32) -> Result<Self::Ok, Self::Error> {
Ok(Value::U32(val))
}
#[inline]
fn serialize_u64(self, val: u64) -> Result<Self::Ok, Self::Error> {
Ok(Value::U64(val))
}
#[inline]
fn serialize_f32(self, val: f32) -> Result<Self::Ok, Self::Error> {
Ok(Value::F32(val))
}
#[inline]
fn serialize_f64(self, val: f64) -> Result<Self::Ok, Self::Error> {
Ok(Value::F64(val))
}
#[inline]
fn serialize_char(self, val: char) -> Result<Self::Ok, Self::Error> {
let mut buf = String::new();
buf.push(val);
self.serialize_str(&buf)
}
#[inline]
fn serialize_str(self, val: &str) -> Result<Self::Ok, Self::Error> {
Ok(Value::String(val.into()))
}
#[inline]
fn serialize_bytes(self, val: &[u8]) -> Result<Self::Ok, Self::Error> {
Ok(Value::Binary(val.into()))
}
#[inline]
fn serialize_unit(self) -> Result<Self::Ok, Self::Error> {
Ok(Value::Null)
}
#[inline]
fn serialize_unit_struct(self, _name: &'static str) -> Result<Self::Ok, Self::Error> {
self.serialize_unit()
}
#[inline]
fn serialize_unit_variant(
self,
_name: &'static str,
_idx: u32,
variant: &'static str,
) -> Result<Self::Ok, Self::Error> {
self.serialize_str(variant)
}
#[inline]
fn serialize_newtype_struct<T: Serialize + ?Sized>(
self,
name: &'static str,
value: &T,
) -> Result<Self::Ok, Self::Error> {
Ok(Value::Ext(name, Box::new(fast_serialize(value)?)))
}
fn serialize_newtype_variant<T: Serialize + ?Sized>(
self,
name: &'static str,
_idx: u32,
variant: &'static str,
_value: &T,
) -> Result<Self::Ok, Self::Error> {
Err(rbs::Error::Syntax(format!(
"Ensemble does not support enums with values: {name}::{variant}",
)))
}
#[inline]
fn serialize_none(self) -> Result<Self::Ok, Self::Error> {
self.serialize_unit()
}
#[inline]
fn serialize_some<T: Serialize + ?Sized>(self, value: &T) -> Result<Self::Ok, Self::Error> {
fast_serialize(value)
}
fn serialize_seq(self, len: Option<usize>) -> Result<Self::SerializeSeq, Self::Error> {
let se = SerializeVec {
vec: Vec::with_capacity(len.unwrap_or(0)),
};
Ok(se)
}
fn serialize_tuple(self, len: usize) -> Result<Self::SerializeTuple, Self::Error> {
self.serialize_seq(Some(len))
}
fn serialize_tuple_struct(
self,
_name: &'static str,
len: usize,
) -> Result<Self::SerializeTupleStruct, Self::Error> {
self.serialize_tuple(len)
}
fn serialize_tuple_variant(
self,
_name: &'static str,
idx: u32,
_variant: &'static str,
len: usize,
) -> Result<Self::SerializeTupleVariant, Self::Error> {
let se = SerializeTupleVariant {
idx,
vec: Vec::with_capacity(len),
};
Ok(se)
}
fn serialize_map(self, len: Option<usize>) -> Result<Self::SerializeMap, Self::Error> {
let se = DefaultSerializeMap {
next_key: None,
map: Vec::with_capacity(len.unwrap_or(0)),
};
Ok(se)
}
#[inline]
fn serialize_struct(
self,
_name: &'static str,
len: usize,
) -> Result<Self::SerializeStruct, Self::Error> {
let se = DefaultSerializeMap {
next_key: None,
map: Vec::with_capacity(len),
};
Ok(se)
}
#[inline]
fn serialize_struct_variant(
self,
_name: &'static str,
_idx: u32,
_variant: &'static str,
len: usize,
) -> Result<Self::SerializeStructVariant, Self::Error> {
let se = DefaultSerializeMap {
map: Vec::with_capacity(len),
next_key: None,
};
Ok(se)
}
}
pub struct SerializeVec {
vec: Vec<Value>,
}
pub struct SerializeTupleVariant {
idx: u32,
vec: Vec<Value>,
}
pub struct DefaultSerializeMap {
map: Vec<(Value, Value)>,
next_key: Option<Value>,
}
pub struct SerializeStructVariant {
idx: u32,
vec: Vec<Value>,
}
impl ser::SerializeSeq for SerializeVec {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_element<T: Serialize + ?Sized>(&mut self, value: &T) -> Result<(), Self::Error> {
self.vec.push(fast_serialize(value)?);
Ok(())
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
Ok(Value::Array(self.vec))
}
}
impl ser::SerializeTuple for SerializeVec {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_element<T: Serialize + ?Sized>(&mut self, value: &T) -> Result<(), Self::Error> {
ser::SerializeSeq::serialize_element(self, value)
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
ser::SerializeSeq::end(self)
}
}
impl ser::SerializeTupleStruct for SerializeVec {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_field<T: Serialize + ?Sized>(&mut self, value: &T) -> Result<(), Self::Error> {
ser::SerializeSeq::serialize_element(self, value)
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
ser::SerializeSeq::end(self)
}
}
impl ser::SerializeTupleVariant for SerializeTupleVariant {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_field<T: Serialize + ?Sized>(&mut self, value: &T) -> Result<(), Self::Error> {
self.vec.push(fast_serialize(value)?);
Ok(())
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
Ok(Value::Array(vec![
Value::from(self.idx),
Value::Array(self.vec),
]))
}
}
impl ser::SerializeMap for DefaultSerializeMap {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_key<T: Serialize + ?Sized>(&mut self, key: &T) -> Result<(), Self::Error> {
self.next_key = Some(fast_serialize(key)?);
Ok(())
}
fn serialize_value<T: Serialize + ?Sized>(&mut self, value: &T) -> Result<(), Self::Error> {
let key = self
.next_key
.take()
.expect("`serialize_value` called before `serialize_key`");
self.map.push((key, fast_serialize(value)?));
Ok(())
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
Ok(Value::Map(ValueMap(self.map)))
}
}
impl ser::SerializeStruct for DefaultSerializeMap {
type Ok = Value;
type Error = rbs::Error;
fn serialize_field<T: Serialize + ?Sized>(
&mut self,
key: &'static str,
value: &T,
) -> Result<(), Self::Error> {
self.map
.push((Value::String(key.to_string()), fast_serialize(value)?));
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(Value::Map(ValueMap(self.map)))
}
}
impl ser::SerializeStructVariant for DefaultSerializeMap {
type Ok = Value;
type Error = rbs::Error;
fn serialize_field<T: Serialize + ?Sized>(
&mut self,
key: &'static str,
value: &T,
) -> Result<(), Self::Error> {
self.map
.push((Value::String(key.to_string()), fast_serialize(value)?));
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(Value::Map(ValueMap(self.map)))
}
}
impl ser::SerializeStruct for SerializeVec {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_field<T: Serialize + ?Sized>(
&mut self,
_key: &'static str,
value: &T,
) -> Result<(), Self::Error> {
ser::SerializeSeq::serialize_element(self, value)
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
ser::SerializeSeq::end(self)
}
}
impl ser::SerializeStructVariant for SerializeStructVariant {
type Ok = Value;
type Error = rbs::Error;
#[inline]
fn serialize_field<T: Serialize + ?Sized>(
&mut self,
_key: &'static str,
value: &T,
) -> Result<(), Self::Error> {
self.vec.push(fast_serialize(value)?);
Ok(())
}
#[inline]
fn end(self) -> Result<Value, Self::Error> {
Ok(Value::Array(vec![
Value::from(self.idx),
Value::Array(self.vec),
]))
}
}
#[cfg(test)]
mod tests {
use crate::types::{DateTime, Hashed, Json, Uuid};
use super::*;
use serde::{Deserialize, Serialize};
use serde_json::json;
#[derive(Debug, PartialEq, Serialize, Deserialize)]
struct Test {
a: i32,
b: String,
c: Vec<u8>,
}
#[test]
fn test_serialize() {
let test = Test {
a: 1,
b: "test".to_string(),
c: vec![1, 2, 3],
};
assert_eq!(
fast_serialize(test).unwrap(),
rbs::to_value! {
"a" : 1,
"b" : "test",
"c" : [1u32, 2u32, 3u32],
}
);
}
#[derive(Debug, PartialEq, Serialize, Deserialize)]
enum Status {
Ok,
Error,
ThirdThing,
}
#[test]
fn test_serialize_enum() {
assert_eq!(fast_serialize(Status::Ok).unwrap(), rbs::to_value!("Ok"));
assert_eq!(
fast_serialize(Status::Error).unwrap(),
rbs::to_value!("Error")
);
assert_eq!(
fast_serialize(Status::ThirdThing).unwrap(),
rbs::to_value!("ThirdThing")
);
}
#[derive(Debug, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
enum StatusV2 {
Ok,
Error,
ThirdThing,
}
#[test]
fn test_serialize_enum_with_custom_config() {
assert_eq!(fast_serialize(StatusV2::Ok).unwrap(), rbs::to_value!("ok"));
assert_eq!(
fast_serialize(StatusV2::Error).unwrap(),
rbs::to_value!("error")
);
assert_eq!(
fast_serialize(StatusV2::ThirdThing).unwrap(),
rbs::to_value!("third_thing")
);
}
#[test]
fn properly_serializes_datetime() {
let datetime = DateTime::now();
assert_eq!(
fast_serialize(&datetime).unwrap(),
Value::Ext("DateTime", Box::new(rbs::to_value!(datetime.0)))
);
}
#[test]
fn properly_serializes_uuid() {
let uuid = Uuid::new();
assert_eq!(
fast_serialize(&uuid).unwrap(),
Value::Ext("Uuid", Box::new(Value::String(uuid.to_string())))
);
}
#[test]
fn properly_serializes_hashed() {
let hashed = Hashed::new("hello-world");
assert_eq!(
fast_serialize(&hashed).unwrap(),
Value::String(hashed.to_string())
);
}
#[test]
fn properly_serializes_json() {
let json = Json(json!({
"hello": "world",
"foo": "bar",
}));
assert_eq!(
fast_serialize(&json).unwrap(),
Value::Ext("Json", Box::new(Value::String(json.to_string())))
);
}
}