Files
sol/src/matrix_utils.rs
Sienna Meridian Satterwhite 4949e70ecc feat: per-user auto-memory with ResponseContext
Three memory channels: hidden tool (sol.memory.set/get in scripts),
pre-response injection (relevant memories loaded into system prompt),
and post-response extraction (ministral-3b extracts facts after each
response). User isolation enforced at Rust level — user_id derived
from Matrix sender, never from script arguments.

New modules: context (ResponseContext), memory (schema, store, extractor).
ResponseContext threaded through responder → tools → script runtime.
OpenSearch index sol_user_memory created on startup alongside archive.
2026-03-21 15:51:31 +00:00

91 lines
3.1 KiB
Rust

use matrix_sdk::room::Room;
use matrix_sdk::RoomMemberships;
use ruma::events::room::message::{
MessageType, OriginalSyncRoomMessageEvent, Relation, RoomMessageEventContent,
};
use ruma::events::relation::{Annotation, InReplyTo};
use ruma::events::reaction::ReactionEventContent;
use ruma::OwnedEventId;
/// Extract the plain-text body from a message event.
pub fn extract_body(event: &OriginalSyncRoomMessageEvent) -> Option<String> {
match &event.content.msgtype {
MessageType::Text(text) => Some(text.body.clone()),
MessageType::Notice(notice) => Some(notice.body.clone()),
MessageType::Emote(emote) => Some(emote.body.clone()),
_ => None,
}
}
/// Check if this event is an edit (m.replace relation) and return the new body.
pub fn extract_edit(event: &OriginalSyncRoomMessageEvent) -> Option<(OwnedEventId, String)> {
if let Some(Relation::Replacement(replacement)) = &event.content.relates_to {
let new_body = match &replacement.new_content.msgtype {
MessageType::Text(text) => text.body.clone(),
MessageType::Notice(notice) => notice.body.clone(),
_ => return None,
};
return Some((replacement.event_id.clone(), new_body));
}
None
}
/// Extract the event ID being replied to, if any.
pub fn extract_reply_to(event: &OriginalSyncRoomMessageEvent) -> Option<OwnedEventId> {
if let Some(Relation::Reply { in_reply_to }) = &event.content.relates_to {
return Some(in_reply_to.event_id.clone());
}
None
}
/// Extract thread root event ID, if any.
pub fn extract_thread_id(event: &OriginalSyncRoomMessageEvent) -> Option<OwnedEventId> {
if let Some(Relation::Thread(thread)) = &event.content.relates_to {
return Some(thread.event_id.clone());
}
None
}
/// Build a reply message content with m.in_reply_to relation and markdown rendering.
pub fn make_reply_content(body: &str, reply_to_event_id: OwnedEventId) -> RoomMessageEventContent {
let mut content = RoomMessageEventContent::text_markdown(body);
content.relates_to = Some(Relation::Reply {
in_reply_to: InReplyTo::new(reply_to_event_id),
});
content
}
/// Send an emoji reaction to a message.
pub async fn send_reaction(
room: &Room,
event_id: OwnedEventId,
emoji: &str,
) -> anyhow::Result<()> {
let annotation = Annotation::new(event_id, emoji.to_string());
let content = ReactionEventContent::new(annotation);
room.send(content).await?;
Ok(())
}
/// Get the display name for a room.
pub fn room_display_name(room: &Room) -> String {
room.cached_display_name()
.map(|n| n.to_string())
.unwrap_or_else(|| room.room_id().to_string())
}
/// Get member display names for a room.
pub async fn room_member_names(room: &Room) -> Vec<String> {
match room.members(RoomMemberships::JOIN).await {
Ok(members) => members
.iter()
.map(|m| {
m.display_name()
.unwrap_or_else(|| m.user_id().as_str())
.to_string()
})
.collect(),
Err(_) => Vec::new(),
}
}