use serde::de::Unexpected::Other;
pub struct U32;
impl<'de> serde_with::DeserializeAs<'de, u32> for U32 {
fn deserialize_as<D>(deserializer: D) -> Result<u32, D::Error>
where
D: serde::de::Deserializer<'de>,
{
deserializer.deserialize_any(U32Visitor)
}
}
visitor_32!(
U32Visitor,
u32,
"a 32-bit unsigned integer in ProtoJSON format"
);
impl serde_with::SerializeAs<u32> for U32 {
fn serialize_as<S>(source: &u32, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_u32(*source)
}
}
#[cfg(test)]
mod tests {
use super::*;
use anyhow::Result;
use serde_json::{Value, json};
use serde_with::{DeserializeAs, SerializeAs};
use test_case::test_case;
#[test_case(0, 0)]
#[test_case(0.0, 0; "zero as f64")]
#[test_case("0", 0; "zero as string")]
#[test_case("0.0", 0; "zero as f64 string")]
#[test_case("2.0", 2)]
#[test_case(3e5, 300_000)]
#[test_case("5e4", 50_000)]
#[test_case(84, 84)]
#[test_case(168.0, 168)]
#[test_case("21", 21)]
#[test_case(u32::MAX, u32::MAX; "max")]
#[test_case(u32::MAX as f64, u32::MAX; "max as f64")]
#[test_case(format!("{}", u32::MAX), u32::MAX; "max as string")]
#[test_case(format!("{}.0", u32::MAX), u32::MAX; "max as f64 string")]
fn deser_and_ser<T: serde::Serialize>(input: T, want: u32) -> Result<()> {
let got = U32::deserialize_as(json!(input))?;
assert_eq!(got, want);
let serialized = U32::serialize_as(&got, serde_json::value::Serializer)?;
assert_eq!(serialized, json!(got));
Ok(())
}
#[test_case(json!(u64::MAX))]
#[test_case(json!(u32::MAX as i64 + 2))]
#[test_case(json!(u32::MIN as i64 - 2))]
#[test_case(json!(format!("{}", u64::MAX)))]
#[test_case(json!("abc"))]
#[test_case(json!(123.4))]
#[test_case(json!("234.5"))]
#[test_case(json!({}))]
fn deser_error(input: Value) {
let got = U32::deserialize_as(input).unwrap_err();
assert!(got.is_data(), "{got:?}");
}
}