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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
//! Integration between hex and serde
//!
//! # Examples
//!
//! ```
//! # extern crate hex_serde;
//! # #[macro_use]
//! # extern crate serde;
//! # #[macro_use]
//! # extern crate serde_derive;
//! #[derive(Serialize, Deserialize)]
//! struct Sha256(#[serde(with = "hex_serde")] [u8; 32]);
//!
//! #[derive(Serialize, Deserialize)]
//! struct MyStruct {
//!     #[serde(with = "hex_serde")] icecream: Vec<u8>,
//!     count: u64,
//! }
//! # fn main() {}
//! ```

extern crate hex;
extern crate serde;

use std::fmt;
use std::marker::PhantomData;
use hex::{FromHex, FromHexError, ToHex};
use serde::{de, Deserializer, Serializer};
use serde::de::Visitor;

/// A serializer that first encodes the argument as a hex-string
pub fn serialize<T, S>(value: &T, serializer: S) -> Result<S::Ok, S::Error>
where
    S: Serializer,
    T: AsRef<[u8]>,
{
    let mut output = String::new();
    value.write_hex(&mut output).expect("Failed to write hex");
    serializer.serialize_str(&output)
}

/// A deserializer that first encodes the argument as a hex-string
pub fn deserialize<'de, D, T>(deserializer: D) -> Result<T, D::Error>
where
    D: Deserializer<'de>,
    T: FromHex<Error = FromHexError>,
{
    struct HexVisitor<T>(PhantomData<T>);

    impl<'de, T> Visitor<'de> for HexVisitor<T>
    where
        T: FromHex<Error = FromHexError>,
    {
        type Value = T;

        fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
            write!(formatter, "hex ASCII text")
        }

        fn visit_str<E>(self, v: &str) -> Result<T, E>
        where
            E: de::Error,
        {
            T::from_hex(v).map_err(|e| match e {
                FromHexError::InvalidHexCharacter { c, index } => E::invalid_value(
                    de::Unexpected::Char(c),
                    &format!("Unexpected character {:?} as position {}", c, index).as_str(),
                ),
                FromHexError::InvalidStringLength => {
                    E::invalid_length(v.len(), &"Unexpected length of hex string")
                }
                FromHexError::OddLength => E::invalid_length(v.len(), &"Odd length of hex string"),
            })
        }
    }

    deserializer.deserialize_str(HexVisitor(PhantomData))
}