use serde::{de::Visitor, Deserialize, Serialize};
use std::collections::HashMap;
use crate::WalleError;
pub type ExtendedMap = HashMap<String, ExtendedValue>;
#[derive(Debug, Clone, Serialize, PartialEq)]
#[serde(untagged)]
pub enum ExtendedValue {
Str(String),
F64(f64),
Int(i64),
Bool(bool),
Map(HashMap<String, ExtendedValue>),
List(Vec<ExtendedValue>),
Bytes(Vec<u8>),
#[serde(serialize_with = "null_serialize")]
Null,
}
macro_rules! impl_from {
($inty: tt,$ty: ty) => {
impl From<$ty> for ExtendedValue {
fn from(v: $ty) -> Self {
ExtendedValue::$inty(v)
}
}
};
($inty: tt, $ty: ty, $ty0:ty) => {
impl From<$ty> for ExtendedValue {
fn from(v: $ty) -> Self {
ExtendedValue::$inty(v as $ty0)
}
}
};
}
impl_from!(Str, String);
impl_from!(Int, i64);
impl_from!(Int, i8, i64);
impl_from!(Int, i16, i64);
impl_from!(Int, i32, i64);
impl_from!(Int, u16, i64);
impl_from!(Int, u32, i64);
impl_from!(F64, f64);
impl_from!(F64, f32, f64);
impl_from!(Bool, bool);
impl_from!(Bytes, Vec<u8>);
impl From<&[u8]> for ExtendedValue {
fn from(v: &[u8]) -> Self {
ExtendedValue::Bytes(v.to_vec())
}
}
impl From<&str> for ExtendedValue {
fn from(v: &str) -> Self {
ExtendedValue::Str(v.to_owned())
}
}
impl<T> From<HashMap<String, T>> for ExtendedValue
where
T: Into<ExtendedValue>,
{
fn from(v: HashMap<String, T>) -> Self {
let mut map = HashMap::new();
for (k, v) in v {
map.insert(k, v.into());
}
ExtendedValue::Map(map)
}
}
impl<T> From<Vec<T>> for ExtendedValue
where
T: Into<ExtendedValue>,
{
fn from(v: Vec<T>) -> Self {
let mut list = Vec::new();
for v in v {
list.push(v.into());
}
ExtendedValue::List(list)
}
}
macro_rules! impl_try_from {
($inty: tt, $ty: ty) => {
impl TryFrom<ExtendedValue> for $ty {
type Error = ExtendedValue;
fn try_from(i: ExtendedValue) -> Result<$ty, Self::Error> {
match i {
ExtendedValue::$inty(v) => Ok(v),
_ => Err(i),
}
}
}
};
}
impl_try_from!(Str, String);
impl_try_from!(Int, i64);
impl_try_from!(F64, f64);
impl_try_from!(Bool, bool);
impl_try_from!(Map, ExtendedMap);
impl_try_from!(Bytes, Vec<u8>);
impl_try_from!(List, Vec<ExtendedValue>);
fn null_serialize<S>(serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_none()
}
struct ValueVisitor;
impl<'de> Visitor<'de> for ValueVisitor {
type Value = ExtendedValue;
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
formatter.write_str("accepts string, f64, int, bool, map, list, null")
}
fn visit_bool<E>(self, v: bool) -> Result<Self::Value, E> {
Ok(ExtendedValue::Bool(v))
}
fn visit_i64<E>(self, v: i64) -> Result<Self::Value, E> {
Ok(ExtendedValue::Int(v))
}
fn visit_u64<E>(self, v: u64) -> Result<Self::Value, E> {
Ok(ExtendedValue::Int(v as i64))
}
fn visit_f64<E>(self, v: f64) -> Result<Self::Value, E> {
Ok(ExtendedValue::F64(v))
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(ExtendedValue::Str(v.to_owned()))
}
fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(ExtendedValue::Bytes(v.to_owned()))
}
fn visit_byte_buf<E>(self, v: Vec<u8>) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(ExtendedValue::Bytes(v))
}
fn visit_seq<V>(self, mut seq: V) -> Result<Self::Value, V::Error>
where
V: serde::de::SeqAccess<'de>,
{
let mut values = Vec::new();
while let Some(value) = seq.next_element()? {
values.push(value);
}
Ok(ExtendedValue::List(values))
}
fn visit_map<V>(self, mut visitor: V) -> Result<Self::Value, V::Error>
where
V: serde::de::MapAccess<'de>,
{
let mut map = HashMap::new();
while let Some((key, value)) = visitor.next_entry()? {
map.insert(key, value);
}
Ok(ExtendedValue::Map(map))
}
fn visit_none<E>(self) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(ExtendedValue::Null)
}
fn visit_unit<E>(self) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(ExtendedValue::Null)
}
}
impl<'de> Deserialize<'de> for ExtendedValue {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
deserializer.deserialize_any(ValueVisitor)
}
}
macro_rules! downcast_fn {
($fname: ident, $ty: ty) => {
pub fn $fname(self) -> Result<$ty, Self> {
self.downcast()
}
};
}
#[allow(dead_code)]
impl ExtendedValue {
pub fn from_value<T: Into<ExtendedValue>>(t: T) -> ExtendedValue {
t.into()
}
pub fn empty_map() -> Self {
Self::Map(ExtendedMap::default())
}
pub fn downcast<T>(self) -> Result<T, Self>
where
T: TryFrom<ExtendedValue, Error = Self>,
{
self.try_into()
}
downcast_fn!(downcast_str, String);
downcast_fn!(downcast_int, i64);
downcast_fn!(downcast_f64, f64);
downcast_fn!(downcast_bool, bool);
downcast_fn!(downcast_bytes, Vec<u8>);
downcast_fn!(downcast_map, ExtendedMap);
downcast_fn!(downcast_list, Vec<ExtendedValue>);
pub fn as_str(&self) -> Option<&str> {
match self {
Self::Str(s) => Some(s),
_ => None,
}
}
pub fn as_f64(&self) -> Option<f64> {
match self {
Self::F64(f) => Some(*f),
_ => None,
}
}
pub fn as_i64(&self) -> Option<i64> {
match self {
Self::Int(i) => Some(*i),
_ => None,
}
}
pub fn as_bool(&self) -> Option<bool> {
match self {
Self::Bool(b) => Some(*b),
_ => None,
}
}
pub fn as_bytes(&self) -> Option<&[u8]> {
match self {
Self::Bytes(b) => Some(b),
_ => None,
}
}
pub fn as_map(&self) -> Option<&ExtendedMap> {
match self {
Self::Map(m) => Some(m),
_ => None,
}
}
pub fn as_list(&self) -> Option<&Vec<ExtendedValue>> {
match self {
Self::List(l) => Some(l),
_ => None,
}
}
pub fn is_str(&self) -> bool {
matches!(self, Self::Str(_))
}
pub fn is_f64(&self) -> bool {
matches!(self, Self::F64(_))
}
pub fn is_int(&self) -> bool {
matches!(self, Self::Int(_))
}
pub fn is_bool(&self) -> bool {
matches!(self, Self::Bool(_))
}
pub fn is_bytes(&self) -> bool {
matches!(self, Self::Bytes(_))
}
pub fn is_map(&self) -> bool {
matches!(self, Self::Map(_))
}
pub fn is_list(&self) -> bool {
matches!(self, Self::List(_))
}
pub fn is_null(&self) -> bool {
matches!(self, Self::Null)
}
}
pub trait ExtendedMapExt {
fn try_remove<T>(&mut self, key: &str) -> Result<T, WalleError>
where
T: TryFrom<ExtendedValue, Error = ExtendedValue>;
}
impl ExtendedMapExt for ExtendedMap {
fn try_remove<T>(&mut self, key: &str) -> Result<T, WalleError>
where
T: TryFrom<ExtendedValue, Error = ExtendedValue>,
{
let value = self
.remove(key)
.ok_or_else(|| WalleError::MapMissedKey(key.to_owned()))?;
T::try_from(value).map_err(|v| {
let msg = format!("{:?}", v);
self.insert(key.to_owned(), v);
WalleError::MapValueTypeMismatch(std::any::type_name::<T>().to_string(), msg)
})
}
}
#[macro_export]
macro_rules! extended_value {
(null) => {
$crate::ExtendedValue::Null
};
([$($tt:tt)*]) => {
$crate::ExtendedValue::List($crate::extended_vec![$($tt)*])
};
({$($tt:tt)*}) => {
$crate::ExtendedValue::Map($crate::extended_map!{$($tt)*})
};
($s:expr) => {
$crate::ExtendedValue::from_value($s.to_owned())
};
}
#[macro_export]
macro_rules! extended_vec {
(@internal [$($elems:expr),*]) => {
vec![$($elems),*]
};
(@internal [$($elems: expr,)*] null $($rest:tt)*) => {
extended_vec![@internal [$($elems,)* ExtendedValue::Null] $($rest)*]
};
(@internal [$($elems: expr,)*] [$($vec: tt)*] $($rest:tt)*) => {
extended_vec![@internal [$($elems,)* $crate::extended_value!([$($vec)*])] $($rest)*]
};
(@internal [$($elems: expr,)*] {$($map: tt)*} $($rest:tt)*) => {
extended_vec![@internal [$($elems,)* $crate::extended_value!({$($map)*})] $($rest)*]
};
(@internal [$($elems: expr,)*] $t:expr, $($rest:tt)*) => {
extended_vec![@internal [$($elems,)* $crate::extended_value!($t),] $($rest)*]
};
(@internal [$($elems: expr,)*] $t:expr) => {
extended_vec![@internal [$($elems,)* $crate::extended_value!($t)]]
};
(@internal [$($elems:expr),*] , $($rest:tt)*) => {
extended_vec![@internal [$($elems,)*] $($rest)*]
};
[$($tt: tt)*] => {
extended_vec!(@internal [] $($tt)*)
};
}
#[macro_export]
macro_rules! extended_map {
(@internal $map: ident {$key: expr} {$value: tt} ($($rest: tt)*)) => {
let _ = $map.insert($key.into(), $crate::extended_value!($value));
extended_map!(@internal $map () ($($rest)*));
};
(@internal $map: ident {$key: expr} {$value: tt}) => {
let _ = $map.insert($key.into(), $crate::extended_value!($value));
};
(@internal $map: ident {$key: expr} (: null $($rest:tt)*)) => {
extended_map!(@internal $map {$key} {null} ($($rest)*));
};
(@internal $map: ident {$key: expr} (: [$($vec: tt)*] $($rest:tt)*)) => {
extended_map!(@internal $map {$key} {[$($vec)*]} ($($rest)*));
};
(@internal $map: ident {$key: expr} (: {$($submap: tt)*} $($rest:tt)*)) => {
extended_map!(@internal $map {$key} {{$($submap)*}} ($($rest)*));
};
(@internal $map: ident {$key: expr} (: $value: expr , $($rest:tt)*)) => {
extended_map!(@internal $map {$key} {$value} ($($rest)*));
};
(@internal $map: ident {$key: expr} (: $value: expr)) => {
extended_map!(@internal $map {$key} {$value});
};
(@internal $map: ident () ($key: tt: $($rest:tt)*)) => {
extended_map!(@internal $map {$key} (: $($rest)*));
};
(@internal $map: ident () (, $($rest: tt)*)) => {
extended_map!(@internal $map () ($($rest)*));
};
(@internal $map: ident () ()) => {};
{$($tt:tt)*} => {
{
let mut map = $crate::ExtendedMap::default();
extended_map!(@internal map () ($($tt)*));
map
}
};
}
#[test]
fn macro_test() {
println!("{:?}", extended_value!(null));
println!(
"{:?}",
extended_vec![true, 1, "c", 3., [1, 2, 3], {"a": 1, "b": 2}, ExtendedValue::Bytes(vec![1, 2, 3])]
);
let a = "a";
println!("{:?}", extended_value!([1, "c", 3.]));
println!(
"{:?}",
extended_map! {
"a": a,
"b": 2,
"c": {
"d": 3,
"e": b"a"[..],
"f": null
}
}
);
}