use std::future::Future;
use eyeball::Subscriber;
use indexmap::IndexMap;
#[cfg(test)]
use matrix_sdk::crypto::{DecryptionSettings, RoomEventDecryptionResult, TrustRequirement};
use matrix_sdk::{
crypto::types::events::CryptoContextInfo, deserialized_responses::TimelineEvent,
event_cache::paginator::PaginableRoom, AsyncTraitDeps, Result, Room, SendOutsideWasm,
};
use matrix_sdk_base::{latest_event::LatestEvent, RoomInfo};
use ruma::{
events::{
fully_read::FullyReadEventContent,
receipt::{Receipt, ReceiptThread, ReceiptType},
AnyMessageLikeEventContent, AnySyncTimelineEvent,
},
push::{PushConditionRoomCtx, Ruleset},
serde::Raw,
EventId, OwnedEventId, OwnedTransactionId, OwnedUserId, RoomVersionId, UserId,
};
use tracing::{debug, error};
use super::{Profile, RedactError, TimelineBuilder};
use crate::timeline::{self, pinned_events_loader::PinnedEventsRoom, Timeline};
pub trait RoomExt {
fn timeline(&self)
-> impl Future<Output = Result<Timeline, timeline::Error>> + SendOutsideWasm;
fn timeline_builder(&self) -> TimelineBuilder;
}
impl RoomExt for Room {
async fn timeline(&self) -> Result<Timeline, timeline::Error> {
self.timeline_builder().build().await
}
fn timeline_builder(&self) -> TimelineBuilder {
Timeline::builder(self).track_read_marker_and_receipts()
}
}
pub(super) trait RoomDataProvider:
Clone + PaginableRoom + PinnedEventsRoom + 'static
{
fn own_user_id(&self) -> &UserId;
fn room_version(&self) -> RoomVersionId;
fn crypto_context_info(&self)
-> impl Future<Output = CryptoContextInfo> + SendOutsideWasm + '_;
fn profile_from_user_id<'a>(
&'a self,
user_id: &'a UserId,
) -> impl Future<Output = Option<Profile>> + SendOutsideWasm + 'a;
fn profile_from_latest_event(&self, latest_event: &LatestEvent) -> Option<Profile>;
fn load_user_receipt<'a>(
&'a self,
receipt_type: ReceiptType,
thread: ReceiptThread,
user_id: &'a UserId,
) -> impl Future<Output = Option<(OwnedEventId, Receipt)>> + SendOutsideWasm + 'a;
fn load_event_receipts<'a>(
&'a self,
event_id: &'a EventId,
) -> impl Future<Output = IndexMap<OwnedUserId, Receipt>> + SendOutsideWasm + 'a;
fn load_fully_read_marker(&self) -> impl Future<Output = Option<OwnedEventId>> + '_;
fn push_rules_and_context(
&self,
) -> impl Future<Output = Option<(Ruleset, PushConditionRoomCtx)>> + SendOutsideWasm + '_;
fn send(
&self,
content: AnyMessageLikeEventContent,
) -> impl Future<Output = Result<(), super::Error>> + SendOutsideWasm + '_;
fn redact<'a>(
&'a self,
event_id: &'a EventId,
reason: Option<&'a str>,
transaction_id: Option<OwnedTransactionId>,
) -> impl Future<Output = Result<(), super::Error>> + SendOutsideWasm + 'a;
fn room_info(&self) -> Subscriber<RoomInfo>;
}
impl RoomDataProvider for Room {
fn own_user_id(&self) -> &UserId {
(**self).own_user_id()
}
fn room_version(&self) -> RoomVersionId {
(**self).clone_info().room_version_or_default()
}
async fn crypto_context_info(&self) -> CryptoContextInfo {
self.crypto_context_info().await
}
async fn profile_from_user_id<'a>(&'a self, user_id: &'a UserId) -> Option<Profile> {
match self.get_member_no_sync(user_id).await {
Ok(Some(member)) => Some(Profile {
display_name: member.display_name().map(ToOwned::to_owned),
display_name_ambiguous: member.name_ambiguous(),
avatar_url: member.avatar_url().map(ToOwned::to_owned),
}),
Ok(None) if self.are_members_synced() => Some(Profile::default()),
Ok(None) => None,
Err(e) => {
error!(%user_id, "Failed to fetch room member information: {e}");
None
}
}
}
fn profile_from_latest_event(&self, latest_event: &LatestEvent) -> Option<Profile> {
if !latest_event.has_sender_profile() {
return None;
}
Some(Profile {
display_name: latest_event.sender_display_name().map(ToOwned::to_owned),
display_name_ambiguous: latest_event.sender_name_ambiguous().unwrap_or(false),
avatar_url: latest_event.sender_avatar_url().map(ToOwned::to_owned),
})
}
async fn load_user_receipt<'a>(
&'a self,
receipt_type: ReceiptType,
thread: ReceiptThread,
user_id: &'a UserId,
) -> Option<(OwnedEventId, Receipt)> {
match self.load_user_receipt(receipt_type.clone(), thread.clone(), user_id).await {
Ok(receipt) => receipt,
Err(e) => {
error!(
?receipt_type,
?thread,
?user_id,
"Failed to get read receipt for user: {e}"
);
None
}
}
}
async fn load_event_receipts<'a>(
&'a self,
event_id: &'a EventId,
) -> IndexMap<OwnedUserId, Receipt> {
let mut unthreaded_receipts = match self
.load_event_receipts(ReceiptType::Read, ReceiptThread::Unthreaded, event_id)
.await
{
Ok(receipts) => receipts.into_iter().collect(),
Err(e) => {
error!(?event_id, "Failed to get unthreaded read receipts for event: {e}");
IndexMap::new()
}
};
let main_thread_receipts = match self
.load_event_receipts(ReceiptType::Read, ReceiptThread::Main, event_id)
.await
{
Ok(receipts) => receipts,
Err(e) => {
error!(?event_id, "Failed to get main thread read receipts for event: {e}");
Vec::new()
}
};
unthreaded_receipts.extend(main_thread_receipts);
unthreaded_receipts
}
async fn push_rules_and_context(&self) -> Option<(Ruleset, PushConditionRoomCtx)> {
match self.push_context().await {
Ok(Some(push_context)) => match self.client().account().push_rules().await {
Ok(push_rules) => Some((push_rules, push_context)),
Err(e) => {
error!("Could not get push rules: {e}");
None
}
},
Ok(None) => {
debug!("Could not aggregate push context");
None
}
Err(e) => {
error!("Could not get push context: {e}");
None
}
}
}
async fn load_fully_read_marker(&self) -> Option<OwnedEventId> {
match self.account_data_static::<FullyReadEventContent>().await {
Ok(Some(fully_read)) => match fully_read.deserialize() {
Ok(fully_read) => Some(fully_read.content.event_id),
Err(e) => {
error!("Failed to deserialize fully-read account data: {e}");
None
}
},
Err(e) => {
error!("Failed to get fully-read account data from the store: {e}");
None
}
_ => None,
}
}
async fn send(&self, content: AnyMessageLikeEventContent) -> Result<(), super::Error> {
let _ = self.send_queue().send(content).await?;
Ok(())
}
async fn redact<'a>(
&'a self,
event_id: &'a EventId,
reason: Option<&'a str>,
transaction_id: Option<OwnedTransactionId>,
) -> Result<(), super::Error> {
let _ = self
.redact(event_id, reason, transaction_id)
.await
.map_err(RedactError::HttpError)
.map_err(super::Error::RedactError)?;
Ok(())
}
fn room_info(&self) -> Subscriber<RoomInfo> {
self.subscribe_info()
}
}
pub(super) trait Decryptor: AsyncTraitDeps + Clone + 'static {
fn decrypt_event_impl(
&self,
raw: &Raw<AnySyncTimelineEvent>,
) -> impl Future<Output = Result<TimelineEvent>> + SendOutsideWasm;
}
impl Decryptor for Room {
async fn decrypt_event_impl(&self, raw: &Raw<AnySyncTimelineEvent>) -> Result<TimelineEvent> {
self.decrypt_event(raw.cast_ref()).await
}
}
#[cfg(test)]
impl Decryptor for (matrix_sdk_base::crypto::OlmMachine, ruma::OwnedRoomId) {
async fn decrypt_event_impl(&self, raw: &Raw<AnySyncTimelineEvent>) -> Result<TimelineEvent> {
let (olm_machine, room_id) = self;
let decryption_settings =
DecryptionSettings { sender_device_trust_requirement: TrustRequirement::Untrusted };
match olm_machine
.try_decrypt_room_event(raw.cast_ref(), room_id, &decryption_settings)
.await?
{
RoomEventDecryptionResult::Decrypted(decrypted) => Ok(decrypted.into()),
RoomEventDecryptionResult::UnableToDecrypt(utd_info) => {
Ok(TimelineEvent::new_utd_event(raw.clone(), utd_info))
}
}
}
}