use as_variant::as_variant;
use eyeball_im::VectorDiff;
pub use matrix_sdk_base::event_cache::{Event, Gap};
use matrix_sdk_base::{
event_cache::store::DEFAULT_CHUNK_CAPACITY,
linked_chunk::{
lazy_loader::{self, LazyLoaderError},
ChunkContent, ChunkIdentifierGenerator, RawChunk,
},
};
use matrix_sdk_common::linked_chunk::{
AsVector, Chunk, ChunkIdentifier, Error, Iter, IterBackward, LinkedChunk, ObservableUpdates,
Position,
};
#[derive(Debug)]
pub struct RoomEvents {
chunks: LinkedChunk<DEFAULT_CHUNK_CAPACITY, Event, Gap>,
chunks_updates_as_vectordiffs: AsVector<Event, Gap>,
}
impl Default for RoomEvents {
fn default() -> Self {
Self::new()
}
}
impl RoomEvents {
pub fn new() -> Self {
Self::with_initial_linked_chunk(None)
}
pub fn with_initial_linked_chunk(
linked_chunk: Option<LinkedChunk<DEFAULT_CHUNK_CAPACITY, Event, Gap>>,
) -> Self {
let mut linked_chunk = linked_chunk.unwrap_or_else(LinkedChunk::new_with_update_history);
let chunks_updates_as_vectordiffs = linked_chunk
.as_vector()
.expect("`LinkedChunk` must have been built with `new_with_update_history`");
Self { chunks: linked_chunk, chunks_updates_as_vectordiffs }
}
pub fn is_empty(&self) -> bool {
self.chunks.num_items() == 0
}
pub fn reset(&mut self) {
self.chunks.clear();
}
pub(super) fn replace_with(
&mut self,
last_chunk: Option<RawChunk<Event, Gap>>,
chunk_identifier_generator: ChunkIdentifierGenerator,
) -> Result<(), LazyLoaderError> {
lazy_loader::replace_with(&mut self.chunks, last_chunk, chunk_identifier_generator)
}
pub fn push_events<I>(&mut self, events: I)
where
I: IntoIterator<Item = Event>,
I::IntoIter: ExactSizeIterator,
{
self.chunks.push_items_back(events);
}
pub fn push_gap(&mut self, gap: Gap) {
self.chunks.push_gap_back(gap);
}
pub fn insert_events_at(
&mut self,
events: Vec<Event>,
position: Position,
) -> Result<(), Error> {
self.chunks.insert_items_at(events, position)?;
Ok(())
}
pub fn insert_gap_at(&mut self, gap: Gap, position: Position) -> Result<(), Error> {
self.chunks.insert_gap_at(gap, position)
}
pub fn remove_empty_chunk_at(
&mut self,
gap: ChunkIdentifier,
) -> Result<Option<Position>, Error> {
self.chunks.remove_empty_chunk_at(gap)
}
pub fn replace_gap_at(
&mut self,
events: Vec<Event>,
gap_identifier: ChunkIdentifier,
) -> Result<Option<Position>, Error> {
let has_only_one_chunk = {
let mut it = self.chunks.chunks();
let _ =
it.next().ok_or(Error::InvalidChunkIdentifier { identifier: gap_identifier })?;
it.next().is_none()
};
let next_pos = if events.is_empty() && !has_only_one_chunk {
self.chunks.remove_empty_chunk_at(gap_identifier)?
} else {
Some(self.chunks.replace_gap_at(events, gap_identifier)?.first_position())
};
Ok(next_pos)
}
pub fn remove_events_by_position(&mut self, mut positions: Vec<Position>) -> Result<(), Error> {
sort_positions_descending(&mut positions);
for position in positions {
self.chunks.remove_item_at(position)?;
}
Ok(())
}
pub fn replace_event_at(&mut self, position: Position, event: Event) -> Result<(), Error> {
self.chunks.replace_item_at(position, event)
}
pub fn chunk_identifier<'a, P>(&'a self, predicate: P) -> Option<ChunkIdentifier>
where
P: FnMut(&'a Chunk<DEFAULT_CHUNK_CAPACITY, Event, Gap>) -> bool,
{
self.chunks.chunk_identifier(predicate)
}
pub fn chunks(&self) -> Iter<'_, DEFAULT_CHUNK_CAPACITY, Event, Gap> {
self.chunks.chunks()
}
pub fn rchunks(&self) -> IterBackward<'_, DEFAULT_CHUNK_CAPACITY, Event, Gap> {
self.chunks.rchunks()
}
pub fn revents(&self) -> impl Iterator<Item = (Position, &Event)> {
self.chunks.ritems()
}
pub fn events(&self) -> impl Iterator<Item = (Position, &Event)> {
self.chunks.items()
}
pub fn updates_as_vector_diffs(&mut self) -> Vec<VectorDiff<Event>> {
self.chunks_updates_as_vectordiffs.take()
}
pub(super) fn store_updates(&mut self) -> &mut ObservableUpdates<Event, Gap> {
self.chunks.updates().expect("this is always built with an update history in the ctor")
}
pub fn debug_string(&self) -> Vec<String> {
let mut result = Vec::new();
for chunk in self.chunks() {
let content = chunk_debug_string(chunk.content());
let lazy_previous = if let Some(cid) = chunk.lazy_previous() {
format!(" (lazy previous = {})", cid.index())
} else {
"".to_owned()
};
let line = format!("chunk #{}{lazy_previous}: {content}", chunk.identifier().index());
result.push(line);
}
result
}
pub fn rgap(&self) -> Option<Gap> {
self.rchunks()
.find_map(|chunk| as_variant!(chunk.content(), ChunkContent::Gap(gap) => gap.clone()))
}
}
impl RoomEvents {
pub(super) fn insert_new_chunk_as_first(
&mut self,
raw_new_first_chunk: RawChunk<Event, Gap>,
) -> Result<(), LazyLoaderError> {
lazy_loader::insert_new_first_chunk(&mut self.chunks, raw_new_first_chunk)
}
}
fn chunk_debug_string(content: &ChunkContent<Event, 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}]")
}
}
}
pub(super) fn sort_positions_descending(positions: &mut [Position]) {
positions.sort_by(|a, b| {
b.chunk_identifier()
.cmp(&a.chunk_identifier())
.then_with(|| a.index().cmp(&b.index()).reverse())
});
}
#[cfg(test)]
mod tests {
use assert_matches::assert_matches;
use assert_matches2::assert_let;
use matrix_sdk_test::{event_factory::EventFactory, ALICE, DEFAULT_TEST_ROOM_ID};
use ruma::{event_id, user_id, EventId, OwnedEventId};
use super::*;
macro_rules! assert_events_eq {
( $events_iterator:expr, [ $( ( $event_id:ident at ( $chunk_identifier:literal, $index:literal ) ) ),* $(,)? ] ) => {
{
let mut events = $events_iterator;
$(
assert_let!(Some((position, event)) = events.next());
assert_eq!(position.chunk_identifier(), $chunk_identifier );
assert_eq!(position.index(), $index );
assert_eq!(event.event_id().unwrap(), $event_id );
)*
assert!(events.next().is_none(), "No more events are expected");
}
};
}
fn new_event(event_id: &str) -> (OwnedEventId, Event) {
let event_id = EventId::parse(event_id).unwrap();
let event = EventFactory::new()
.text_msg("")
.sender(user_id!("@mnt_io:matrix.org"))
.event_id(&event_id)
.into_event();
(event_id, event)
}
#[test]
fn test_new_room_events_has_zero_events() {
let room_events = RoomEvents::new();
assert_eq!(room_events.events().count(), 0);
}
#[test]
fn test_push_events() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let (event_id_2, event_2) = new_event("$ev2");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
room_events.push_events([event_2]);
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_1 at (0, 1)),
(event_id_2 at (0, 2)),
]
);
}
#[test]
fn test_push_gap() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0]);
room_events.push_gap(Gap { prev_token: "hello".to_owned() });
room_events.push_events([event_1]);
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_1 at (2, 0)),
]
);
{
let mut chunks = room_events.chunks();
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_gap());
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert!(chunks.next().is_none());
}
}
#[test]
fn test_insert_events_at() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let (event_id_2, event_2) = new_event("$ev2");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
let position_of_event_1 = room_events
.events()
.find_map(|(position, event)| {
(event.event_id().unwrap() == event_id_1).then_some(position)
})
.unwrap();
room_events.insert_events_at(vec![event_2], position_of_event_1).unwrap();
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_2 at (0, 1)),
(event_id_1 at (0, 2)),
]
);
}
#[test]
fn test_insert_gap_at() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
let position_of_event_1 = room_events
.events()
.find_map(|(position, event)| {
(event.event_id().unwrap() == event_id_1).then_some(position)
})
.unwrap();
room_events
.insert_gap_at(Gap { prev_token: "hello".to_owned() }, position_of_event_1)
.unwrap();
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_1 at (2, 0)),
]
);
{
let mut chunks = room_events.chunks();
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_gap());
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert!(chunks.next().is_none());
}
}
#[test]
fn test_replace_gap_at() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let (event_id_2, event_2) = new_event("$ev2");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0]);
room_events.push_gap(Gap { prev_token: "hello".to_owned() });
let chunk_identifier_of_gap = room_events
.chunks()
.find_map(|chunk| chunk.is_gap().then_some(chunk.identifier()))
.unwrap();
room_events.replace_gap_at(vec![event_1, event_2], chunk_identifier_of_gap).unwrap();
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_1 at (2, 0)),
(event_id_2 at (2, 1)),
]
);
{
let mut chunks = room_events.chunks();
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert_let!(Some(chunk) = chunks.next());
assert!(chunk.is_items());
assert!(chunks.next().is_none());
}
}
#[test]
fn test_replace_gap_at_with_no_new_events() {
let (_, event_0) = new_event("$ev0");
let (_, event_1) = new_event("$ev1");
let (_, event_2) = new_event("$ev2");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
room_events.push_gap(Gap { prev_token: "middle".to_owned() });
room_events.push_events([event_2]);
room_events.push_gap(Gap { prev_token: "end".to_owned() });
let first_gap_id = room_events
.chunks()
.find_map(|chunk| chunk.is_gap().then_some(chunk.identifier()))
.unwrap();
let pos = room_events.replace_gap_at(vec![], first_gap_id).unwrap();
assert_eq!(pos, Some(Position::new(ChunkIdentifier::new(2), 0)));
let second_gap_id = room_events
.chunks()
.find_map(|chunk| chunk.is_gap().then_some(chunk.identifier()))
.unwrap();
let pos = room_events.replace_gap_at(vec![], second_gap_id).unwrap();
assert!(pos.is_none());
}
#[test]
fn test_remove_events() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let (event_id_2, event_2) = new_event("$ev2");
let (event_id_3, event_3) = new_event("$ev3");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
room_events.push_gap(Gap { prev_token: "hello".to_owned() });
room_events.push_events([event_2, event_3]);
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_1 at (0, 1)),
(event_id_2 at (2, 0)),
(event_id_3 at (2, 1)),
]
);
assert_eq!(room_events.chunks().count(), 3);
room_events
.remove_events_by_position(vec![
Position::new(ChunkIdentifier::new(2), 1),
Position::new(ChunkIdentifier::new(0), 1),
])
.unwrap();
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
(event_id_2 at (2, 0)),
]
);
room_events
.remove_events_by_position(vec![Position::new(ChunkIdentifier::new(2), 0)])
.unwrap();
assert_events_eq!(
room_events.events(),
[
(event_id_0 at (0, 0)),
]
);
assert_eq!(room_events.chunks().count(), 2);
}
#[test]
fn test_remove_events_unknown_event() {
let mut room_events = RoomEvents::new();
assert_events_eq!(room_events.events(), []);
room_events
.remove_events_by_position(vec![Position::new(ChunkIdentifier::new(42), 153)])
.unwrap_err();
assert_events_eq!(room_events.events(), []);
let mut events = room_events.events();
assert!(events.next().is_none());
}
#[test]
fn test_reset() {
let (event_id_0, event_0) = new_event("$ev0");
let (event_id_1, event_1) = new_event("$ev1");
let (event_id_2, event_2) = new_event("$ev2");
let (event_id_3, event_3) = new_event("$ev3");
let mut room_events = RoomEvents::new();
room_events.push_events([event_0, event_1]);
room_events.push_gap(Gap { prev_token: "raclette".to_owned() });
room_events.push_events([event_2]);
let diffs = room_events.updates_as_vector_diffs();
assert_eq!(diffs.len(), 2);
assert_matches!(
&diffs[0],
VectorDiff::Append { values } => {
assert_eq!(values.len(), 2);
assert_eq!(values[0].event_id(), Some(event_id_0));
assert_eq!(values[1].event_id(), Some(event_id_1));
}
);
assert_matches!(
&diffs[1],
VectorDiff::Append { values } => {
assert_eq!(values.len(), 1);
assert_eq!(values[0].event_id(), Some(event_id_2));
}
);
room_events.reset();
room_events.push_events([event_3]);
let diffs = room_events.updates_as_vector_diffs();
assert_eq!(diffs.len(), 2);
assert_matches!(&diffs[0], VectorDiff::Clear);
assert_matches!(
&diffs[1],
VectorDiff::Append { values } => {
assert_eq!(values.len(), 1);
assert_eq!(values[0].event_id(), Some(event_id_3));
}
);
}
#[test]
fn test_debug_string() {
let event_factory = EventFactory::new().room(&DEFAULT_TEST_ROOM_ID).sender(*ALICE);
let mut room_events = RoomEvents::new();
room_events.push_events(vec![
event_factory
.text_msg("hey")
.event_id(event_id!("$123456789101112131415617181920"))
.into_event(),
event_factory.text_msg("you").event_id(event_id!("$2")).into_event(),
]);
room_events.push_gap(Gap { prev_token: "raclette".to_owned() });
let output = room_events.debug_string();
assert_eq!(output.len(), 2);
assert_eq!(&output[0], "chunk #0: events[$12345678, $2]");
assert_eq!(&output[1], "chunk #1: gap['raclette']");
}
#[test]
fn test_sort_positions_descending() {
let mut positions = vec![
Position::new(ChunkIdentifier::new(2), 1),
Position::new(ChunkIdentifier::new(1), 0),
Position::new(ChunkIdentifier::new(2), 0),
Position::new(ChunkIdentifier::new(1), 1),
Position::new(ChunkIdentifier::new(0), 0),
];
sort_positions_descending(&mut positions);
assert_eq!(
positions,
&[
Position::new(ChunkIdentifier::new(2), 1),
Position::new(ChunkIdentifier::new(2), 0),
Position::new(ChunkIdentifier::new(1), 1),
Position::new(ChunkIdentifier::new(1), 0),
Position::new(ChunkIdentifier::new(0), 0),
]
);
}
}