1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
extern crate redis;
extern crate serde;

// `encode` and `decode` are used instead of `ser` and `de` to avoid confusion with the serder
// Serializer and Deserializer traits which occupy a similar namespace.
pub mod decode;
pub mod encode;

pub use decode::Deserializer;
pub use encode::Serializer;

/// Use serde Deserialize to build `T` from a `redis::Value`
pub fn from_redis_value<'de, T>(rv: ::redis::Value) -> decode::Result<T>
where
    T: serde::de::Deserialize<'de>,
{
    ::serde::de::Deserialize::deserialize(Deserializer::new(rv))
}

pub trait RedisDeserialize<'de, T>
where
    T: serde::de::Deserialize<'de>,
{
    fn deserialize(self) -> decode::Result<T>;
}

impl<'de, T> RedisDeserialize<'de, T> for redis::Value
where
    T: serde::de::Deserialize<'de>,
{
    fn deserialize(self) -> decode::Result<T> {
        ::serde::de::Deserialize::deserialize(Deserializer::new(self))
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use redis::Value;

    #[test]
    fn chain_deserialize_works() {
        let v = Value::Bulk(vec![Value::Int(5), Value::Data(b"hello".to_vec())]);

        let actual: (u8, String) = v.deserialize().unwrap();
        let expected = (5, "hello".into());

        assert_eq!(expected, actual);
    }

    #[test]
    fn from_redis_value_works() {
        let v = Value::Bulk(vec![Value::Int(5), Value::Data(b"hello".to_vec())]);

        let actual: (u8, String) = from_redis_value(v).unwrap();
        let expected = (5, "hello".into());

        assert_eq!(expected, actual);
    }
}