ruma_appservice_api/event/
push_events.rs

1//! `PUT /_matrix/app/*/transactions/{txnId}`
2//!
3//! Endpoint to push an event (or batch of events) to the application service.
4
5pub mod v1 {
6    //! `/v1/` ([spec])
7    //!
8    //! [spec]: https://spec.matrix.org/latest/application-service-api/#put_matrixappv1transactionstxnid
9
10    use std::borrow::Cow;
11    #[cfg(feature = "unstable-msc3202")]
12    use std::collections::BTreeMap;
13
14    #[cfg(feature = "unstable-msc3202")]
15    use js_int::UInt;
16    #[cfg(feature = "unstable-msc3202")]
17    use ruma_common::OwnedUserId;
18    use ruma_common::{
19        api::{request, response, Metadata},
20        metadata,
21        serde::{from_raw_json_value, JsonObject, Raw},
22        OwnedTransactionId,
23    };
24    #[cfg(feature = "unstable-msc3202")]
25    use ruma_common::{OneTimeKeyAlgorithm, OwnedDeviceId};
26    #[cfg(feature = "unstable-msc4203")]
27    use ruma_events::AnyToDeviceEvent;
28    use ruma_events::{
29        presence::PresenceEvent, receipt::ReceiptEvent, typing::TypingEvent, AnyTimelineEvent,
30    };
31    use serde::{Deserialize, Deserializer, Serialize};
32    use serde_json::value::{RawValue as RawJsonValue, Value as JsonValue};
33
34    const METADATA: Metadata = metadata! {
35        method: PUT,
36        rate_limited: false,
37        authentication: AccessToken,
38        history: {
39            1.0 => "/_matrix/app/v1/transactions/:txn_id",
40        }
41    };
42
43    /// Request type for the `push_events` endpoint.
44    #[request]
45    pub struct Request {
46        /// The transaction ID for this set of events.
47        ///
48        /// Homeservers generate these IDs and they are used to ensure idempotency of results.
49        #[ruma_api(path)]
50        pub txn_id: OwnedTransactionId,
51
52        /// A list of events.
53        pub events: Vec<Raw<AnyTimelineEvent>>,
54
55        /// Information on E2E device updates.
56        #[cfg(feature = "unstable-msc3202")]
57        #[serde(
58            default,
59            skip_serializing_if = "DeviceLists::is_empty",
60            rename = "org.matrix.msc3202.device_lists"
61        )]
62        pub device_lists: DeviceLists,
63
64        /// The number of unclaimed one-time keys currently held on the server for this device, for
65        /// each algorithm.
66        #[cfg(feature = "unstable-msc3202")]
67        #[serde(
68            default,
69            skip_serializing_if = "BTreeMap::is_empty",
70            rename = "org.matrix.msc3202.device_one_time_keys_count"
71        )]
72        pub device_one_time_keys_count:
73            BTreeMap<OwnedUserId, BTreeMap<OwnedDeviceId, BTreeMap<OneTimeKeyAlgorithm, UInt>>>,
74
75        /// A list of key algorithms for which the server has an unused fallback key for the
76        /// device.
77        #[cfg(feature = "unstable-msc3202")]
78        #[serde(
79            default,
80            skip_serializing_if = "BTreeMap::is_empty",
81            rename = "org.matrix.msc3202.device_unused_fallback_key_types"
82        )]
83        pub device_unused_fallback_key_types:
84            BTreeMap<OwnedUserId, BTreeMap<OwnedDeviceId, Vec<OneTimeKeyAlgorithm>>>,
85
86        /// A list of ephemeral data.
87        #[serde(default, skip_serializing_if = "<[_]>::is_empty")]
88        pub ephemeral: Vec<EphemeralData>,
89
90        /// A list of to-device messages.
91        #[cfg(feature = "unstable-msc4203")]
92        #[serde(
93            default,
94            skip_serializing_if = "<[_]>::is_empty",
95            rename = "de.sorunome.msc2409.to_device"
96        )]
97        pub to_device: Vec<Raw<AnyToDeviceEvent>>,
98    }
99
100    /// Response type for the `push_events` endpoint.
101    #[response]
102    #[derive(Default)]
103    pub struct Response {}
104
105    impl Request {
106        /// Creates an `Request` with the given transaction ID and list of events.
107        pub fn new(txn_id: OwnedTransactionId, events: Vec<Raw<AnyTimelineEvent>>) -> Request {
108            Request {
109                txn_id,
110                events,
111                #[cfg(feature = "unstable-msc3202")]
112                device_lists: DeviceLists::new(),
113                #[cfg(feature = "unstable-msc3202")]
114                device_one_time_keys_count: BTreeMap::new(),
115                #[cfg(feature = "unstable-msc3202")]
116                device_unused_fallback_key_types: BTreeMap::new(),
117                ephemeral: Vec::new(),
118                #[cfg(feature = "unstable-msc4203")]
119                to_device: Vec::new(),
120            }
121        }
122    }
123
124    impl Response {
125        /// Creates an empty `Response`.
126        pub fn new() -> Self {
127            Self {}
128        }
129    }
130
131    /// Information on E2E device updates.
132    #[derive(Clone, Debug, Default, Deserialize, Serialize)]
133    #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
134    #[cfg(feature = "unstable-msc3202")]
135    pub struct DeviceLists {
136        /// List of users who have updated their device identity keys or who now
137        /// share an encrypted room with the client since the previous sync.
138        #[serde(default, skip_serializing_if = "Vec::is_empty")]
139        pub changed: Vec<OwnedUserId>,
140
141        /// List of users who no longer share encrypted rooms since the previous sync
142        /// response.
143        #[serde(default, skip_serializing_if = "Vec::is_empty")]
144        pub left: Vec<OwnedUserId>,
145    }
146
147    #[cfg(feature = "unstable-msc3202")]
148    impl DeviceLists {
149        /// Creates an empty `DeviceLists`.
150        pub fn new() -> Self {
151            Default::default()
152        }
153
154        /// Returns true if there are no device list updates.
155        pub fn is_empty(&self) -> bool {
156            self.changed.is_empty() && self.left.is_empty()
157        }
158    }
159
160    /// Type for passing ephemeral data to application services.
161    #[derive(Clone, Debug, Serialize)]
162    #[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
163    #[serde(untagged)]
164    pub enum EphemeralData {
165        /// A presence update for a user.
166        Presence(PresenceEvent),
167
168        /// A receipt update for a room.
169        Receipt(ReceiptEvent),
170
171        /// A typing notification update for a room.
172        Typing(TypingEvent),
173
174        #[doc(hidden)]
175        _Custom(_CustomEphemeralData),
176    }
177
178    impl EphemeralData {
179        /// A reference to the `type` string of the data.
180        pub fn data_type(&self) -> &str {
181            match self {
182                Self::Presence(_) => "m.presence",
183                Self::Receipt(_) => "m.receipt",
184                Self::Typing(_) => "m.typing",
185                Self::_Custom(c) => &c.data_type,
186            }
187        }
188
189        /// The data as a JSON object.
190        ///
191        /// Prefer to use the public variants of `EphemeralData` where possible; this method is
192        /// meant to be used for unsupported data types only.
193        pub fn data(&self) -> Cow<'_, JsonObject> {
194            fn serialize<T: Serialize>(obj: &T) -> JsonObject {
195                match serde_json::to_value(obj).expect("ephemeral data serialization to succeed") {
196                    JsonValue::Object(obj) => obj,
197                    _ => panic!("all ephemeral data types must serialize to objects"),
198                }
199            }
200
201            match self {
202                Self::Presence(d) => Cow::Owned(serialize(d)),
203                Self::Receipt(d) => Cow::Owned(serialize(d)),
204                Self::Typing(d) => Cow::Owned(serialize(d)),
205                Self::_Custom(c) => Cow::Borrowed(&c.data),
206            }
207        }
208    }
209
210    impl<'de> Deserialize<'de> for EphemeralData {
211        fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
212        where
213            D: Deserializer<'de>,
214        {
215            #[derive(Deserialize)]
216            struct EphemeralDataDeHelper {
217                /// The data type.
218                #[serde(rename = "type")]
219                data_type: String,
220            }
221
222            let json = Box::<RawJsonValue>::deserialize(deserializer)?;
223            let EphemeralDataDeHelper { data_type } = from_raw_json_value(&json)?;
224
225            Ok(match data_type.as_ref() {
226                "m.presence" => Self::Presence(from_raw_json_value(&json)?),
227                "m.receipt" => Self::Receipt(from_raw_json_value(&json)?),
228                "m.typing" => Self::Typing(from_raw_json_value(&json)?),
229                _ => Self::_Custom(_CustomEphemeralData {
230                    data_type,
231                    data: from_raw_json_value(&json)?,
232                }),
233            })
234        }
235    }
236
237    /// Ephemeral data with an unknown type.
238    #[doc(hidden)]
239    #[derive(Debug, Clone)]
240    pub struct _CustomEphemeralData {
241        /// The type of the data.
242        data_type: String,
243        /// The data.
244        data: JsonObject,
245    }
246
247    impl Serialize for _CustomEphemeralData {
248        fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
249        where
250            S: serde::Serializer,
251        {
252            self.data.serialize(serializer)
253        }
254    }
255
256    #[cfg(test)]
257    mod tests {
258        use assert_matches2::assert_matches;
259        use js_int::uint;
260        use ruma_common::{event_id, room_id, user_id, MilliSecondsSinceUnixEpoch};
261        use ruma_events::receipt::ReceiptType;
262        use serde_json::{from_value as from_json_value, json, to_value as to_json_value};
263
264        use super::EphemeralData;
265
266        #[cfg(feature = "client")]
267        #[test]
268        fn request_contains_events_field() {
269            use ruma_common::api::{OutgoingRequest, SendAccessToken};
270
271            let dummy_event_json = json!({
272                "type": "m.room.message",
273                "event_id": "$143273582443PhrSn:example.com",
274                "origin_server_ts": 1,
275                "room_id": "!roomid:room.com",
276                "sender": "@user:example.com",
277                "content": {
278                    "body": "test",
279                    "msgtype": "m.text",
280                },
281            });
282            let dummy_event = from_json_value(dummy_event_json.clone()).unwrap();
283            let events = vec![dummy_event];
284
285            let req = super::Request::new("any_txn_id".into(), events)
286                .try_into_http_request::<Vec<u8>>(
287                    "https://homeserver.tld",
288                    SendAccessToken::IfRequired("auth_tok"),
289                    &[ruma_common::api::MatrixVersion::V1_1],
290                )
291                .unwrap();
292            let json_body: serde_json::Value = serde_json::from_slice(req.body()).unwrap();
293
294            assert_eq!(
295                json_body,
296                json!({
297                    "events": [
298                        dummy_event_json,
299                    ]
300                })
301            );
302        }
303
304        #[test]
305        fn serde_ephemeral_data() {
306            let room_id = room_id!("!jEsUZKDJdhlrceRyVU:server.local");
307            let user_id = user_id!("@alice:server.local");
308            let event_id = event_id!("$1435641916114394fHBL");
309
310            // Test m.typing serde.
311            let typing_json = json!({
312                "type": "m.typing",
313                "room_id": room_id,
314                "content": {
315                    "user_ids": [user_id],
316                },
317            });
318
319            let data = from_json_value::<EphemeralData>(typing_json.clone()).unwrap();
320            assert_matches!(&data, EphemeralData::Typing(typing));
321            assert_eq!(typing.room_id, room_id);
322            assert_eq!(typing.content.user_ids, &[user_id.to_owned()]);
323
324            let serialized_data = to_json_value(data).unwrap();
325            assert_eq!(serialized_data, typing_json);
326
327            // Test m.receipt serde.
328            let receipt_json = json!({
329                "type": "m.receipt",
330                "room_id": room_id,
331                "content": {
332                    event_id: {
333                        "m.read": {
334                            user_id: {
335                                "ts": 453,
336                            },
337                        },
338                    },
339                },
340            });
341
342            let data = from_json_value::<EphemeralData>(receipt_json.clone()).unwrap();
343            assert_matches!(&data, EphemeralData::Receipt(receipt));
344            assert_eq!(receipt.room_id, room_id);
345            let event_receipts = receipt.content.get(event_id).unwrap();
346            let event_read_receipts = event_receipts.get(&ReceiptType::Read).unwrap();
347            let event_user_read_receipt = event_read_receipts.get(user_id).unwrap();
348            assert_eq!(event_user_read_receipt.ts, Some(MilliSecondsSinceUnixEpoch(uint!(453))));
349
350            let serialized_data = to_json_value(data).unwrap();
351            assert_eq!(serialized_data, receipt_json);
352
353            // Test m.presence serde.
354            let presence_json = json!({
355                "type": "m.presence",
356                "sender": user_id,
357                "content": {
358                    "avatar_url": "mxc://localhost/wefuiwegh8742w",
359                    "currently_active": false,
360                    "last_active_ago": 785,
361                    "presence": "online",
362                    "status_msg": "Making cupcakes",
363                },
364            });
365
366            let data = from_json_value::<EphemeralData>(presence_json.clone()).unwrap();
367            assert_matches!(&data, EphemeralData::Presence(presence));
368            assert_eq!(presence.sender, user_id);
369            assert_eq!(presence.content.currently_active, Some(false));
370
371            let serialized_data = to_json_value(data).unwrap();
372            assert_eq!(serialized_data, presence_json);
373
374            // Test custom serde.
375            let custom_json = json!({
376                "type": "dev.ruma.custom",
377                "key": "value",
378                "content": {
379                    "foo": "bar",
380                },
381            });
382
383            let data = from_json_value::<EphemeralData>(custom_json.clone()).unwrap();
384
385            let serialized_data = to_json_value(data).unwrap();
386            assert_eq!(serialized_data, custom_json);
387        }
388    }
389}