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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
use crate::hex::FromHex;
use std::{fmt::Display, str};

pub trait Deserialize<'de>: Sized + FromHex + TryFrom<&'de [u8]> {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: serde::Deserializer<'de>;
}

impl<'de, T: FromHex + TryFrom<&'de [u8]>> Deserialize<'de> for T
where
    <T as TryFrom<&'de [u8]>>::Error: Display,
{
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: serde::Deserializer<'de>,
    {
        if deserializer.is_human_readable() {
            deserializer.deserialize_str(FromHexVisitor::default())
        } else {
            // serde::Deserialize for &[u8] is already optimized, so simply forward to that.
            serde::Deserialize::deserialize(deserializer).and_then(|bts| T::try_from(bts).map_err(serde::de::Error::custom))
        }
    }
}

pub struct FromHexVisitor<'de, T: FromHex> {
    marker: std::marker::PhantomData<T>,
    lifetime: std::marker::PhantomData<&'de ()>,
}

impl<'de, T: FromHex> Default for FromHexVisitor<'de, T> {
    fn default() -> Self {
        Self { marker: Default::default(), lifetime: Default::default() }
    }
}

impl<'de, T: FromHex> serde::de::Visitor<'de> for FromHexVisitor<'de, T> {
    type Value = T;

    fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
        write!(formatter, "string, str or slice, vec of bytes")
    }
    #[inline]
    fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        FromHex::from_hex(v).map_err(serde::de::Error::custom)
    }

    #[inline]
    fn visit_borrowed_str<E>(self, v: &'de str) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        FromHex::from_hex(v).map_err(serde::de::Error::custom)
    }

    #[inline]
    fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        FromHex::from_hex(&v).map_err(serde::de::Error::custom)
    }

    #[inline]
    fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        let str = str::from_utf8(v).map_err(serde::de::Error::custom)?;
        FromHex::from_hex(str).map_err(serde::de::Error::custom)
    }

    #[inline]
    fn visit_borrowed_bytes<E>(self, v: &'de [u8]) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        let str = str::from_utf8(v).map_err(serde::de::Error::custom)?;
        FromHex::from_hex(str).map_err(serde::de::Error::custom)
    }

    #[inline]
    fn visit_byte_buf<E>(self, v: Vec<u8>) -> Result<Self::Value, E>
    where
        E: serde::de::Error,
    {
        let str = str::from_utf8(&v).map_err(serde::de::Error::custom)?;
        FromHex::from_hex(str).map_err(serde::de::Error::custom)
    }
}