use std::convert::TryFrom;
use std::fmt;
use std::io::{Read as _, Write as _};
use std::net::Shutdown;
#[cfg(unix)]
use std::os::unix::io::{AsRawFd, RawFd};
#[cfg(windows)]
use std::os::windows::io::{AsRawSocket, RawSocket};
use std::path::Path;
use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, Poll};
#[doc(no_inline)]
pub use std::os::unix::net::SocketAddr;
use async_io::Async;
use futures_lite::*;
#[derive(Clone, Debug)]
pub struct UnixListener {
inner: Arc<Async<std::os::unix::net::UnixListener>>,
}
impl UnixListener {
fn new(inner: Arc<Async<std::os::unix::net::UnixListener>>) -> UnixListener {
UnixListener { inner }
}
pub fn bind<P: AsRef<Path>>(path: P) -> io::Result<UnixListener> {
let listener = Async::<std::os::unix::net::UnixListener>::bind(path)?;
Ok(UnixListener::new(Arc::new(listener)))
}
pub async fn accept(&self) -> io::Result<(UnixStream, SocketAddr)> {
let (stream, addr) = self.inner.accept().await?;
Ok((UnixStream::new(Arc::new(stream)), addr))
}
pub fn incoming(&self) -> Incoming<'_> {
Incoming { listener: self }
}
pub fn local_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().local_addr()
}
}
impl From<Async<std::os::unix::net::UnixListener>> for UnixListener {
fn from(listener: Async<std::os::unix::net::UnixListener>) -> UnixListener {
UnixListener::new(Arc::new(listener))
}
}
impl TryFrom<std::os::unix::net::UnixListener> for UnixListener {
type Error = io::Error;
fn try_from(listener: std::os::unix::net::UnixListener) -> io::Result<UnixListener> {
Ok(UnixListener::new(Arc::new(Async::new(listener)?)))
}
}
#[cfg(unix)]
impl AsRawFd for UnixListener {
fn as_raw_fd(&self) -> RawFd {
self.inner.as_raw_fd()
}
}
#[cfg(windows)]
impl AsRawSocket for UnixListener {
fn as_raw_socket(&self) -> RawSocket {
self.inner.as_raw_socket()
}
}
#[derive(Debug)]
pub struct Incoming<'a> {
listener: &'a UnixListener,
}
impl Stream for Incoming<'_> {
type Item = io::Result<UnixStream>;
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
let future = self.listener.accept();
pin!(future);
let (socket, _) = ready!(future.poll(cx))?;
Poll::Ready(Some(Ok(socket)))
}
}
pub struct UnixStream {
inner: Arc<Async<std::os::unix::net::UnixStream>>,
readable: Option<Pin<Box<dyn Future<Output = io::Result<()>> + Send + Sync>>>,
writable: Option<Pin<Box<dyn Future<Output = io::Result<()>> + Send + Sync>>>,
}
impl UnixStream {
fn new(inner: Arc<Async<std::os::unix::net::UnixStream>>) -> UnixStream {
UnixStream {
inner,
readable: None,
writable: None,
}
}
pub async fn connect<P: AsRef<Path>>(path: P) -> io::Result<UnixStream> {
let stream = Async::<std::os::unix::net::UnixStream>::connect(path).await?;
Ok(UnixStream::new(Arc::new(stream)))
}
pub fn pair() -> io::Result<(UnixStream, UnixStream)> {
let (a, b) = Async::<std::os::unix::net::UnixStream>::pair()?;
Ok((UnixStream::new(Arc::new(a)), UnixStream::new(Arc::new(b))))
}
pub fn local_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().local_addr()
}
pub fn peer_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().peer_addr()
}
pub fn shutdown(&self, how: Shutdown) -> io::Result<()> {
self.inner.get_ref().shutdown(how)
}
}
impl fmt::Debug for UnixStream {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.inner.fmt(f)
}
}
impl Clone for UnixStream {
fn clone(&self) -> UnixStream {
UnixStream::new(self.inner.clone())
}
}
impl From<Async<std::os::unix::net::UnixStream>> for UnixStream {
fn from(stream: Async<std::os::unix::net::UnixStream>) -> UnixStream {
UnixStream::new(Arc::new(stream))
}
}
impl TryFrom<std::os::unix::net::UnixStream> for UnixStream {
type Error = io::Error;
fn try_from(stream: std::os::unix::net::UnixStream) -> io::Result<UnixStream> {
Ok(UnixStream::new(Arc::new(Async::new(stream)?)))
}
}
#[cfg(unix)]
impl AsRawFd for UnixStream {
fn as_raw_fd(&self) -> RawFd {
self.inner.as_raw_fd()
}
}
#[cfg(windows)]
impl AsRawSocket for UnixStream {
fn as_raw_socket(&self) -> RawSocket {
self.inner.as_raw_socket()
}
}
impl AsyncRead for UnixStream {
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut [u8],
) -> Poll<io::Result<usize>> {
loop {
ready!(crate::maybe_yield(cx));
match self.inner.get_ref().read(buf) {
Err(err) if err.kind() == io::ErrorKind::WouldBlock => {}
res => {
self.readable = None;
return Poll::Ready(res);
}
}
if self.readable.is_none() {
let inner = self.inner.clone();
self.readable = Some(Box::pin(async move { inner.readable().await }));
}
if let Some(f) = &mut self.readable {
ready!(f.as_mut().poll(cx))?;
self.readable = None;
}
}
}
}
impl AsyncWrite for UnixStream {
fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<io::Result<usize>> {
loop {
ready!(crate::maybe_yield(cx));
match self.inner.get_ref().write(buf) {
Err(err) if err.kind() == io::ErrorKind::WouldBlock => {}
res => {
self.writable = None;
return Poll::Ready(res);
}
}
if self.writable.is_none() {
let inner = self.inner.clone();
self.writable = Some(Box::pin(async move { inner.writable().await }));
}
if let Some(f) = &mut self.writable {
ready!(f.as_mut().poll(cx))?;
self.writable = None;
}
}
}
fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
loop {
ready!(crate::maybe_yield(cx));
match self.inner.get_ref().flush() {
Err(err) if err.kind() == io::ErrorKind::WouldBlock => {}
res => {
self.writable = None;
return Poll::Ready(res);
}
}
if self.writable.is_none() {
let inner = self.inner.clone();
self.writable = Some(Box::pin(async move { inner.writable().await }));
}
if let Some(f) = &mut self.writable {
ready!(f.as_mut().poll(cx))?;
self.writable = None;
}
}
}
fn poll_close(self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll<io::Result<()>> {
Poll::Ready(self.inner.get_ref().shutdown(Shutdown::Write))
}
}
#[derive(Clone, Debug)]
pub struct UnixDatagram {
inner: Arc<Async<std::os::unix::net::UnixDatagram>>,
}
impl UnixDatagram {
fn new(inner: Arc<Async<std::os::unix::net::UnixDatagram>>) -> UnixDatagram {
UnixDatagram { inner }
}
pub fn bind<P: AsRef<Path>>(path: P) -> io::Result<UnixDatagram> {
let socket = Async::<std::os::unix::net::UnixDatagram>::bind(path)?;
Ok(UnixDatagram::new(Arc::new(socket)))
}
pub fn unbound() -> io::Result<UnixDatagram> {
let socket = Async::<std::os::unix::net::UnixDatagram>::unbound()?;
Ok(UnixDatagram::new(Arc::new(socket)))
}
pub fn pair() -> io::Result<(UnixDatagram, UnixDatagram)> {
let (a, b) = Async::<std::os::unix::net::UnixDatagram>::pair()?;
Ok((
UnixDatagram::new(Arc::new(a)),
UnixDatagram::new(Arc::new(b)),
))
}
pub fn connect<P: AsRef<Path>>(&self, path: P) -> io::Result<()> {
let p = path.as_ref();
self.inner.get_ref().connect(p)
}
pub fn local_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().local_addr()
}
pub fn peer_addr(&self) -> io::Result<SocketAddr> {
self.inner.get_ref().peer_addr()
}
pub async fn recv_from(&self, buf: &mut [u8]) -> io::Result<(usize, SocketAddr)> {
self.inner.recv_from(buf).await
}
pub async fn send_to<P: AsRef<Path>>(&self, buf: &[u8], path: P) -> io::Result<usize> {
self.inner.send_to(buf, path.as_ref()).await
}
pub async fn recv(&self, buf: &mut [u8]) -> io::Result<usize> {
self.inner.recv(buf).await
}
pub async fn send(&self, buf: &[u8]) -> io::Result<usize> {
self.inner.send(buf).await
}
pub fn shutdown(&self, how: Shutdown) -> io::Result<()> {
self.inner.get_ref().shutdown(how)
}
}
impl From<Async<std::os::unix::net::UnixDatagram>> for UnixDatagram {
fn from(socket: Async<std::os::unix::net::UnixDatagram>) -> UnixDatagram {
UnixDatagram::new(Arc::new(socket))
}
}
impl TryFrom<std::os::unix::net::UnixDatagram> for UnixDatagram {
type Error = io::Error;
fn try_from(socket: std::os::unix::net::UnixDatagram) -> io::Result<UnixDatagram> {
Ok(UnixDatagram::new(Arc::new(Async::new(socket)?)))
}
}
#[cfg(unix)]
impl AsRawFd for UnixDatagram {
fn as_raw_fd(&self) -> RawFd {
self.inner.as_raw_fd()
}
}
#[cfg(windows)]
impl AsRawSocket for UnixDatagram {
fn as_raw_socket(&self) -> RawSocket {
self.inner.as_raw_socket()
}
}