use byteorder::{BigEndian, ByteOrder};
use stun_types::{attribute::*, message::StunParseError};
#[derive(Debug, Clone)]
pub struct ReservationToken {
token: u64,
}
impl AttributeStaticType for ReservationToken {
const TYPE: AttributeType = AttributeType::new(0x0022);
}
impl Attribute for ReservationToken {
fn get_type(&self) -> AttributeType {
Self::TYPE
}
fn length(&self) -> u16 {
8
}
}
impl AttributeWrite for ReservationToken {
fn to_raw(&self) -> RawAttribute<'_> {
let mut data = [0; 8];
BigEndian::write_u64(&mut data, self.token);
RawAttribute::new(self.get_type(), &data).into_owned()
}
fn write_into_unchecked(&self, dest: &mut [u8]) {
self.write_header_unchecked(dest);
BigEndian::write_u64(&mut dest[4..12], self.token);
}
}
impl AttributeFromRaw<'_> for ReservationToken {
fn from_raw_ref(raw: &RawAttribute) -> Result<Self, StunParseError>
where
Self: Sized,
{
Self::try_from(raw)
}
}
impl TryFrom<&RawAttribute<'_>> for ReservationToken {
type Error = StunParseError;
fn try_from(raw: &RawAttribute) -> Result<Self, Self::Error> {
raw.check_type_and_len(Self::TYPE, 8..=8)?;
Ok(Self {
token: BigEndian::read_u64(&raw.value),
})
}
}
impl ReservationToken {
pub fn new(token: u64) -> Self {
Self { token }
}
pub fn token(&self) -> u64 {
self.token
}
}
impl core::fmt::Display for ReservationToken {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
write!(f, "{}: 0x{:#x}", self.get_type(), self.token())
}
}
#[cfg(test)]
mod tests {
use super::*;
use alloc::{vec, vec::Vec};
use byteorder::{BigEndian, ByteOrder};
use tracing::trace;
#[test]
fn reservation_token() {
let _log = crate::tests::test_init_log();
let token = ReservationToken::new(200);
assert_eq!(token.get_type(), ReservationToken::TYPE);
assert_eq!(token.token(), 200);
}
#[test]
fn reservation_token_raw() {
let _log = crate::tests::test_init_log();
let token = ReservationToken::new(200);
let raw: RawAttribute = token.to_raw();
trace!("{}", raw);
assert_eq!(raw.get_type(), ReservationToken::TYPE);
let token2 = ReservationToken::try_from(&raw).unwrap();
assert_eq!(token2.get_type(), ReservationToken::TYPE);
assert_eq!(token2.token(), 200);
}
#[test]
fn reservation_token_raw_wrong_type() {
let _log = crate::tests::test_init_log();
let token = ReservationToken::new(200);
let raw: RawAttribute = token.to_raw();
let mut data: Vec<_> = raw.into();
BigEndian::write_u16(&mut data[0..2], 0);
assert!(matches!(
ReservationToken::try_from(&RawAttribute::from_bytes(data.as_ref()).unwrap()),
Err(StunParseError::WrongAttributeImplementation)
));
}
#[test]
fn reservation_token_write_into() {
let _log = crate::tests::test_init_log();
let token = ReservationToken::new(200);
let raw: RawAttribute = token.to_raw();
let mut dest = vec![0; raw.padded_len()];
token.write_into(&mut dest).unwrap();
let raw = RawAttribute::from_bytes(&dest).unwrap();
let token2 = ReservationToken::try_from(&raw).unwrap();
assert_eq!(token2.get_type(), ReservationToken::TYPE);
assert_eq!(token2.token(), 200);
}
#[test]
#[should_panic = "out of range"]
fn reservation_token_write_into_unchecked() {
let _log = crate::tests::test_init_log();
let token = ReservationToken::new(200);
let raw: RawAttribute = token.to_raw();
let mut dest = vec![0; raw.padded_len() - 1];
token.write_into_unchecked(&mut dest);
}
}