webrtc_ice/control/
mod.rs

1#[cfg(test)]
2mod control_test;
3
4use std::fmt;
5
6use stun::attributes::*;
7use stun::checks::*;
8use stun::message::*;
9
10/// Common helper for ICE-{CONTROLLED,CONTROLLING} and represents the so-called Tiebreaker number.
11#[derive(Default, PartialEq, Eq, Debug, Copy, Clone)]
12pub struct TieBreaker(pub u64);
13
14pub(crate) const TIE_BREAKER_SIZE: usize = 8; // 64 bit
15
16impl TieBreaker {
17    /// Adds Tiebreaker value to m as t attribute.
18    pub fn add_to_as(self, m: &mut Message, t: AttrType) -> Result<(), stun::Error> {
19        let mut v = vec![0; TIE_BREAKER_SIZE];
20        v.copy_from_slice(&self.0.to_be_bytes());
21        m.add(t, &v);
22        Ok(())
23    }
24
25    /// Decodes Tiebreaker value in message getting it as for t type.
26    pub fn get_from_as(&mut self, m: &Message, t: AttrType) -> Result<(), stun::Error> {
27        let v = m.get(t)?;
28        check_size(t, v.len(), TIE_BREAKER_SIZE)?;
29        self.0 = u64::from_be_bytes([v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7]]);
30        Ok(())
31    }
32}
33/// Represents ICE-CONTROLLED attribute.
34#[derive(Default, PartialEq, Eq, Debug, Copy, Clone)]
35pub struct AttrControlled(pub u64);
36
37impl Setter for AttrControlled {
38    /// Adds ICE-CONTROLLED to message.
39    fn add_to(&self, m: &mut Message) -> Result<(), stun::Error> {
40        TieBreaker(self.0).add_to_as(m, ATTR_ICE_CONTROLLED)
41    }
42}
43
44impl Getter for AttrControlled {
45    /// Decodes ICE-CONTROLLED from message.
46    fn get_from(&mut self, m: &Message) -> Result<(), stun::Error> {
47        let mut t = TieBreaker::default();
48        t.get_from_as(m, ATTR_ICE_CONTROLLED)?;
49        self.0 = t.0;
50        Ok(())
51    }
52}
53
54/// Represents ICE-CONTROLLING attribute.
55#[derive(Default, PartialEq, Eq, Debug, Copy, Clone)]
56pub struct AttrControlling(pub u64);
57
58impl Setter for AttrControlling {
59    // add_to adds ICE-CONTROLLING to message.
60    fn add_to(&self, m: &mut Message) -> Result<(), stun::Error> {
61        TieBreaker(self.0).add_to_as(m, ATTR_ICE_CONTROLLING)
62    }
63}
64
65impl Getter for AttrControlling {
66    // get_from decodes ICE-CONTROLLING from message.
67    fn get_from(&mut self, m: &Message) -> Result<(), stun::Error> {
68        let mut t = TieBreaker::default();
69        t.get_from_as(m, ATTR_ICE_CONTROLLING)?;
70        self.0 = t.0;
71        Ok(())
72    }
73}
74
75/// Helper that wraps ICE-{CONTROLLED,CONTROLLING}.
76#[derive(Default, PartialEq, Eq, Debug, Copy, Clone)]
77pub struct AttrControl {
78    role: Role,
79    tie_breaker: TieBreaker,
80}
81
82impl Setter for AttrControl {
83    // add_to adds ICE-CONTROLLED or ICE-CONTROLLING attribute depending on Role.
84    fn add_to(&self, m: &mut Message) -> Result<(), stun::Error> {
85        if self.role == Role::Controlling {
86            self.tie_breaker.add_to_as(m, ATTR_ICE_CONTROLLING)
87        } else {
88            self.tie_breaker.add_to_as(m, ATTR_ICE_CONTROLLED)
89        }
90    }
91}
92
93impl Getter for AttrControl {
94    // get_from decodes Role and Tiebreaker value from message.
95    fn get_from(&mut self, m: &Message) -> Result<(), stun::Error> {
96        if m.contains(ATTR_ICE_CONTROLLING) {
97            self.role = Role::Controlling;
98            return self.tie_breaker.get_from_as(m, ATTR_ICE_CONTROLLING);
99        }
100        if m.contains(ATTR_ICE_CONTROLLED) {
101            self.role = Role::Controlled;
102            return self.tie_breaker.get_from_as(m, ATTR_ICE_CONTROLLED);
103        }
104
105        Err(stun::Error::ErrAttributeNotFound)
106    }
107}
108
109/// Represents ICE agent role, which can be controlling or controlled.
110/// Possible ICE agent roles.
111#[derive(PartialEq, Eq, Copy, Clone, Debug)]
112pub enum Role {
113    Controlling,
114    Controlled,
115    Unspecified,
116}
117
118impl Default for Role {
119    fn default() -> Self {
120        Self::Controlling
121    }
122}
123
124impl From<&str> for Role {
125    fn from(raw: &str) -> Self {
126        match raw {
127            "controlling" => Self::Controlling,
128            "controlled" => Self::Controlled,
129            _ => Self::Unspecified,
130        }
131    }
132}
133
134impl fmt::Display for Role {
135    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
136        let s = match *self {
137            Self::Controlling => "controlling",
138            Self::Controlled => "controlled",
139            Self::Unspecified => "unspecified",
140        };
141        write!(f, "{s}")
142    }
143}