Skip to content
Snippets Groups Projects
message.rs 8.87 KiB
Newer Older
  • Learn to ignore specific revisions
  • Timo Kösters's avatar
    Timo Kösters committed
    use crate::{
        service::{pdu::PduBuilder, rooms::timeline::PduCount},
        services, utils, Error, Result, Ruma,
    };
    
    use ruma::{
        api::client::{
            error::ErrorKind,
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
            message::{get_message_events, send_message_event},
    
    Kévin Commaille's avatar
    Kévin Commaille committed
        events::{StateEventType, TimelineEventType},
    
    Timo Kösters's avatar
    Timo Kösters committed
    use std::{
        collections::{BTreeMap, HashSet},
        sync::Arc,
    };
    
    /// # `PUT /_matrix/client/v3/rooms/{roomId}/send/{eventType}/{txnId}`
    
    ///
    /// Send a message event into the room.
    ///
    /// - Is a NOOP if the txn id was already used before and returns the same event id again
    /// - The only requirement for the content is that it has to be valid json
    /// - Tries to send the event into the room, auth rules will determine if it is allowed
    
    pub async fn send_message_event_route(
    
    Jonas Platte's avatar
    Jonas Platte committed
        body: Ruma<send_message_event::v3::Request>,
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
    ) -> Result<send_message_event::v3::Response> {
    
        let sender_user = body.sender_user.as_ref().expect("user is authenticated");
    
        let sender_device = body.sender_device.as_deref();
    
    Timo's avatar
    Timo committed
    
    
    Timo Kösters's avatar
    Timo Kösters committed
        let mutex_state = Arc::clone(
    
    Timo Kösters's avatar
    Timo Kösters committed
            services()
                .globals
    
    Timo Kösters's avatar
    Timo Kösters committed
                .roomid_mutex_state
    
    Timo Kösters's avatar
    Timo Kösters committed
                .write()
                .unwrap()
                .entry(body.room_id.clone())
                .or_default(),
        );
    
    Timo Kösters's avatar
    Timo Kösters committed
        let state_lock = mutex_state.lock().await;
    
    Timo Kösters's avatar
    Timo Kösters committed
    
    
        // Forbid m.room.encrypted if encryption is disabled
    
    Kévin Commaille's avatar
    Kévin Commaille committed
        if TimelineEventType::RoomEncrypted == body.event_type.to_string().into()
    
            && !services().globals.allow_encryption()
    
    Timo Kösters's avatar
    Timo Kösters committed
        {
    
            return Err(Error::BadRequest(
                ErrorKind::Forbidden,
                "Encryption has been disabled",
            ));
        }
    
    
    Timo's avatar
    Timo committed
        // Check if this is a new transaction id
    
        if let Some(response) =
    
    Timo Kösters's avatar
    Timo Kösters committed
            services()
                .transaction_ids
    
                .existing_txnid(sender_user, sender_device, &body.txn_id)?
    
    Timo's avatar
    Timo committed
        {
            // The client might have sent a txnid of the /sendToDevice endpoint
            // This txnid has no response associated with it
            if response.is_empty() {
                return Err(Error::BadRequest(
                    ErrorKind::InvalidParam,
                    "Tried to use txn id already used for an incompatible endpoint.",
                ));
            }
    
    
            let event_id = utils::string_from_bytes(&response)
                .map_err(|_| Error::bad_database("Invalid txnid bytes in database."))?
                .try_into()
                .map_err(|_| Error::bad_database("Invalid event id in txnid data."))?;
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
            return Ok(send_message_event::v3::Response { event_id });
    
    Timo's avatar
    Timo committed
        }
    
        let mut unsigned = BTreeMap::new();
    
        unsigned.insert("transaction_id".to_owned(), body.txn_id.to_string().into());
    
        let event_id = services()
            .rooms
            .timeline
            .build_and_append_pdu(
                PduBuilder {
                    event_type: body.event_type.to_string().into(),
                    content: serde_json::from_str(body.body.body.json().get())
                        .map_err(|_| Error::BadRequest(ErrorKind::BadJson, "Invalid JSON body."))?,
                    unsigned: Some(unsigned),
                    state_key: None,
                    redacts: None,
                },
                sender_user,
                &body.room_id,
                &state_lock,
            )
            .await?;
    
        services().transaction_ids.add_txnid(
    
            sender_user,
            sender_device,
            &body.txn_id,
            event_id.as_bytes(),
        )?;
    
    Timo Kösters's avatar
    Timo Kösters committed
        drop(state_lock);
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
        Ok(send_message_event::v3::Response::new(
            (*event_id).to_owned(),
        ))
    
    /// # `GET /_matrix/client/r0/rooms/{roomId}/messages`
    ///
    /// Allows paginating through room history.
    ///
    /// - Only works if the user is joined (TODO: always allow, but only show events where the user was
    /// joined, depending on history_visibility)
    
    pub async fn get_message_events_route(
    
    Jonas Platte's avatar
    Jonas Platte committed
        body: Ruma<get_message_events::v3::Request>,
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
    ) -> Result<get_message_events::v3::Response> {
    
        let sender_user = body.sender_user.as_ref().expect("user is authenticated");
    
    Timo Kösters's avatar
    Timo Kösters committed
        let sender_device = body.sender_device.as_ref().expect("user is authenticated");
    
    Timo Kösters's avatar
    Timo Kösters committed
        let from = match body.from.clone() {
    
    Timo Kösters's avatar
    Timo Kösters committed
            Some(from) => PduCount::try_from_string(&from)?,
    
    Timo Kösters's avatar
    Timo Kösters committed
            None => match body.dir {
    
    Kévin Commaille's avatar
    Kévin Commaille committed
                ruma::api::Direction::Forward => PduCount::min(),
                ruma::api::Direction::Backward => PduCount::max(),
    
    Timo Kösters's avatar
    Timo Kösters committed
            },
        };
    
    Timo Kösters's avatar
    Timo Kösters committed
        let to = body
            .to
            .as_ref()
    
            .and_then(|t| PduCount::try_from_string(t).ok());
    
    Timo Kösters's avatar
    Timo Kösters committed
        services().rooms.lazy_loading.lazy_load_confirm_delivery(
            sender_user,
            sender_device,
            &body.room_id,
            from,
        )?;
    
        let limit = u64::from(body.limit).min(100) as usize;
    
    Timo Kösters's avatar
    Timo Kösters committed
        let next_token;
    
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
        let mut resp = get_message_events::v3::Response::new();
    
    Timo Kösters's avatar
    Timo Kösters committed
    
        let mut lazy_loaded = HashSet::new();
    
    
        match body.dir {
    
    Kévin Commaille's avatar
    Kévin Commaille committed
            ruma::api::Direction::Forward => {
    
                let events_after: Vec<_> = services()
    
    Timo Kösters's avatar
    Timo Kösters committed
                    .timeline
    
                    .pdus_after(sender_user, &body.room_id, from)?
    
                    .take(limit)
                    .filter_map(|r| r.ok()) // Filter out buggy events
    
                    .filter(|(_, pdu)| {
                        services()
                            .rooms
                            .state_accessor
                            .user_can_see_event(sender_user, &body.room_id, &pdu.event_id)
                            .unwrap_or(false)
                    })
    
    Timo Kösters's avatar
    Timo Kösters committed
                    .take_while(|&(k, _)| Some(k) != to) // Stop at `to`
    
    Jonas Platte's avatar
    Jonas Platte committed
                    .collect();
    
    Timo Kösters's avatar
    Timo Kösters committed
                for (_, event) in &events_after {
    
                    /* TODO: Remove this when these are resolved:
                     * https://github.com/vector-im/element-android/issues/3417
                     * https://github.com/vector-im/element-web/issues/21034
    
    Timo Kösters's avatar
    Timo Kösters committed
                    if !services().rooms.lazy_loading.lazy_load_was_sent_before(
    
                        sender_user,
                        sender_device,
    
    Timo Kösters's avatar
    Timo Kösters committed
                        &body.room_id,
                        &event.sender,
                    )? {
                        lazy_loaded.insert(event.sender.clone());
                    }
    
                    */
                    lazy_loaded.insert(event.sender.clone());
    
    Timo Kösters's avatar
    Timo Kösters committed
                }
    
                next_token = events_after.last().map(|(count, _)| count).copied();
    
    Jonas Platte's avatar
    Jonas Platte committed
                let events_after: Vec<_> = events_after
    
                    .into_iter()
                    .map(|(_, pdu)| pdu.to_room_event())
    
    Jonas Platte's avatar
    Jonas Platte committed
                    .collect();
    
    Timo Kösters's avatar
    Timo Kösters committed
                resp.start = from.stringify();
                resp.end = next_token.map(|count| count.stringify());
    
    Timo Kösters's avatar
    Timo Kösters committed
                resp.chunk = events_after;
    
    Kévin Commaille's avatar
    Kévin Commaille committed
            ruma::api::Direction::Backward => {
    
    Timo Kösters's avatar
    Timo Kösters committed
                services()
                    .rooms
                    .timeline
                    .backfill_if_required(&body.room_id, from)
                    .await?;
    
                let events_before: Vec<_> = services()
    
    Timo Kösters's avatar
    Timo Kösters committed
                    .timeline
    
                    .pdus_until(sender_user, &body.room_id, from)?
    
                    .take(limit)
                    .filter_map(|r| r.ok()) // Filter out buggy events
    
                    .filter(|(_, pdu)| {
                        services()
                            .rooms
                            .state_accessor
                            .user_can_see_event(sender_user, &body.room_id, &pdu.event_id)
                            .unwrap_or(false)
                    })
    
    Timo Kösters's avatar
    Timo Kösters committed
                    .take_while(|&(k, _)| Some(k) != to) // Stop at `to`
    
    Jonas Platte's avatar
    Jonas Platte committed
                    .collect();
    
    Timo Kösters's avatar
    Timo Kösters committed
                for (_, event) in &events_before {
    
                    /* TODO: Remove this when these are resolved:
                     * https://github.com/vector-im/element-android/issues/3417
                     * https://github.com/vector-im/element-web/issues/21034
    
    Timo Kösters's avatar
    Timo Kösters committed
                    if !services().rooms.lazy_loading.lazy_load_was_sent_before(
    
                        sender_user,
                        sender_device,
    
    Timo Kösters's avatar
    Timo Kösters committed
                        &body.room_id,
                        &event.sender,
                    )? {
                        lazy_loaded.insert(event.sender.clone());
                    }
    
                    */
                    lazy_loaded.insert(event.sender.clone());
    
    Timo Kösters's avatar
    Timo Kösters committed
                }
    
                next_token = events_before.last().map(|(count, _)| count).copied();
    
    Jonas Platte's avatar
    Jonas Platte committed
                let events_before: Vec<_> = events_before
    
                    .into_iter()
                    .map(|(_, pdu)| pdu.to_room_event())
    
    Jonas Platte's avatar
    Jonas Platte committed
                    .collect();
    
    Timo Kösters's avatar
    Timo Kösters committed
                resp.start = from.stringify();
                resp.end = next_token.map(|count| count.stringify());
    
    Timo Kösters's avatar
    Timo Kösters committed
                resp.chunk = events_before;
            }
        }
    
    Timo Kösters's avatar
    Timo Kösters committed
        resp.state = Vec::new();
        for ll_id in &lazy_loaded {
    
    Timo Kösters's avatar
    Timo Kösters committed
            if let Some(member_event) = services().rooms.state_accessor.room_state_get(
                &body.room_id,
                &StateEventType::RoomMember,
                ll_id.as_str(),
            )? {
    
    Timo Kösters's avatar
    Timo Kösters committed
                resp.state.push(member_event.to_state_event());
    
        // TODO: enable again when we are sure clients can handle it
        /*
    
    Timo Kösters's avatar
    Timo Kösters committed
        if let Some(next_token) = next_token {
    
    Timo Kösters's avatar
    Timo Kösters committed
            services().rooms.lazy_loading.lazy_load_mark_sent(
    
                sender_user,
                sender_device,
    
    Timo Kösters's avatar
    Timo Kösters committed
                &body.room_id,
    
    Timo Kösters's avatar
    Timo Kösters committed
                next_token,
            );
        }