use serde::ser::{Serialize, Serializer};
use crate::language::{TomlArray, TomlTable, TomlValue};
use oak_core::OakError;
pub fn serialize<S, T>(value: &T, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
T: Serialize,
{
value.serialize(serializer)
}
pub fn to_string<T>(value: &T) -> Result<String, OakError>
where
T: Serialize,
{
let serializer = TomlSerializer {};
let toml_value = value.serialize(serializer)?;
Ok(format!("{}", toml_value))
}
struct TomlSerializer {}
impl Default for TomlSerializer {
fn default() -> Self {
Self {}
}
}
impl Serializer for TomlSerializer {
type Ok = TomlValue;
type Error = OakError;
type SerializeSeq = TomlSeqSerializer;
type SerializeTuple = TomlSeqSerializer;
type SerializeTupleStruct = TomlSeqSerializer;
type SerializeTupleVariant = TomlSeqSerializer;
type SerializeMap = TomlMapSerializer;
type SerializeStruct = TomlMapSerializer;
type SerializeStructVariant = TomlMapSerializer;
fn serialize_bool(self, v: bool) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Boolean(v))
}
fn serialize_i8(self, v: i8) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_i16(self, v: i16) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_i32(self, v: i32) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_i64(self, v: i64) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v))
}
fn serialize_u8(self, v: u8) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_u16(self, v: u16) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_u32(self, v: u32) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_u64(self, v: u64) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Integer(v as i64))
}
fn serialize_f32(self, v: f32) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Float(v as f64))
}
fn serialize_f64(self, v: f64) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Float(v))
}
fn serialize_char(self, v: char) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String(v.to_string()))
}
fn serialize_str(self, v: &str) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String(v.to_string()))
}
fn serialize_bytes(self, v: &[u8]) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String(String::from_utf8_lossy(v).to_string()))
}
fn serialize_none(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String("null".to_string()))
}
fn serialize_some<T>(self, value: &T) -> Result<Self::Ok, Self::Error>
where
T: ?Sized + Serialize,
{
value.serialize(self)
}
fn serialize_unit(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String("null".to_string()))
}
fn serialize_unit_struct(self, _name: &'static str) -> Result<Self::Ok, Self::Error> {
self.serialize_unit()
}
fn serialize_unit_variant(self, _name: &'static str, _variant_index: u32, variant: &'static str) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::String(variant.to_string()))
}
fn serialize_newtype_struct<T>(self, _name: &'static str, value: &T) -> Result<Self::Ok, Self::Error>
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<Self::Ok, Self::Error>
where
T: ?Sized + Serialize,
{
value.serialize(self)
}
fn serialize_seq(self, len: Option<usize>) -> Result<Self::SerializeSeq, Self::Error> {
Ok(TomlSeqSerializer { items: Vec::with_capacity(len.unwrap_or(0)) })
}
fn serialize_tuple(self, len: usize) -> Result<Self::SerializeTuple, Self::Error> {
Ok(TomlSeqSerializer { items: Vec::with_capacity(len) })
}
fn serialize_tuple_struct(self, _name: &'static str, len: usize) -> Result<Self::SerializeTupleStruct, Self::Error> {
Ok(TomlSeqSerializer { items: Vec::with_capacity(len) })
}
fn serialize_tuple_variant(self, _name: &'static str, _variant_index: u32, _variant: &'static str, len: usize) -> Result<Self::SerializeTupleVariant, Self::Error> {
Ok(TomlSeqSerializer { items: Vec::with_capacity(len) })
}
fn serialize_map(self, len: Option<usize>) -> Result<Self::SerializeMap, Self::Error> {
Ok(TomlMapSerializer { entries: std::collections::HashMap::with_capacity(len.unwrap_or(0)), current_key: None })
}
fn serialize_struct(self, _name: &'static str, len: usize) -> Result<Self::SerializeStruct, Self::Error> {
Ok(TomlMapSerializer { entries: std::collections::HashMap::with_capacity(len), current_key: None })
}
fn serialize_struct_variant(self, _name: &'static str, _variant_index: u32, _variant: &'static str, len: usize) -> Result<Self::SerializeStructVariant, Self::Error> {
Ok(TomlMapSerializer { entries: std::collections::HashMap::with_capacity(len), current_key: None })
}
}
struct TomlSeqSerializer {
items: Vec<TomlValue>,
}
impl serde::ser::SerializeSeq for TomlSeqSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_element<T>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let item = value.serialize(serializer)?;
self.items.push(item);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Array(TomlArray { list: self.items }))
}
}
impl serde::ser::SerializeTuple for TomlSeqSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_element<T>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let item = value.serialize(serializer)?;
self.items.push(item);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Array(TomlArray { list: self.items }))
}
}
impl serde::ser::SerializeTupleStruct for TomlSeqSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_field<T>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let item = value.serialize(serializer)?;
self.items.push(item);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Array(TomlArray { list: self.items }))
}
}
impl serde::ser::SerializeTupleVariant for TomlSeqSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_field<T>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let item = value.serialize(serializer)?;
self.items.push(item);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Array(TomlArray { list: self.items }))
}
}
struct TomlMapSerializer {
entries: std::collections::HashMap<String, TomlValue>,
current_key: Option<String>,
}
impl serde::ser::SerializeMap for TomlMapSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_key<T>(&mut self, key: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let key_value = key.serialize(serializer)?;
let key_str = key_value.to_string();
self.current_key = Some(key_str);
Ok(())
}
fn serialize_value<T>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let value_value = value.serialize(serializer)?;
if let Some(key) = self.current_key.take() {
self.entries.insert(key, value_value);
}
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Table(TomlTable { dict: self.entries }))
}
}
impl serde::ser::SerializeStruct for TomlMapSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_field<T>(&mut self, key: &'static str, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let value_value = value.serialize(serializer)?;
self.entries.insert(key.to_string(), value_value);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Table(TomlTable { dict: self.entries }))
}
}
impl serde::ser::SerializeStructVariant for TomlMapSerializer {
type Ok = TomlValue;
type Error = OakError;
fn serialize_field<T>(&mut self, key: &'static str, value: &T) -> Result<(), Self::Error>
where
T: ?Sized + Serialize,
{
let serializer = TomlSerializer {};
let value_value = value.serialize(serializer)?;
self.entries.insert(key.to_string(), value_value);
Ok(())
}
fn end(self) -> Result<Self::Ok, Self::Error> {
Ok(TomlValue::Table(TomlTable { dict: self.entries }))
}
}