use std::{collections::BTreeMap, fmt, sync::Arc};
use events::Gap;
use matrix_sdk_base::{
deserialized_responses::{AmbiguityChange, SyncTimelineEvent},
linked_chunk::ChunkContent,
sync::{JoinedRoomUpdate, LeftRoomUpdate, Timeline},
};
use ruma::{
events::{relation::RelationType, AnyRoomAccountDataEvent, AnySyncEphemeralRoomEvent},
serde::Raw,
EventId, OwnedEventId, OwnedRoomId, RoomVersionId,
};
use tokio::sync::{
broadcast::{Receiver, Sender},
Notify, RwLock,
};
use tracing::{debug, trace, warn};
use super::{
paginator::{Paginator, PaginatorState},
AllEventsCache, EventsOrigin, Result, RoomEventCacheUpdate, RoomPagination,
};
use crate::{client::WeakClient, room::WeakRoom};
pub(super) mod events;
#[derive(Clone)]
pub struct RoomEventCache {
pub(super) inner: Arc<RoomEventCacheInner>,
}
impl fmt::Debug for RoomEventCache {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("RoomEventCache").finish_non_exhaustive()
}
}
impl RoomEventCache {
pub(super) fn new(
client: WeakClient,
state: RoomEventCacheState,
room_id: OwnedRoomId,
room_version: RoomVersionId,
all_events_cache: Arc<RwLock<AllEventsCache>>,
) -> Self {
Self {
inner: Arc::new(RoomEventCacheInner::new(
client,
state,
room_id,
room_version,
all_events_cache,
)),
}
}
pub async fn subscribe(
&self,
) -> Result<(Vec<SyncTimelineEvent>, Receiver<RoomEventCacheUpdate>)> {
let state = self.inner.state.read().await;
let events = state.events().events().map(|(_position, item)| item.clone()).collect();
Ok((events, self.inner.sender.subscribe()))
}
pub fn pagination(&self) -> RoomPagination {
RoomPagination { inner: self.inner.clone() }
}
pub async fn event(&self, event_id: &EventId) -> Option<SyncTimelineEvent> {
if let Some((room_id, event)) =
self.inner.all_events.read().await.events.get(event_id).cloned()
{
if room_id == self.inner.room_id {
return Some(event);
}
}
let state = self.inner.state.read().await;
for (_pos, event) in state.events().revents() {
if event.event_id().as_deref() == Some(event_id) {
return Some(event.clone());
}
}
None
}
pub async fn event_with_relations(
&self,
event_id: &EventId,
filter: Option<Vec<RelationType>>,
) -> Option<(SyncTimelineEvent, Vec<SyncTimelineEvent>)> {
let cache = self.inner.all_events.read().await;
if let Some((_, event)) = cache.events.get(event_id) {
let related_events = cache.collect_related_events(event_id, filter.as_deref());
Some((event.clone(), related_events))
} else {
None
}
}
pub async fn clear(&self) -> Result<()> {
self.inner.state.write().await.reset().await?;
self.inner.all_events.write().await.clear();
let _ = self.inner.paginator.set_idle_state(PaginatorState::Initial, None, None);
let _ = self.inner.sender.send(RoomEventCacheUpdate::Clear);
Ok(())
}
pub(crate) async fn save_event(&self, event: SyncTimelineEvent) {
if let Some(event_id) = event.event_id() {
let mut cache = self.inner.all_events.write().await;
cache.append_related_event(&event);
cache.events.insert(event_id, (self.inner.room_id.clone(), event));
} else {
warn!("couldn't save event without event id in the event cache");
}
}
pub(crate) async fn save_events(&self, events: impl IntoIterator<Item = SyncTimelineEvent>) {
let mut cache = self.inner.all_events.write().await;
for event in events {
if let Some(event_id) = event.event_id() {
cache.append_related_event(&event);
cache.events.insert(event_id, (self.inner.room_id.clone(), event));
} else {
warn!("couldn't save event without event id in the event cache");
}
}
}
pub async fn debug_string(&self) -> Vec<String> {
self.inner.state.read().await.events().debug_string()
}
}
pub(super) struct RoomEventCacheInner {
room_id: OwnedRoomId,
pub(crate) room_version: RoomVersionId,
pub sender: Sender<RoomEventCacheUpdate>,
pub state: RwLock<RoomEventCacheState>,
all_events: Arc<RwLock<AllEventsCache>>,
pub pagination_batch_token_notifier: Notify,
pub paginator: Paginator<WeakRoom>,
}
impl RoomEventCacheInner {
fn new(
client: WeakClient,
state: RoomEventCacheState,
room_id: OwnedRoomId,
room_version: RoomVersionId,
all_events_cache: Arc<RwLock<AllEventsCache>>,
) -> Self {
let sender = Sender::new(32);
let weak_room = WeakRoom::new(client, room_id);
Self {
room_id: weak_room.room_id().to_owned(),
room_version,
state: RwLock::new(state),
all_events: all_events_cache,
sender,
pagination_batch_token_notifier: Default::default(),
paginator: Paginator::new(weak_room),
}
}
fn handle_account_data(&self, account_data: Vec<Raw<AnyRoomAccountDataEvent>>) {
if account_data.is_empty() {
return;
}
let mut handled_read_marker = false;
trace!("Handling account data");
for raw_event in account_data {
match raw_event.deserialize() {
Ok(AnyRoomAccountDataEvent::FullyRead(ev)) => {
if handled_read_marker {
continue;
}
handled_read_marker = true;
let _ = self.sender.send(RoomEventCacheUpdate::MoveReadMarkerTo {
event_id: ev.content.event_id,
});
}
Ok(_) => {
}
Err(e) => {
let event_type = raw_event.get_field::<String>("type").ok().flatten();
warn!(event_type, "Failed to deserialize account data: {e}");
}
}
}
}
pub(super) async fn handle_joined_room_update(
&self,
has_storage: bool,
updates: JoinedRoomUpdate,
) -> Result<()> {
self.handle_timeline(
has_storage,
updates.timeline,
updates.ephemeral.clone(),
updates.ambiguity_changes,
)
.await?;
self.handle_account_data(updates.account_data);
Ok(())
}
async fn handle_timeline(
&self,
has_storage: bool,
timeline: Timeline,
ephemeral_events: Vec<Raw<AnySyncEphemeralRoomEvent>>,
ambiguity_changes: BTreeMap<OwnedEventId, AmbiguityChange>,
) -> Result<()> {
if !has_storage && timeline.limited {
trace!("limited timeline, clearing all previous events and pushing new events");
self.replace_all_events_by(
timeline.events,
timeline.prev_batch,
ephemeral_events,
ambiguity_changes,
)
.await?;
} else {
trace!("adding new events");
let prev_batch =
if has_storage && !timeline.limited { None } else { timeline.prev_batch };
let mut state = self.state.write().await;
self.append_events_locked(
&mut state,
timeline.events,
prev_batch,
ephemeral_events,
ambiguity_changes,
)
.await?;
}
Ok(())
}
pub(super) async fn handle_left_room_update(
&self,
has_storage: bool,
updates: LeftRoomUpdate,
) -> Result<()> {
self.handle_timeline(has_storage, updates.timeline, Vec::new(), updates.ambiguity_changes)
.await?;
Ok(())
}
pub(super) async fn replace_all_events_by(
&self,
sync_timeline_events: Vec<SyncTimelineEvent>,
prev_batch: Option<String>,
ephemeral_events: Vec<Raw<AnySyncEphemeralRoomEvent>>,
ambiguity_changes: BTreeMap<OwnedEventId, AmbiguityChange>,
) -> Result<()> {
let mut state = self.state.write().await;
state.reset().await?;
let _ = self.sender.send(RoomEventCacheUpdate::Clear);
self.append_events_locked(
&mut state,
sync_timeline_events,
prev_batch.clone(),
ephemeral_events,
ambiguity_changes,
)
.await?;
self.paginator.set_idle_state(PaginatorState::Initial, prev_batch, None)?;
Ok(())
}
async fn append_events_locked(
&self,
state: &mut RoomEventCacheState,
sync_timeline_events: Vec<SyncTimelineEvent>,
prev_batch: Option<String>,
ephemeral_events: Vec<Raw<AnySyncEphemeralRoomEvent>>,
ambiguity_changes: BTreeMap<OwnedEventId, AmbiguityChange>,
) -> Result<()> {
if sync_timeline_events.is_empty()
&& prev_batch.is_none()
&& ephemeral_events.is_empty()
&& ambiguity_changes.is_empty()
{
return Ok(());
}
let sync_timeline_events_diffs = {
let (_, sync_timeline_events_diffs) = state
.with_events_mut(|room_events| {
if let Some(prev_token) = &prev_batch {
room_events.push_gap(Gap { prev_token: prev_token.clone() });
}
let added_unique_events = room_events.push_events(sync_timeline_events.clone());
if !added_unique_events {
debug!(
"not storing previous batch token, because we deduplicated all new sync events"
);
if let Some(prev_token) = &prev_batch {
trace!("removing gap we just inserted");
let prev_gap_id = room_events
.rchunks()
.find_map(|c| {
let gap = as_variant::as_variant!(c.content(), ChunkContent::Gap)?;
(gap.prev_token == *prev_token).then_some(c.identifier())
})
.expect("we just inserted the gap beforehand");
room_events
.replace_gap_at([], prev_gap_id)
.expect("we obtained the valid position beforehand");
}
}
room_events.on_new_events(&self.room_version, sync_timeline_events.iter());
})
.await?;
let mut all_events = self.all_events.write().await;
for sync_timeline_event in sync_timeline_events {
if let Some(event_id) = sync_timeline_event.event_id() {
all_events.append_related_event(&sync_timeline_event);
all_events
.events
.insert(event_id.to_owned(), (self.room_id.clone(), sync_timeline_event));
}
}
sync_timeline_events_diffs
};
if prev_batch.is_some() {
self.pagination_batch_token_notifier.notify_one();
}
{
if !sync_timeline_events_diffs.is_empty() {
let _ = self.sender.send(RoomEventCacheUpdate::UpdateTimelineEvents {
diffs: sync_timeline_events_diffs,
origin: EventsOrigin::Sync,
});
}
if !ephemeral_events.is_empty() {
let _ = self
.sender
.send(RoomEventCacheUpdate::AddEphemeralEvents { events: ephemeral_events });
}
if !ambiguity_changes.is_empty() {
let _ = self.sender.send(RoomEventCacheUpdate::UpdateMembers { ambiguity_changes });
}
}
Ok(())
}
}
fn chunk_debug_string(content: &ChunkContent<SyncTimelineEvent, Gap>) -> String {
match content {
ChunkContent::Gap(Gap { prev_token }) => {
format!("gap['{prev_token}']")
}
ChunkContent::Items(vec) => {
let items = vec
.iter()
.map(|event| {
event.event_id().map_or_else(
|| "<no event id>".to_owned(),
|id| id.as_str().chars().take(1 + 8).collect(),
)
})
.collect::<Vec<_>>()
.join(", ");
format!("events[{items}]")
}
}
}
mod private {
use std::sync::Arc;
use eyeball_im::VectorDiff;
use matrix_sdk_base::{
deserialized_responses::{SyncTimelineEvent, TimelineEventKind},
event_cache::{
store::{
EventCacheStoreError, EventCacheStoreLock, EventCacheStoreLockGuard,
DEFAULT_CHUNK_CAPACITY,
},
Event, Gap,
},
linked_chunk::{LinkedChunk, LinkedChunkBuilder, RawChunk, Update},
};
use once_cell::sync::OnceCell;
use ruma::{serde::Raw, OwnedRoomId, RoomId};
use tracing::{error, instrument, trace};
use super::{chunk_debug_string, events::RoomEvents};
use crate::event_cache::EventCacheError;
pub struct RoomEventCacheState {
room: OwnedRoomId,
store: Arc<OnceCell<EventCacheStoreLock>>,
events: RoomEvents,
pub waited_for_initial_prev_token: bool,
}
impl RoomEventCacheState {
async fn try_reload_linked_chunk(
room: &RoomId,
locked: &EventCacheStoreLockGuard<'_>,
) -> Result<Option<LinkedChunk<DEFAULT_CHUNK_CAPACITY, Event, Gap>>, EventCacheError>
{
let raw_chunks = locked.reload_linked_chunk(room).await?;
let mut builder = LinkedChunkBuilder::from_raw_parts(raw_chunks.clone());
builder.with_update_history();
Ok(builder.build().map_err(|err| {
if tracing::enabled!(tracing::Level::TRACE) {
trace!("couldn't build a linked chunk from the raw parts. Raw chunks:");
for line in raw_chunks_debug_string(raw_chunks) {
trace!("{line}");
}
}
EventCacheStoreError::InvalidData { details: err.to_string() }
})?)
}
pub async fn new(
room: OwnedRoomId,
store: Arc<OnceCell<EventCacheStoreLock>>,
) -> Result<Self, EventCacheError> {
let events = if let Some(store) = store.get() {
let locked = store.lock().await?;
let linked_chunk = match Self::try_reload_linked_chunk(&room, &locked).await {
Ok(linked_chunk) => linked_chunk,
Err(err) => {
error!("error when reloading a linked chunk from memory: {err}");
locked.handle_linked_chunk_updates(&room, vec![Update::Clear]).await?;
None
}
};
RoomEvents::with_initial_chunks(linked_chunk)
} else {
RoomEvents::default()
};
Ok(Self { room, store, events, waited_for_initial_prev_token: false })
}
fn strip_relations_if_present<T>(event: &mut Raw<T>) {
let mut closure = || -> Option<()> {
let mut val: serde_json::Value = event.deserialize_as().ok()?;
let unsigned = val.get_mut("unsigned")?;
let unsigned_obj = unsigned.as_object_mut()?;
if unsigned_obj.remove("m.relations").is_some() {
*event = Raw::new(&val).ok()?.cast();
}
None
};
let _ = closure();
}
fn strip_relations_from_event(ev: &mut SyncTimelineEvent) {
match &mut ev.kind {
TimelineEventKind::Decrypted(decrypted) => {
decrypted.unsigned_encryption_info = None;
Self::strip_relations_if_present(&mut decrypted.event);
}
TimelineEventKind::UnableToDecrypt { event, .. }
| TimelineEventKind::PlainText { event } => {
Self::strip_relations_if_present(event);
}
}
}
fn strip_relations_from_events(items: &mut [SyncTimelineEvent]) {
for ev in items.iter_mut() {
Self::strip_relations_from_event(ev);
}
}
#[instrument(skip_all)]
async fn propagate_changes(&mut self) -> Result<(), EventCacheError> {
let mut updates = self.events.updates().take();
if updates.is_empty() {
return Ok(());
}
let Some(store) = self.store.get() else {
return Ok(());
};
trace!("propagating {} updates", updates.len());
for up in updates.iter_mut() {
match up {
Update::PushItems { items, .. } => Self::strip_relations_from_events(items),
Update::ReplaceItem { item, .. } => Self::strip_relations_from_event(item),
Update::NewItemsChunk { .. }
| Update::NewGapChunk { .. }
| Update::RemoveChunk(_)
| Update::RemoveItem { .. }
| Update::DetachLastItems { .. }
| Update::StartReattachItems
| Update::EndReattachItems
| Update::Clear => {}
}
}
let store = store.clone();
let room_id = self.room.clone();
matrix_sdk_common::executor::spawn(async move {
let locked = store.lock().await?;
if let Err(err) = locked.handle_linked_chunk_updates(&room_id, updates).await {
error!("unable to handle linked chunk updates: {err}");
}
super::Result::Ok(())
})
.await
.expect("joining failed")?;
trace!("done propagating store changes");
Ok(())
}
pub async fn reset(&mut self) -> Result<(), EventCacheError> {
self.events.reset();
self.propagate_changes().await?;
self.waited_for_initial_prev_token = false;
Ok(())
}
pub fn events(&self) -> &RoomEvents {
&self.events
}
pub async fn with_events_mut<O, F: FnOnce(&mut RoomEvents) -> O>(
&mut self,
func: F,
) -> Result<(O, Vec<VectorDiff<SyncTimelineEvent>>), EventCacheError> {
let output = func(&mut self.events);
self.propagate_changes().await?;
let updates_as_vector_diffs = self.events.updates_as_vector_diffs();
Ok((output, updates_as_vector_diffs))
}
}
fn raw_chunks_debug_string(mut raw_chunks: Vec<RawChunk<Event, Gap>>) -> Vec<String> {
let mut result = Vec::new();
raw_chunks.sort_by_key(|c| c.identifier.index());
for c in raw_chunks {
let content = chunk_debug_string(&c.content);
let prev =
c.previous.map_or_else(|| "<none>".to_owned(), |prev| prev.index().to_string());
let next = c.next.map_or_else(|| "<none>".to_owned(), |prev| prev.index().to_string());
let line =
format!("chunk #{} (prev={prev}, next={next}): {content}", c.identifier.index());
result.push(line);
}
result
}
#[cfg(test)]
mod tests {
use matrix_sdk_base::{
event_cache::Gap,
linked_chunk::{ChunkContent, ChunkIdentifier as CId, RawChunk},
};
use matrix_sdk_test::{event_factory::EventFactory, ALICE, DEFAULT_TEST_ROOM_ID};
use ruma::event_id;
use super::raw_chunks_debug_string;
#[test]
fn test_raw_chunks_debug_string() {
let mut raws = Vec::new();
let f = EventFactory::new().room(&DEFAULT_TEST_ROOM_ID).sender(*ALICE);
raws.push(RawChunk {
content: ChunkContent::Items(vec![
f.text_msg("hey")
.event_id(event_id!("$123456789101112131415617181920"))
.into_sync(),
f.text_msg("you").event_id(event_id!("$2")).into_sync(),
]),
identifier: CId::new(1),
previous: Some(CId::new(0)),
next: None,
});
raws.push(RawChunk {
content: ChunkContent::Gap(Gap { prev_token: "prev-token".to_owned() }),
identifier: CId::new(0),
previous: None,
next: Some(CId::new(1)),
});
let output = raw_chunks_debug_string(raws);
assert_eq!(output.len(), 2);
assert_eq!(&output[0], "chunk #0 (prev=<none>, next=1): gap['prev-token']");
assert_eq!(&output[1], "chunk #1 (prev=0, next=<none>): events[$12345678, $2]");
}
}
}
pub(super) use private::RoomEventCacheState;
#[cfg(test)]
mod tests {
use std::sync::Arc;
use assert_matches::assert_matches;
use assert_matches2::assert_let;
use matrix_sdk_base::{
event_cache::{
store::{EventCacheStore as _, MemoryStore},
Gap,
},
linked_chunk::{ChunkContent, ChunkIdentifier, Position, Update},
store::StoreConfig,
sync::{JoinedRoomUpdate, Timeline},
};
use matrix_sdk_common::deserialized_responses::SyncTimelineEvent;
use matrix_sdk_test::{async_test, event_factory::EventFactory, ALICE, BOB};
use ruma::{
event_id,
events::{
relation::RelationType, room::message::RoomMessageEventContentWithoutRelation,
AnySyncMessageLikeEvent, AnySyncTimelineEvent,
},
room_id, user_id, RoomId,
};
use crate::test_utils::{client::MockClientBuilder, logged_in_client};
#[async_test]
async fn test_event_with_redaction_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.text_msg("Original event").event_id(original_id).into(),
f.redaction(original_id).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_edit_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.text_msg("Original event").event_id(original_id).into(),
f.text_msg("* An edited event")
.edit(
original_id,
RoomMessageEventContentWithoutRelation::text_plain("And edited event"),
)
.event_id(related_id)
.into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_reply_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.text_msg("Original event").event_id(original_id).into(),
f.text_msg("A reply").reply_to(original_id).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_thread_reply_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.text_msg("Original event").event_id(original_id).into(),
f.text_msg("A reply").in_thread(original_id, related_id).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_reaction_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.text_msg("Original event").event_id(original_id).into(),
f.reaction(original_id, ":D".to_owned()).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_poll_response_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.poll_start("Poll start event", "A poll question", vec!["An answer"])
.event_id(original_id)
.into(),
f.poll_response("1", original_id).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_poll_end_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
assert_relations(
room_id,
f.poll_start("Poll start event", "A poll question", vec!["An answer"])
.event_id(original_id)
.into(),
f.poll_end("Poll ended", original_id).event_id(related_id).into(),
f,
)
.await;
}
#[async_test]
async fn test_event_with_filtered_relationships() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let associated_related_id = event_id!("$recursive_related");
let room_id = room_id!("!galette:saucisse.bzh");
let event_factory = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let original_event = event_factory.text_msg("Original event").event_id(original_id).into();
let related_event = event_factory
.text_msg("* Edited event")
.edit(original_id, RoomMessageEventContentWithoutRelation::text_plain("Edited event"))
.event_id(related_id)
.into();
let associated_related_event =
event_factory.redaction(related_id).event_id(associated_related_id).into();
let client = logged_in_client(None).await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
room_event_cache.save_event(original_event).await;
room_event_cache.save_event(related_event).await;
room_event_cache.save_event(associated_related_event).await;
let filter = Some(vec![RelationType::Replacement]);
let (event, related_events) =
room_event_cache.event_with_relations(original_id, filter).await.unwrap();
let cached_event_id = event.event_id().unwrap();
assert_eq!(cached_event_id, original_id);
assert_eq!(related_events.len(), 2);
let related_event_id = related_events[0].event_id().unwrap();
assert_eq!(related_event_id, related_id);
let related_event_id = related_events[1].event_id().unwrap();
assert_eq!(related_event_id, associated_related_id);
let filter = Some(vec![RelationType::Thread]);
let (event, related_events) =
room_event_cache.event_with_relations(original_id, filter).await.unwrap();
let cached_event_id = event.event_id().unwrap();
assert_eq!(cached_event_id, original_id);
assert!(related_events.is_empty());
}
#[async_test]
async fn test_event_with_recursive_relation() {
let original_id = event_id!("$original");
let related_id = event_id!("$related");
let associated_related_id = event_id!("$recursive_related");
let room_id = room_id!("!galette:saucisse.bzh");
let event_factory = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let original_event = event_factory.text_msg("Original event").event_id(original_id).into();
let related_event = event_factory
.text_msg("* Edited event")
.edit(original_id, RoomMessageEventContentWithoutRelation::text_plain("Edited event"))
.event_id(related_id)
.into();
let associated_related_event =
event_factory.redaction(related_id).event_id(associated_related_id).into();
let client = logged_in_client(None).await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
room_event_cache.save_event(original_event).await;
room_event_cache.save_event(related_event).await;
room_event_cache.save_event(associated_related_event).await;
let (event, related_events) =
room_event_cache.event_with_relations(original_id, None).await.unwrap();
let cached_event_id = event.event_id().unwrap();
assert_eq!(cached_event_id, original_id);
assert_eq!(related_events.len(), 2);
let related_event_id = related_events[0].event_id().unwrap();
assert_eq!(related_event_id, related_id);
let related_event_id = related_events[1].event_id().unwrap();
assert_eq!(related_event_id, associated_related_id);
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_write_to_storage() {
use matrix_sdk_base::linked_chunk::LinkedChunkBuilder;
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let event_cache_store = Arc::new(MemoryStore::new());
let client = MockClientBuilder::new("http://localhost".to_owned())
.store_config(
StoreConfig::new("hodlor".to_owned()).event_cache_store(event_cache_store.clone()),
)
.build()
.await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let timeline = Timeline {
limited: true,
prev_batch: Some("raclette".to_owned()),
events: vec![f.text_msg("hey yo").sender(*ALICE).into_sync()],
};
room_event_cache
.inner
.handle_joined_room_update(true, JoinedRoomUpdate { timeline, ..Default::default() })
.await
.unwrap();
let raws = event_cache_store.reload_linked_chunk(room_id).await.unwrap();
let linked_chunk =
LinkedChunkBuilder::<3, _, _>::from_raw_parts(raws).build().unwrap().unwrap();
assert_eq!(linked_chunk.chunks().count(), 3);
let mut chunks = linked_chunk.chunks();
assert_matches!(chunks.next().unwrap().content(), ChunkContent::Items(events) => {
assert_eq!(events.len(), 0)
});
assert_matches!(chunks.next().unwrap().content(), ChunkContent::Gap(gap) => {
assert_eq!(gap.prev_token, "raclette");
});
assert_matches!(chunks.next().unwrap().content(), ChunkContent::Items(events) => {
assert_eq!(events.len(), 1);
let deserialized = events[0].raw().deserialize().unwrap();
assert_let!(AnySyncTimelineEvent::MessageLike(AnySyncMessageLikeEvent::RoomMessage(msg)) = deserialized);
assert_eq!(msg.as_original().unwrap().content.body(), "hey yo");
});
assert!(chunks.next().is_none());
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_write_to_storage_strips_bundled_relations() {
use matrix_sdk_base::linked_chunk::LinkedChunkBuilder;
use ruma::events::BundledMessageLikeRelations;
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let event_cache_store = Arc::new(MemoryStore::new());
let client = MockClientBuilder::new("http://localhost".to_owned())
.store_config(
StoreConfig::new("hodlor".to_owned()).event_cache_store(event_cache_store.clone()),
)
.build()
.await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let mut relations = BundledMessageLikeRelations::new();
relations.replace =
Some(Box::new(f.text_msg("Hello, Kind Sir").sender(*ALICE).into_raw_sync()));
let ev = f.text_msg("hey yo").sender(*ALICE).bundled_relations(relations).into_sync();
let timeline = Timeline { limited: false, prev_batch: None, events: vec![ev] };
room_event_cache
.inner
.handle_joined_room_update(true, JoinedRoomUpdate { timeline, ..Default::default() })
.await
.unwrap();
{
let (events, _) = room_event_cache.subscribe().await.unwrap();
assert_eq!(events.len(), 1);
let ev = events[0].raw().deserialize().unwrap();
assert_let!(
AnySyncTimelineEvent::MessageLike(AnySyncMessageLikeEvent::RoomMessage(msg)) = ev
);
let original = msg.as_original().unwrap();
assert_eq!(original.content.body(), "hey yo");
assert!(original.unsigned.relations.replace.is_some());
}
let raws = event_cache_store.reload_linked_chunk(room_id).await.unwrap();
let linked_chunk =
LinkedChunkBuilder::<3, _, _>::from_raw_parts(raws).build().unwrap().unwrap();
assert_eq!(linked_chunk.chunks().count(), 1);
let mut chunks = linked_chunk.chunks();
assert_matches!(chunks.next().unwrap().content(), ChunkContent::Items(events) => {
assert_eq!(events.len(), 1);
let ev = events[0].raw().deserialize().unwrap();
assert_let!(AnySyncTimelineEvent::MessageLike(AnySyncMessageLikeEvent::RoomMessage(msg)) = ev);
let original = msg.as_original().unwrap();
assert_eq!(original.content.body(), "hey yo");
assert!(original.unsigned.relations.replace.is_none());
});
assert!(chunks.next().is_none());
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_clear() {
use matrix_sdk_base::linked_chunk::LinkedChunkBuilder;
use crate::{assert_let_timeout, event_cache::RoomEventCacheUpdate};
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let event_cache_store = Arc::new(MemoryStore::new());
let event_id1 = event_id!("$1");
let event_id2 = event_id!("$2");
let ev1 = f.text_msg("hello world").sender(*ALICE).event_id(event_id1).into_sync();
let ev2 = f.text_msg("how's it going").sender(*BOB).event_id(event_id2).into_sync();
event_cache_store
.handle_linked_chunk_updates(
room_id,
vec![
Update::NewItemsChunk {
previous: None,
new: ChunkIdentifier::new(0),
next: None,
},
Update::NewGapChunk {
previous: Some(ChunkIdentifier::new(0)),
new: ChunkIdentifier::new(42),
next: None,
gap: Gap { prev_token: "cheddar".to_owned() },
},
Update::NewItemsChunk {
previous: Some(ChunkIdentifier::new(42)),
new: ChunkIdentifier::new(1),
next: None,
},
Update::PushItems {
at: Position::new(ChunkIdentifier::new(1), 0),
items: vec![ev1.clone()],
},
Update::NewItemsChunk {
previous: Some(ChunkIdentifier::new(1)),
new: ChunkIdentifier::new(2),
next: None,
},
Update::PushItems {
at: Position::new(ChunkIdentifier::new(2), 0),
items: vec![ev2.clone()],
},
],
)
.await
.unwrap();
let client = MockClientBuilder::new("http://localhost".to_owned())
.store_config(
StoreConfig::new("hodlor".to_owned()).event_cache_store(event_cache_store.clone()),
)
.build()
.await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let (items, mut stream) = room_event_cache.subscribe().await.unwrap();
assert!(room_event_cache.event(event_id1).await.is_some());
assert_eq!(items.len(), 2);
assert_eq!(items[0].event_id().unwrap(), event_id1);
assert_eq!(items[1].event_id().unwrap(), event_id2);
assert!(stream.is_empty());
room_event_cache.clear().await.unwrap();
assert_let_timeout!(Ok(RoomEventCacheUpdate::Clear) = stream.recv());
assert!(room_event_cache.event(event_id1).await.is_none());
let (items, _) = room_event_cache.subscribe().await.unwrap();
assert!(items.is_empty());
let raws = event_cache_store.reload_linked_chunk(room_id).await.unwrap();
let linked_chunk = LinkedChunkBuilder::<3, _, _>::from_raw_parts(raws).build().unwrap();
let linked_chunk = linked_chunk.unwrap();
assert_eq!(linked_chunk.num_items(), 0);
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_load_from_storage() {
let room_id = room_id!("!galette:saucisse.bzh");
let f = EventFactory::new().room(room_id).sender(user_id!("@ben:saucisse.bzh"));
let event_cache_store = Arc::new(MemoryStore::new());
let event_id1 = event_id!("$1");
let event_id2 = event_id!("$2");
let ev1 = f.text_msg("hello world").sender(*ALICE).event_id(event_id1).into_sync();
let ev2 = f.text_msg("how's it going").sender(*BOB).event_id(event_id2).into_sync();
event_cache_store
.handle_linked_chunk_updates(
room_id,
vec![
Update::NewItemsChunk {
previous: None,
new: ChunkIdentifier::new(0),
next: None,
},
Update::NewGapChunk {
previous: Some(ChunkIdentifier::new(0)),
new: ChunkIdentifier::new(42),
next: None,
gap: Gap { prev_token: "cheddar".to_owned() },
},
Update::NewItemsChunk {
previous: Some(ChunkIdentifier::new(42)),
new: ChunkIdentifier::new(1),
next: None,
},
Update::PushItems {
at: Position::new(ChunkIdentifier::new(1), 0),
items: vec![ev1.clone()],
},
Update::NewItemsChunk {
previous: Some(ChunkIdentifier::new(1)),
new: ChunkIdentifier::new(2),
next: None,
},
Update::PushItems {
at: Position::new(ChunkIdentifier::new(2), 0),
items: vec![ev2.clone()],
},
],
)
.await
.unwrap();
let client = MockClientBuilder::new("http://localhost".to_owned())
.store_config(
StoreConfig::new("hodlor".to_owned()).event_cache_store(event_cache_store.clone()),
)
.build()
.await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let (items, _stream) = room_event_cache.subscribe().await.unwrap();
assert_eq!(items.len(), 2);
assert_eq!(items[0].event_id().unwrap(), event_id1);
assert_eq!(items[1].event_id().unwrap(), event_id2);
let timeline = Timeline { limited: false, prev_batch: None, events: vec![ev2] };
room_event_cache
.inner
.handle_joined_room_update(true, JoinedRoomUpdate { timeline, ..Default::default() })
.await
.unwrap();
let (items, _stream) = room_event_cache.subscribe().await.unwrap();
assert_eq!(items.len(), 2);
assert_eq!(items[0].event_id().unwrap(), event_id1);
assert_eq!(items[1].event_id().unwrap(), event_id2);
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_load_from_storage_resilient_to_failure() {
let room_id = room_id!("!galette:saucisse.bzh");
let event_cache_store = Arc::new(MemoryStore::new());
event_cache_store
.handle_linked_chunk_updates(
room_id,
vec![
Update::NewItemsChunk {
previous: None,
new: ChunkIdentifier::new(0),
next: None,
},
Update::NewItemsChunk {
previous: Some(ChunkIdentifier::new(0)),
new: ChunkIdentifier::new(1),
next: Some(ChunkIdentifier::new(0)),
},
],
)
.await
.unwrap();
let client = MockClientBuilder::new("http://localhost".to_owned())
.store_config(
StoreConfig::new("holder".to_owned()).event_cache_store(event_cache_store.clone()),
)
.build()
.await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let (items, _stream) = room_event_cache.subscribe().await.unwrap();
assert!(items.is_empty());
let raw_chunks = event_cache_store.reload_linked_chunk(room_id).await.unwrap();
assert!(raw_chunks.is_empty());
}
#[cfg(not(target_arch = "wasm32"))] #[async_test]
async fn test_no_useless_gaps() {
let room_id = room_id!("!galette:saucisse.bzh");
let client = MockClientBuilder::new("http://localhost".to_owned()).build().await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
let has_storage = true; event_cache.enable_storage().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let f = EventFactory::new().room(room_id).sender(*ALICE);
room_event_cache
.inner
.handle_joined_room_update(
has_storage,
JoinedRoomUpdate {
timeline: Timeline {
limited: true,
prev_batch: Some("raclette".to_owned()),
events: vec![f.text_msg("hey yo").into_sync()],
},
..Default::default()
},
)
.await
.unwrap();
{
let state = room_event_cache.inner.state.read().await;
let mut num_gaps = 0;
let mut num_events = 0;
for c in state.events().chunks() {
match c.content() {
ChunkContent::Items(items) => num_events += items.len(),
ChunkContent::Gap(_) => num_gaps += 1,
}
}
assert_eq!(num_gaps, 1);
assert_eq!(num_events, 1);
}
room_event_cache
.inner
.handle_joined_room_update(
has_storage,
JoinedRoomUpdate {
timeline: Timeline {
limited: false,
prev_batch: Some("fondue".to_owned()),
events: vec![f.text_msg("sup").into_sync()],
},
..Default::default()
},
)
.await
.unwrap();
{
let state = room_event_cache.inner.state.read().await;
let mut num_gaps = 0;
let mut num_events = 0;
for c in state.events().chunks() {
match c.content() {
ChunkContent::Items(items) => num_events += items.len(),
ChunkContent::Gap(gap) => {
assert_eq!(gap.prev_token, "raclette");
num_gaps += 1;
}
}
}
assert_eq!(num_gaps, 1);
assert_eq!(num_events, 2);
}
}
async fn assert_relations(
room_id: &RoomId,
original_event: SyncTimelineEvent,
related_event: SyncTimelineEvent,
event_factory: EventFactory,
) {
let client = logged_in_client(None).await;
let event_cache = client.event_cache();
event_cache.subscribe().unwrap();
client.base_client().get_or_create_room(room_id, matrix_sdk_base::RoomState::Joined);
let room = client.get_room(room_id).unwrap();
let (room_event_cache, _drop_handles) = room.event_cache().await.unwrap();
let original_event_id = original_event.event_id().unwrap();
room_event_cache.save_event(original_event).await;
let unrelated_id = event_id!("$2");
room_event_cache
.save_event(event_factory.text_msg("An unrelated event").event_id(unrelated_id).into())
.await;
let related_id = related_event.event_id().unwrap();
room_event_cache.save_event(related_event).await;
let (event, related_events) =
room_event_cache.event_with_relations(&original_event_id, None).await.unwrap();
let cached_event_id = event.event_id().unwrap();
assert_eq!(cached_event_id, original_event_id);
assert_eq!(related_events.len(), 1);
let related_event_id = related_events[0].event_id().unwrap();
assert_eq!(related_event_id, related_id);
}
}