use serde::{ de::{MapAccess, Visitor}, ser::SerializeMap, Deserialize, Serialize, }; use std::{ cmp::Ordering, collections::BTreeMap, fmt::{self, Display, Formatter}, marker::PhantomData, sync::{Arc, RwLock, RwLockReadGuard, RwLockWriteGuard}, }; use crate::{value::Value, Result, Structure, Type}; /// A collection dust variables comprised of key-value pairs. /// /// The inner value is a BTreeMap in order to allow VariableMap instances to be sorted and compared /// to one another. #[derive(Clone, Debug)] pub struct Map { variables: Arc>>, structure: Option, } impl Map { /// Creates a new instace. pub fn new() -> Self { Map { variables: Arc::new(RwLock::new(BTreeMap::new())), structure: None, } } pub fn clone_from(other: &Self) -> Result { let mut new_map = BTreeMap::new(); for (key, (value, r#type)) in other.variables()?.iter() { new_map.insert(key.clone(), (value.clone(), r#type.clone())); } Ok(Map { variables: Arc::new(RwLock::new(new_map)), structure: other.structure.clone(), }) } pub fn clone_complex_values_from(&self, other: &Self) -> Result<()> { for (key, (value, r#type)) in other.variables()?.iter() { if value.is_function() { self.variables_mut()? .insert(key.clone(), (value.clone(), r#type.clone())); } } Ok(()) } pub fn variables(&self) -> Result>> { Ok(self.variables.read()?) } pub fn variables_mut(&self) -> Result>> { Ok(self.variables.write()?) } pub fn set( &self, key: String, value: Value, r#type: Option, ) -> Result> { let value_type = r#type.unwrap_or(value.r#type()); let previous = self .variables .write()? .insert(key, (value, value_type.clone())); Ok(previous) } pub fn unset_all(&self) -> Result<()> { for (_key, (value, r#_type)) in self.variables.write()?.iter_mut() { *value = Value::none(); } Ok(()) } pub fn clear(&self) -> Result<()> { self.variables.write()?.clear(); Ok(()) } } impl Default for Map { fn default() -> Self { Self::new() } } impl Eq for Map {} impl PartialEq for Map { fn eq(&self, other: &Self) -> bool { let left = self.variables.read().unwrap().clone().into_iter(); let right = other.variables.read().unwrap().clone().into_iter(); left.eq(right) } } impl Ord for Map { fn cmp(&self, other: &Self) -> Ordering { let left = self.variables.read().unwrap().clone().into_iter(); let right = other.variables.read().unwrap().clone().into_iter(); left.cmp(right) } } impl PartialOrd for Map { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } } impl Display for Map { fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result { writeln!(f, "{{")?; let variables = self.variables.read().unwrap().clone().into_iter(); for (key, (value, _)) in variables { writeln!(f, " {key} = {value}")?; } write!(f, "}}") } } impl Serialize for Map { fn serialize(&self, serializer: S) -> std::result::Result where S: serde::Serializer, { let variables = self.variables.read().unwrap(); let mut map = serializer.serialize_map(Some(variables.len()))?; for (key, (value, _type)) in variables.iter() { map.serialize_entry(key, value)?; } map.end() } } struct MapVisitor { marker: PhantomData Map>, } impl MapVisitor { fn new() -> Self { MapVisitor { marker: PhantomData, } } } impl<'de> Visitor<'de> for MapVisitor { type Value = Map; fn expecting(&self, formatter: &mut Formatter) -> fmt::Result { formatter.write_str("Any valid whale data.") } fn visit_map(self, mut access: M) -> std::result::Result where M: MapAccess<'de>, { let map = Map::new(); { while let Some((key, value)) = access.next_entry::()? { map.set(key, value, None).unwrap(); } } Ok(map) } } impl<'de> Deserialize<'de> for Map { fn deserialize(deserializer: D) -> std::result::Result where D: serde::Deserializer<'de>, { deserializer.deserialize_any(MapVisitor::new()) } }