use tokio::io::{AsyncRead, AsyncReadExt, AsyncWrite, ReadBuf};
use tokio::net::TcpStream;
use std::pin::Pin;
use std::task::{Context, Poll};
use bytes::BytesMut;
use std::io::{Error, ErrorKind};
#[derive(Debug)]
pub struct ManagedStream {
stream: TcpStream,
buffer: BytesMut,
sealed: bool
}
impl ManagedStream {
pub fn new(stream: TcpStream) -> Self {
Self {
stream,
sealed: false,
buffer: BytesMut::with_capacity(4096)
}
}
pub fn seal(&mut self) {
self.sealed = true;
}
pub async fn peek_async(&mut self) -> Result<(bool,Vec<u8>), Error> {
if self.sealed {
return Err(Error::new(ErrorKind::Other, "Stream is sealed"));
}
if let Ok(Some(e)) = self.stream.take_error() {
return Err(e);
}
let mut temp_buf = [0u8; 1024]; match self.stream.read(&mut temp_buf).await {
Ok(0) => {
return Ok((true,self.buffer.to_vec()));
}
Ok(n) => {
self.buffer.extend_from_slice(&temp_buf[..n]);
}
Err(e) if e.kind() == std::io::ErrorKind::WouldBlock => {
}
Err(e) => return Err(e),
}
Ok((false,self.buffer.to_vec()))
}
}
impl AsyncRead for ManagedStream {
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut ReadBuf<'_>,
) -> Poll<Result<(), Error>> {
if !self.sealed {
return Poll::Ready(Err(Error::new(
ErrorKind::Other,
"Stream has not been properly sealed",
)));
}
if !self.buffer.is_empty() {
let to_read = std::cmp::min(buf.remaining(), self.buffer.len());
buf.put_slice(&self.buffer.split_to(to_read));
if buf.remaining() == 0 {
return Poll::Ready(Ok(()));
}
}
match Pin::new(&mut self.stream).poll_read(cx, buf) {
Poll::Pending => {
if buf.filled().is_empty() {
Poll::Pending
} else {
Poll::Ready(Ok(()))
}
}
Poll::Ready(Ok(())) => {
Poll::Ready(Ok(()))
}
Poll::Ready(Err(e)) => {
if buf.filled().is_empty() {
Poll::Ready(Err(e))
} else {
Poll::Ready(Ok(()))
}
}
}
}
}
impl AsyncWrite for ManagedStream {
fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<Result<usize, Error>> {
Pin::new(&mut self.stream).poll_write(cx, buf)
}
fn poll_flush(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Result<(), Error>> {
Pin::new(&mut self.stream).poll_flush(cx)
}
fn poll_shutdown(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Result<(), Error>> {
Pin::new(&mut self.stream).poll_shutdown(cx)
}
}