2
0
Fork 1
mirror of https://github.com/async-rs/async-std.git synced 2025-01-16 10:49:55 +00:00

Merge pull request #772 from jbr/unixstream-clone

This commit is contained in:
Friedel Ziegelmayer 2020-05-13 18:03:47 +02:00 committed by GitHub
commit 2b6c7fedff
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 42 additions and 11 deletions

View file

@ -13,6 +13,7 @@ use crate::io;
use crate::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd}; use crate::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd};
use crate::path::Path; use crate::path::Path;
use crate::stream::Stream; use crate::stream::Stream;
use crate::sync::Arc;
use crate::task::{Context, Poll}; use crate::task::{Context, Poll};
/// A Unix domain socket server, listening for connections. /// A Unix domain socket server, listening for connections.
@ -92,7 +93,7 @@ impl UnixListener {
pub async fn accept(&self) -> io::Result<(UnixStream, SocketAddr)> { pub async fn accept(&self) -> io::Result<(UnixStream, SocketAddr)> {
let (stream, addr) = self.watcher.accept().await?; let (stream, addr) = self.watcher.accept().await?;
Ok((UnixStream { watcher: stream }, addr)) Ok((UnixStream { watcher: Arc::new(stream) }, addr))
} }
/// Returns a stream of incoming connections. /// Returns a stream of incoming connections.

View file

@ -11,6 +11,7 @@ use super::SocketAddr;
use crate::io::{self, Read, Write}; use crate::io::{self, Read, Write};
use crate::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd}; use crate::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd};
use crate::path::Path; use crate::path::Path;
use crate::sync::Arc;
use crate::task::{Context, Poll}; use crate::task::{Context, Poll};
/// A Unix stream socket. /// A Unix stream socket.
@ -36,8 +37,9 @@ use crate::task::{Context, Poll};
/// # /// #
/// # Ok(()) }) } /// # Ok(()) }) }
/// ``` /// ```
#[derive(Clone)]
pub struct UnixStream { pub struct UnixStream {
pub(super) watcher: Async<StdUnixStream>, pub(super) watcher: Arc<Async<StdUnixStream>>,
} }
impl UnixStream { impl UnixStream {
@ -56,7 +58,7 @@ impl UnixStream {
/// ``` /// ```
pub async fn connect<P: AsRef<Path>>(path: P) -> io::Result<UnixStream> { pub async fn connect<P: AsRef<Path>>(path: P) -> io::Result<UnixStream> {
let path = path.as_ref().to_owned(); let path = path.as_ref().to_owned();
let stream = Async::<StdUnixStream>::connect(path).await?; let stream = Arc::new(Async::<StdUnixStream>::connect(path).await?);
Ok(UnixStream { watcher: stream }) Ok(UnixStream { watcher: stream })
} }
@ -78,8 +80,12 @@ impl UnixStream {
/// ``` /// ```
pub fn pair() -> io::Result<(UnixStream, UnixStream)> { pub fn pair() -> io::Result<(UnixStream, UnixStream)> {
let (a, b) = Async::<StdUnixStream>::pair()?; let (a, b) = Async::<StdUnixStream>::pair()?;
let a = UnixStream { watcher: a }; let a = UnixStream {
let b = UnixStream { watcher: b }; watcher: Arc::new(a),
};
let b = UnixStream {
watcher: Arc::new(b),
};
Ok((a, b)) Ok((a, b))
} }
@ -158,7 +164,7 @@ impl Read for &UnixStream {
cx: &mut Context<'_>, cx: &mut Context<'_>,
buf: &mut [u8], buf: &mut [u8],
) -> Poll<io::Result<usize>> { ) -> Poll<io::Result<usize>> {
Pin::new(&mut &self.watcher).poll_read(cx, buf) Pin::new(&mut &*self.watcher).poll_read(cx, buf)
} }
} }
@ -186,15 +192,15 @@ impl Write for &UnixStream {
cx: &mut Context<'_>, cx: &mut Context<'_>,
buf: &[u8], buf: &[u8],
) -> Poll<io::Result<usize>> { ) -> Poll<io::Result<usize>> {
Pin::new(&mut &self.watcher).poll_write(cx, buf) Pin::new(&mut &*self.watcher).poll_write(cx, buf)
} }
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> { fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
Pin::new(&mut &self.watcher).poll_flush(cx) Pin::new(&mut &*self.watcher).poll_flush(cx)
} }
fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> { fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
Pin::new(&mut &self.watcher).poll_close(cx) Pin::new(&mut &*self.watcher).poll_close(cx)
} }
} }
@ -219,7 +225,7 @@ impl From<StdUnixStream> for UnixStream {
/// Converts a `std::os::unix::net::UnixStream` into its asynchronous equivalent. /// Converts a `std::os::unix::net::UnixStream` into its asynchronous equivalent.
fn from(stream: StdUnixStream) -> UnixStream { fn from(stream: StdUnixStream) -> UnixStream {
let stream = Async::new(stream).expect("UnixStream is known to be good"); let stream = Async::new(stream).expect("UnixStream is known to be good");
UnixStream { watcher: stream } UnixStream { watcher: Arc::new(stream) }
} }
} }
@ -238,6 +244,6 @@ impl FromRawFd for UnixStream {
impl IntoRawFd for UnixStream { impl IntoRawFd for UnixStream {
fn into_raw_fd(self) -> RawFd { fn into_raw_fd(self) -> RawFd {
self.watcher.into_raw_fd() self.as_raw_fd()
} }
} }

View file

@ -94,3 +94,27 @@ async fn ping_pong_client(socket: &std::path::PathBuf, iterations: u32) -> std::
} }
Ok(()) Ok(())
} }
#[test]
fn uds_clone() -> io::Result<()> {
task::block_on(async {
let tmp_dir = TempDir::new("socket_ping_pong").expect("Temp dir not created");
let sock_path = tmp_dir.as_ref().join("sock");
let input = UnixListener::bind(&sock_path).await?;
let mut writer = UnixStream::connect(&sock_path).await?;
let mut reader = input.incoming().next().await.unwrap()?;
writer.write(b"original").await.unwrap();
let mut original_buf = [0; 8];
reader.read(&mut original_buf).await?;
assert_eq!(&original_buf, b"original");
writer.clone().write(b"clone").await.unwrap();
let mut clone_buf = [0; 5];
reader.clone().read(&mut clone_buf).await?;
assert_eq!(&clone_buf, b"clone");
Ok(())
})
}