ruma_events/
message.rs

1//! Types for extensible text message events ([MSC1767]).
2//!
3//! # Extensible events
4//!
5//! [MSC1767] defines a new structure for events that is made of two parts: a type and zero or more
6//! reusable content blocks.
7//!
8//! This allows to construct new event types from a list of known content blocks that allows in turn
9//! clients to be able to render unknown event types by using the known content blocks as a
10//! fallback. When a new type is defined, all the content blocks it can or must contain are defined
11//! too.
12//!
13//! There are also some content blocks called "mixins" that can apply to any event when they are
14//! defined.
15//!
16//! # MSCs
17//!
18//! This is a list of MSCs defining the extensible events and deprecating the corresponding legacy
19//! types. Note that "primary type" means the `type` field at the root of the event and "message
20//! type" means the `msgtype` field in the content of the `m.room.message` primary type.
21//!
22//! - [MSC1767][]: Text messages, where the `m.message` primary type replaces the `m.text` message
23//!   type.
24//! - [MSC3954][]: Emotes, where the `m.emote` primary type replaces the `m.emote` message type.
25//! - [MSC3955][]: Automated events, where the `m.automated` mixin replaces the `m.notice` message
26//!   type.
27//! - [MSC3956][]: Encrypted events, where the `m.encrypted` primary type replaces the
28//!   `m.room.encrypted` primary type.
29//! - [MSC3551][]: Files, where the `m.file` primary type replaces the `m.file` message type.
30//! - [MSC3552][]: Images and Stickers, where the `m.image` primary type replaces the `m.image`
31//!   message type and the `m.sticker` primary type.
32//! - [MSC3553][]: Videos, where the `m.video` primary type replaces the `m.video` message type.
33//! - [MSC3927][]: Audio, where the `m.audio` primary type replaces the `m.audio` message type.
34//! - [MSC3488][]: Location, where the `m.location` primary type replaces the `m.location` message
35//!   type.
36//!
37//! There are also the following MSCs that introduce new features with extensible events:
38//!
39//! - [MSC3245][]: Voice Messages.
40//! - [MSC3246][]: Audio Waveform.
41//! - [MSC3381][]: Polls.
42//!
43//! # How to use them in Matrix
44//!
45//! The extensible events types are meant to be used separately than the legacy types. As such,
46//! their use is reserved for room versions that support it.
47//!
48//! Currently no stable room version supports extensible events so they can only be sent with
49//! unstable room versions that support them.
50//!
51//! An exception is made for some new extensible events types that don't have a legacy type. They
52//! can be used with stable room versions without support for extensible types, but they might be
53//! ignored by clients that have no support for extensible events. The types that support this must
54//! advertise it in their MSC.
55//!
56//! Note that if a room version supports extensible events, it doesn't support the legacy types
57//! anymore and those should be ignored. There is not yet a definition of the deprecated legacy
58//! types in extensible events rooms.
59//!
60//! # How to use them in Ruma
61//!
62//! First, you can enable the `unstable-extensible-events` feature from the `ruma` crate, that
63//! will enable all the MSCs for the extensible events that correspond to the legacy types. It
64//! is also possible to enable only the MSCs you want with the `unstable-mscXXXX` features (where
65//! `XXXX` is the number of the MSC). When enabling an MSC, all MSC dependencies are enabled at the
66//! same time to avoid issues.
67//!
68//! Currently the extensible events use the unstable prefixes as defined in the corresponding MSCs.
69//!
70//! [MSC1767]: https://github.com/matrix-org/matrix-spec-proposals/pull/1767
71//! [MSC3954]: https://github.com/matrix-org/matrix-spec-proposals/pull/3954
72//! [MSC3955]: https://github.com/matrix-org/matrix-spec-proposals/pull/3955
73//! [MSC3956]: https://github.com/matrix-org/matrix-spec-proposals/pull/3956
74//! [MSC3551]: https://github.com/matrix-org/matrix-spec-proposals/pull/3551
75//! [MSC3552]: https://github.com/matrix-org/matrix-spec-proposals/pull/3552
76//! [MSC3553]: https://github.com/matrix-org/matrix-spec-proposals/pull/3553
77//! [MSC3927]: https://github.com/matrix-org/matrix-spec-proposals/pull/3927
78//! [MSC3488]: https://github.com/matrix-org/matrix-spec-proposals/pull/3488
79//! [MSC3245]: https://github.com/matrix-org/matrix-spec-proposals/pull/3245
80//! [MSC3246]: https://github.com/matrix-org/matrix-spec-proposals/pull/3246
81//! [MSC3381]: https://github.com/matrix-org/matrix-spec-proposals/pull/3381
82use std::ops::Deref;
83
84use ruma_macros::EventContent;
85use serde::{Deserialize, Serialize};
86
87use super::room::message::Relation;
88#[cfg(feature = "unstable-msc4095")]
89use super::room::message::UrlPreview;
90
91pub(super) mod historical_serde;
92
93/// The payload for an extensible text message.
94///
95/// This is the new primary type introduced in [MSC1767] and should only be sent in rooms with a
96/// version that supports it. See the documentation of the [`message`] module for more information.
97///
98/// To construct a `MessageEventContent` with a custom [`TextContentBlock`], convert it with
99/// `MessageEventContent::from()` / `.into()`.
100///
101/// [MSC1767]: https://github.com/matrix-org/matrix-spec-proposals/pull/1767
102/// [`message`]: super::message
103#[derive(Clone, Debug, Serialize, Deserialize, EventContent)]
104#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
105#[ruma_event(type = "org.matrix.msc1767.message", kind = MessageLike, without_relation)]
106pub struct MessageEventContent {
107    /// The message's text content.
108    #[serde(rename = "org.matrix.msc1767.text", alias = "m.text")]
109    pub text: TextContentBlock,
110
111    /// Whether this message is automated.
112    #[cfg(feature = "unstable-msc3955")]
113    #[serde(
114        default,
115        skip_serializing_if = "ruma_common::serde::is_default",
116        rename = "org.matrix.msc1767.automated"
117    )]
118    pub automated: bool,
119
120    /// Information about related messages.
121    #[serde(
122        flatten,
123        skip_serializing_if = "Option::is_none",
124        deserialize_with = "crate::room::message::relation_serde::deserialize_relation"
125    )]
126    pub relates_to: Option<Relation<MessageEventContentWithoutRelation>>,
127
128    /// [MSC4095](https://github.com/matrix-org/matrix-spec-proposals/pull/4095)-style bundled url previews
129    #[cfg(feature = "unstable-msc4095")]
130    #[serde(
131        rename = "com.beeper.linkpreviews",
132        skip_serializing_if = "Option::is_none",
133        alias = "m.url_previews"
134    )]
135    pub url_previews: Option<Vec<UrlPreview>>,
136}
137
138impl MessageEventContent {
139    /// A convenience constructor to create a plain text message.
140    pub fn plain(body: impl Into<String>) -> Self {
141        Self {
142            text: TextContentBlock::plain(body),
143            #[cfg(feature = "unstable-msc3955")]
144            automated: false,
145            relates_to: None,
146            #[cfg(feature = "unstable-msc4095")]
147            url_previews: None,
148        }
149    }
150
151    /// A convenience constructor to create an HTML message.
152    pub fn html(body: impl Into<String>, html_body: impl Into<String>) -> Self {
153        Self {
154            text: TextContentBlock::html(body, html_body),
155            #[cfg(feature = "unstable-msc3955")]
156            automated: false,
157            relates_to: None,
158            #[cfg(feature = "unstable-msc4095")]
159            url_previews: None,
160        }
161    }
162
163    /// A convenience constructor to create a message from Markdown.
164    ///
165    /// The content includes an HTML message if some Markdown formatting was detected, otherwise
166    /// only a plain text message is included.
167    #[cfg(feature = "markdown")]
168    pub fn markdown(body: impl AsRef<str> + Into<String>) -> Self {
169        Self {
170            text: TextContentBlock::markdown(body),
171            #[cfg(feature = "unstable-msc3955")]
172            automated: false,
173            relates_to: None,
174            #[cfg(feature = "unstable-msc4095")]
175            url_previews: None,
176        }
177    }
178}
179
180impl From<TextContentBlock> for MessageEventContent {
181    fn from(text: TextContentBlock) -> Self {
182        Self {
183            text,
184            #[cfg(feature = "unstable-msc3955")]
185            automated: false,
186            relates_to: None,
187            #[cfg(feature = "unstable-msc4095")]
188            url_previews: None,
189        }
190    }
191}
192
193/// A block for text content with optional markup.
194///
195/// This is an array of [`TextRepresentation`].
196///
197/// To construct a `TextContentBlock` with custom MIME types, construct a `Vec<TextRepresentation>`
198/// first and use its `::from()` / `.into()` implementation.
199#[derive(Clone, Debug, Default, Serialize, Deserialize)]
200pub struct TextContentBlock(Vec<TextRepresentation>);
201
202impl TextContentBlock {
203    /// A convenience constructor to create a plain text message.
204    pub fn plain(body: impl Into<String>) -> Self {
205        Self(vec![TextRepresentation::plain(body)])
206    }
207
208    /// A convenience constructor to create an HTML message with a plain text fallback.
209    pub fn html(body: impl Into<String>, html_body: impl Into<String>) -> Self {
210        Self(vec![TextRepresentation::html(html_body), TextRepresentation::plain(body)])
211    }
212
213    /// A convenience constructor to create a message from Markdown.
214    ///
215    /// The content includes an HTML message if some Markdown formatting was detected, otherwise
216    /// only a plain text message is included.
217    #[cfg(feature = "markdown")]
218    pub fn markdown(body: impl AsRef<str> + Into<String>) -> Self {
219        let mut message = Vec::with_capacity(2);
220        if let Some(html_body) = TextRepresentation::markdown(&body) {
221            message.push(html_body);
222        }
223        message.push(TextRepresentation::plain(body));
224        Self(message)
225    }
226
227    /// Whether this content block is empty.
228    pub fn is_empty(&self) -> bool {
229        self.0.is_empty()
230    }
231
232    /// Get the plain text representation of this message.
233    pub fn find_plain(&self) -> Option<&str> {
234        self.iter()
235            .find(|content| content.mimetype == "text/plain")
236            .map(|content| content.body.as_ref())
237    }
238
239    /// Get the HTML representation of this message.
240    pub fn find_html(&self) -> Option<&str> {
241        self.iter()
242            .find(|content| content.mimetype == "text/html")
243            .map(|content| content.body.as_ref())
244    }
245}
246
247impl From<Vec<TextRepresentation>> for TextContentBlock {
248    fn from(representations: Vec<TextRepresentation>) -> Self {
249        Self(representations)
250    }
251}
252
253impl FromIterator<TextRepresentation> for TextContentBlock {
254    fn from_iter<T: IntoIterator<Item = TextRepresentation>>(iter: T) -> Self {
255        Self(iter.into_iter().collect())
256    }
257}
258
259impl Deref for TextContentBlock {
260    type Target = [TextRepresentation];
261
262    fn deref(&self) -> &Self::Target {
263        &self.0
264    }
265}
266
267/// Text content with optional markup.
268#[derive(Clone, Debug, Serialize, Deserialize)]
269#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
270pub struct TextRepresentation {
271    /// The MIME type of the `body`.
272    ///
273    /// This must follow the format defined in [RFC6838].
274    ///
275    /// [RFC6838]: https://datatracker.ietf.org/doc/html/rfc6838
276    #[serde(
277        default = "TextRepresentation::default_mimetype",
278        skip_serializing_if = "TextRepresentation::is_default_mimetype"
279    )]
280    pub mimetype: String,
281
282    /// The text content.
283    pub body: String,
284
285    /// The language of the text ([MSC3554]).
286    ///
287    /// This must be a valid language code according to [BCP 47](https://www.rfc-editor.org/rfc/bcp/bcp47.txt).
288    ///
289    /// This is optional and defaults to `en`.
290    ///
291    /// [MSC3554]: https://github.com/matrix-org/matrix-spec-proposals/pull/3554
292    #[cfg(feature = "unstable-msc3554")]
293    #[serde(
294        rename = "org.matrix.msc3554.lang",
295        default = "TextRepresentation::default_lang",
296        skip_serializing_if = "TextRepresentation::is_default_lang"
297    )]
298    pub lang: String,
299}
300
301impl TextRepresentation {
302    /// Creates a new `TextRepresentation` with the given MIME type and body.
303    pub fn new(mimetype: impl Into<String>, body: impl Into<String>) -> Self {
304        Self {
305            mimetype: mimetype.into(),
306            body: body.into(),
307            #[cfg(feature = "unstable-msc3554")]
308            lang: Self::default_lang(),
309        }
310    }
311
312    /// Creates a new plain text message body.
313    pub fn plain(body: impl Into<String>) -> Self {
314        Self::new("text/plain", body)
315    }
316
317    /// Creates a new HTML-formatted message body.
318    pub fn html(body: impl Into<String>) -> Self {
319        Self::new("text/html", body)
320    }
321
322    /// Creates a new HTML-formatted message body by parsing the Markdown in `body`.
323    ///
324    /// Returns `None` if no Markdown formatting was found.
325    #[cfg(feature = "markdown")]
326    pub fn markdown(body: impl AsRef<str>) -> Option<Self> {
327        use super::room::message::parse_markdown;
328
329        parse_markdown(body.as_ref()).map(Self::html)
330    }
331
332    fn default_mimetype() -> String {
333        "text/plain".to_owned()
334    }
335
336    fn is_default_mimetype(mime: &str) -> bool {
337        mime == "text/plain"
338    }
339
340    #[cfg(feature = "unstable-msc3554")]
341    fn default_lang() -> String {
342        "en".to_owned()
343    }
344
345    #[cfg(feature = "unstable-msc3554")]
346    fn is_default_lang(lang: &str) -> bool {
347        lang == "en"
348    }
349}