use std::future::Future;
use matrix_sdk_base::{deserialized_responses::SyncTimelineEvent, SendOutsideWasm};
use ruma::{
events::{
poll::unstable_start::{
ReplacementUnstablePollStartEventContent, UnstablePollStartContentBlock,
UnstablePollStartEventContent,
},
room::message::{
FormattedBody, MessageType, Relation, ReplacementMetadata, RoomMessageEventContent,
RoomMessageEventContentWithoutRelation,
},
AnyMessageLikeEvent, AnyMessageLikeEventContent, AnySyncMessageLikeEvent,
AnySyncTimelineEvent, AnyTimelineEvent, Mentions, MessageLikeEvent,
OriginalMessageLikeEvent, SyncMessageLikeEvent,
},
EventId, RoomId, UserId,
};
use thiserror::Error;
use tracing::{debug, instrument, trace, warn};
use crate::Room;
pub enum EditedContent {
RoomMessage(RoomMessageEventContentWithoutRelation),
MediaCaption {
caption: Option<String>,
formatted_caption: Option<FormattedBody>,
mentions: Option<Mentions>,
},
PollStart {
fallback_text: String,
new_content: UnstablePollStartContentBlock,
},
}
#[cfg(not(tarpaulin_include))]
impl std::fmt::Debug for EditedContent {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Self::RoomMessage(_) => f.debug_tuple("RoomMessage").finish(),
Self::MediaCaption { .. } => f.debug_tuple("MediaCaption").finish(),
Self::PollStart { .. } => f.debug_tuple("PollStart").finish(),
}
}
}
#[derive(Debug, Error)]
pub enum EditError {
#[error("State events can't be edited")]
StateEvent,
#[error("You're not the author of the event you'd like to edit.")]
NotAuthor,
#[error("Couldn't fetch the remote event: {0}")]
Fetch(Box<crate::Error>),
#[error(transparent)]
Deserialize(#[from] serde_json::Error),
#[error("The original event type ({target}) isn't the same as the parameter's new content type ({new_content})")]
IncompatibleEditType {
target: String,
new_content: &'static str,
},
}
impl Room {
#[instrument(skip(self, new_content), fields(room = %self.room_id()))]
pub async fn make_edit_event(
&self,
event_id: &EventId,
new_content: EditedContent,
) -> Result<AnyMessageLikeEventContent, EditError> {
make_edit_event(self, self.room_id(), self.own_user_id(), event_id, new_content).await
}
}
trait EventSource {
fn get_event(
&self,
event_id: &EventId,
) -> impl Future<Output = Result<SyncTimelineEvent, EditError>> + SendOutsideWasm;
}
impl EventSource for &Room {
async fn get_event(&self, event_id: &EventId) -> Result<SyncTimelineEvent, EditError> {
match self.event_cache().await {
Ok((event_cache, _drop_handles)) => {
if let Some(event) = event_cache.event(event_id).await {
return Ok(event);
}
}
Err(err) => {
debug!("error when getting the event cache: {err}");
}
}
trace!("trying with /event now");
self.event(event_id, None)
.await
.map(Into::into)
.map_err(|err| EditError::Fetch(Box::new(err)))
}
}
async fn make_edit_event<S: EventSource>(
source: S,
room_id: &RoomId,
own_user_id: &UserId,
event_id: &EventId,
new_content: EditedContent,
) -> Result<AnyMessageLikeEventContent, EditError> {
let target = source.get_event(event_id).await?;
let event = target.raw().deserialize().map_err(EditError::Deserialize)?;
let AnySyncTimelineEvent::MessageLike(message_like_event) = event else {
return Err(EditError::StateEvent);
};
if message_like_event.sender() != own_user_id {
return Err(EditError::NotAuthor);
}
match new_content {
EditedContent::RoomMessage(new_content) => {
let AnySyncMessageLikeEvent::RoomMessage(SyncMessageLikeEvent::Original(original)) =
message_like_event
else {
return Err(EditError::IncompatibleEditType {
target: message_like_event.event_type().to_string(),
new_content: "room message",
});
};
let mentions = original.content.mentions.clone();
let replied_to_original_room_msg =
extract_replied_to(source, room_id, original.content.relates_to).await;
let replacement = new_content.make_replacement(
ReplacementMetadata::new(event_id.to_owned(), mentions),
replied_to_original_room_msg.as_ref(),
);
Ok(replacement.into())
}
EditedContent::MediaCaption { caption, formatted_caption, mentions } => {
let AnySyncMessageLikeEvent::RoomMessage(SyncMessageLikeEvent::Original(original)) =
message_like_event
else {
return Err(EditError::IncompatibleEditType {
target: message_like_event.event_type().to_string(),
new_content: "caption for a media room message",
});
};
let original_mentions = original.content.mentions.clone();
let replied_to_original_room_msg =
extract_replied_to(source, room_id, original.content.relates_to.clone()).await;
let mut prev_content = original.content;
if !update_media_caption(&mut prev_content, caption, formatted_caption, mentions) {
return Err(EditError::IncompatibleEditType {
target: prev_content.msgtype.msgtype().to_owned(),
new_content: "caption for a media room message",
});
}
let replacement = prev_content.make_replacement(
ReplacementMetadata::new(event_id.to_owned(), original_mentions),
replied_to_original_room_msg.as_ref(),
);
Ok(replacement.into())
}
EditedContent::PollStart { fallback_text, new_content } => {
if !matches!(
message_like_event,
AnySyncMessageLikeEvent::UnstablePollStart(SyncMessageLikeEvent::Original(_))
) {
return Err(EditError::IncompatibleEditType {
target: message_like_event.event_type().to_string(),
new_content: "poll start",
});
}
let replacement = UnstablePollStartEventContent::Replacement(
ReplacementUnstablePollStartEventContent::plain_text(
fallback_text,
new_content,
event_id.to_owned(),
),
);
Ok(replacement.into())
}
}
}
macro_rules! set_caption {
($event:expr, $caption:expr) => {
let filename = $event.filename().to_owned();
if let Some(caption) = $caption {
$event.filename = Some(filename);
$event.body = caption;
} else {
$event.filename = None;
$event.body = filename;
}
};
}
pub(crate) fn update_media_caption(
content: &mut RoomMessageEventContent,
caption: Option<String>,
formatted_caption: Option<FormattedBody>,
mentions: Option<Mentions>,
) -> bool {
content.mentions = mentions;
match &mut content.msgtype {
MessageType::Audio(event) => {
set_caption!(event, caption);
event.formatted = formatted_caption;
true
}
MessageType::File(event) => {
set_caption!(event, caption);
event.formatted = formatted_caption;
true
}
MessageType::Image(event) => {
set_caption!(event, caption);
event.formatted = formatted_caption;
true
}
MessageType::Video(event) => {
set_caption!(event, caption);
event.formatted = formatted_caption;
true
}
_ => false,
}
}
async fn extract_replied_to<S: EventSource>(
source: S,
room_id: &RoomId,
relates_to: Option<Relation<RoomMessageEventContentWithoutRelation>>,
) -> Option<OriginalMessageLikeEvent<RoomMessageEventContent>> {
let replied_to_sync_timeline_event = if let Some(Relation::Reply { in_reply_to }) = relates_to {
source
.get_event(&in_reply_to.event_id)
.await
.map_err(|err| {
warn!("couldn't fetch the replied-to event, when editing: {err}");
err
})
.ok()
} else {
None
};
replied_to_sync_timeline_event
.and_then(|sync_timeline_event| {
sync_timeline_event
.raw()
.deserialize()
.map_err(|err| warn!("unable to deserialize replied-to event: {err}"))
.ok()
})
.and_then(|event| {
if let AnyTimelineEvent::MessageLike(AnyMessageLikeEvent::RoomMessage(
MessageLikeEvent::Original(original),
)) = event.into_full_event(room_id.to_owned())
{
Some(original)
} else {
None
}
})
}
#[cfg(test)]
mod tests {
use std::collections::BTreeMap;
use assert_matches2::{assert_let, assert_matches};
use matrix_sdk_base::deserialized_responses::SyncTimelineEvent;
use matrix_sdk_test::{async_test, event_factory::EventFactory};
use ruma::{
event_id,
events::{
room::message::{MessageType, Relation, RoomMessageEventContentWithoutRelation},
AnyMessageLikeEventContent, AnySyncTimelineEvent, Mentions,
},
owned_mxc_uri, owned_user_id, room_id,
serde::Raw,
user_id, EventId, OwnedEventId,
};
use serde_json::json;
use super::{make_edit_event, EditError, EventSource};
use crate::room::edit::EditedContent;
#[derive(Default)]
struct TestEventCache {
events: BTreeMap<OwnedEventId, SyncTimelineEvent>,
}
impl EventSource for TestEventCache {
async fn get_event(&self, event_id: &EventId) -> Result<SyncTimelineEvent, EditError> {
Ok(self.events.get(event_id).unwrap().clone())
}
}
#[async_test]
async fn test_edit_state_event() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let mut cache = TestEventCache::default();
cache.events.insert(
event_id.to_owned(),
SyncTimelineEvent::new(
Raw::<AnySyncTimelineEvent>::from_json_string(
json!({
"content": {
"name": "The room name"
},
"event_id": event_id,
"sender": own_user_id,
"state_key": "",
"origin_server_ts": 1,
"type": "m.room.name",
})
.to_string(),
)
.unwrap(),
),
);
let room_id = room_id!("!galette:saucisse.bzh");
let new_content = RoomMessageEventContentWithoutRelation::text_plain("the edit");
assert_matches!(
make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::RoomMessage(new_content),
)
.await,
Err(EditError::StateEvent)
);
}
#[async_test]
async fn test_edit_event_other_user() {
let event_id = event_id!("$1");
let f = EventFactory::new();
let mut cache = TestEventCache::default();
cache.events.insert(
event_id.to_owned(),
f.text_msg("hi").event_id(event_id).sender(user_id!("@other:saucisse.bzh")).into(),
);
let room_id = room_id!("!galette:saucisse.bzh");
let own_user_id = user_id!("@me:saucisse.bzh");
let new_content = RoomMessageEventContentWithoutRelation::text_plain("the edit");
assert_matches!(
make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::RoomMessage(new_content),
)
.await,
Err(EditError::NotAuthor)
);
}
#[async_test]
async fn test_make_edit_event_success() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let mut cache = TestEventCache::default();
let f = EventFactory::new();
cache.events.insert(
event_id.to_owned(),
f.text_msg("hi").event_id(event_id).sender(own_user_id).into(),
);
let room_id = room_id!("!galette:saucisse.bzh");
let new_content = RoomMessageEventContentWithoutRelation::text_plain("the edit");
let edit_event = make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::RoomMessage(new_content),
)
.await
.unwrap();
assert_let!(AnyMessageLikeEventContent::RoomMessage(msg) = &edit_event);
assert_eq!(msg.body(), "* the edit");
assert_let!(Some(Relation::Replacement(repl)) = &msg.relates_to);
assert_eq!(repl.event_id, event_id);
assert_eq!(repl.new_content.msgtype.body(), "the edit");
}
#[async_test]
async fn test_make_edit_caption_for_non_media_room_message() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let mut cache = TestEventCache::default();
let f = EventFactory::new();
cache.events.insert(
event_id.to_owned(),
f.text_msg("hello world").event_id(event_id).sender(own_user_id).into(),
);
let room_id = room_id!("!galette:saucisse.bzh");
let err = make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::MediaCaption {
caption: Some("yo".to_owned()),
formatted_caption: None,
mentions: None,
},
)
.await
.unwrap_err();
assert_let!(EditError::IncompatibleEditType { target, new_content } = err);
assert_eq!(target, "m.text");
assert_eq!(new_content, "caption for a media room message");
}
#[async_test]
async fn test_add_caption_for_media() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let filename = "rickroll.gif";
let mut cache = TestEventCache::default();
let f = EventFactory::new();
cache.events.insert(
event_id.to_owned(),
f.image(filename.to_owned(), owned_mxc_uri!("mxc://sdk.rs/rickroll"))
.event_id(event_id)
.sender(own_user_id)
.into(),
);
let room_id = room_id!("!galette:saucisse.bzh");
let edit_event = make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::MediaCaption {
caption: Some("Best joke ever".to_owned()),
formatted_caption: None,
mentions: None,
},
)
.await
.unwrap();
assert_let!(AnyMessageLikeEventContent::RoomMessage(msg) = edit_event);
assert_let!(MessageType::Image(image) = msg.msgtype);
assert_eq!(image.filename(), filename);
assert_eq!(image.caption(), Some("* Best joke ever")); assert!(image.formatted_caption().is_none());
assert_let!(Some(Relation::Replacement(repl)) = msg.relates_to);
assert_let!(MessageType::Image(new_image) = repl.new_content.msgtype);
assert_eq!(new_image.filename(), filename);
assert_eq!(new_image.caption(), Some("Best joke ever"));
assert!(new_image.formatted_caption().is_none());
}
#[async_test]
async fn test_remove_caption_for_media() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let filename = "rickroll.gif";
let mut cache = TestEventCache::default();
let f = EventFactory::new();
let event = f
.image(filename.to_owned(), owned_mxc_uri!("mxc://sdk.rs/rickroll"))
.caption(Some("caption".to_owned()), None)
.event_id(event_id)
.sender(own_user_id)
.into_sync();
{
let event = event.raw().deserialize().unwrap();
assert_let!(AnySyncTimelineEvent::MessageLike(event) = event);
assert_let!(
AnyMessageLikeEventContent::RoomMessage(msg) = event.original_content().unwrap()
);
assert_let!(MessageType::Image(image) = msg.msgtype);
assert_eq!(image.filename(), filename);
assert_eq!(image.caption(), Some("caption"));
assert!(image.formatted_caption().is_none());
}
cache.events.insert(event_id.to_owned(), event);
let room_id = room_id!("!galette:saucisse.bzh");
let edit_event = make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::MediaCaption { caption: None, formatted_caption: None, mentions: None },
)
.await
.unwrap();
assert_let!(AnyMessageLikeEventContent::RoomMessage(msg) = edit_event);
assert_let!(MessageType::Image(image) = msg.msgtype);
assert_eq!(image.filename(), "* rickroll.gif"); assert!(image.caption().is_none());
assert!(image.formatted_caption().is_none());
assert_let!(Some(Relation::Replacement(repl)) = msg.relates_to);
assert_let!(MessageType::Image(new_image) = repl.new_content.msgtype);
assert_eq!(new_image.filename(), "rickroll.gif");
assert!(new_image.caption().is_none());
assert!(new_image.formatted_caption().is_none());
}
#[async_test]
async fn test_add_media_caption_mention() {
let event_id = event_id!("$1");
let own_user_id = user_id!("@me:saucisse.bzh");
let filename = "rickroll.gif";
let mut cache = TestEventCache::default();
let f = EventFactory::new();
let event = f
.image(filename.to_owned(), owned_mxc_uri!("mxc://sdk.rs/rickroll"))
.event_id(event_id)
.sender(own_user_id)
.into_sync();
{
let event = event.raw().deserialize().unwrap();
assert_let!(AnySyncTimelineEvent::MessageLike(event) = event);
assert_let!(
AnyMessageLikeEventContent::RoomMessage(msg) = event.original_content().unwrap()
);
assert_matches!(msg.mentions, None);
}
cache.events.insert(event_id.to_owned(), event);
let room_id = room_id!("!galette:saucisse.bzh");
let mentioned_user_id = owned_user_id!("@crepe:saucisse.bzh");
let edit_event = {
let mentions = Mentions::with_user_ids([mentioned_user_id.clone()]);
make_edit_event(
cache,
room_id,
own_user_id,
event_id,
EditedContent::MediaCaption {
caption: None,
formatted_caption: None,
mentions: Some(mentions),
},
)
.await
.unwrap()
};
assert_let!(AnyMessageLikeEventContent::RoomMessage(msg) = edit_event);
assert_let!(MessageType::Image(image) = msg.msgtype);
assert!(image.caption().is_none());
assert!(image.formatted_caption().is_none());
assert_let!(Some(mentions) = msg.mentions);
assert!(!mentions.room);
assert_eq!(
mentions.user_ids.into_iter().collect::<Vec<_>>(),
vec![mentioned_user_id.clone()]
);
assert_let!(Some(Relation::Replacement(repl)) = msg.relates_to);
assert_let!(MessageType::Image(new_image) = repl.new_content.msgtype);
assert!(new_image.caption().is_none());
assert!(new_image.formatted_caption().is_none());
assert_let!(Some(mentions) = repl.new_content.mentions);
assert!(!mentions.room);
assert_eq!(mentions.user_ids.into_iter().collect::<Vec<_>>(), vec![mentioned_user_id]);
}
#[async_test]
async fn test_make_edit_event_success_with_response() {
let event_id = event_id!("$1");
let resp_event_id = event_id!("$resp");
let own_user_id = user_id!("@me:saucisse.bzh");
let mut cache = TestEventCache::default();
let f = EventFactory::new();
cache.events.insert(
event_id.to_owned(),
f.text_msg("hi").event_id(event_id).sender(user_id!("@steb:saucisse.bzh")).into(),
);
cache.events.insert(
resp_event_id.to_owned(),
f.text_msg("you're the hi")
.event_id(resp_event_id)
.sender(own_user_id)
.reply_to(event_id)
.into(),
);
let room_id = room_id!("!galette:saucisse.bzh");
let new_content = RoomMessageEventContentWithoutRelation::text_plain("uh i mean hi too");
let edit_event = make_edit_event(
cache,
room_id,
own_user_id,
resp_event_id,
EditedContent::RoomMessage(new_content),
)
.await
.unwrap();
assert_let!(AnyMessageLikeEventContent::RoomMessage(msg) = &edit_event);
assert_eq!(
msg.body(),
r#"> <@steb:saucisse.bzh> hi
* uh i mean hi too"#
);
assert_let!(Some(Relation::Replacement(repl)) = &msg.relates_to);
assert_eq!(repl.event_id, resp_event_id);
assert_eq!(repl.new_content.msgtype.body(), "uh i mean hi too");
}
}