use crate::Error;
use futures_io::AsyncWrite;
use futures_util::AsyncWriteExt;
use minicbor::Encode;
use std::io;
#[derive(Debug)]
pub struct AsyncWriter<W> {
writer: W,
buffer: Vec<u8>,
max_len: usize,
state: State
}
#[derive(Debug)]
enum State {
None,
WriteFrom(usize)
}
impl<W> AsyncWriter<W> {
pub fn new(writer: W) -> Self {
Self::with_buffer(writer, Vec::new())
}
pub fn with_buffer(writer: W, buffer: Vec<u8>) -> Self {
Self { writer, buffer, max_len: 512 * 1024, state: State::None }
}
pub fn set_max_len(&mut self, val: u32) {
self.max_len = val as usize
}
pub fn writer(&self) -> &W {
&self.writer
}
pub fn writer_mut(&mut self) -> &mut W {
&mut self.writer
}
pub fn into_parts(self) -> (W, Vec<u8>) {
(self.writer, self.buffer)
}
}
impl<W: AsyncWrite + Unpin> AsyncWriter<W> {
pub async fn write<T: Encode<()>>(&mut self, val: T) -> Result<usize, Error> {
self.write_with(val, &mut ()).await
}
pub async fn write_with<C, T: Encode<C>>(&mut self, val: T, ctx: &mut C) -> Result<usize, Error> {
self.buffer.resize(4, 0u8);
minicbor::encode_with(val, &mut self.buffer, ctx)?;
if self.buffer.len() - 4 > self.max_len {
return Err(Error::InvalidLen)
}
let prefix = (self.buffer.len() as u32 - 4).to_be_bytes();
self.buffer[.. 4].copy_from_slice(&prefix);
self.state = State::WriteFrom(0);
self.sync().await?;
Ok(self.buffer.len() - 4)
}
pub async fn sync(&mut self) -> Result<(), Error> {
loop {
match self.state {
State::None => {
return Ok(())
}
State::WriteFrom(o) if o >= self.buffer.len() => {
self.state = State::None;
return Ok(())
}
State::WriteFrom(ref mut o) => {
let n = self.writer.write(&self.buffer[*o ..]).await?;
if n == 0 {
return Err(Error::Io(io::ErrorKind::WriteZero.into()))
}
*o += n
}
}
}
}
pub async fn flush(&mut self) -> Result<(), Error> {
self.writer.flush().await?;
Ok(())
}
}