1//! Types for persistent data unit schemas
2//!
3//! The differences between the `RoomV1Pdu` schema and the `RoomV3Pdu` schema are that the
4//! `RoomV1Pdu` takes an `event_id` field (`RoomV3Pdu` does not), and `auth_events` and
5//! `prev_events` take `Vec<(OwnedEventId, EventHash)>` rather than `Vec<OwnedEventId>` in
6//! `RoomV3Pdu`.
78use std::collections::BTreeMap;
910use js_int::UInt;
11use ruma_common::{
12 MilliSecondsSinceUnixEpoch, OwnedEventId, OwnedRoomId, OwnedUserId, ServerSignatures,
13};
14use serde::{
15 de::{Error as _, IgnoredAny},
16 Deserialize, Deserializer, Serialize,
17};
18use serde_json::{from_str as from_json_str, value::RawValue as RawJsonValue};
1920use super::TimelineEventType;
2122/// Enum for PDU schemas
23#[derive(Clone, Debug, Serialize)]
24#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
25#[serde(untagged)]
26pub enum Pdu {
27/// PDU for room versions 1 and 2.
28RoomV1Pdu(RoomV1Pdu),
2930/// PDU for room versions 3 and above.
31RoomV3Pdu(RoomV3Pdu),
32}
3334/// A 'persistent data unit' (event) for room versions 1 and 2.
35#[derive(Clone, Debug, Deserialize, Serialize)]
36#[allow(clippy::exhaustive_structs)]
37pub struct RoomV1Pdu {
38/// Event ID for the PDU.
39pub event_id: OwnedEventId,
4041/// The room this event belongs to.
42pub room_id: OwnedRoomId,
4344/// The user id of the user who sent this event.
45pub sender: OwnedUserId,
4647/// Timestamp (milliseconds since the UNIX epoch) on originating homeserver
48 /// of when this event was created.
49pub origin_server_ts: MilliSecondsSinceUnixEpoch,
5051// TODO: Encode event type as content enum variant, like event enums do
52/// The event's type.
53#[serde(rename = "type")]
54pub kind: TimelineEventType,
5556/// The event's content.
57pub content: Box<RawJsonValue>,
5859/// A key that determines which piece of room state the event represents.
60#[serde(skip_serializing_if = "Option::is_none")]
61pub state_key: Option<String>,
6263/// Event IDs for the most recent events in the room that the homeserver was
64 /// aware of when it created this event.
65#[serde(skip_serializing_if = "Vec::is_empty")]
66pub prev_events: Vec<(OwnedEventId, EventHash)>,
6768/// The maximum depth of the `prev_events`, plus one.
69pub depth: UInt,
7071/// Event IDs for the authorization events that would allow this event to be
72 /// in the room.
73#[serde(skip_serializing_if = "Vec::is_empty")]
74pub auth_events: Vec<(OwnedEventId, EventHash)>,
7576/// For redaction events, the ID of the event being redacted.
77#[serde(skip_serializing_if = "Option::is_none")]
78pub redacts: Option<OwnedEventId>,
7980/// Additional data added by the origin server but not covered by the
81 /// signatures.
82#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
83pub unsigned: BTreeMap<String, Box<RawJsonValue>>,
8485/// Content hashes of the PDU.
86pub hashes: EventHash,
8788/// Signatures for the PDU.
89pub signatures: ServerSignatures,
90}
9192/// A 'persistent data unit' (event) for room versions 3 and beyond.
93#[derive(Clone, Debug, Deserialize, Serialize)]
94#[allow(clippy::exhaustive_structs)]
95pub struct RoomV3Pdu {
96/// The room this event belongs to.
97pub room_id: OwnedRoomId,
9899/// The user id of the user who sent this event.
100pub sender: OwnedUserId,
101102/// Timestamp (milliseconds since the UNIX epoch) on originating homeserver
103 /// of when this event was created.
104pub origin_server_ts: MilliSecondsSinceUnixEpoch,
105106// TODO: Encode event type as content enum variant, like event enums do
107/// The event's type.
108#[serde(rename = "type")]
109pub kind: TimelineEventType,
110111/// The event's content.
112pub content: Box<RawJsonValue>,
113114/// A key that determines which piece of room state the event represents.
115#[serde(skip_serializing_if = "Option::is_none")]
116pub state_key: Option<String>,
117118/// Event IDs for the most recent events in the room that the homeserver was
119 /// aware of when it created this event.
120pub prev_events: Vec<OwnedEventId>,
121122/// The maximum depth of the `prev_events`, plus one.
123pub depth: UInt,
124125/// Event IDs for the authorization events that would allow this event to be
126 /// in the room.
127pub auth_events: Vec<OwnedEventId>,
128129/// For redaction events, the ID of the event being redacted.
130#[serde(skip_serializing_if = "Option::is_none")]
131pub redacts: Option<OwnedEventId>,
132133/// Additional data added by the origin server but not covered by the
134 /// signatures.
135#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
136pub unsigned: BTreeMap<String, Box<RawJsonValue>>,
137138/// Content hashes of the PDU.
139pub hashes: EventHash,
140141/// Signatures for the PDU.
142pub signatures: ServerSignatures,
143}
144145/// Content hashes of a PDU.
146#[derive(Clone, Debug, Deserialize, Serialize)]
147#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
148pub struct EventHash {
149/// The SHA-256 hash.
150pub sha256: String,
151}
152153impl EventHash {
154/// Create a new `EventHash` with the given SHA256 hash.
155pub fn new(sha256: String) -> Self {
156Self { sha256 }
157 }
158}
159160impl<'de> Deserialize<'de> for Pdu {
161fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
162where
163D: Deserializer<'de>,
164 {
165#[derive(Deserialize)]
166struct GetEventId {
167 event_id: Option<IgnoredAny>,
168 }
169170let json = Box::<RawJsonValue>::deserialize(deserializer)?;
171if from_json_str::<GetEventId>(json.get()).map_err(D::Error::custom)?.event_id.is_some() {
172 from_json_str(json.get()).map(Self::RoomV1Pdu).map_err(D::Error::custom)
173 } else {
174 from_json_str(json.get()).map(Self::RoomV3Pdu).map_err(D::Error::custom)
175 }
176 }
177}