pub mod candidate_base;
pub mod candidate_host;
pub mod candidate_peer_reflexive;
pub mod candidate_relay;
pub mod candidate_server_reflexive;
use crate::candidate::candidate_base::*;
use crate::network_type::*;
use crate::tcp_type::*;
use util::Error;
use async_trait::async_trait;
use std::fmt;
use std::net::{IpAddr, SocketAddr};
use std::sync::Arc;
use std::time::SystemTime;
use tokio::sync::broadcast;
pub(crate) const RECEIVE_MTU: usize = 8192;
pub(crate) const DEFAULT_LOCAL_PREFERENCE: u16 = 65535;
pub(crate) const COMPONENT_RTP: u16 = 1;
pub(crate) const COMPONENT_RTCP: u16 = 0;
#[async_trait]
pub trait Candidate: fmt::Display {
fn foundation(&self) -> String;
fn id(&self) -> String;
fn component(&self) -> u16;
fn set_component(&self, c: u16);
fn last_received(&self) -> SystemTime;
fn last_sent(&self) -> SystemTime;
fn network_type(&self) -> NetworkType;
fn address(&self) -> String;
fn port(&self) -> u16;
fn priority(&self) -> u32;
fn related_address(&self) -> Option<CandidateRelatedAddress>;
fn candidate_type(&self) -> CandidateType;
fn tcp_type(&self) -> TCPType;
fn marshal(&self) -> String;
fn addr(&self) -> SocketAddr;
async fn close(&self) -> Result<(), Error>;
fn seen(&self, outbound: bool);
async fn write_to(
&self,
raw: &[u8],
dst: &(dyn Candidate + Send + Sync),
) -> Result<usize, Error>;
fn equal(&self, other: &dyn Candidate) -> bool;
fn clone(&self) -> Arc<dyn Candidate + Send + Sync>;
fn clone_with_ip(&self, ip: &IpAddr) -> Arc<dyn Candidate + Send + Sync>;
async fn start(&self, initialized_ch: Option<broadcast::Receiver<()>>);
}
#[derive(PartialEq, Debug, Copy, Clone)]
pub enum CandidateType {
Unspecified,
Host,
ServerReflexive,
PeerReflexive,
Relay,
}
impl fmt::Display for CandidateType {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let s = match *self {
CandidateType::Host => "host",
CandidateType::ServerReflexive => "srflx",
CandidateType::PeerReflexive => "prflx",
CandidateType::Relay => "relay",
CandidateType::Unspecified => "Unknown candidate type",
};
write!(f, "{}", s)
}
}
impl Default for CandidateType {
fn default() -> Self {
CandidateType::Unspecified
}
}
impl CandidateType {
pub fn preference(&self) -> u16 {
match *self {
CandidateType::Host => 126,
CandidateType::PeerReflexive => 110,
CandidateType::ServerReflexive => 100,
CandidateType::Relay | CandidateType::Unspecified => 0,
}
}
}
pub(crate) fn contains_candidate_type(
candidate_type: CandidateType,
candidate_type_list: &[CandidateType],
) -> bool {
if candidate_type_list.is_empty() {
return false;
}
for ct in candidate_type_list {
if *ct == candidate_type {
return true;
}
}
false
}
#[derive(PartialEq, Debug, Clone)]
pub struct CandidateRelatedAddress {
pub address: String,
pub port: u16,
}
impl fmt::Display for CandidateRelatedAddress {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, " related {}:{}", self.address, self.port)
}
}
#[derive(PartialEq, Debug, Copy, Clone)]
pub enum CandidatePairState {
Waiting,
InProgress,
Failed,
Succeeded,
}
impl Default for CandidatePairState {
fn default() -> Self {
CandidatePairState::Waiting
}
}
impl fmt::Display for CandidatePairState {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let s = match *self {
CandidatePairState::Waiting => "waiting",
CandidatePairState::InProgress => "in-progress",
CandidatePairState::Failed => "failed",
CandidatePairState::Succeeded => "succeeded",
};
write!(f, "{}", s)
}
}
pub(crate) struct CandidatePair {
pub(crate) ice_role_controlling: bool,
pub(crate) remote: Arc<dyn Candidate + Send + Sync>,
pub(crate) local: Arc<dyn Candidate + Send + Sync>,
pub(crate) binding_request_count: u16,
pub(crate) state: CandidatePairState,
pub(crate) nominated: bool,
}
impl Clone for CandidatePair {
fn clone(&self) -> Self {
CandidatePair {
ice_role_controlling: self.ice_role_controlling,
remote: self.remote.clone(),
local: self.local.clone(),
state: self.state,
binding_request_count: self.binding_request_count,
nominated: self.nominated,
}
}
}
impl Default for CandidatePair {
fn default() -> Self {
CandidatePair {
ice_role_controlling: false,
remote: Arc::new(CandidateBase::default()),
local: Arc::new(CandidateBase::default()),
state: CandidatePairState::Waiting,
binding_request_count: 0,
nominated: false,
}
}
}
impl fmt::Debug for CandidatePair {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"prio {} (local, prio {}) {} <-> {} (remote, prio {})",
self.priority(),
self.local.priority(),
self.local,
self.remote,
self.remote.priority()
)
}
}
impl fmt::Display for CandidatePair {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"prio {} (local, prio {}) {} <-> {} (remote, prio {})",
self.priority(),
self.local.priority(),
self.local,
self.remote,
self.remote.priority()
)
}
}
impl PartialEq for CandidatePair {
fn eq(&self, other: &CandidatePair) -> bool {
self.local.equal(&*other.local) && self.remote.equal(&*other.remote)
}
}
impl CandidatePair {
pub fn new(
local: Arc<dyn Candidate + Send + Sync>,
remote: Arc<dyn Candidate + Send + Sync>,
controlling: bool,
) -> Self {
CandidatePair {
ice_role_controlling: controlling,
remote,
local,
state: CandidatePairState::Waiting,
binding_request_count: 0,
nominated: false,
}
}
pub fn priority(&self) -> u64 {
let (g, d) = if self.ice_role_controlling {
(self.local.priority(), self.remote.priority())
} else {
(self.remote.priority(), self.local.priority())
};
((1 << 32u64) - 1) * std::cmp::min(g, d) as u64
+ 2 * std::cmp::max(g, d) as u64
+ if g > d { 1 } else { 0 }
}
pub async fn write(&self, b: &[u8]) -> Result<usize, Error> {
self.local.write_to(b, &*self.remote).await
}
}