matrix-sdk 0.16.0

A high level Matrix client-server library.
Documentation
// Copyright 2025 The Matrix.org Foundation C.I.C.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

//!  Facilities to handle incoming calls.

use ruma::{
    EventId, OwnedUserId, UserId,
    events::{
        AnySyncMessageLikeEvent, AnySyncTimelineEvent,
        rtc::decline::{RtcDeclineEventContent, SyncRtcDeclineEvent},
    },
};
use thiserror::Error;
use tokio::sync::broadcast;
use tracing::instrument;

use crate::{Room, event_handler::EventHandlerDropGuard, room::EventSource};

/// An error occurring while interacting with a call/rtc event.
#[derive(Debug, Error)]
pub enum CallError {
    /// We couldn't fetch the remote notification event.
    #[error("Couldn't fetch the remote event: {0}")]
    Fetch(Box<crate::Error>),

    /// We tried to decline an event which is not of type m.rtc.notification.
    #[error("You cannot decline this event type.")]
    BadEventType,

    /// We tried to decline a call started by ourselves.
    #[error("You cannot decline your own call.")]
    DeclineOwnCall,

    /// We couldn't properly deserialize the target event.
    #[error(transparent)]
    Deserialize(#[from] serde_json::Error),
}

impl Room {
    /// Create a new decline call event for the target notification event id .
    ///
    /// The event can then be sent with [`Room::send`] or a
    /// [`crate::send_queue::RoomSendQueue`].
    #[instrument(skip(self), fields(room = %self.room_id()))]
    pub async fn make_decline_call_event(
        &self,
        notification_event_id: &EventId,
    ) -> Result<RtcDeclineEventContent, CallError> {
        make_call_decline_event(self, self.own_user_id(), notification_event_id).await
    }

    /// Subscribe to decline call event for this room.
    ///
    /// The returned receiver will receive the sender UserID for each decline
    /// for the matching notify event.
    /// Example:
    /// - A push is received for an `m.rtc.notification` event.
    /// - The app starts ringing on this device.
    /// - The app subscribes to decline events for that notify event and stops
    ///   ringing if another device declines the call.
    ///
    /// In case of outgoing call, you can subscribe to see if your call was
    /// denied from the other side.
    ///
    /// ```rust
    /// # async fn start_ringing() {}
    /// # async fn stop_ringing() {}
    /// # async fn show_incoming_call_ui() {}
    /// # async fn dismiss_incoming_call_ui() {}
    /// #
    /// # async fn on_push_for_call_notify(room: matrix_sdk::Room, notify_event_id: &ruma::EventId) {
    ///     // 1) We just received a push for an `m.rtc.notification` in `room`.
    ///     show_incoming_call_ui().await;
    ///     start_ringing().await;
    ///
    ///     // 2) Subscribe to declines for this notify event, in case the call is declined from another device.
    ///     let (drop_guard, mut declines) = room.subscribe_to_call_decline_events(notify_event_id);
    ///
    ///     // Keep the subscription alive while we wait for a decline.
    ///     // You might store `drop_guard` alongside your call state.
    ///     tokio::spawn(async move {
    ///         loop {
    ///             match declines.recv().await {
    ///                 Ok(_decliner) => {
    ///                     // 3) Check the mxID -> I declined this call from another device.
    ///                     stop_ringing().await;
    ///                     dismiss_incoming_call_ui().await;
    ///                     // Exiting ends the task; dropping the guard unsubscribes the handler.
    ///                     drop(drop_guard);
    ///                     break;
    ///                 }
    ///                 Err(broadcast_err) => {
    ///                     // Channel closed or lagged; stop waiting.
    ///                     // In practice you might want to handle Lagged specifically.
    ///                     eprintln!("decline subscription ended: {broadcast_err}");
    ///                     drop(drop_guard);
    ///                     break;
    ///                 }
    ///             }
    ///         }
    ///     });
    /// # }
    /// ```
    pub fn subscribe_to_call_decline_events(
        &self,
        notification_event_id: &EventId,
    ) -> (EventHandlerDropGuard, broadcast::Receiver<OwnedUserId>) {
        let (sender, receiver) = broadcast::channel(16);

        let decline_call_event_handler_handle =
            self.client.add_room_event_handler(self.room_id(), {
                let own_notification_event_id = notification_event_id.to_owned();
                move |event: SyncRtcDeclineEvent| async move {
                    // Ignore decline for other unrelated notification events.
                    if let Some(declined_event_id) =
                        event.as_original().map(|ev| ev.content.relates_to.event_id.clone())
                        && declined_event_id == own_notification_event_id
                    {
                        let _ = sender.send(event.sender().to_owned());
                    }
                }
            });
        let drop_guard = self.client().event_handler_drop_guard(decline_call_event_handler_handle);
        (drop_guard, receiver)
    }
}

async fn make_call_decline_event(
    room: &Room,
    own_user_id: &UserId,
    notification_event_id: &EventId,
) -> Result<RtcDeclineEventContent, CallError> {
    let target = room
        .get_event(notification_event_id)
        .await
        .map_err(|err| CallError::Fetch(Box::new(err)))?;

    let event = target.raw().deserialize().map_err(CallError::Deserialize)?;

    // The event must be RtcNotification-like.
    if let AnySyncTimelineEvent::MessageLike(AnySyncMessageLikeEvent::RtcNotification(notify)) =
        event
    {
        if notify.sender() == own_user_id {
            // Cannot decline own call.
            Err(CallError::DeclineOwnCall)
        } else {
            Ok(RtcDeclineEventContent::new(notification_event_id))
        }
    } else {
        Err(CallError::BadEventType)
    }
}