use std::fmt;
use data_encoding::BASE64URL_NOPAD;
use num_bigint::BigUint;
use serde::de;
use serde::{Deserializer, Serializer};
pub fn serialize<S>(value: &BigUint, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let bytes = value.to_bytes_be();
let base64 = BASE64URL_NOPAD.encode(bytes.as_slice());
serializer.serialize_str(&base64)
}
pub fn deserialize<'de, D>(deserializer: D) -> Result<BigUint, D::Error>
where
D: Deserializer<'de>,
{
struct BigUintVisitor;
impl<'de> de::Visitor<'de> for BigUintVisitor {
type Value = BigUint;
fn expecting(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.write_str("a Base64urlUInt string")
}
fn visit_str<E>(self, value: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
let bytes = BASE64URL_NOPAD
.decode(value.as_bytes())
.map_err(E::custom)?;
Ok(BigUint::from_bytes_be(&bytes))
}
}
deserializer.deserialize_str(BigUintVisitor)
}
pub struct Wrapper<'a>(&'a BigUint);
pub fn wrap(data: &BigUint) -> Wrapper<'_> {
Wrapper(data)
}
impl<'a> serde::Serialize for Wrapper<'a> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serialize(self.0, serializer)
}
}
#[cfg(test)]
mod tests {
use num_bigint::BigUint;
use num_traits::cast::FromPrimitive;
use serde::{Deserialize, Serialize};
use serde_test::{assert_tokens, Token};
#[derive(Serialize, Deserialize, Eq, PartialEq, Debug)]
struct TestStruct {
#[serde(with = "super")]
bytes: BigUint,
}
#[test]
fn serialization_round_trip() {
let test_value = TestStruct {
bytes: BigUint::from_u64(12345).unwrap(),
};
assert_tokens(
&test_value,
&[
Token::Struct {
name: "TestStruct",
len: 1,
},
Token::Str("bytes"),
Token::Str("MDk"),
Token::StructEnd,
],
);
}
}