use serde::de; use Value; use super::{Decoder, DecodeError, DecodeErrorKind}; use std::collections::BTreeMap; fn se2toml(err: de::value::Error, ty: &'static str) -> DecodeError { match err { de::value::Error::SyntaxError => de::Error::syntax(ty), de::value::Error::EndOfStreamError => de::Error::end_of_stream(), de::value::Error::MissingFieldError(s) => { DecodeError { field: Some(s.to_string()), kind: DecodeErrorKind::ExpectedField(Some(ty)), } }, de::value::Error::UnknownFieldError(s) => { DecodeError { field: Some(s.to_string()), kind: DecodeErrorKind::UnknownField, } }, } } impl de::Deserializer for Decoder { type Error = DecodeError; fn visit(&mut self, mut visitor: V) -> Result where V: de::Visitor { match self.toml.take() { Some(Value::String(s)) => { visitor.visit_string(s).map_err(|e| se2toml(e, "string")) } Some(Value::Integer(i)) => { visitor.visit_i64(i).map_err(|e| se2toml(e, "integer")) } Some(Value::Float(f)) => { visitor.visit_f64(f).map_err(|e| se2toml(e, "float")) } Some(Value::Boolean(b)) => { visitor.visit_bool(b).map_err(|e| se2toml(e, "bool")) } Some(Value::Datetime(s)) => { visitor.visit_string(s).map_err(|e| se2toml(e, "date")) } Some(Value::Array(a)) => { let len = a.len(); let iter = a.into_iter(); visitor.visit_seq(SeqDeserializer::new(iter, len, &mut self.toml)) } Some(Value::Table(t)) => { visitor.visit_map(MapVisitor { iter: t.into_iter(), de: self, key: None, value: None, }) } None => Err(de::Error::end_of_stream()), } } fn visit_isize(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_i8(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_i16(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_i32(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_i64(&mut self, mut visitor: V) -> Result where V: de::Visitor { match self.toml.take() { Some(Value::Integer(f)) => { visitor.visit_i64(f).map_err(|e| se2toml(e, "integer")) } ref found => Err(self.mismatch("integer", found)), } } fn visit_usize(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_u8(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_u16(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_u32(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_u64(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_i64(visitor) } fn visit_f32(&mut self, visitor: V) -> Result where V: de::Visitor { self.visit_f64(visitor) } fn visit_f64(&mut self, mut visitor: V) -> Result where V: de::Visitor { match self.toml.take() { Some(Value::Float(f)) => { visitor.visit_f64(f).map_err(|e| se2toml(e, "float")) } ref found => Err(self.mismatch("float", found)), } } fn visit_option(&mut self, mut visitor: V) -> Result where V: de::Visitor { if self.toml.is_none() { visitor.visit_none() } else { visitor.visit_some(self) } } fn visit_seq(&mut self, mut visitor: V) -> Result where V: de::Visitor, { if self.toml.is_none() { let iter = None::.into_iter(); let e = visitor.visit_seq(de::value::SeqDeserializer::new(iter, 0)); e.map_err(|e| se2toml(e, "array")) } else { self.visit(visitor) } } fn visit_enum(&mut self, _enum: &str, variants: &[&str], mut visitor: V) -> Result where V: de::EnumVisitor, { // When decoding enums, this crate takes the strategy of trying to // decode the current TOML as all of the possible variants, returning // success on the first one that succeeds. // // Note that fidelity of the errors returned here is a little nebulous, // but we try to return the error that had the relevant field as the // longest field. This way we hopefully match an error against what was // most likely being written down without losing too much info. let mut first_error = None::; for variant in 0..variants.len() { let mut de = VariantVisitor { de: self.sub_decoder(self.toml.clone(), ""), variant: variant, }; match visitor.visit(&mut de) { Ok(value) => { self.toml = de.de.toml; return Ok(value); } Err(e) => { if let Some(ref first) = first_error { let my_len = e.field.as_ref().map(|s| s.len()); let first_len = first.field.as_ref().map(|s| s.len()); if my_len <= first_len { continue } } first_error = Some(e); } } } Err(first_error.unwrap_or_else(|| self.err(DecodeErrorKind::NoEnumVariants))) } } struct VariantVisitor { de: Decoder, variant: usize, } impl de::VariantVisitor for VariantVisitor { type Error = DecodeError; fn visit_variant(&mut self) -> Result where V: de::Deserialize { use serde::de::value::ValueDeserializer; let mut de = self.variant.into_deserializer(); de::Deserialize::deserialize(&mut de).map_err(|e| se2toml(e, "variant")) } fn visit_unit(&mut self) -> Result<(), DecodeError> { de::Deserialize::deserialize(&mut self.de) } fn visit_newtype(&mut self) -> Result where T: de::Deserialize, { de::Deserialize::deserialize(&mut self.de) } fn visit_tuple(&mut self, _len: usize, visitor: V) -> Result where V: de::Visitor, { de::Deserializer::visit(&mut self.de, visitor) } fn visit_struct(&mut self, _fields: &'static [&'static str], visitor: V) -> Result where V: de::Visitor, { de::Deserializer::visit(&mut self.de, visitor) } } struct SeqDeserializer<'a, I> { iter: I, len: usize, toml: &'a mut Option, } impl<'a, I> SeqDeserializer<'a, I> where I: Iterator { fn new(iter: I, len: usize, toml: &'a mut Option) -> Self { SeqDeserializer { iter: iter, len: len, toml: toml, } } fn put_value_back(&mut self, v: Value) { *self.toml = self.toml.take().or(Some(Value::Array(Vec::new()))); match self.toml.as_mut().unwrap() { &mut Value::Array(ref mut a) => { a.push(v); }, _ => unreachable!(), } } } impl<'a, I> de::Deserializer for SeqDeserializer<'a, I> where I: Iterator, { type Error = DecodeError; fn visit(&mut self, mut visitor: V) -> Result where V: de::Visitor, { visitor.visit_seq(self) } } impl<'a, I> de::SeqVisitor for SeqDeserializer<'a, I> where I: Iterator { type Error = DecodeError; fn visit(&mut self) -> Result, DecodeError> where V: de::Deserialize { match self.iter.next() { Some(value) => { self.len -= 1; let mut de = Decoder::new(value); let v = try!(de::Deserialize::deserialize(&mut de)); if let Some(t) = de.toml { self.put_value_back(t); } Ok(Some(v)) } None => Ok(None), } } fn end(&mut self) -> Result<(), DecodeError> { if self.len == 0 { Ok(()) } else { Err(de::Error::end_of_stream()) } } fn size_hint(&self) -> (usize, Option) { (self.len, Some(self.len)) } } impl de::Error for DecodeError { fn syntax(_: &str) -> DecodeError { DecodeError { field: None, kind: DecodeErrorKind::SyntaxError } } fn end_of_stream() -> DecodeError { DecodeError { field: None, kind: DecodeErrorKind::EndOfStream } } fn missing_field(name: &'static str) -> DecodeError { DecodeError { field: Some(name.to_string()), kind: DecodeErrorKind::ExpectedField(None), } } fn unknown_field(name: &str) -> DecodeError { DecodeError { field: Some(name.to_string()), kind: DecodeErrorKind::UnknownField, } } } struct MapVisitor<'a, I> { iter: I, de: &'a mut Decoder, key: Option, value: Option, } impl<'a, I> MapVisitor<'a, I> { fn put_value_back(&mut self, v: Value) { self.de.toml = self.de.toml.take().or_else(|| { Some(Value::Table(BTreeMap::new())) }); match self.de.toml.as_mut().unwrap() { &mut Value::Table(ref mut t) => { t.insert(self.key.take().unwrap(), v); }, _ => unreachable!(), } } } impl<'a, I> de::MapVisitor for MapVisitor<'a, I> where I: Iterator { type Error = DecodeError; fn visit_key(&mut self) -> Result, DecodeError> where K: de::Deserialize { while let Some((k, v)) = self.iter.next() { let mut dec = self.de.sub_decoder(Some(Value::String(k.clone())), &k); self.key = Some(k); match de::Deserialize::deserialize(&mut dec) { Ok(val) => { self.value = Some(v); return Ok(Some(val)) } // If this was an unknown field, then we put the toml value // back into the map and keep going. Err(DecodeError {kind: DecodeErrorKind::UnknownField, ..}) => { self.put_value_back(v); } Err(e) => return Err(e), } } Ok(None) } fn visit_value(&mut self) -> Result where V: de::Deserialize { match self.value.take() { Some(t) => { let mut dec = { // Borrowing the key here because Rust doesn't have // non-lexical borrows yet. let key = match self.key { Some(ref key) => &**key, None => "" }; self.de.sub_decoder(Some(t), key) }; let v = try!(de::Deserialize::deserialize(&mut dec)); if let Some(t) = dec.toml { self.put_value_back(t); } Ok(v) }, None => Err(de::Error::end_of_stream()) } } fn end(&mut self) -> Result<(), DecodeError> { Ok(()) } fn missing_field(&mut self, field_name: &'static str) -> Result where V: de::Deserialize { // See if the type can deserialize from a unit. match de::Deserialize::deserialize(&mut UnitDeserializer) { Err(DecodeError { kind: DecodeErrorKind::SyntaxError, field, }) => Err(DecodeError { field: field.or(Some(field_name.to_string())), kind: DecodeErrorKind::ExpectedField(None), }), v => v, } } } struct UnitDeserializer; impl de::Deserializer for UnitDeserializer { type Error = DecodeError; fn visit(&mut self, mut visitor: V) -> Result where V: de::Visitor, { visitor.visit_unit() } fn visit_option(&mut self, mut visitor: V) -> Result where V: de::Visitor, { visitor.visit_none() } } impl de::Deserialize for Value { fn deserialize(deserializer: &mut D) -> Result where D: de::Deserializer { struct ValueVisitor; impl de::Visitor for ValueVisitor { type Value = Value; fn visit_bool(&mut self, value: bool) -> Result { Ok(Value::Boolean(value)) } fn visit_i64(&mut self, value: i64) -> Result { Ok(Value::Integer(value)) } fn visit_f64(&mut self, value: f64) -> Result { Ok(Value::Float(value)) } fn visit_str(&mut self, value: &str) -> Result { Ok(Value::String(value.into())) } fn visit_string(&mut self, value: String) -> Result { Ok(Value::String(value)) } fn visit_seq(&mut self, visitor: V) -> Result where V: de::SeqVisitor { let values = try!(de::impls::VecVisitor::new().visit_seq(visitor)); Ok(Value::Array(values)) } fn visit_map(&mut self, visitor: V) -> Result where V: de::MapVisitor { let mut v = de::impls::BTreeMapVisitor::new(); let values = try!(v.visit_map(visitor)); Ok(Value::Table(values)) } } deserializer.visit(ValueVisitor) } }