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
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
use std::{
    ops::{Deref, Index},
    slice::SliceIndex,
};

/// A size-adating buffer to store bytes in. The buffer grows when data is
/// stored into it. The contents can then be consumed which results in
/// the buffer dropping the consumed data before new data are appended.
#[derive(Debug, Default)]
pub struct BlobBuffer {
    buf: Vec<u8>,
    start: usize,
    offset: usize,
}

impl BlobBuffer {
    /// Constructs a new `BlobBuffer`.
    pub fn new() -> BlobBuffer {
        BlobBuffer::default()
    }

    /// Provide additional data to the internal buffer.
    pub fn extend_from_slice(&mut self, data: &[u8]) {
        if self.start > 0 {
            drop(self.buf.drain(0..self.start));
            self.start = 0;
        }

        self.buf.extend_from_slice(data);
    }

    /// Consume the given amount of data from the internal buffer.
    pub fn consume(&mut self, length: usize) {
        self.start += length;
        self.offset += length;
    }

    /// Returns the unconsumed byte length of the internal buffer.
    pub fn len(&self) -> usize {
        self.buf.len() - self.start
    }

    /// Returns whether the internal buffer is empty.
    pub fn is_empty(&self) -> bool {
        self.buf.len() == self.start
    }

    /// Get amount of already consumed bytes.
    pub fn offset(&self) -> usize {
        self.offset
    }
}

impl Deref for BlobBuffer {
    type Target = [u8];

    fn deref(&self) -> &[u8] {
        &self.buf[self.start..]
    }
}

impl<I> Index<I> for BlobBuffer
where
    I: SliceIndex<[u8]>,
{
    type Output = I::Output;

    fn index(&self, index: I) -> &Self::Output {
        &self.deref()[index]
    }
}

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

    #[test]
    fn test() {
        let mut bb = BlobBuffer::new();

        assert_eq!(0, bb.buf.len());
        assert_eq!(0, bb.start);
        assert_eq!(0, bb.offset);
        assert_eq!(0, bb.len());
        assert_eq!(true, bb.is_empty());

        bb.extend_from_slice(&[0x00, 0x01, 0x02, 0x03]);

        assert_eq!(4, bb.buf.len());
        assert_eq!(0, bb.start);
        assert_eq!(0, bb.offset);
        assert_eq!(4, bb.len());
        assert_eq!(false, bb.is_empty());
        assert_eq!(&[0x00, 0x01, 0x02, 0x03], &*bb);

        bb.consume(2);

        assert_eq!(4, bb.buf.len());
        assert_eq!(2, bb.start);
        assert_eq!(2, bb.offset);
        assert_eq!(2, bb.len());
        assert_eq!(false, bb.is_empty());
        assert_eq!(&[0x02, 0x03], &*bb);

        bb.consume(1);

        assert_eq!(4, bb.buf.len());
        assert_eq!(3, bb.start);
        assert_eq!(3, bb.offset);
        assert_eq!(1, bb.len());
        assert_eq!(false, bb.is_empty());
        assert_eq!(&[0x03], &*bb);

        bb.extend_from_slice(&[0x04, 0x05, 0x06, 0x07]);

        assert_eq!(5, bb.buf.len());
        assert_eq!(0, bb.start);
        assert_eq!(3, bb.offset);
        assert_eq!(5, bb.len());
        assert_eq!(false, bb.is_empty());
        assert_eq!(&[0x03, 0x04, 0x05, 0x06, 0x07], &*bb);

        // Deref trait impl
        assert_eq!(&[0x03, 0x04, 0x05, 0x06, 0x07], &(*bb));

        // Index trait impl
        assert_eq!(0x05, bb[2]);
        assert_eq!(&[0x05, 0x06], &bb[2..4]);
    }
}