1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
use super::*;
use bytes::{Buf, BufMut, BytesMut};

/// The request_n frame.
///
/// # Frame Contents
///
/// ```text
///  0                   1                   2                   3
///  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
/// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
/// |                           Stream ID                           |
/// +-----------+-+-+---------------+-------------------------------+
/// |Frame Type |0|0|     Flags     |
/// +-------------------------------+-------------------------------+
/// |0|                         Request N                           |
/// +---------------------------------------------------------------+
/// ```
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct RequestNFrame {
    stream_id: u32,
    request_n: u32,
}

impl RequestNFrame {
    /// Type of this frame.
    pub const TYPE: FrameType = FrameType::REQUEST_N;

    /// Create a new `RequestN` frame.
    ///
    /// - `stream_id` MUST be <= [`MAX_U31`].
    /// - `request_n` represents the number of items to request. Value MUST be > 0 and
    /// <= [`MAX_U31`].
    pub fn new(stream_id: u32, request_n: u32) -> Self {
        debug_assert_max_u31!(stream_id, request_n);
        debug_assert_non_zero!(request_n);
        let stream_id = stream_id & MAX_U31;
        let request_n = request_n & MAX_U31;
        RequestNFrame { stream_id, request_n }
    }

    /// Returns the stream ID of this frame.
    pub fn stream_id(&self) -> u32 {
        self.stream_id
    }

    /// Returns the number of items to request.
    pub fn request_n(&self) -> u32 {
        self.request_n
    }
}

impl Encode for RequestNFrame {
    fn encode(&self, buf: &mut BytesMut) {
        buf.put_u32(self.stream_id);
        buf.put_u16(FrameType::REQUEST_N.bits());
        buf.put_u32(self.request_n);
    }

    fn len(&self) -> usize {
        // len(stream_id): 4
        // len(flags): 2
        // len(request_n): 4
        10
    }
}

impl Decode for RequestNFrame {
    type Value = Self;

    fn decode<B: Buf>(
        buf: &mut B,
        stream_id: u32,
        _flags: Flags,
    ) -> Result<Self::Value> {
        let request_n = eat_u31(buf)?;
        Ok(RequestNFrame { stream_id, request_n })
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_codec() {
        let lease = RequestNFrame::new(1, 2);

        let mut buf = BytesMut::new();
        lease.encode(&mut buf);
        let mut buf = buf.freeze();

        // len(stream_id): 4
        // len(flags): 2
        // len(request_n): 4
        let buf_len = buf.len();
        assert_eq!(buf_len, 4 + 2 + 4);

        // Eat the stream_id and flags before decoding bytes.
        let stream_id = eat_stream_id(&mut buf).unwrap();
        let (frame_type, flags) = eat_flags(&mut buf).unwrap();
        assert_eq!(frame_type, FrameType::REQUEST_N);
        assert_eq!(flags, Flags::empty());

        let decoded =
            RequestNFrame::decode(&mut buf, stream_id, flags).unwrap();

        assert_eq!(decoded, lease);
        assert_eq!(lease.len(), buf_len);
        assert_eq!(decoded.len(), buf_len);
    }
}