mod data; use std::collections::{HashMap, HashSet}; pub(crate) use data::Data; use ruma::{DeviceId, OwnedDeviceId, OwnedRoomId, OwnedUserId, RoomId, UserId}; use tokio::sync::Mutex; use super::timeline::PduCount; use crate::Result; pub(crate) struct Service { pub(crate) db: &'static dyn Data, #[allow(clippy::type_complexity)] pub(crate) lazy_load_waiting: Mutex>>, } impl Service { #[tracing::instrument(skip(self))] pub(crate) fn lazy_load_was_sent_before( &self, user_id: &UserId, device_id: &DeviceId, room_id: &RoomId, ll_user: &UserId, ) -> Result { self.db .lazy_load_was_sent_before(user_id, device_id, room_id, ll_user) } #[tracing::instrument(skip(self))] pub(crate) async fn lazy_load_mark_sent( &self, user_id: &UserId, device_id: &DeviceId, room_id: &RoomId, lazy_load: HashSet, count: PduCount, ) { self.lazy_load_waiting .lock() .await .insert((user_id.to_owned(), device_id.to_owned(), room_id.to_owned(), count), lazy_load); } #[tracing::instrument(skip(self))] pub(crate) async fn lazy_load_confirm_delivery( &self, user_id: &UserId, device_id: &DeviceId, room_id: &RoomId, since: PduCount, ) -> Result<()> { if let Some(user_ids) = self.lazy_load_waiting.lock().await.remove(&( user_id.to_owned(), device_id.to_owned(), room_id.to_owned(), since, )) { self.db .lazy_load_confirm_delivery(user_id, device_id, room_id, &mut user_ids.iter().map(|u| &**u))?; } else { // Ignore } Ok(()) } #[tracing::instrument(skip(self))] pub(crate) fn lazy_load_reset(&self, user_id: &UserId, device_id: &DeviceId, room_id: &RoomId) -> Result<()> { self.db.lazy_load_reset(user_id, device_id, room_id) } }