1use as_variant::as_variant;
6use js_int::Int;
7#[cfg(feature = "canonical-json")]
8use ruma_common::canonical_json::RedactionEvent;
9use ruma_common::{
10 room_version_rules::RedactionRules, serde::CanBeEmpty, EventId, MilliSecondsSinceUnixEpoch,
11 OwnedEventId, OwnedRoomId, OwnedTransactionId, OwnedUserId, RoomId, RoomVersionId, UserId,
12};
13use ruma_macros::{Event, EventContent};
14use serde::{Deserialize, Serialize};
15use tracing::error;
16
17use crate::{
18 BundledMessageLikeRelations, EventContent, MessageLikeEventType, RedactContent,
19 RedactedMessageLikeEventContent, RedactedUnsigned, StaticEventContent,
20};
21
22mod event_serde;
23
24#[allow(clippy::exhaustive_enums)]
26#[derive(Clone, Debug)]
27pub enum RoomRedactionEvent {
28 Original(OriginalRoomRedactionEvent),
30
31 Redacted(RedactedRoomRedactionEvent),
33}
34
35#[allow(clippy::exhaustive_enums)]
37#[derive(Clone, Debug)]
38pub enum SyncRoomRedactionEvent {
39 Original(OriginalSyncRoomRedactionEvent),
41
42 Redacted(RedactedSyncRoomRedactionEvent),
44}
45
46#[derive(Clone, Debug)]
48#[allow(clippy::exhaustive_structs)]
49pub struct OriginalRoomRedactionEvent {
50 pub content: RoomRedactionEventContent,
52
53 pub redacts: Option<OwnedEventId>,
57
58 pub event_id: OwnedEventId,
60
61 pub sender: OwnedUserId,
63
64 pub origin_server_ts: MilliSecondsSinceUnixEpoch,
66
67 pub room_id: OwnedRoomId,
69
70 pub unsigned: RoomRedactionUnsigned,
72}
73
74impl From<OriginalRoomRedactionEvent> for OriginalSyncRoomRedactionEvent {
75 fn from(value: OriginalRoomRedactionEvent) -> Self {
76 let OriginalRoomRedactionEvent {
77 content,
78 redacts,
79 event_id,
80 sender,
81 origin_server_ts,
82 unsigned,
83 ..
84 } = value;
85
86 Self { content, redacts, event_id, sender, origin_server_ts, unsigned }
87 }
88}
89
90#[derive(Clone, Debug, Event)]
92#[allow(clippy::exhaustive_structs)]
93pub struct RedactedRoomRedactionEvent {
94 pub content: RedactedRoomRedactionEventContent,
96
97 pub event_id: OwnedEventId,
99
100 pub sender: OwnedUserId,
102
103 pub origin_server_ts: MilliSecondsSinceUnixEpoch,
105
106 pub room_id: OwnedRoomId,
108
109 pub unsigned: RedactedUnsigned,
111}
112
113#[derive(Clone, Debug)]
115#[allow(clippy::exhaustive_structs)]
116pub struct OriginalSyncRoomRedactionEvent {
117 pub content: RoomRedactionEventContent,
119
120 pub redacts: Option<OwnedEventId>,
124
125 pub event_id: OwnedEventId,
127
128 pub sender: OwnedUserId,
130
131 pub origin_server_ts: MilliSecondsSinceUnixEpoch,
133
134 pub unsigned: RoomRedactionUnsigned,
136}
137
138impl OriginalSyncRoomRedactionEvent {
139 pub fn into_full_event(self, room_id: OwnedRoomId) -> OriginalRoomRedactionEvent {
141 let Self { content, redacts, event_id, sender, origin_server_ts, unsigned } = self;
142
143 OriginalRoomRedactionEvent {
144 content,
145 redacts,
146 event_id,
147 sender,
148 origin_server_ts,
149 room_id,
150 unsigned,
151 }
152 }
153
154 pub(crate) fn into_maybe_redacted(self) -> SyncRoomRedactionEvent {
155 SyncRoomRedactionEvent::Original(self)
156 }
157}
158
159#[derive(Clone, Debug, Event)]
161#[allow(clippy::exhaustive_structs)]
162pub struct RedactedSyncRoomRedactionEvent {
163 pub content: RedactedRoomRedactionEventContent,
165
166 pub event_id: OwnedEventId,
168
169 pub sender: OwnedUserId,
171
172 pub origin_server_ts: MilliSecondsSinceUnixEpoch,
174
175 pub unsigned: RedactedUnsigned,
177}
178
179#[derive(Clone, Debug, Default, Deserialize, Serialize, EventContent)]
181#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
182#[ruma_event(type = "m.room.redaction", kind = MessageLike, custom_redacted)]
183pub struct RoomRedactionEventContent {
184 #[serde(skip_serializing_if = "Option::is_none")]
188 pub redacts: Option<OwnedEventId>,
189
190 #[serde(skip_serializing_if = "Option::is_none")]
192 pub reason: Option<String>,
193}
194
195impl RoomRedactionEventContent {
196 pub fn new_v1() -> Self {
198 Self::default()
199 }
200
201 pub fn new_v11(redacts: OwnedEventId) -> Self {
204 Self { redacts: Some(redacts), ..Default::default() }
205 }
206
207 pub fn with_reason(mut self, reason: String) -> Self {
209 self.reason = Some(reason);
210 self
211 }
212}
213
214impl RedactContent for RoomRedactionEventContent {
215 type Redacted = RedactedRoomRedactionEventContent;
216
217 fn redact(self, rules: &RedactionRules) -> Self::Redacted {
218 let redacts = self.redacts.filter(|_| rules.keep_room_redaction_redacts);
219 RedactedRoomRedactionEventContent { redacts }
220 }
221}
222
223#[derive(Clone, Debug, Default, Deserialize, Serialize)]
225#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
226pub struct RedactedRoomRedactionEventContent {
227 #[serde(skip_serializing_if = "Option::is_none")]
231 pub redacts: Option<OwnedEventId>,
232}
233
234impl EventContent for RedactedRoomRedactionEventContent {
235 type EventType = MessageLikeEventType;
236
237 fn event_type(&self) -> Self::EventType {
238 MessageLikeEventType::RoomRedaction
239 }
240}
241
242impl StaticEventContent for RedactedRoomRedactionEventContent {
243 const TYPE: &'static str = "m.room.redaction";
244}
245
246impl RedactedMessageLikeEventContent for RedactedRoomRedactionEventContent {}
247
248impl RoomRedactionEvent {
249 pub fn event_type(&self) -> MessageLikeEventType {
251 match self {
252 Self::Original(ev) => ev.content.event_type(),
253 Self::Redacted(ev) => ev.content.event_type(),
254 }
255 }
256
257 pub fn event_id(&self) -> &EventId {
259 match self {
260 Self::Original(ev) => &ev.event_id,
261 Self::Redacted(ev) => &ev.event_id,
262 }
263 }
264
265 pub fn sender(&self) -> &UserId {
267 match self {
268 Self::Original(ev) => &ev.sender,
269 Self::Redacted(ev) => &ev.sender,
270 }
271 }
272
273 pub fn origin_server_ts(&self) -> MilliSecondsSinceUnixEpoch {
275 match self {
276 Self::Original(ev) => ev.origin_server_ts,
277 Self::Redacted(ev) => ev.origin_server_ts,
278 }
279 }
280
281 pub fn room_id(&self) -> &RoomId {
283 match self {
284 Self::Original(ev) => &ev.room_id,
285 Self::Redacted(ev) => &ev.room_id,
286 }
287 }
288
289 pub fn redacts(&self, room_version: &RoomVersionId) -> Option<&EventId> {
296 match self {
297 Self::Original(ev) => Some(ev.redacts(room_version)),
298 Self::Redacted(ev) => ev.content.redacts.as_deref(),
299 }
300 }
301
302 pub fn as_original(&self) -> Option<&OriginalRoomRedactionEvent> {
304 as_variant!(self, Self::Original)
305 }
306}
307
308impl SyncRoomRedactionEvent {
309 pub fn event_type(&self) -> MessageLikeEventType {
311 match self {
312 Self::Original(ev) => ev.content.event_type(),
313 Self::Redacted(ev) => ev.content.event_type(),
314 }
315 }
316
317 pub fn event_id(&self) -> &EventId {
319 match self {
320 Self::Original(ev) => &ev.event_id,
321 Self::Redacted(ev) => &ev.event_id,
322 }
323 }
324
325 pub fn sender(&self) -> &UserId {
327 match self {
328 Self::Original(ev) => &ev.sender,
329 Self::Redacted(ev) => &ev.sender,
330 }
331 }
332
333 pub fn origin_server_ts(&self) -> MilliSecondsSinceUnixEpoch {
335 match self {
336 Self::Original(ev) => ev.origin_server_ts,
337 Self::Redacted(ev) => ev.origin_server_ts,
338 }
339 }
340
341 pub fn redacts(&self, room_version: &RoomVersionId) -> Option<&EventId> {
348 match self {
349 Self::Original(ev) => Some(ev.redacts(room_version)),
350 Self::Redacted(ev) => ev.content.redacts.as_deref(),
351 }
352 }
353
354 pub fn as_original(&self) -> Option<&OriginalSyncRoomRedactionEvent> {
356 as_variant!(self, Self::Original)
357 }
358
359 pub fn into_full_event(self, room_id: OwnedRoomId) -> RoomRedactionEvent {
361 match self {
362 Self::Original(ev) => RoomRedactionEvent::Original(ev.into_full_event(room_id)),
363 Self::Redacted(ev) => RoomRedactionEvent::Redacted(ev.into_full_event(room_id)),
364 }
365 }
366}
367
368impl From<RoomRedactionEvent> for SyncRoomRedactionEvent {
369 fn from(full: RoomRedactionEvent) -> Self {
370 match full {
371 RoomRedactionEvent::Original(ev) => Self::Original(ev.into()),
372 RoomRedactionEvent::Redacted(ev) => Self::Redacted(ev.into()),
373 }
374 }
375}
376
377impl OriginalRoomRedactionEvent {
378 pub fn redacts(&self, room_version: &RoomVersionId) -> &EventId {
389 redacts(room_version, self.redacts.as_deref(), self.content.redacts.as_deref())
390 }
391}
392
393impl OriginalSyncRoomRedactionEvent {
394 pub fn redacts(&self, room_version: &RoomVersionId) -> &EventId {
405 redacts(room_version, self.redacts.as_deref(), self.content.redacts.as_deref())
406 }
407}
408
409#[cfg(feature = "canonical-json")]
410impl RedactionEvent for OriginalRoomRedactionEvent {}
411#[cfg(feature = "canonical-json")]
412impl RedactionEvent for OriginalSyncRoomRedactionEvent {}
413#[cfg(feature = "canonical-json")]
414impl RedactionEvent for RoomRedactionEvent {}
415#[cfg(feature = "canonical-json")]
416impl RedactionEvent for SyncRoomRedactionEvent {}
417
418#[derive(Clone, Debug, Deserialize)]
420#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
421pub struct RoomRedactionUnsigned {
422 pub age: Option<Int>,
428
429 pub transaction_id: Option<OwnedTransactionId>,
432
433 #[serde(rename = "m.relations", default)]
437 pub relations: BundledMessageLikeRelations<OriginalSyncRoomRedactionEvent>,
438}
439
440impl RoomRedactionUnsigned {
441 pub fn new() -> Self {
443 Self { age: None, transaction_id: None, relations: BundledMessageLikeRelations::default() }
444 }
445}
446
447impl Default for RoomRedactionUnsigned {
448 fn default() -> Self {
449 Self::new()
450 }
451}
452
453impl CanBeEmpty for RoomRedactionUnsigned {
454 fn is_empty(&self) -> bool {
460 self.age.is_none() && self.transaction_id.is_none() && self.relations.is_empty()
461 }
462}
463
464fn redacts<'a>(
473 room_version: &'_ RoomVersionId,
474 redacts: Option<&'a EventId>,
475 content_redacts: Option<&'a EventId>,
476) -> &'a EventId {
477 match room_version {
478 RoomVersionId::V1
479 | RoomVersionId::V2
480 | RoomVersionId::V3
481 | RoomVersionId::V4
482 | RoomVersionId::V5
483 | RoomVersionId::V6
484 | RoomVersionId::V7
485 | RoomVersionId::V8
486 | RoomVersionId::V9
487 | RoomVersionId::V10 => redacts
488 .or_else(|| {
489 error!("Redacts field at event level not available, falling back to the one inside content");
490 content_redacts
491 })
492 .expect("At least one redacts field is set"),
493 _ => content_redacts
494 .or_else(|| {
495 error!("Redacts field inside content not available, falling back to the one at the event level");
496 redacts
497 })
498 .expect("At least one redacts field is set"),
499 }
500}