webrtc_stun/
attributes.rs1#[cfg(test)]
2mod attributes_test;
3
4use crate::message::*;
5
6use util::Error;
7
8use std::fmt;
9
10#[derive(Default, PartialEq, Eq, Debug, Clone)]
12pub struct Attributes(pub Vec<RawAttribute>);
13
14impl Attributes {
15 pub fn get(&self, t: AttrType) -> (RawAttribute, bool) {
20 for candidate in &self.0 {
21 if candidate.typ == t {
22 return (candidate.clone(), true);
23 }
24 }
25
26 (RawAttribute::default(), false)
27 }
28}
29
30#[derive(PartialEq, Debug, Eq, Default, Copy, Clone)]
32pub struct AttrType(pub u16);
33
34impl fmt::Display for AttrType {
35 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
36 let other = format!("0x{:x}", self.0);
37
38 let s = match *self {
39 ATTR_MAPPED_ADDRESS => "MAPPED-ADDRESS",
40 ATTR_USERNAME => "USERNAME",
41 ATTR_ERROR_CODE => "ERROR-CODE",
42 ATTR_MESSAGE_INTEGRITY => "MESSAGE-INTEGRITY",
43 ATTR_UNKNOWN_ATTRIBUTES => "UNKNOWN-ATTRIBUTES",
44 ATTR_REALM => "REALM",
45 ATTR_NONCE => "NONCE",
46 ATTR_XORMAPPED_ADDRESS => "XOR-MAPPED-ADDRESS",
47 ATTR_SOFTWARE => "SOFTWARE",
48 ATTR_ALTERNATE_SERVER => "ALTERNATE-SERVER",
49 ATTR_FINGERPRINT => "FINGERPRINT",
50 ATTR_PRIORITY => "PRIORITY",
51 ATTR_USE_CANDIDATE => "USE-CANDIDATE",
52 ATTR_ICE_CONTROLLED => "ICE-CONTROLLED",
53 ATTR_ICE_CONTROLLING => "ICE-CONTROLLING",
54 ATTR_CHANNEL_NUMBER => "CHANNEL-NUMBER",
55 ATTR_LIFETIME => "LIFETIME",
56 ATTR_XOR_PEER_ADDRESS => "XOR-PEER-ADDRESS",
57 ATTR_DATA => "DATA",
58 ATTR_XOR_RELAYED_ADDRESS => "XOR-RELAYED-ADDRESS",
59 ATTR_EVEN_PORT => "EVEN-PORT",
60 ATTR_REQUESTED_TRANSPORT => "REQUESTED-TRANSPORT",
61 ATTR_DONT_FRAGMENT => "DONT-FRAGMENT",
62 ATTR_RESERVATION_TOKEN => "RESERVATION-TOKEN",
63 ATTR_CONNECTION_ID => "CONNECTION-ID",
64 ATTR_REQUESTED_ADDRESS_FAMILY => "REQUESTED-ADDRESS-FAMILY",
65 ATTR_MESSAGE_INTEGRITY_SHA256 => "MESSAGE-INTEGRITY-SHA256",
66 ATTR_PASSWORD_ALGORITHM => "PASSWORD-ALGORITHM",
67 ATTR_USER_HASH => "USERHASH",
68 ATTR_PASSWORD_ALGORITHMS => "PASSWORD-ALGORITHMS",
69 ATTR_ALTERNATE_DOMAIN => "ALTERNATE-DOMAIN",
70 _ => other.as_str(),
71 };
72
73 write!(f, "{}", s)
74 }
75}
76
77impl AttrType {
78 pub fn required(&self) -> bool {
80 self.0 <= 0x7FFF
81 }
82
83 pub fn optional(&self) -> bool {
85 self.0 >= 0x8000
86 }
87
88 pub fn value(&self) -> u16 {
90 self.0
91 }
92}
93
94pub const ATTR_MAPPED_ADDRESS: AttrType = AttrType(0x0001); pub const ATTR_USERNAME: AttrType = AttrType(0x0006); pub const ATTR_MESSAGE_INTEGRITY: AttrType = AttrType(0x0008); pub const ATTR_ERROR_CODE: AttrType = AttrType(0x0009); pub const ATTR_UNKNOWN_ATTRIBUTES: AttrType = AttrType(0x000A); pub const ATTR_REALM: AttrType = AttrType(0x0014); pub const ATTR_NONCE: AttrType = AttrType(0x0015); pub const ATTR_XORMAPPED_ADDRESS: AttrType = AttrType(0x0020); pub const ATTR_SOFTWARE: AttrType = AttrType(0x8022); pub const ATTR_ALTERNATE_SERVER: AttrType = AttrType(0x8023); pub const ATTR_FINGERPRINT: AttrType = AttrType(0x8028); pub const ATTR_PRIORITY: AttrType = AttrType(0x0024); pub const ATTR_USE_CANDIDATE: AttrType = AttrType(0x0025); pub const ATTR_ICE_CONTROLLED: AttrType = AttrType(0x8029); pub const ATTR_ICE_CONTROLLING: AttrType = AttrType(0x802A); pub const ATTR_CHANNEL_NUMBER: AttrType = AttrType(0x000C); pub const ATTR_LIFETIME: AttrType = AttrType(0x000D); pub const ATTR_XOR_PEER_ADDRESS: AttrType = AttrType(0x0012); pub const ATTR_DATA: AttrType = AttrType(0x0013); pub const ATTR_XOR_RELAYED_ADDRESS: AttrType = AttrType(0x0016); pub const ATTR_EVEN_PORT: AttrType = AttrType(0x0018); pub const ATTR_REQUESTED_TRANSPORT: AttrType = AttrType(0x0019); pub const ATTR_DONT_FRAGMENT: AttrType = AttrType(0x001A); pub const ATTR_RESERVATION_TOKEN: AttrType = AttrType(0x0022); pub const ATTR_CHANGE_REQUEST: AttrType = AttrType(0x0003); pub const ATTR_PADDING: AttrType = AttrType(0x0026); pub const ATTR_RESPONSE_PORT: AttrType = AttrType(0x0027); pub const ATTR_CACHE_TIMEOUT: AttrType = AttrType(0x8027); pub const ATTR_RESPONSE_ORIGIN: AttrType = AttrType(0x802b); pub const ATTR_OTHER_ADDRESS: AttrType = AttrType(0x802C); pub const ATTR_SOURCE_ADDRESS: AttrType = AttrType(0x0004); pub const ATTR_CHANGED_ADDRESS: AttrType = AttrType(0x0005); pub const ATTR_CONNECTION_ID: AttrType = AttrType(0x002a); pub const ATTR_REQUESTED_ADDRESS_FAMILY: AttrType = AttrType(0x0017); pub const ATTR_ORIGIN: AttrType = AttrType(0x802F);
148
149pub const ATTR_MESSAGE_INTEGRITY_SHA256: AttrType = AttrType(0x001C); pub const ATTR_PASSWORD_ALGORITHM: AttrType = AttrType(0x001D); pub const ATTR_USER_HASH: AttrType = AttrType(0x001E); pub const ATTR_PASSWORD_ALGORITHMS: AttrType = AttrType(0x8002); pub const ATTR_ALTERNATE_DOMAIN: AttrType = AttrType(0x8003); #[derive(Default, Debug, Clone, PartialEq, Eq)]
164pub struct RawAttribute {
165 pub typ: AttrType,
166 pub length: u16, pub value: Vec<u8>,
168}
169
170impl fmt::Display for RawAttribute {
171 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
172 write!(f, "{}: {:?}", self.typ, self.value)
173 }
174}
175
176impl Setter for RawAttribute {
177 fn add_to(&self, m: &mut Message) -> Result<(), Error> {
180 m.add(self.typ, &self.value);
181 Ok(())
182 }
183}
184
185pub(crate) const PADDING: usize = 4;
192
193pub(crate) fn nearest_padded_value_length(l: usize) -> usize {
194 let mut n = PADDING * (l / PADDING);
195 if n < l {
196 n += PADDING
197 }
198 n
199}
200
201pub(crate) fn compat_attr_type(val: u16) -> AttrType {
205 if val == 0x8020 {
206 ATTR_XORMAPPED_ADDRESS } else {
209 AttrType(val)
210 }
211}