use super::State; use crate::{ConduitResult, Database, Error, Ruma}; use ruma::api::client::{error::ErrorKind, r0::search::search_events}; #[cfg(feature = "conduit_bin")] use rocket::post; use search_events::{EventContextResult, ResultCategories, ResultRoomEvents, SearchResult}; use std::collections::BTreeMap; #[cfg_attr( feature = "conduit_bin", post("/_matrix/client/r0/search", data = "") )] pub async fn search_events_route( db: State<'_, Database>, body: Ruma>, ) -> ConduitResult { let sender_user = body.sender_user.as_ref().expect("user is authenticated"); let search_criteria = body.search_categories.room_events.as_ref().unwrap(); let filter = search_criteria.filter.as_ref().unwrap(); let room_id = filter.rooms.as_ref().unwrap().first().unwrap(); let limit = filter.limit.map_or(10, |l| u64::from(l) as usize); if !db.rooms.is_joined(sender_user, &room_id)? { return Err(Error::BadRequest( ErrorKind::Forbidden, "You don't have permission to view this room.", )); } let skip = match body.next_batch.as_ref().map(|s| s.parse()) { Some(Ok(s)) => s, Some(Err(_)) => { return Err(Error::BadRequest( ErrorKind::InvalidParam, "Invalid next_batch token.", )) } None => 0, // Default to the start }; let search = db .rooms .search_pdus(&room_id, &search_criteria.search_term)?; let results = search .0 .map(|result| { Ok::<_, Error>(SearchResult { context: EventContextResult { end: None, events_after: Vec::new(), events_before: Vec::new(), profile_info: BTreeMap::new(), start: None, }, rank: None, result: db .rooms .get_pdu_from_id(&result)? .map(|pdu| pdu.to_room_event()), }) }) .filter_map(|r| r.ok()) .skip(skip) .take(limit) .collect::>(); let next_batch = if results.len() < limit as usize { None } else { Some((skip + limit).to_string()) }; Ok(search_events::Response::new(ResultCategories { room_events: ResultRoomEvents { count: None, // TODO? maybe not groups: BTreeMap::new(), // TODO next_batch, results, state: BTreeMap::new(), // TODO highlights: search.1, }, }) .into()) }