use std::{
io, net,
pin::Pin,
task::{Context, Poll},
};
use futures::{
future::{BoxFuture, FutureExt},
prelude::*,
};
use super::{Incoming, Provider};
pub type Transport = crate::Transport<Tcp>;
#[derive(Copy, Clone)]
#[doc(hidden)]
pub enum Tcp {}
impl Provider for Tcp {
type Stream = TcpStream;
type Listener = tokio::net::TcpListener;
type IfWatcher = if_watch::tokio::IfWatcher;
fn new_if_watcher() -> io::Result<Self::IfWatcher> {
Self::IfWatcher::new()
}
fn addrs(if_watcher: &Self::IfWatcher) -> Vec<if_watch::IpNet> {
if_watcher.iter().copied().collect()
}
fn new_listener(l: net::TcpListener) -> io::Result<Self::Listener> {
tokio::net::TcpListener::try_from(l)
}
fn new_stream(s: net::TcpStream) -> BoxFuture<'static, io::Result<Self::Stream>> {
async move {
let stream = tokio::net::TcpStream::try_from(s)?;
stream.writable().await?;
if let Some(e) = stream.take_error()? {
return Err(e);
}
Ok(TcpStream(stream))
}
.boxed()
}
fn poll_accept(
l: &mut Self::Listener,
cx: &mut Context<'_>,
) -> Poll<io::Result<Incoming<Self::Stream>>> {
let (stream, remote_addr) = match l.poll_accept(cx) {
Poll::Pending => return Poll::Pending,
Poll::Ready(Err(e)) => return Poll::Ready(Err(e)),
Poll::Ready(Ok((stream, remote_addr))) => (stream, remote_addr),
};
let local_addr = stream.local_addr()?;
let stream = TcpStream(stream);
Poll::Ready(Ok(Incoming {
stream,
local_addr,
remote_addr,
}))
}
}
#[derive(Debug)]
pub struct TcpStream(pub tokio::net::TcpStream);
impl From<TcpStream> for tokio::net::TcpStream {
fn from(t: TcpStream) -> tokio::net::TcpStream {
t.0
}
}
impl AsyncRead for TcpStream {
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context,
buf: &mut [u8],
) -> Poll<Result<usize, io::Error>> {
let mut read_buf = tokio::io::ReadBuf::new(buf);
futures::ready!(tokio::io::AsyncRead::poll_read(
Pin::new(&mut self.0),
cx,
&mut read_buf
))?;
Poll::Ready(Ok(read_buf.filled().len()))
}
}
impl AsyncWrite for TcpStream {
fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context,
buf: &[u8],
) -> Poll<Result<usize, io::Error>> {
tokio::io::AsyncWrite::poll_write(Pin::new(&mut self.0), cx, buf)
}
fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<Result<(), io::Error>> {
tokio::io::AsyncWrite::poll_flush(Pin::new(&mut self.0), cx)
}
fn poll_close(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<Result<(), io::Error>> {
tokio::io::AsyncWrite::poll_shutdown(Pin::new(&mut self.0), cx)
}
fn poll_write_vectored(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
bufs: &[io::IoSlice<'_>],
) -> Poll<io::Result<usize>> {
tokio::io::AsyncWrite::poll_write_vectored(Pin::new(&mut self.0), cx, bufs)
}
}