rune 0.14.2

The Rune Language, an embeddable dynamic programming language for Rust.
Documentation
use core::fmt;

use crate::alloc;
use crate::alloc::prelude::*;
use crate::runtime::{self, Bytes, Inline, Object, OwnedTuple, Repr, RttiKind, Vec};
use crate::TypeHash;

use serde::de::{self, Deserialize as _, Error as _};
use serde::ser::{self, Error as _, SerializeMap as _, SerializeSeq as _};

use super::Value;

/// Deserialize implementation for value pointers.
impl<'de> de::Deserialize<'de> for Value {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: de::Deserializer<'de>,
    {
        deserializer.deserialize_any(VmVisitor)
    }
}

/// Serialize implementation for value pointers.
impl ser::Serialize for Value {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: ser::Serializer,
    {
        match self.as_ref() {
            Repr::Inline(value) => match *value {
                Inline::Empty => Err(ser::Error::custom("cannot serialize empty values")),
                Inline::Unit => serializer.serialize_unit(),
                Inline::Bool(value) => serializer.serialize_bool(value),
                Inline::Char(value) => serializer.serialize_char(value),
                Inline::Unsigned(value) => serializer.serialize_u64(value),
                Inline::Signed(value) => serializer.serialize_i64(value),
                Inline::Float(value) => serializer.serialize_f64(value),
                Inline::Type(..) => Err(ser::Error::custom("cannot serialize types")),
                Inline::Ordering(..) => Err(ser::Error::custom("cannot serialize orderings")),
                Inline::Hash(..) => Err(ser::Error::custom("cannot serialize type hashes")),
            },
            Repr::Dynamic(value) => match value.rtti().kind {
                RttiKind::Empty => Err(ser::Error::custom(format!(
                    "cannot serialize empty struct {}",
                    value.rtti().item
                ))),
                RttiKind::Tuple => Err(ser::Error::custom(format!(
                    "cannot serialize tuple struct {}",
                    value.rtti().item
                ))),
                RttiKind::Struct => Err(ser::Error::custom(format!(
                    "cannot serialize struct {}",
                    value.rtti().item
                ))),
            },
            Repr::Any(value) => match value.type_hash() {
                Option::<Value>::HASH => {
                    let option = value
                        .borrow_ref::<Option<Value>>()
                        .map_err(S::Error::custom)?;
                    <Option<Value>>::serialize(&option, serializer)
                }
                String::HASH => {
                    let string = value.borrow_ref::<String>().map_err(S::Error::custom)?;
                    serializer.serialize_str(string.as_str())
                }
                Bytes::HASH => {
                    let bytes = value.borrow_ref::<Bytes>().map_err(S::Error::custom)?;
                    serializer.serialize_bytes(bytes.as_slice())
                }
                runtime::Vec::HASH => {
                    let vec = value.borrow_ref::<Vec>().map_err(S::Error::custom)?;
                    let mut serializer = serializer.serialize_seq(Some(vec.len()))?;

                    for value in vec.iter() {
                        serializer.serialize_element(value)?;
                    }

                    serializer.end()
                }
                OwnedTuple::HASH => {
                    let tuple = value.borrow_ref::<OwnedTuple>().map_err(S::Error::custom)?;
                    let mut serializer = serializer.serialize_seq(Some(tuple.len()))?;

                    for value in tuple.iter() {
                        serializer.serialize_element(value)?;
                    }

                    serializer.end()
                }
                Object::HASH => {
                    let object = value.borrow_ref::<Object>().map_err(S::Error::custom)?;
                    let mut serializer = serializer.serialize_map(Some(object.len()))?;

                    for (key, value) in object.iter() {
                        serializer.serialize_entry(key, value)?;
                    }

                    serializer.end()
                }
                _ => Err(ser::Error::custom("cannot serialize external references")),
            },
        }
    }
}

struct VmVisitor;

impl<'de> de::Visitor<'de> for VmVisitor {
    type Value = Value;

    #[inline]
    fn expecting(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
        fmt.write_str("any valid value")
    }

    #[inline]
    fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        let v = v.try_to_owned().map_err(E::custom)?;
        Value::try_from(v).map_err(E::custom)
    }

    #[inline]
    fn visit_string<E>(self, v: ::rust_alloc::string::String) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        let v = alloc::String::try_from(v).map_err(E::custom)?;
        Value::try_from(v).map_err(E::custom)
    }

    #[inline]
    fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        let v = alloc::Vec::try_from(v).map_err(E::custom)?;
        let v = Bytes::from_vec(v);
        Value::try_from(v).map_err(E::custom)
    }

    #[inline]
    fn visit_byte_buf<E>(self, v: ::rust_alloc::vec::Vec<u8>) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        let v = alloc::Vec::try_from(v).map_err(E::custom)?;
        let v = Bytes::from_vec(v);
        Value::try_from(v).map_err(E::custom)
    }

    #[inline]
    fn visit_i8<E>(self, v: i8) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_i16<E>(self, v: i16) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_i32<E>(self, v: i32) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_i64<E>(self, v: i64) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v))
    }

    #[inline]
    fn visit_i128<E>(self, v: i128) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_u8<E>(self, v: u8) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_u16<E>(self, v: u16) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_u32<E>(self, v: u32) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_u64<E>(self, v: u64) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_u128<E>(self, v: u128) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as i64))
    }

    #[inline]
    fn visit_f32<E>(self, v: f32) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v as f64))
    }

    #[inline]
    fn visit_f64<E>(self, v: f64) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v))
    }

    #[inline]
    fn visit_bool<E>(self, v: bool) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::from(v))
    }

    #[inline]
    fn visit_some<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
    where
        D: serde::Deserializer<'de>,
    {
        let some = Some(Value::deserialize(deserializer)?);
        Value::try_from(some).map_err(D::Error::custom)
    }

    #[inline]
    fn visit_none<E>(self) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Value::try_from(None).map_err(E::custom)
    }

    #[inline]
    fn visit_unit<E>(self) -> Result<Self::Value, E>
    where
        E: de::Error,
    {
        Ok(Value::unit())
    }

    #[inline]
    fn visit_seq<V>(self, mut visitor: V) -> Result<Self::Value, V::Error>
    where
        V: de::SeqAccess<'de>,
    {
        let mut vec = if let Some(hint) = visitor.size_hint() {
            alloc::Vec::try_with_capacity(hint).map_err(V::Error::custom)?
        } else {
            alloc::Vec::new()
        };

        while let Some(elem) = visitor.next_element()? {
            vec.try_push(elem).map_err(V::Error::custom)?;
        }

        let vec = Vec::from(vec);
        Value::try_from(vec).map_err(V::Error::custom)
    }

    #[inline]
    fn visit_map<V>(self, mut visitor: V) -> Result<Self::Value, V::Error>
    where
        V: de::MapAccess<'de>,
    {
        let mut object = Object::new();

        while let Some((key, value)) = visitor.next_entry()? {
            object.insert(key, value).map_err(V::Error::custom)?;
        }

        Value::try_from(object).map_err(V::Error::custom)
    }
}