1#[cfg(test)]
2mod candidate_pair_test;
3#[cfg(test)]
4mod candidate_relay_test;
5#[cfg(test)]
6mod candidate_server_reflexive_test;
7#[cfg(test)]
8mod candidate_test;
9
10pub mod candidate_base;
11pub mod candidate_host;
12pub mod candidate_peer_reflexive;
13pub mod candidate_relay;
14pub mod candidate_server_reflexive;
15
16use std::fmt;
17use std::net::{IpAddr, SocketAddr};
18use std::sync::atomic::Ordering;
19use std::sync::Arc;
20use std::time::SystemTime;
21
22use async_trait::async_trait;
23use candidate_base::*;
24use portable_atomic::{AtomicBool, AtomicU16, AtomicU8};
25use serde::{Deserialize, Serialize};
26use tokio::sync::{broadcast, Mutex};
27
28use crate::error::Result;
29use crate::network_type::*;
30use crate::tcp_type::*;
31
32pub(crate) const RECEIVE_MTU: usize = 8192;
33pub(crate) const DEFAULT_LOCAL_PREFERENCE: u16 = 65535;
34
35pub(crate) const COMPONENT_RTP: u16 = 1;
37pub(crate) const COMPONENT_RTCP: u16 = 0;
39
40#[async_trait]
42pub trait Candidate: fmt::Display {
43 fn foundation(&self) -> String;
48
49 fn id(&self) -> String;
52
53 fn component(&self) -> u16;
56 fn set_component(&self, c: u16);
57
58 fn last_received(&self) -> SystemTime;
60
61 fn last_sent(&self) -> SystemTime;
63
64 fn network_type(&self) -> NetworkType;
65 fn address(&self) -> String;
66 fn port(&self) -> u16;
67
68 fn priority(&self) -> u32;
69
70 fn related_address(&self) -> Option<CandidateRelatedAddress>;
73
74 fn candidate_type(&self) -> CandidateType;
75 fn tcp_type(&self) -> TcpType;
76
77 fn marshal(&self) -> String;
78
79 fn addr(&self) -> SocketAddr;
80
81 async fn close(&self) -> Result<()>;
82 fn seen(&self, outbound: bool);
83
84 async fn write_to(&self, raw: &[u8], dst: &(dyn Candidate + Send + Sync)) -> Result<usize>;
85 fn equal(&self, other: &dyn Candidate) -> bool;
86 fn set_ip(&self, ip: &IpAddr) -> Result<()>;
87 fn get_conn(&self) -> Option<&Arc<dyn util::Conn + Send + Sync>>;
88 fn get_closed_ch(&self) -> Arc<Mutex<Option<broadcast::Sender<()>>>>;
89}
90
91#[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)]
93pub enum CandidateType {
94 #[serde(rename = "unspecified")]
95 Unspecified,
96 #[serde(rename = "host")]
97 Host,
98 #[serde(rename = "srflx")]
99 ServerReflexive,
100 #[serde(rename = "prflx")]
101 PeerReflexive,
102 #[serde(rename = "relay")]
103 Relay,
104}
105
106impl fmt::Display for CandidateType {
108 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
109 let s = match *self {
110 CandidateType::Host => "host",
111 CandidateType::ServerReflexive => "srflx",
112 CandidateType::PeerReflexive => "prflx",
113 CandidateType::Relay => "relay",
114 CandidateType::Unspecified => "Unknown candidate type",
115 };
116 write!(f, "{s}")
117 }
118}
119
120impl Default for CandidateType {
121 fn default() -> Self {
122 Self::Unspecified
123 }
124}
125
126impl CandidateType {
127 #[must_use]
134 pub const fn preference(self) -> u16 {
135 match self {
136 Self::Host => 126,
137 Self::PeerReflexive => 110,
138 Self::ServerReflexive => 100,
139 Self::Relay | CandidateType::Unspecified => 0,
140 }
141 }
142}
143
144pub(crate) fn contains_candidate_type(
145 candidate_type: CandidateType,
146 candidate_type_list: &[CandidateType],
147) -> bool {
148 if candidate_type_list.is_empty() {
149 return false;
150 }
151 for ct in candidate_type_list {
152 if *ct == candidate_type {
153 return true;
154 }
155 }
156 false
157}
158
159#[derive(PartialEq, Eq, Debug, Clone)]
161pub struct CandidateRelatedAddress {
162 pub address: String,
163 pub port: u16,
164}
165
166impl fmt::Display for CandidateRelatedAddress {
168 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
169 write!(f, " related {}:{}", self.address, self.port)
170 }
171}
172
173#[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)]
175pub enum CandidatePairState {
176 #[serde(rename = "unspecified")]
177 Unspecified = 0,
178
179 #[serde(rename = "waiting")]
181 Waiting = 1,
182
183 #[serde(rename = "in-progress")]
185 InProgress = 2,
186
187 #[serde(rename = "failed")]
190 Failed = 3,
191
192 #[serde(rename = "succeeded")]
194 Succeeded = 4,
195}
196
197impl From<u8> for CandidatePairState {
198 fn from(v: u8) -> Self {
199 match v {
200 1 => Self::Waiting,
201 2 => Self::InProgress,
202 3 => Self::Failed,
203 4 => Self::Succeeded,
204 _ => Self::Unspecified,
205 }
206 }
207}
208
209impl Default for CandidatePairState {
210 fn default() -> Self {
211 Self::Unspecified
212 }
213}
214
215impl fmt::Display for CandidatePairState {
216 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
217 let s = match *self {
218 Self::Waiting => "waiting",
219 Self::InProgress => "in-progress",
220 Self::Failed => "failed",
221 Self::Succeeded => "succeeded",
222 Self::Unspecified => "unspecified",
223 };
224
225 write!(f, "{s}")
226 }
227}
228
229pub struct CandidatePair {
231 pub(crate) ice_role_controlling: AtomicBool,
232 pub remote: Arc<dyn Candidate + Send + Sync>,
233 pub local: Arc<dyn Candidate + Send + Sync>,
234 pub(crate) binding_request_count: AtomicU16,
235 pub(crate) state: AtomicU8, pub(crate) nominated: AtomicBool,
237}
238
239impl Default for CandidatePair {
240 fn default() -> Self {
241 Self {
242 ice_role_controlling: AtomicBool::new(false),
243 remote: Arc::new(CandidateBase::default()),
244 local: Arc::new(CandidateBase::default()),
245 state: AtomicU8::new(CandidatePairState::Waiting as u8),
246 binding_request_count: AtomicU16::new(0),
247 nominated: AtomicBool::new(false),
248 }
249 }
250}
251
252impl fmt::Debug for CandidatePair {
253 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
254 write!(
255 f,
256 "prio {} (local, prio {}) {} <-> {} (remote, prio {})",
257 self.priority(),
258 self.local.priority(),
259 self.local,
260 self.remote,
261 self.remote.priority()
262 )
263 }
264}
265
266impl fmt::Display for CandidatePair {
267 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
268 write!(
269 f,
270 "prio {} (local, prio {}) {} <-> {} (remote, prio {})",
271 self.priority(),
272 self.local.priority(),
273 self.local,
274 self.remote,
275 self.remote.priority()
276 )
277 }
278}
279
280impl PartialEq for CandidatePair {
281 fn eq(&self, other: &Self) -> bool {
282 self.local.equal(&*other.local) && self.remote.equal(&*other.remote)
283 }
284}
285
286impl CandidatePair {
287 #[must_use]
288 pub fn new(
289 local: Arc<dyn Candidate + Send + Sync>,
290 remote: Arc<dyn Candidate + Send + Sync>,
291 controlling: bool,
292 ) -> Self {
293 Self {
294 ice_role_controlling: AtomicBool::new(controlling),
295 remote,
296 local,
297 state: AtomicU8::new(CandidatePairState::Waiting as u8),
298 binding_request_count: AtomicU16::new(0),
299 nominated: AtomicBool::new(false),
300 }
301 }
302
303 pub fn priority(&self) -> u64 {
309 let (g, d) = if self.ice_role_controlling.load(Ordering::SeqCst) {
310 (self.local.priority(), self.remote.priority())
311 } else {
312 (self.remote.priority(), self.local.priority())
313 };
314
315 ((1 << 32_u64) - 1) * u64::from(std::cmp::min(g, d))
318 + 2 * u64::from(std::cmp::max(g, d))
319 + u64::from(g > d)
320 }
321
322 pub async fn write(&self, b: &[u8]) -> Result<usize> {
323 self.local.write_to(b, &*self.remote).await
324 }
325}