//! Deserializer of empty string values into optionals. use std::marker::PhantomData; use serde::{Deserialize, Deserializer, Error}; use serde::de::Visitor; use serde_json::Value; use serde_json::value; /// Deserializer of empty string values into optionals. #[derive(Debug, PartialEq)] pub enum MaybeEmpty { /// Some. Some(T), /// None. None, } impl Deserialize for MaybeEmpty where T: Deserialize { fn deserialize(deserializer: &mut D) -> Result where D: Deserializer { deserializer.deserialize(MaybeEmptyVisitor::new()) } } struct MaybeEmptyVisitor { _phantom: PhantomData } impl MaybeEmptyVisitor { fn new() -> Self { MaybeEmptyVisitor { _phantom: PhantomData } } } impl Visitor for MaybeEmptyVisitor where T: Deserialize { type Value = MaybeEmpty; fn visit_str(&mut self, value: &str) -> Result where E: Error { self.visit_string(value.to_owned()) } fn visit_string(&mut self, value: String) -> Result where E: Error { match value.is_empty() { true => Ok(MaybeEmpty::None), false => { let value = Value::String(value); T::deserialize(&mut value::Deserializer::new(value)).map(MaybeEmpty::Some).map_err(|_| Error::custom("failed")) } } } } impl Into> for MaybeEmpty { fn into(self) -> Option { match self { MaybeEmpty::Some(s) => Some(s), MaybeEmpty::None => None } } } #[cfg(test)] mod tests { use std::str::FromStr; use serde_json; use util::hash; use hash::H256; use maybe::MaybeEmpty; #[test] fn maybe_deserialization() { let s = r#"["", "5a39ed1020c04d4d84539975b893a4e7c53eab6c2965db8bc3468093a31bc5ae"]"#; let deserialized: Vec> = serde_json::from_str(s).unwrap(); assert_eq!(deserialized, vec![ MaybeEmpty::None, MaybeEmpty::Some(H256(hash::H256::from_str("5a39ed1020c04d4d84539975b893a4e7c53eab6c2965db8bc3468093a31bc5ae").unwrap())) ]); } }