use crate::{frame::Tag, varint::VarInt};
use core::{convert::TryInto, mem::size_of};
use s2n_codec::{decoder_invariant, decoder_parameterized_value, Encoder, EncoderValue};
macro_rules! new_connection_id_tag {
() => {
0x18u8
};
}
pub const STATELESS_RESET_TOKEN_LEN: usize = size_of::<u128>();
#[derive(Debug, PartialEq, Eq)]
pub struct NewConnectionId<'a> {
pub sequence_number: VarInt,
pub retire_prior_to: VarInt,
pub connection_id: &'a [u8],
pub stateless_reset_token: &'a [u8; STATELESS_RESET_TOKEN_LEN],
}
impl NewConnectionId<'_> {
pub const fn tag(&self) -> u8 {
new_connection_id_tag!()
}
}
decoder_parameterized_value!(
impl<'a> NewConnectionId<'a> {
fn decode(_tag: Tag, buffer: Buffer) -> Result<Self> {
let (sequence_number, buffer) = buffer.decode()?;
let (retire_prior_to, buffer) = buffer.decode()?;
decoder_invariant!(
retire_prior_to <= sequence_number,
"invalid retire prior to value"
);
let (connection_id_len, buffer) = buffer.decode::<u8>()?;
decoder_invariant!(
(1..=20).contains(&connection_id_len),
"invalid connection id length"
);
let (connection_id, buffer) = buffer.decode_slice(connection_id_len.into())?;
let connection_id = connection_id.into_less_safe_slice();
let (stateless_reset_token, buffer) = buffer.decode_slice(STATELESS_RESET_TOKEN_LEN)?;
let stateless_reset_token: &[u8] = stateless_reset_token.into_less_safe_slice();
let stateless_reset_token = stateless_reset_token
.try_into()
.expect("Length has been already verified");
let frame = NewConnectionId {
sequence_number,
retire_prior_to,
connection_id,
stateless_reset_token,
};
Ok((frame, buffer))
}
}
);
impl EncoderValue for NewConnectionId<'_> {
fn encode<E: Encoder>(&self, buffer: &mut E) {
buffer.encode(&self.tag());
buffer.encode(&self.sequence_number);
buffer.encode(&self.retire_prior_to);
buffer.encode_with_len_prefix::<u8, _>(&self.connection_id);
buffer.encode(&self.stateless_reset_token.as_ref());
}
}