use std::{ io, pin::Pin, task::{Context, Poll}, }; use futures_io::AsyncWrite; use crate::{encode, MAX_DATA_LEN, U16_HEX_BYTES}; pin_project_lite::pin_project! { /// An implementor of [`Write`][io::Write] which passes all input to an inner `Write` in packet line data encoding, /// one line per `write(…)` call or as many lines as it takes if the data doesn't fit into the maximum allowed line length. pub struct Writer { #[pin] inner: encode::LineWriter<'static, T>, state: State, } } enum State { Idle, WriteData(usize), } impl Writer { /// Create a new instance from the given `write` pub fn new(write: T) -> Self { Writer { inner: encode::LineWriter::new(write, &[], &[]), state: State::Idle, } } /// Return the inner writer, consuming self. pub fn into_inner(self) -> T { self.inner.into_inner() } /// Return a mutable reference to the inner writer, useful if packet lines should be serialized directly. pub fn inner_mut(&mut self) -> &mut T { &mut self.inner.writer } } /// Non-IO methods impl Writer { /// If called, each call to [`write()`][io::Write::write()] will write bytes as is. pub fn enable_binary_mode(&mut self) { self.inner.suffix = &[]; } /// If called, each call to [`write()`][io::Write::write()] will write the input as text, appending a trailing newline /// if needed before writing. pub fn enable_text_mode(&mut self) { self.inner.suffix = &[b'\n']; } } impl AsyncWrite for Writer { fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll> { let mut this = self.project(); loop { match this.state { State::Idle => { if buf.is_empty() { return Poll::Ready(Err(io::Error::new( io::ErrorKind::Other, "empty packet lines are not permitted as '0004' is invalid", ))); } *this.state = State::WriteData(0) } State::WriteData(written) => { while *written != buf.len() { let data = &buf[*written..*written + (buf.len() - *written).min(MAX_DATA_LEN)]; let n = futures_lite::ready!(this.inner.as_mut().poll_write(cx, data))?; if n == 0 { return Poll::Ready(Err(io::ErrorKind::WriteZero.into())); } *written += n; *written -= U16_HEX_BYTES + this.inner.suffix.len(); } *this.state = State::Idle; return Poll::Ready(Ok(buf.len())); } } } } fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll> { self.project().inner.poll_flush(cx) } fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll> { self.project().inner.poll_close(cx) } }