use axum::extract::State; use conduwuit::{ PduCount, Result, at, utils::{IterStream, ReadyExt, result::FlatOk, stream::WidebandExt}, }; use futures::StreamExt; use ruma::{ EventId, RoomId, UInt, UserId, api::{ Direction, client::relations::{ get_relating_events, get_relating_events_with_rel_type, get_relating_events_with_rel_type_and_event_type, }, }, events::{TimelineEventType, relation::RelationType}, }; use service::{Services, rooms::timeline::PdusIterItem}; use crate::Ruma; /// # `GET /_matrix/client/r0/rooms/{roomId}/relations/{eventId}/{relType}/{eventType}` pub(crate) async fn get_relating_events_with_rel_type_and_event_type_route( State(services): State, body: Ruma, ) -> Result { paginate_relations_with_filter( &services, body.sender_user(), &body.room_id, &body.event_id, body.event_type.clone().into(), body.rel_type.clone().into(), body.from.as_deref(), body.to.as_deref(), body.limit, body.recurse, body.dir, ) .await .map(|res| get_relating_events_with_rel_type_and_event_type::v1::Response { chunk: res.chunk, next_batch: res.next_batch, prev_batch: res.prev_batch, recursion_depth: res.recursion_depth, }) } /// # `GET /_matrix/client/r0/rooms/{roomId}/relations/{eventId}/{relType}` pub(crate) async fn get_relating_events_with_rel_type_route( State(services): State, body: Ruma, ) -> Result { paginate_relations_with_filter( &services, body.sender_user(), &body.room_id, &body.event_id, None, body.rel_type.clone().into(), body.from.as_deref(), body.to.as_deref(), body.limit, body.recurse, body.dir, ) .await .map(|res| get_relating_events_with_rel_type::v1::Response { chunk: res.chunk, next_batch: res.next_batch, prev_batch: res.prev_batch, recursion_depth: res.recursion_depth, }) } /// # `GET /_matrix/client/r0/rooms/{roomId}/relations/{eventId}` pub(crate) async fn get_relating_events_route( State(services): State, body: Ruma, ) -> Result { paginate_relations_with_filter( &services, body.sender_user(), &body.room_id, &body.event_id, None, None, body.from.as_deref(), body.to.as_deref(), body.limit, body.recurse, body.dir, ) .await } #[allow(clippy::too_many_arguments)] async fn paginate_relations_with_filter( services: &Services, sender_user: &UserId, room_id: &RoomId, target: &EventId, filter_event_type: Option, filter_rel_type: Option, from: Option<&str>, to: Option<&str>, limit: Option, recurse: bool, dir: Direction, ) -> Result { let start: PduCount = from .map(str::parse) .transpose()? .unwrap_or_else(|| match dir { | Direction::Forward => PduCount::min(), | Direction::Backward => PduCount::max(), }); let to: Option = to.map(str::parse).flat_ok(); // Use limit or else 30, with maximum 100 let limit: usize = limit .map(TryInto::try_into) .flat_ok() .unwrap_or(30) .min(100); // Spec (v1.10) recommends depth of at least 3 let depth: u8 = if recurse { 3 } else { 1 }; let events: Vec = services .rooms .pdu_metadata .get_relations(sender_user, room_id, target, start, limit, depth, dir) .await .into_iter() .filter(|(_, pdu)| { filter_event_type .as_ref() .is_none_or(|kind| *kind == pdu.kind) }) .filter(|(_, pdu)| { filter_rel_type .as_ref() .is_none_or(|rel_type| pdu.relation_type_equal(rel_type)) }) .stream() .ready_take_while(|(count, _)| Some(*count) != to) .wide_filter_map(|item| visibility_filter(services, sender_user, item)) .take(limit) .collect() .await; let next_batch = match dir { | Direction::Forward => events.last(), | Direction::Backward => events.first(), } .map(at!(0)) .as_ref() .map(ToString::to_string); Ok(get_relating_events::v1::Response { next_batch, prev_batch: from.map(Into::into), recursion_depth: recurse.then_some(depth.into()), chunk: events .into_iter() .map(at!(1)) .map(|pdu| pdu.to_message_like_event()) .collect(), }) } async fn visibility_filter( services: &Services, sender_user: &UserId, item: PdusIterItem, ) -> Option { let (_, pdu) = &item; services .rooms .state_accessor .user_can_see_event(sender_user, &pdu.room_id, &pdu.event_id) .await .then_some(item) }