xmpp: Receive message corrections, unchecked

This commit is contained in:
xmppftw 2024-12-19 17:20:11 +01:00 committed by xmpp ftw
parent b8af0d8fa2
commit bfac3e8893
5 changed files with 131 additions and 47 deletions

View file

@ -16,6 +16,9 @@ XXXX-YY-ZZ [ RELEASER <admin@localhost> ]
* Added: * Added:
- Agent::send_room_message takes RoomMessageSettings argument (!483) - Agent::send_room_message takes RoomMessageSettings argument (!483)
- Agent::send_raw_message takes RawMessageSettings for any message type (!487) - Agent::send_raw_message takes RawMessageSettings for any message type (!487)
- Event::ChatMessageCorrection, Event::RoomMessageCorrection, and
Event::RoomPrivateMessageCorrection signal XEP-0308 message corrections; they're
not checked how old the corrected entry is, which has security concerns (!496)
* Fixes: * Fixes:
- Use tokio::sync::RwLock not std::sync::RwLock (!432) - Use tokio::sync::RwLock not std::sync::RwLock (!432)
- Agent::wait_for_events now return Vec<Event> and sets inner tokio_xmpp Client - Agent::wait_for_events now return Vec<Event> and sets inner tokio_xmpp Client

View file

@ -26,9 +26,22 @@ pub enum Event {
/// - The [`Body`] is the message body. /// - The [`Body`] is the message body.
/// - The [`StanzaTimeInfo`] about when message was received, and when the message was claimed sent. /// - The [`StanzaTimeInfo`] about when message was received, and when the message was claimed sent.
ChatMessage(Id, BareJid, Body, StanzaTimeInfo), ChatMessage(Id, BareJid, Body, StanzaTimeInfo),
/// A message in a one-to-one chat was corrected/edited.
/// - The [`Id`] is the ID of the message that was corrected (always Some)
/// - The [`BareJid`] is the JID of the other participant in the chat.
/// - The [`Body`] is the new body of the message, to replace the old one.
/// - The [`StanzaTimeInfo`] is the time the message correction was sent/received
ChatMessageCorrection(Id, BareJid, Body, StanzaTimeInfo),
RoomJoined(BareJid), RoomJoined(BareJid),
RoomLeft(BareJid), RoomLeft(BareJid),
RoomMessage(Id, BareJid, RoomNick, Body, StanzaTimeInfo), RoomMessage(Id, BareJid, RoomNick, Body, StanzaTimeInfo),
/// A message in a MUC was corrected/edited.
/// - The [`Id`] is the ID of the message that was corrected (always Some)
/// - The [`BareJid`] is the JID of the room where the message was sent.
/// - The [`RoomNick`] is the nickname of the sender of the message.
/// - The [`Body`] is the new body of the message, to replace the old one.
/// - The [`StanzaTimeInfo`] is the time the message correction was sent/received
RoomMessageCorrection(Id, BareJid, RoomNick, Body, StanzaTimeInfo),
/// The subject of a room was received. /// The subject of a room was received.
/// - The BareJid is the room's address. /// - The BareJid is the room's address.
/// - The RoomNick is the nickname of the room member who set the subject. /// - The RoomNick is the nickname of the room member who set the subject.
@ -37,6 +50,13 @@ pub enum Event {
/// A private message received from a room, containing the message ID, the room's BareJid, /// A private message received from a room, containing the message ID, the room's BareJid,
/// the sender's nickname, and the message body. /// the sender's nickname, and the message body.
RoomPrivateMessage(Id, BareJid, RoomNick, Body, StanzaTimeInfo), RoomPrivateMessage(Id, BareJid, RoomNick, Body, StanzaTimeInfo),
/// A private message in a MUC was corrected/edited.
/// - The [`Id`] is the ID of the message that was corrected (always Some)
/// - The [`BareJid`] is the JID of the room where the message was sent.
/// - The [`RoomNick`] is the nickname of the sender of the message.
/// - The [`Body`] is the new body of the message, to replace the old one.
/// - The [`StanzaTimeInfo`] is the time the message correction was sent/received
RoomPrivateMessageCorrection(Id, BareJid, RoomNick, Body, StanzaTimeInfo),
ServiceMessage(Id, BareJid, Body, StanzaTimeInfo), ServiceMessage(Id, BareJid, Body, StanzaTimeInfo),
HttpUploadedFile(String), HttpUploadedFile(String),
} }

View file

@ -6,7 +6,7 @@
use tokio_xmpp::{ use tokio_xmpp::{
jid::Jid, jid::Jid,
parsers::{message::Message, muc::user::MucUser}, parsers::{message::Message, message_correct::Replace, muc::user::MucUser},
}; };
use crate::{delay::StanzaTimeInfo, Agent, Event, RoomNick}; use crate::{delay::StanzaTimeInfo, Agent, Event, RoomNick};
@ -15,44 +15,56 @@ pub async fn handle_message_chat(
agent: &mut Agent, agent: &mut Agent,
events: &mut Vec<Event>, events: &mut Vec<Event>,
from: Jid, from: Jid,
message: &Message, message: &mut Message,
time_info: StanzaTimeInfo, time_info: StanzaTimeInfo,
) { ) {
let langs: Vec<&str> = agent.lang.iter().map(String::as_str).collect(); let langs: Vec<&str> = agent.lang.iter().map(String::as_str).collect();
if let Some((_lang, body)) = message.get_best_body(langs) {
let mut found_special_message = false;
for payload in &message.payloads { let Some((_lang, body)) = message.get_best_body_cloned(langs) else {
if let Ok(_) = MucUser::try_from(payload.clone()) { debug!("Received normal/chat message without body:\n{:#?}", message);
let event = match from.clone().try_into_full() { return;
Err(bare) => {
// TODO: Can a service message be of type Chat/Normal and not Groupchat?
warn!("Received malformed MessageType::Chat in muc#user namespace from a bare JID.");
Event::ServiceMessage(
message.id.clone(),
bare,
body.clone(),
time_info.clone(),
)
}
Ok(full) => Event::RoomPrivateMessage(
message.id.clone(),
full.to_bare(),
RoomNick::from_resource_ref(full.resource()),
body.clone(),
time_info.clone(),
),
}; };
found_special_message = true; let is_muc_pm = message.extract_valid_payload::<MucUser>().is_some();
events.push(event); let correction = message.extract_valid_payload::<Replace>();
}
}
if !found_special_message { if is_muc_pm {
let event = if from.resource().is_none() {
Event::ChatMessage(message.id.clone(), from.to_bare(), body.clone(), time_info); warn!("Received malformed MessageType::Chat in muc#user namespace from a bare JID:\n{:#?}", message);
} else {
let full_from = from.clone().try_into_full().unwrap();
let event = if let Some(correction) = correction {
Event::RoomPrivateMessageCorrection(
Some(correction.id),
full_from.to_bare(),
RoomNick::from_resource_ref(full_from.resource()),
body.clone(),
time_info,
)
} else {
Event::RoomPrivateMessage(
message.id.clone(),
from.to_bare(),
RoomNick::from_resource_ref(full_from.resource()),
body.clone(),
time_info,
)
};
events.push(event); events.push(event);
} }
} else {
let event = if let Some(correction) = correction {
// TODO: Check that correction is valid (only for last N minutes or last N messages)
Event::ChatMessageCorrection(
Some(correction.id),
from.to_bare(),
body.clone(),
time_info,
)
} else {
Event::ChatMessage(message.id.clone(), from.to_bare(), body.clone(), time_info)
};
events.push(event);
} }
} }

View file

@ -4,18 +4,22 @@
// License, v. 2.0. If a copy of the MPL was not distributed with this // License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/. // file, You can obtain one at http://mozilla.org/MPL/2.0/.
use tokio_xmpp::{jid::Jid, parsers::message::Message}; use crate::{
delay::StanzaTimeInfo,
use crate::{delay::StanzaTimeInfo, Agent, Event, RoomNick}; jid::Jid,
parsers::{message::Message, message_correct::Replace},
Agent, Event, RoomNick,
};
pub async fn handle_message_group_chat( pub async fn handle_message_group_chat(
agent: &mut Agent, agent: &mut Agent,
events: &mut Vec<Event>, events: &mut Vec<Event>,
from: Jid, from: Jid,
message: &Message, message: &mut Message,
time_info: StanzaTimeInfo, time_info: StanzaTimeInfo,
) { ) {
let langs: Vec<&str> = agent.lang.iter().map(String::as_str).collect(); let langs: Vec<&str> = agent.lang.iter().map(String::as_str).collect();
let mut found_subject = false;
if let Some((_lang, subject)) = message.get_best_subject(langs.clone()) { if let Some((_lang, subject)) = message.get_best_subject(langs.clone()) {
events.push(Event::RoomSubject( events.push(Event::RoomSubject(
@ -24,19 +28,60 @@ pub async fn handle_message_group_chat(
subject.0.clone(), subject.0.clone(),
time_info.clone(), time_info.clone(),
)); ));
found_subject = true;
} }
if let Some((_lang, body)) = message.get_best_body(langs) { let Some((_lang, body)) = message.get_best_body_cloned(langs) else {
let event = match from.clone().try_into_full() { if !found_subject {
Ok(full) => Event::RoomMessage( debug!(
message.id.clone(), "Received groupchat message without body/subject:\n{:#?}",
message
);
}
return;
};
let correction = message.extract_payload::<Replace>().unwrap_or_else(|e| {
warn!("Failed to parse <replace> payload: {e}");
None
});
// Now we have a groupchat message... which can be:
//
// - a normal MUC message from a user in a room
// - a MUC message correction from a user in a room
// - a service message from a MUC channel (barejid)
//
// In theory we can have service message correction but nope nope nope
if let Some(resource) = from.resource() {
// User message/correction
let event = if let Some(correction) = correction {
Event::RoomMessageCorrection(
Some(correction.id),
from.to_bare(), from.to_bare(),
RoomNick::from_resource_ref(full.resource()), RoomNick::from_resource_ref(resource),
body.clone(), body.clone(),
time_info, time_info,
), )
Err(bare) => Event::ServiceMessage(message.id.clone(), bare, body.clone(), time_info), } else {
Event::RoomMessage(
message.id.clone(),
from.to_bare(),
RoomNick::from_resource_ref(resource),
body.clone(),
time_info,
)
}; };
events.push(event) events.push(event);
} else {
// Service message
if correction.is_some() {
warn!("Found correction in service message:\n{:#?}", message);
} else {
let event = Event::ServiceMessage(message.id.clone(), from.to_bare(), body, time_info);
events.push(event);
}
} }
} }

View file

@ -14,7 +14,7 @@ use crate::{delay::message_time_info, pubsub, Agent, Event};
pub mod chat; pub mod chat;
pub mod group_chat; pub mod group_chat;
pub async fn handle_message(agent: &mut Agent, message: Message) -> Vec<Event> { pub async fn handle_message(agent: &mut Agent, mut message: Message) -> Vec<Event> {
let mut events = vec![]; let mut events = vec![];
let from = message.from.clone().unwrap(); let from = message.from.clone().unwrap();
let time_info = message_time_info(&message); let time_info = message_time_info(&message);
@ -25,17 +25,21 @@ pub async fn handle_message(agent: &mut Agent, message: Message) -> Vec<Event> {
agent, agent,
&mut events, &mut events,
from.clone(), from.clone(),
&message, &mut message,
time_info, time_info,
) )
.await; .await;
} }
MessageType::Chat | MessageType::Normal => { MessageType::Chat | MessageType::Normal => {
chat::handle_message_chat(agent, &mut events, from.clone(), &message, time_info).await; chat::handle_message_chat(agent, &mut events, from.clone(), &mut message, time_info)
.await;
} }
_ => {} _ => {}
} }
// TODO: should this be here or in specific branch of messagetype?
// We may drop some payloads in branches before (&mut message), but
// that's ok because pubsub only wants the pubsub payload.
for child in message.payloads { for child in message.payloads {
if child.is("event", ns::PUBSUB_EVENT) { if child.is("event", ns::PUBSUB_EVENT) {
let new_events = pubsub::handle_event(&from, child, agent).await; let new_events = pubsub::handle_event(&from, child, agent).await;