//! De/Serialization of hexadecimal encoded bytes //! //! This modules is only available when using the `hex` feature of the crate. //! //! Please check the documentation on the [`Hex`] type for details. use crate::{ de::DeserializeAs, formats::{Format, Lowercase, Uppercase}, ser::SerializeAs, }; use alloc::{borrow::Cow, format, vec::Vec}; use core::{ convert::{TryFrom, TryInto}, marker::PhantomData, }; use serde::{de::Error, Deserialize, Deserializer, Serializer}; /// Serialize bytes as a hex string /// /// The type serializes a sequence of bytes as a hexadecimal string. /// It works on any type implementing `AsRef<[u8]>` for serialization and `TryFrom>` for deserialization. /// /// The format type parameter specifies if the hex string should use lower- or uppercase characters. /// Valid options are the types [`Lowercase`] and [`Uppercase`]. /// Deserialization always supports lower- and uppercase characters, even mixed in one string. /// /// # Example /// /// ```rust /// # #[cfg(feature = "macros")] { /// # use serde::{Deserialize, Serialize}; /// # use serde_json::json; /// # use serde_with::serde_as; /// # /// #[serde_as] /// # #[derive(Debug, PartialEq, Eq)] /// #[derive(Deserialize, Serialize)] /// struct BytesLowercase( /// // Equivalent to serde_with::hex::Hex /// #[serde_as(as = "serde_with::hex::Hex")] /// Vec /// ); /// /// #[serde_as] /// # #[derive(Debug, PartialEq, Eq)] /// #[derive(Deserialize, Serialize)] /// struct BytesUppercase( /// #[serde_as(as = "serde_with::hex::Hex")] /// Vec /// ); /// /// let b = b"Hello World!"; /// /// // Hex with lowercase letters /// assert_eq!( /// json!("48656c6c6f20576f726c6421"), /// serde_json::to_value(BytesLowercase(b.to_vec())).unwrap() /// ); /// // Hex with uppercase letters /// assert_eq!( /// json!("48656C6C6F20576F726C6421"), /// serde_json::to_value(BytesUppercase(b.to_vec())).unwrap() /// ); /// /// // Serialization always work from lower- and uppercase characters, even mixed case. /// assert_eq!( /// BytesLowercase(vec![0x00, 0xaa, 0xbc, 0x99, 0xff]), /// serde_json::from_value(json!("00aAbc99FF")).unwrap() /// ); /// assert_eq!( /// BytesUppercase(vec![0x00, 0xaa, 0xbc, 0x99, 0xff]), /// serde_json::from_value(json!("00aAbc99FF")).unwrap() /// ); /// /// #[serde_as] /// # #[derive(Debug, PartialEq, Eq)] /// #[derive(Deserialize, Serialize)] /// struct ByteArray( /// // Equivalent to serde_with::hex::Hex /// #[serde_as(as = "serde_with::hex::Hex")] /// [u8; 12] /// ); /// /// let b = b"Hello World!"; /// /// assert_eq!( /// json!("48656c6c6f20576f726c6421"), /// serde_json::to_value(ByteArray(b.clone())).unwrap() /// ); /// /// // Serialization always work from lower- and uppercase characters, even mixed case. /// assert_eq!( /// ByteArray([0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0xaa, 0xbc, 0x99, 0xff]), /// serde_json::from_value(json!("0011223344556677aAbc99FF")).unwrap() /// ); /// /// // Remember that the conversion may fail. (The following errors are specific to fixed-size arrays) /// let error_result: Result = serde_json::from_value(json!("42")); // Too short /// error_result.unwrap_err(); /// /// let error_result: Result = /// serde_json::from_value(json!("000000000000000000000000000000")); // Too long /// error_result.unwrap_err(); /// # } /// ``` #[derive(Copy, Clone, Debug, Default)] pub struct Hex(PhantomData); impl SerializeAs for Hex where T: AsRef<[u8]>, { fn serialize_as(source: &T, serializer: S) -> Result where S: Serializer, { serializer.serialize_str(&hex::encode(source)) } } impl SerializeAs for Hex where T: AsRef<[u8]>, { fn serialize_as(source: &T, serializer: S) -> Result where S: Serializer, { serializer.serialize_str(&hex::encode_upper(source)) } } impl<'de, T, FORMAT> DeserializeAs<'de, T> for Hex where T: TryFrom>, FORMAT: Format, { fn deserialize_as(deserializer: D) -> Result where D: Deserializer<'de>, { as Deserialize<'de>>::deserialize(deserializer) .and_then(|s| hex::decode(&*s).map_err(Error::custom)) .and_then(|vec: Vec| { let length = vec.len(); vec.try_into().map_err(|_e: T::Error| { Error::custom(format!( "Can't convert a Byte Vector of length {} to the output type.", length )) }) }) } }