use std::io;
use std::task::{Context, Poll};
use std::pin::Pin;
use tokio::io::{AsyncRead, AsyncWrite, ReadBuf};
use crate::http::hyper::upgrade::Upgraded;
pub struct IoStream {
kind: IoStreamKind,
}
enum IoStreamKind {
Upgraded(Upgraded)
}
#[crate::async_trait]
pub trait IoHandler: Send {
async fn io(self: Pin<Box<Self>>, io: IoStream) -> io::Result<()>;
}
#[doc(hidden)]
impl From<Upgraded> for IoStream {
fn from(io: Upgraded) -> Self {
IoStream { kind: IoStreamKind::Upgraded(io) }
}
}
pub trait AsyncReadWrite: AsyncRead + AsyncWrite + Unpin { }
impl<T: AsyncRead + AsyncWrite + Unpin> AsyncReadWrite for T { }
impl IoStream {
fn inner_mut(&mut self) -> Pin<&mut dyn AsyncReadWrite> {
match self.kind {
IoStreamKind::Upgraded(ref mut io) => Pin::new(io),
}
}
fn inner_is_write_vectored(&self) -> bool {
match self.kind {
IoStreamKind::Upgraded(ref io) => io.is_write_vectored(),
}
}
}
impl AsyncRead for IoStream {
fn poll_read(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut ReadBuf<'_>,
) -> Poll<io::Result<()>> {
self.get_mut().inner_mut().poll_read(cx, buf)
}
}
impl AsyncWrite for IoStream {
fn poll_write(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<io::Result<usize>> {
self.get_mut().inner_mut().poll_write(cx, buf)
}
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
self.get_mut().inner_mut().poll_flush(cx)
}
fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
self.get_mut().inner_mut().poll_shutdown(cx)
}
fn poll_write_vectored(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
bufs: &[io::IoSlice<'_>],
) -> Poll<io::Result<usize>> {
self.get_mut().inner_mut().poll_write_vectored(cx, bufs)
}
fn is_write_vectored(&self) -> bool {
self.inner_is_write_vectored()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn is_unpin() {
fn check_traits<T: AsyncRead + AsyncWrite + Unpin + Send>() {}
check_traits::<IoStream>();
}
}