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}