ruma_federation_api/serde/
v1_pdu.rsuse std::{fmt, marker::PhantomData};
use serde::{
de::{Deserialize, Deserializer, Error, IgnoredAny, SeqAccess, Visitor},
ser::{Serialize, SerializeSeq, Serializer},
};
pub(crate) fn serialize<T, S>(val: &T, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
T: Serialize,
{
let mut seq = serializer.serialize_seq(Some(2))?;
seq.serialize_element(&200)?;
seq.serialize_element(val)?;
seq.end()
}
pub(crate) fn deserialize<'de, T, D>(deserializer: D) -> Result<T, D::Error>
where
D: Deserializer<'de>,
T: Deserialize<'de>,
{
deserializer.deserialize_seq(PduVisitor { phantom: PhantomData })
}
struct PduVisitor<T> {
phantom: PhantomData<T>,
}
impl<'de, T> Visitor<'de> for PduVisitor<T>
where
T: Deserialize<'de>,
{
type Value = T;
fn expecting(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.write_str("a PDU wrapped in an array.")
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: SeqAccess<'de>,
{
let expected = "a two-element list in the response";
if seq.next_element::<IgnoredAny>()?.is_none() {
return Err(A::Error::invalid_length(0, &expected));
}
let val = seq.next_element()?.ok_or_else(|| A::Error::invalid_length(1, &expected))?;
while let Some(IgnoredAny) = seq.next_element()? {
}
Ok(val)
}
}
#[cfg(not(feature = "unstable-unspecified"))]
#[cfg(test)]
mod tests {
use assert_matches2::assert_matches;
use serde_json::json;
use super::{deserialize, serialize};
#[allow(deprecated)]
use crate::membership::create_join_event::v1::RoomState;
#[test]
fn deserialize_response() {
let response = json!([
200,
{
"origin": "example.com",
"auth_chain": [],
"state": []
}
]);
#[allow(deprecated)]
let RoomState { origin, auth_chain, state, event } = deserialize(response).unwrap();
assert_eq!(origin, "example.com");
assert_matches!(auth_chain.as_slice(), []);
assert_matches!(state.as_slice(), []);
assert_matches!(event, None);
}
#[test]
fn serialize_response() {
#[allow(deprecated)]
let room_state = RoomState {
origin: "matrix.org".into(),
auth_chain: Vec::new(),
state: Vec::new(),
event: None,
};
let serialized = serialize(&room_state, serde_json::value::Serializer).unwrap();
let expected = json!(
[
200,
{
"origin": "matrix.org",
"auth_chain": [],
"state": []
}
]
);
assert_eq!(serialized, expected);
}
#[test]
fn too_short_array() {
let json = json!([200]);
#[allow(deprecated)]
let failed_room_state = deserialize::<RoomState, _>(json);
assert_eq!(
failed_room_state.unwrap_err().to_string(),
"invalid length 1, expected a two-element list in the response"
);
}
#[test]
fn not_an_array() {
let json = json!({
"origin": "matrix.org",
"auth_chain": [],
"state": []
});
#[allow(deprecated)]
let failed_room_state = deserialize::<RoomState, _>(json);
assert_eq!(
failed_room_state.unwrap_err().to_string(),
"invalid type: map, expected a PDU wrapped in an array.",
);
}
#[test]
fn too_long_array() {
let json = json!([200, { "origin": "", "auth_chain": [], "state": [] }, 200]);
#[allow(deprecated)]
let RoomState { origin, auth_chain, state, event } = deserialize(json).unwrap();
assert_eq!(origin, "");
assert_matches!(auth_chain.as_slice(), []);
assert_matches!(state.as_slice(), []);
assert_matches!(event, None);
}
}