1 // SPDX-License-Identifier: LGPL-2.1-or-later
2 // See Notices.txt for copyright information
4 use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
6 pub(crate) trait SerdeHex {
7 fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error>;
8 fn deserialize<'de, D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error>
13 #[derive(Deserialize, Serialize)]
14 struct SerdeHexWrapper<T: SerdeHex>(#[serde(with = "SerdeHex")] T);
16 fn serialize_ref_helper<T: SerdeHex, S: Serializer>(
19 ) -> Result<S::Ok, S::Error> {
20 v.serialize(serializer)
24 struct SerdeHexRefWrapper<'a, T: SerdeHex>(#[serde(serialize_with = "serialize_ref_helper")] &'a T);
26 impl<T: SerdeHex> SerdeHex for Option<T> {
27 fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
28 self.as_ref().map(SerdeHexRefWrapper).serialize(serializer)
30 fn deserialize<'de, D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error>
34 Ok(Option::<SerdeHexWrapper<T>>::deserialize(deserializer)?.map(|v| v.0))
38 macro_rules! impl_hex_for_uint {
40 impl SerdeHex for $ty {
41 fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
42 serializer.serialize_str(&format!("{:#X}", self))
44 fn deserialize<'de, D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
45 let text: &str = Deserialize::deserialize(deserializer)?;
46 const PREFIX: &str = "0x";
47 if text.starts_with(PREFIX) {
48 let hex_digits = &text[PREFIX.len()..];
49 Self::from_str_radix(hex_digits, 16).map_err(de::Error::custom)
51 Err(de::Error::custom("hexadecimal field must start with 0x"))
58 impl_hex_for_uint!(u8);
59 impl_hex_for_uint!(u16);
60 impl_hex_for_uint!(u32);
61 impl_hex_for_uint!(u64);
62 impl_hex_for_uint!(u128);