use alloc::collections::VecDeque;
use core::cmp::min;
use core::ops::RangeBounds;
use std::io::{BufRead, BufReader, Read};
use crate::bits::Bits;
#[derive()]
pub struct Buffer<T> {
reader: BufReader<T>,
buffer: VecDeque<u8>,
}
impl<T: Read> Buffer<T> {
pub fn new(reader: T) -> Self {
Buffer {
reader: BufReader::new(reader),
buffer: VecDeque::new(),
}
}
pub fn consume_read_buffer_up_to(&mut self, len: usize) -> Vec<u8> {
let len = min(len, self.buffer.len());
self.buffer.drain(0..len).collect()
}
pub fn consume_read_buffer(&mut self) -> VecDeque<u8> {
core::mem::take(&mut self.buffer)
}
pub fn drain<R: RangeBounds<usize>>(&mut self, range: R) {
self.buffer.drain(range);
}
}
impl<T: Read> Read for Buffer<T> {
fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> {
if self.buffer.is_empty() {
let mut buf: [u8; 4096] = [0; 4096];
let read = self.reader.read(&mut buf)?;
self.buffer.extend(buf.get(0..read).unwrap_or_default());
}
let read = self.buffer.read(buf)?;
Ok(read)
}
}
impl<T: Read> Iterator for Buffer<T> {
type Item = u8;
fn next(&mut self) -> Option<Self::Item> {
if let Ok(Some(val)) = self.reader.next_u8() {
self.buffer.push_back(val);
return Some(val);
};
None
}
}
impl<T: Read> BufRead for Buffer<T> {
fn fill_buf(&mut self) -> std::io::Result<&[u8]> {
let buf = self.reader.fill_buf()?;
self.buffer.extend(buf.iter());
Ok(buf)
}
fn consume(&mut self, amt: usize) {
self.reader.consume(amt);
self.consume_read_buffer_up_to(amt);
}
}