use std::future::Future;
use std::io;
use std::net::{self, SocketAddr, ToSocketAddrs};
use std::os::unix::io::{AsRawFd, RawFd};
use std::pin::{pin, Pin};
use std::task::{ready, Context, Poll};
use futures_core::stream::Stream;
use super::stream::TcpStream;
use crate::io::Async;
pub struct TcpListener {
inner: Async<net::TcpListener>,
}
impl TcpListener {
pub async fn bind<A: ToSocketAddrs>(addr: A) -> io::Result<TcpListener> {
Ok(TcpListener {
inner: Async::new(net::TcpListener::bind(addr)?)?,
})
}
pub fn from_std(listener: net::TcpListener) -> io::Result<TcpListener> {
Ok(TcpListener {
inner: Async::new(listener)?,
})
}
pub async fn accept(&self) -> io::Result<(TcpStream, SocketAddr)> {
let (stream, addr) = self.inner.read_with(|io| io.accept()).await?;
Ok((TcpStream::from_std(stream)?, addr))
}
pub fn incoming(&self) -> Incoming<'_> {
Incoming { inner: self }
}
pub fn local_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().local_addr()
}
}
pub struct Incoming<'a> {
inner: &'a TcpListener,
}
impl<'a> Stream for Incoming<'a> {
type Item = io::Result<TcpStream>;
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
let fut = self.inner.accept();
let fut = pin!(fut);
let (stream, _) = ready!(fut.poll(cx))?;
Poll::Ready(Some(Ok(stream)))
}
}
impl AsRawFd for TcpListener {
fn as_raw_fd(&self) -> RawFd {
self.inner.get_ref().as_raw_fd()
}
}