use crate::{
frame::{GetFrameType, io::WriteFrameType},
sid::{Dir, MAX_STREAMS_LIMIT},
varint::{VarInt, WriteVarInt, be_varint},
};
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum MaxStreamsFrame {
Bi(VarInt),
Uni(VarInt),
}
impl MaxStreamsFrame {
pub fn with(dir: Dir, max_streams: VarInt) -> Self {
match dir {
Dir::Bi => MaxStreamsFrame::Bi(max_streams),
Dir::Uni => MaxStreamsFrame::Uni(max_streams),
}
}
}
impl super::GetFrameType for MaxStreamsFrame {
fn frame_type(&self) -> super::FrameType {
super::FrameType::MaxStreams(match self {
MaxStreamsFrame::Bi(_) => Dir::Bi,
MaxStreamsFrame::Uni(_) => Dir::Uni,
})
}
}
impl super::EncodeSize for MaxStreamsFrame {
fn max_encoding_size(&self) -> usize {
1 + 8
}
fn encoding_size(&self) -> usize {
1 + match self {
MaxStreamsFrame::Bi(max_streams) => max_streams.encoding_size(),
MaxStreamsFrame::Uni(max_streams) => max_streams.encoding_size(),
}
}
}
pub fn max_streams_frame_with_dir(
dir: Dir,
) -> impl Fn(&[u8]) -> nom::IResult<&[u8], MaxStreamsFrame> {
move |input: &[u8]| {
let (remain, max_streams) = be_varint(input)?;
if max_streams > MAX_STREAMS_LIMIT {
Err(nom::Err::Error(nom::error::Error::new(
input,
nom::error::ErrorKind::TooLarge,
)))
} else {
Ok((
remain,
match dir {
Dir::Bi => MaxStreamsFrame::Bi(max_streams),
Dir::Uni => MaxStreamsFrame::Uni(max_streams),
},
))
}
}
}
impl<T: bytes::BufMut> super::io::WriteFrame<MaxStreamsFrame> for T {
fn put_frame(&mut self, frame: &MaxStreamsFrame) {
match frame {
MaxStreamsFrame::Bi(max_streams) => {
self.put_frame_type(frame.frame_type());
self.put_varint(max_streams);
}
MaxStreamsFrame::Uni(max_streams) => {
self.put_frame_type(frame.frame_type());
self.put_varint(max_streams);
}
}
}
}
#[cfg(test)]
mod tests {
use nom::{Parser, combinator::flat_map};
use super::{MaxStreamsFrame, max_streams_frame_with_dir};
use crate::{
frame::{
EncodeSize, FrameType, GetFrameType,
io::{WriteFrame, WriteFrameType},
},
sid::Dir,
varint::{VarInt, be_varint},
};
#[test]
fn test_max_streams_frame() {
let frame = MaxStreamsFrame::Bi(VarInt::from_u32(0x1234));
assert_eq!(frame.frame_type(), FrameType::MaxStreams(Dir::Bi));
assert_eq!(frame.max_encoding_size(), 1 + 8);
assert_eq!(frame.encoding_size(), 1 + 2);
let frame = MaxStreamsFrame::Uni(VarInt::from_u32(0x1236));
assert_eq!(frame.frame_type(), FrameType::MaxStreams(Dir::Uni));
assert_eq!(frame.max_encoding_size(), 1 + 8);
assert_eq!(frame.encoding_size(), 1 + 2);
}
#[test]
fn test_read_max_streams_frame() {
let max_streams_bi_type = VarInt::from(FrameType::MaxStreams(Dir::Bi));
let max_streams_uni_type = VarInt::from(FrameType::MaxStreams(Dir::Uni));
let buf = vec![max_streams_bi_type.into_inner() as u8, 0x52, 0x34];
let (input, frame) = flat_map(be_varint, |frame_type| {
if frame_type == max_streams_bi_type {
max_streams_frame_with_dir(Dir::Bi)
} else {
panic!("wrong frame type: {frame_type}")
}
})
.parse(buf.as_ref())
.unwrap();
assert!(input.is_empty());
assert_eq!(frame, MaxStreamsFrame::Bi(VarInt::from_u32(0x1234)));
let buf = vec![max_streams_uni_type.into_inner() as u8, 0x52, 0x36];
let (input, frame) = flat_map(be_varint, |frame_type| {
if frame_type == max_streams_uni_type {
max_streams_frame_with_dir(Dir::Uni)
} else {
panic!("wrong frame type: {frame_type}")
}
})
.parse(buf.as_ref())
.unwrap();
assert!(input.is_empty());
assert_eq!(frame, MaxStreamsFrame::Uni(VarInt::from_u32(0x1236)));
}
#[test]
fn test_read_too_large_max_streams_frame() {
let mut buf = Vec::new();
buf.put_frame_type(FrameType::MaxStreams(Dir::Bi));
buf.extend_from_slice(&[0xd0, 0x34, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80]);
let result = flat_map(be_varint, |frame_type| {
if frame_type == VarInt::from(FrameType::MaxStreams(Dir::Bi)) {
max_streams_frame_with_dir(Dir::Bi)
} else {
panic!("wrong frame type: {frame_type}")
}
})
.parse(buf.as_ref());
assert_eq!(
result,
Err(nom::Err::Error(nom::error::Error::new(
&buf[1..],
nom::error::ErrorKind::TooLarge,
)))
);
}
#[test]
fn test_write_max_streams_frame() {
let mut buf = Vec::new();
buf.put_frame(&MaxStreamsFrame::Bi(VarInt::from_u32(0x1234)));
let mut expected = Vec::new();
expected.put_frame_type(FrameType::MaxStreams(Dir::Bi));
expected.extend_from_slice(&[0x52, 0x34]);
assert_eq!(buf, expected);
buf.clear();
buf.put_frame(&MaxStreamsFrame::Uni(VarInt::from_u32(0x1236)));
expected.clear();
expected.put_frame_type(FrameType::MaxStreams(Dir::Uni));
expected.extend_from_slice(&[0x52, 0x36]);
assert_eq!(buf, expected);
}
}