use crate::{
ast::{TomlRoot, TomlValueNode},
language::{TomlArray, TomlTable, TomlValue},
parse,
};
use oak_core::OakError;
use serde::de::{self, DeserializeSeed, Deserializer, MapAccess, SeqAccess, Visitor};
use std::{collections::HashMap, fmt::Formatter};
pub fn deserialize<'de, D>(deserializer: D) -> Result<TomlValue, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_any(TomlDeserializer {})
}
struct TomlDeserializer {}
impl<'de> Visitor<'de> for TomlDeserializer {
type Value = TomlValue;
fn expecting(&self, formatter: &mut Formatter) -> std::fmt::Result {
formatter.write_str("a valid TOML value")
}
fn visit_bool<E>(self, v: bool) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Boolean(v))
}
fn visit_i8<E>(self, v: i8) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_i16<E>(self, v: i16) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_i32<E>(self, v: i32) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_i64<E>(self, v: i64) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v))
}
fn visit_u8<E>(self, v: u8) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_u16<E>(self, v: u16) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_u32<E>(self, v: u32) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_u64<E>(self, v: u64) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Integer(v as i64))
}
fn visit_f32<E>(self, v: f32) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Float(v as f64))
}
fn visit_f64<E>(self, v: f64) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::Float(v))
}
fn visit_char<E>(self, v: char) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String(v.to_string()))
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String(v.to_string()))
}
fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String(v))
}
fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String(String::from_utf8_lossy(v).to_string()))
}
fn visit_byte_buf<E>(self, v: Vec<u8>) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String(String::from_utf8_lossy(&v).to_string()))
}
fn visit_none<E>(self) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String("null".to_string()))
}
fn visit_some<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_any(self)
}
fn visit_unit<E>(self) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(TomlValue::String("null".to_string()))
}
fn visit_newtype_struct<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_any(self)
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: SeqAccess<'de>,
{
let mut items = Vec::new();
while let Some(elem) = seq.next_element()? {
items.push(elem);
}
Ok(TomlValue::Array(TomlArray { list: items }))
}
fn visit_map<M>(self, mut map: M) -> Result<Self::Value, M::Error>
where
M: MapAccess<'de>,
{
let mut entries = HashMap::new();
while let Some((key, value)) = map.next_entry()? {
entries.insert(key, value);
}
Ok(TomlValue::Table(TomlTable { dict: entries }))
}
fn visit_enum<A>(self, _variant: A) -> Result<Self::Value, A::Error>
where
A: de::EnumAccess<'de>,
{
Err(de::Error::custom("enums are not supported"))
}
}
pub fn from_str<'de, T>(toml: &'de str) -> Result<T, OakError>
where
T: serde::Deserialize<'de>,
{
let root = crate::parse(toml)?;
let toml_value = root_into_value(root);
let deserializer = ValueDeserializerWrapper::new(toml_value);
T::deserialize(deserializer)
}
fn root_into_value(root: TomlRoot) -> TomlValue {
let mut table = HashMap::new();
for item in root.items {
if let crate::ast::TomlItem::KeyValue(kv) = item {
let key = kv.key.to_string();
let value = toml_value_node_into_value(kv.value);
table.insert(key, value);
}
}
TomlValue::Table(TomlTable { dict: table })
}
fn toml_value_node_into_value(value: TomlValueNode) -> TomlValue {
match value {
TomlValueNode::String(s) => TomlValue::String(s.value),
TomlValueNode::Integer(i) => TomlValue::Integer(i.value),
TomlValueNode::Float(f) => TomlValue::Float(f.value),
TomlValueNode::Boolean(b) => TomlValue::Boolean(b.value),
TomlValueNode::DateTime(dt) => TomlValue::DateTime(dt.value),
TomlValueNode::Array(a) => {
let items = a.items.into_iter().map(toml_value_node_into_value).collect();
TomlValue::Array(TomlArray { list: items })
}
TomlValueNode::InlineTable(t) => {
let mut table = HashMap::new();
for kv in t.items {
let key = kv.key.to_string();
let value = toml_value_node_into_value(kv.value);
table.insert(key, value);
}
TomlValue::Table(TomlTable { dict: table })
}
}
}
struct ValueDeserializerWrapper {
value: TomlValue,
}
impl ValueDeserializerWrapper {
fn new(value: TomlValue) -> Self {
Self { value }
}
}
impl<'de> serde::de::Deserializer<'de> for ValueDeserializerWrapper {
type Error = OakError;
fn deserialize_any<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
match self.value {
TomlValue::String(s) => visitor.visit_str(&s),
TomlValue::Integer(i) => visitor.visit_i64(i),
TomlValue::Float(f) => visitor.visit_f64(f),
TomlValue::Boolean(b) => visitor.visit_bool(b),
TomlValue::DateTime(dt) => visitor.visit_str(&dt),
TomlValue::Array(TomlArray { list: seq }) => visitor.visit_seq(ValueArrayAccess::new(seq)),
TomlValue::Table(TomlTable { dict: map }) => visitor.visit_map(ValueMapAccess::new(map)),
}
}
fn deserialize_bool<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_i8<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_i16<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_i32<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_i64<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_u8<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_u16<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_u32<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_u64<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_f32<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_f64<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_char<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_str<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_string<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_bytes<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_byte_buf<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_unit<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_unit_struct<V>(self, _name: &'static str, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_newtype_struct<V>(self, _name: &'static str, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_seq<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_tuple<V>(self, _len: usize, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_tuple_struct<V>(self, _name: &'static str, _len: usize, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_map<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_struct<V>(self, _name: &'static str, _fields: &'static [&'static str], visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_enum<V>(self, _name: &'static str, _variants: &'static [&'static str], _visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
Err(OakError::custom_error("Enums are not supported"))
}
fn deserialize_identifier<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
fn deserialize_ignored_any<V>(self, visitor: V) -> Result<V::Value, Self::Error>
where
V: Visitor<'de>,
{
self.deserialize_any(visitor)
}
}
struct ValueArrayAccess {
items: Vec<TomlValue>,
index: usize,
}
impl ValueArrayAccess {
fn new(items: Vec<TomlValue>) -> Self {
Self { items, index: 0 }
}
}
impl<'de> SeqAccess<'de> for ValueArrayAccess {
type Error = OakError;
fn next_element_seed<T>(&mut self, seed: T) -> Result<Option<T::Value>, Self::Error>
where
T: DeserializeSeed<'de>,
{
if self.index < self.items.len() {
let value = self.items[self.index].clone();
self.index += 1;
seed.deserialize(ValueDeserializerWrapper::new(value)).map(Some)
}
else {
Ok(None)
}
}
fn size_hint(&self) -> Option<usize> {
Some(self.items.len() - self.index)
}
}
struct ValueMapAccess {
entries: Vec<(String, TomlValue)>,
index: usize,
}
impl ValueMapAccess {
fn new(map: HashMap<String, TomlValue>) -> Self {
Self { entries: map.into_iter().collect(), index: 0 }
}
}
impl<'de> MapAccess<'de> for ValueMapAccess {
type Error = OakError;
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Self::Error>
where
K: DeserializeSeed<'de>,
{
if self.index < self.entries.len() {
let key = &self.entries[self.index].0;
seed.deserialize(serde::de::value::StrDeserializer::new(key)).map(Some)
}
else {
Ok(None)
}
}
fn next_value_seed<V>(&mut self, seed: V) -> Result<V::Value, Self::Error>
where
V: DeserializeSeed<'de>,
{
let value = self.entries[self.index].1.clone();
self.index += 1;
seed.deserialize(ValueDeserializerWrapper::new(value))
}
fn size_hint(&self) -> Option<usize> {
Some(self.entries.len() - self.index)
}
}