diff options
| author | Alex Crichton <alex@alexcrichton.com> | 2017-04-28 15:01:08 -0500 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2017-04-28 15:01:08 -0500 | 
| commit | 95b3545938f67ca98d313be5c9c8930ee2407a30 (patch) | |
| tree | 4ec4a2484d45132da883a11b7a1a6700ba06480d | |
| parent | d8b6c83da660db49949bb484c5e3eeea9642cd8e (diff) | |
| parent | 082ee7090212e8a377b2145fe82712cc41431fee (diff) | |
| download | milf-rs-95b3545938f67ca98d313be5c9c8930ee2407a30.tar.gz milf-rs-95b3545938f67ca98d313be5c9c8930ee2407a30.zip | |
Merge pull request #165 from Keats/master
Allow to deserialize/serialize into enums
| -rw-r--r-- | src/de.rs | 48 | ||||
| -rw-r--r-- | src/ser.rs | 4 | ||||
| -rw-r--r-- | src/value.rs | 21 | ||||
| -rw-r--r-- | tests/serde.rs | 19 | 
4 files changed, 86 insertions, 6 deletions
| @@ -11,6 +11,7 @@ use std::str;  use std::vec;  use serde::de; +use serde::de::IntoDeserializer;  use tokens::{Tokenizer, Token, Error as TokenError};  use datetime::{SERDE_STRUCT_FIELD_NAME, SERDE_STRUCT_NAME}; @@ -121,6 +122,9 @@ enum ErrorKind {      /// type.      Custom, +    /// A struct was expected but something else was found +    ExpectedString, +      #[doc(hidden)]      __Nonexhaustive,  } @@ -145,6 +149,7 @@ impl<'de, 'b> de::Deserializer<'de> for &'b mut Deserializer<'de> {              values: None,              array: false,          }; +          while let Some(line) = self.line()? {              match line {                  Line::Table { at, mut header, array } => { @@ -192,9 +197,30 @@ impl<'de, 'b> de::Deserializer<'de> for &'b mut Deserializer<'de> {          })      } +    fn deserialize_enum<V>( +        self, +        _name: &'static str, +        _variants: &'static [&'static str], +        visitor: V +    ) -> Result<V::Value, Error> +        where V: de::Visitor<'de> +    { +        if let Some(next) = self.next()? { +            match next { +                Token::String { val, .. } => { +                    visitor.visit_enum(val.into_deserializer()) +                }, +                _ => Err(Error::from_kind(ErrorKind::ExpectedString)) +            } +        } else { +            Err(Error::from_kind(ErrorKind::UnexpectedEof)) +        } +    } + +      forward_to_deserialize_any! {          bool u8 u16 u32 u64 i8 i16 i32 i64 f32 f64 char str string seq -        bytes byte_buf map struct unit enum newtype_struct +        bytes byte_buf map struct unit newtype_struct          ignored_any unit_struct tuple_struct tuple option identifier      }  } @@ -489,10 +515,24 @@ impl<'de> de::Deserializer<'de> for ValueDeserializer<'de> {          visitor.visit_some(self)      } +    fn deserialize_enum<V>( +        self, +        _name: &'static str, +        _variants: &'static [&'static str], +        visitor: V +    ) -> Result<V::Value, Error> +        where V: de::Visitor<'de> +    { +        match self.value { +            Value::String(val) => visitor.visit_enum(val.into_deserializer()), +            _ => Err(Error::from_kind(ErrorKind::ExpectedString)) +        } +    } +      forward_to_deserialize_any! {          bool u8 u16 u32 u64 i8 i16 i32 i64 f32 f64 char str string seq          bytes byte_buf map unit newtype_struct identifier -        ignored_any unit_struct tuple_struct tuple enum +        ignored_any unit_struct tuple_struct tuple      }  } @@ -574,6 +614,7 @@ impl<'de> de::MapAccess<'de> for InlineTableDeserializer<'de> {      }  } +  impl<'a> Deserializer<'a> {      /// Creates a new deserializer which will be deserializing the string      /// provided. @@ -1092,6 +1133,7 @@ impl fmt::Display for Error {              ErrorKind::RedefineAsArray => "table redefined as array".fmt(f)?,              ErrorKind::EmptyTableKey => "empty table key found".fmt(f)?,              ErrorKind::Custom => self.inner.message.fmt(f)?, +            ErrorKind::ExpectedString => "expected string".fmt(f)?,              ErrorKind::__Nonexhaustive => panic!(),          } @@ -1134,6 +1176,7 @@ impl error::Error for Error {              ErrorKind::RedefineAsArray => "table redefined as array",              ErrorKind::EmptyTableKey => "empty table key found",              ErrorKind::Custom => "a custom error", +            ErrorKind::ExpectedString => "expected string",              ErrorKind::__Nonexhaustive => panic!(),          }      } @@ -1193,6 +1236,7 @@ impl<'a> Header<'a> {      }  } +#[derive(Debug)]  enum Value<'a> {      Integer(i64),      Float(f64), @@ -421,9 +421,9 @@ impl<'a, 'b> ser::Serializer for &'b mut Serializer<'a> {      fn serialize_unit_variant(self,                                _name: &'static str,                                _variant_index: u32, -                              _variant: &'static str) +                              variant: &'static str)                                -> Result<(), Self::Error> { -        Err(Error::UnsupportedType) +        self.serialize_str(variant)      }      fn serialize_newtype_struct<T: ?Sized>(self, _name: &'static str, value: &T) diff --git a/src/value.rs b/src/value.rs index 7c2b3eb..dcc3567 100644 --- a/src/value.rs +++ b/src/value.rs @@ -8,6 +8,7 @@ use std::vec;  use serde::ser;  use serde::de; +use serde::de::IntoDeserializer;  pub use datetime::{Datetime, DatetimeParseError};  use datetime::{DatetimeFromString, SERDE_STRUCT_FIELD_NAME}; @@ -505,6 +506,22 @@ impl<'de> de::Deserializer<'de> for Value {          }      } +    #[inline] +    fn deserialize_enum<V>( +        self, +        _name: &str, +        _variants: &'static [&'static str], +        visitor: V, +    ) -> Result<V::Value, ::de::Error> +    where +        V: de::Visitor<'de>, +    { +        match self { +            Value::String(variant) => visitor.visit_enum(variant.into_deserializer()), +            _ => Err(de::Error::invalid_type(de::Unexpected::UnitVariant, &"string only")), +        } +    } +      // `None` is interpreted as a missing field so be sure to implement `Some`      // as a present field.      fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, ::de::Error> @@ -516,7 +533,7 @@ impl<'de> de::Deserializer<'de> for Value {      forward_to_deserialize_any! {          bool u8 u16 u32 u64 i8 i16 i32 i64 f32 f64 char str string unit seq          bytes byte_buf map unit_struct tuple_struct struct -        tuple ignored_any enum newtype_struct identifier +        tuple ignored_any newtype_struct identifier      }  } @@ -694,7 +711,7 @@ impl ser::Serializer for Serializer {                                _variant_index: u32,                                _variant: &'static str)                                -> Result<Value, ::ser::Error> { -        Err(::ser::Error::UnsupportedType) +        self.serialize_str(_variant)      }      fn serialize_newtype_struct<T: ?Sized>(self, diff --git a/tests/serde.rs b/tests/serde.rs index bfbdc6f..0f4c37a 100644 --- a/tests/serde.rs +++ b/tests/serde.rs @@ -329,6 +329,25 @@ fn parse_enum() {      }  } +#[test] +fn parse_enum_string() { +    #[derive(Serialize, Deserialize, PartialEq, Debug, Clone)] +    struct Foo { a: Sort } + +    #[derive(Serialize, Deserialize, PartialEq, Debug, Clone)] +    #[serde(rename_all = "lowercase")] +    enum Sort { +        Asc, +        Desc, +    } + +    equivalent! { +        Foo { a: Sort::Desc }, +        Table(map! { a: Value::String("desc".to_string()) }), +    } + +} +  // #[test]  // fn unused_fields() {  //     #[derive(Serialize, Deserialize, PartialEq, Debug)] |