use crate::Error;
use futures_io::AsyncRead;
use futures_util::AsyncReadExt;
use minicbor::Decode;
use std::io;
#[derive(Debug)]
pub struct AsyncReader<R> {
reader: R,
buffer: Vec<u8>,
max_len: usize,
state: State
}
#[derive(Debug)]
enum State {
ReadLen([u8; 4], u8),
ReadVal(usize)
}
impl State {
fn new() -> Self {
State::ReadLen([0; 4], 0)
}
}
impl<R> AsyncReader<R> {
pub fn new(reader: R) -> Self {
Self::with_buffer(reader, Vec::new())
}
pub fn with_buffer(reader: R, buffer: Vec<u8>) -> Self {
Self { reader, buffer, max_len: 512 * 1024, state: State::new() }
}
pub fn set_max_len(&mut self, val: u32) {
self.max_len = val as usize
}
pub fn reader(&self) -> &R {
&self.reader
}
pub fn reader_mut(&mut self) -> &mut R {
&mut self.reader
}
pub fn into_parts(self) -> (R, Vec<u8>) {
(self.reader, self.buffer)
}
}
impl<R: AsyncRead + Unpin> AsyncReader<R> {
pub async fn read<'a, T: Decode<'a, ()>>(&'a mut self) -> Result<Option<T>, Error> {
self.read_with(&mut ()).await
}
pub async fn read_with<'a, C, T: Decode<'a, C>>(&'a mut self, ctx: &mut C) -> Result<Option<T>, Error> {
loop {
match self.state {
State::ReadLen(buf, 4) => {
let len = u32::from_be_bytes(buf) as usize;
if len > self.max_len {
return Err(Error::InvalidLen)
}
self.buffer.clear();
self.buffer.resize(len, 0u8);
self.state = State::ReadVal(0)
}
State::ReadLen(ref mut buf, ref mut o) => {
let n = self.reader.read(&mut buf[usize::from(*o) ..]).await?;
if n == 0 {
return if *o == 0 {
Ok(None)
} else {
Err(Error::Io(io::ErrorKind::UnexpectedEof.into()))
}
}
*o += n as u8
}
State::ReadVal(o) if o >= self.buffer.len() => {
self.state = State::new();
return minicbor::decode_with(&self.buffer, ctx).map_err(Error::Decode).map(Some)
}
State::ReadVal(ref mut o) => {
let n = self.reader.read(&mut self.buffer[*o ..]).await?;
if n == 0 {
return Err(Error::Io(io::ErrorKind::UnexpectedEof.into()))
}
*o += n
}
}
}
}
}