Skip to content
Snippets Groups Projects
report.rs 4.04 KiB
Newer Older
  • Learn to ignore specific revisions
  • 🥺's avatar
    🥺 committed
    use std::time::Duration;
    
    
    Timo Kösters's avatar
    Timo Kösters committed
    use crate::{services, utils::HtmlEscape, Error, Result, Ruma};
    
    🥺's avatar
    🥺 committed
    use rand::Rng;
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
        api::client::{error::ErrorKind, room::report_content},
    
        events::room::message,
    
    🥺's avatar
    🥺 committed
    use tokio::time::sleep;
    use tracing::{debug, info};
    
    /// # `POST /_matrix/client/v3/rooms/{roomId}/report/{eventId}`
    
    ///
    /// Reports an inappropriate event to homeserver admins
    ///
    pub async fn report_event_route(
    
    Jonas Platte's avatar
    Jonas Platte committed
        body: Ruma<report_content::v3::Request>,
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
    ) -> Result<report_content::v3::Response> {
    
    🥺's avatar
    🥺 committed
        // user authentication
    
        let sender_user = body.sender_user.as_ref().expect("user is authenticated");
    
    
    🥺's avatar
    🥺 committed
        info!("Received /report request by user {}", sender_user);
    
        // check if we know about the reported event ID or if it's invalid
    
    Timo Kösters's avatar
    Timo Kösters committed
        let pdu = match services().rooms.timeline.get_pdu(&body.event_id)? {
    
            _ => {
                return Err(Error::BadRequest(
    
    🥺's avatar
    🥺 committed
                    ErrorKind::NotFound,
                    "Event ID is not known to us or Event ID is invalid",
    
    🥺's avatar
    🥺 committed
        // check if the room ID from the URI matches the PDU's room ID
        if body.room_id != pdu.room_id {
            return Err(Error::BadRequest(
                ErrorKind::NotFound,
                "Event ID does not belong to the reported room",
            ));
        }
    
    
        // check if reporting user is in the reporting room
        if !services()
            .rooms
            .state_cache
            .room_members(&pdu.room_id)
            .filter_map(|r| r.ok())
            .any(|user_id| user_id == *sender_user)
        {
            return Err(Error::BadRequest(
                ErrorKind::NotFound,
                "You are not in the room you are reporting.",
            ));
        }
    
    
    🥺's avatar
    🥺 committed
        // check if score is in valid range
    
    chenyuqide's avatar
    chenyuqide committed
        if let Some(true) = body.score.map(|s| s > int!(0) || s < int!(-100)) {
    
            return Err(Error::BadRequest(
                ErrorKind::InvalidParam,
                "Invalid score, must be within 0 to -100",
            ));
        };
    
    
    🥺's avatar
    🥺 committed
        // check if report reasoning is less than or equal to 750 characters
        if let Some(true) = body.reason.clone().map(|s| s.chars().count() >= 750) {
    
            return Err(Error::BadRequest(
                ErrorKind::InvalidParam,
    
    🥺's avatar
    🥺 committed
                "Reason too long, should be 750 characters or fewer",
    
    🥺's avatar
    🥺 committed
        // send admin room message that we received the report with an @room ping for urgency
    
        services().admin
    
            .send_message(message::RoomMessageEventContent::text_html(
    
    🥺's avatar
    🥺 committed
                    "@room Report received from: {}\n\n\
    
    chenyuqide's avatar
    chenyuqide committed
                    Event ID: {:?}\n\
                    Room ID: {:?}\n\
                    Sent By: {:?}\n\n\
    
    🥺's avatar
    🥺 committed
                    Report Score: {:#?}\n\
    
    chenyuqide's avatar
    chenyuqide committed
                    Report Reason: {:?}",
    
    🥺's avatar
    🥺 committed
                    sender_user.to_owned(), pdu.event_id, pdu.room_id, pdu.sender, body.score, body.reason
    
    Jonas Platte's avatar
    Jonas Platte committed
                ),
    
    🥺's avatar
    🥺 committed
                    "<details><summary>@room Report received from: <a href=\"https://matrix.to/#/{0}\">{0}\
    
    chenyuqide's avatar
    chenyuqide committed
                    </a></summary><ul><li>Event Info<ul><li>Event ID: <code>{1:?}</code>\
                    <a href=\"https://matrix.to/#/{2:?}/{1:?}\">🔗</a></li><li>Room ID: <code>{2:?}</code>\
                    </li><li>Sent By: <a href=\"https://matrix.to/#/{3:?}\">{3:?}</a></li></ul></li><li>\
                    Report Info<ul><li>Report Score: {4:?}</li><li>Report Reason: {5}</li></ul></li>\
    
    🥺's avatar
    🥺 committed
                    sender_user.to_owned(),
    
                    HtmlEscape(body.reason.as_deref().unwrap_or(""))
    
    Jonas Platte's avatar
    Jonas Platte committed
                ),
    
    🥺's avatar
    🥺 committed
        // even though this is kinda security by obscurity, let's still make a small random delay sending a successful response
        // per spec suggestion regarding enumerating for potential events existing in our server.
        let time_to_wait = rand::thread_rng().gen_range(8..21);
        debug!(
            "Got successful /report request, waiting {} seconds before sending successful response.",
            time_to_wait
        );
        sleep(Duration::from_secs(time_to_wait)).await;
    
    
    Jonathan de Jong's avatar
    Jonathan de Jong committed
        Ok(report_content::v3::Response {})