ruma_events/call/
candidates.rs

1//! Types for the [`m.call.candidates`] event.
2//!
3//! [`m.call.candidates`]: https://spec.matrix.org/latest/client-server-api/#mcallcandidates
4
5use js_int::UInt;
6use ruma_common::{OwnedVoipId, VoipVersionId};
7use ruma_macros::EventContent;
8use serde::{Deserialize, Serialize};
9
10/// The content of an `m.call.candidates` event.
11///
12/// This event is sent by callers after sending an invite and by the callee after answering. Its
13/// purpose is to give the other party additional ICE candidates to try using to communicate.
14#[derive(Clone, Debug, Deserialize, Serialize, EventContent)]
15#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
16#[ruma_event(type = "m.call.candidates", kind = MessageLike)]
17pub struct CallCandidatesEventContent {
18    /// A unique identifier for the call.
19    pub call_id: OwnedVoipId,
20
21    /// **Required in VoIP version 1.** The unique ID for this session for the duration of the
22    /// call.
23    ///
24    /// Must be the same as the one sent by the previous invite or answer from
25    /// this session.
26    #[serde(skip_serializing_if = "Option::is_none")]
27    pub party_id: Option<OwnedVoipId>,
28
29    /// A list of candidates.
30    ///
31    /// In VoIP version 1, this list should end with a `Candidate` with an empty `candidate` field
32    /// when no more candidates will be sent.
33    pub candidates: Vec<Candidate>,
34
35    /// The version of the VoIP specification this messages adheres to.
36    pub version: VoipVersionId,
37}
38
39impl CallCandidatesEventContent {
40    /// Creates a new `CallCandidatesEventContent` with the given call id, candidate list and VoIP
41    /// version.
42    pub fn new(call_id: OwnedVoipId, candidates: Vec<Candidate>, version: VoipVersionId) -> Self {
43        Self { call_id, candidates, version, party_id: None }
44    }
45
46    /// Convenience method to create a VoIP version 0 `CallCandidatesEventContent` with all the
47    /// required fields.
48    pub fn version_0(call_id: OwnedVoipId, candidates: Vec<Candidate>) -> Self {
49        Self::new(call_id, candidates, VoipVersionId::V0)
50    }
51
52    /// Convenience method to create a VoIP version 1 `CallCandidatesEventContent` with all the
53    /// required fields.
54    pub fn version_1(
55        call_id: OwnedVoipId,
56        party_id: OwnedVoipId,
57        candidates: Vec<Candidate>,
58    ) -> Self {
59        Self { call_id, party_id: Some(party_id), candidates, version: VoipVersionId::V1 }
60    }
61}
62
63/// An ICE (Interactive Connectivity Establishment) candidate.
64#[derive(Clone, Debug, Deserialize, Serialize)]
65#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
66#[serde(rename_all = "camelCase")]
67pub struct Candidate {
68    /// The SDP "a" line of the candidate.
69    pub candidate: String,
70
71    /// The SDP media type this candidate is intended for.
72    ///
73    /// At least one of `sdp_mid` or `sdp_m_line_index` is required, unless
74    /// `candidate` is empty.
75    #[serde(skip_serializing_if = "Option::is_none")]
76    pub sdp_mid: Option<String>,
77
78    /// The index of the SDP "m" line this candidate is intended for.
79    ///
80    /// At least one of `sdp_mid` or `sdp_m_line_index` is required, unless
81    /// `candidate` is empty.
82    #[serde(skip_serializing_if = "Option::is_none")]
83    pub sdp_m_line_index: Option<UInt>,
84}
85
86impl Candidate {
87    /// Creates a new `Candidate` with the given "a" line.
88    pub fn new(candidate: String) -> Self {
89        Self { candidate, sdp_mid: None, sdp_m_line_index: None }
90    }
91
92    /// Creates a new `Candidate` with all the required fields in VoIP version 0.
93    pub fn version_0(candidate: String, sdp_mid: String, sdp_m_line_index: UInt) -> Self {
94        Self { candidate, sdp_mid: Some(sdp_mid), sdp_m_line_index: Some(sdp_m_line_index) }
95    }
96}