| #![unstable(feature = "windows_unix_domain_sockets", issue = "150487")] |
| use crate::net::Shutdown; |
| use crate::os::windows::io::{ |
| AsRawSocket, AsSocket, BorrowedSocket, FromRawSocket, IntoRawSocket, RawSocket, |
| }; |
| use crate::os::windows::net::SocketAddr; |
| use crate::path::Path; |
| #[cfg(not(doc))] |
| use crate::sys::c::{ |
| AF_UNIX, SO_RCVTIMEO, SO_SNDTIMEO, SOCK_STREAM, connect, getpeername, getsockname, |
| }; |
| use crate::sys::net::Socket; |
| #[cfg(not(doc))] |
| use crate::sys::winsock::startup; |
| use crate::sys::{AsInner, cvt_nz}; |
| use crate::time::Duration; |
| use crate::{fmt, io}; |
| /// A Unix stream socket. |
| /// |
| /// Under Windows, it will only work starting from Windows 10 17063. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::io::prelude::*; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let mut stream = UnixStream::connect("/path/to/my/socket")?; |
| /// stream.write_all(b"hello world")?; |
| /// let mut response = String::new(); |
| /// stream.read_to_string(&mut response)?; |
| /// println!("{response}"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub struct UnixStream(pub(super) Socket); |
| impl fmt::Debug for UnixStream { |
| fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result { |
| let mut builder = fmt.debug_struct("UnixStream"); |
| builder.field("sock", self.0.as_inner()); |
| if let Ok(addr) = self.local_addr() { |
| builder.field("local", &addr); |
| } |
| if let Ok(addr) = self.peer_addr() { |
| builder.field("peer", &addr); |
| } |
| builder.finish() |
| } |
| } |
| impl UnixStream { |
| /// Connects to the socket named by `path`. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// let socket = match UnixStream::connect("/tmp/sock") { |
| /// Ok(sock) => sock, |
| /// Err(e) => { |
| /// println!("Couldn't connect: {e:?}"); |
| /// return |
| /// } |
| /// }; |
| /// ``` |
| pub fn connect<P: AsRef<Path>>(path: P) -> io::Result<UnixStream> { |
| let socket_addr = SocketAddr::from_pathname(path)?; |
| Self::connect_addr(&socket_addr) |
| } |
| |
| /// Connects to the socket specified by [`address`]. |
| /// |
| /// [`address`]: crate::os::windows::net::SocketAddr |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::{UnixListener, UnixStream}; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let listener = UnixListener::bind("/path/to/the/socket")?; |
| /// let addr = listener.local_addr()?; |
| /// |
| /// let sock = match UnixStream::connect_addr(&addr) { |
| /// Ok(sock) => sock, |
| /// Err(e) => { |
| /// println!("Couldn't connect: {e:?}"); |
| /// return Err(e) |
| /// } |
| /// }; |
| /// Ok(()) |
| /// } |
| /// ```` |
| pub fn connect_addr(socket_addr: &SocketAddr) -> io::Result<UnixStream> { |
| startup(); |
| let inner = Socket::new(AF_UNIX as _, SOCK_STREAM)?; |
| unsafe { |
| cvt_nz(connect( |
| inner.as_raw(), |
| &raw const socket_addr.addr as *const _, |
| socket_addr.len as _, |
| ))?; |
| } |
| Ok(UnixStream(inner)) |
| } |
| |
| /// Returns the socket address of the local half of this connection. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// let addr = socket.local_addr().expect("Couldn't get local address"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn local_addr(&self) -> io::Result<SocketAddr> { |
| SocketAddr::new(|addr, len| unsafe { getsockname(self.0.as_raw(), addr, len) }) |
| } |
| |
| /// Returns the socket address of the remote half of this connection. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// let addr = socket.peer_addr().expect("Couldn't get peer address"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn peer_addr(&self) -> io::Result<SocketAddr> { |
| SocketAddr::new(|addr, len| unsafe { getpeername(self.0.as_raw(), addr, len) }) |
| } |
| |
| /// Returns the read timeout of this socket. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.set_read_timeout(Some(Duration::new(1, 0))).expect("Couldn't set read timeout"); |
| /// assert_eq!(socket.read_timeout()?, Some(Duration::new(1, 0))); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn read_timeout(&self) -> io::Result<Option<Duration>> { |
| self.0.timeout(SO_RCVTIMEO) |
| } |
| |
| /// Moves the socket into or out of nonblocking mode. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.set_nonblocking(true).expect("Couldn't set nonblocking"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> { |
| self.0.set_nonblocking(nonblocking) |
| } |
| |
| /// Sets the read timeout for the socket. |
| /// |
| /// If the provided value is [`None`], then [`read`] calls will block |
| /// indefinitely. An [`Err`] is returned if the zero [`Duration`] is passed to this |
| /// method. |
| /// |
| /// [`read`]: io::Read::read |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.set_read_timeout(Some(Duration::new(1, 0))).expect("Couldn't set read timeout"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| /// |
| /// An [`Err`] is returned if the zero [`Duration`] is passed to this |
| /// method: |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::io; |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// let result = socket.set_read_timeout(Some(Duration::new(0, 0))); |
| /// let err = result.unwrap_err(); |
| /// assert_eq!(err.kind(), io::ErrorKind::InvalidInput); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn set_read_timeout(&self, dur: Option<Duration>) -> io::Result<()> { |
| self.0.set_timeout(dur, SO_RCVTIMEO) |
| } |
| |
| /// Sets the write timeout for the socket. |
| /// |
| /// If the provided value is [`None`], then [`write`] calls will block |
| /// indefinitely. An [`Err`] is returned if the zero [`Duration`] is |
| /// passed to this method. |
| /// |
| /// [`read`]: io::Read::read |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.set_write_timeout(Some(Duration::new(1, 0))) |
| /// .expect("Couldn't set write timeout"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| /// |
| /// An [`Err`] is returned if the zero [`Duration`] is passed to this |
| /// method: |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::io; |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// let result = socket.set_write_timeout(Some(Duration::new(0, 0))); |
| /// let err = result.unwrap_err(); |
| /// assert_eq!(err.kind(), io::ErrorKind::InvalidInput); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn set_write_timeout(&self, dur: Option<Duration>) -> io::Result<()> { |
| self.0.set_timeout(dur, SO_SNDTIMEO) |
| } |
| |
| /// Shuts down the read, write, or both halves of this connection. |
| /// |
| /// This function will cause all pending and future I/O calls on the |
| /// specified portions to immediately return with an appropriate value |
| /// (see the documentation of [`Shutdown`]). |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::net::Shutdown; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.shutdown(Shutdown::Both).expect("shutdown function failed"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn shutdown(&self, how: Shutdown) -> io::Result<()> { |
| self.0.shutdown(how) |
| } |
| |
| /// Returns the value of the `SO_ERROR` option. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// if let Ok(Some(err)) = socket.take_error() { |
| /// println!("Got error: {err:?}"); |
| /// } |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn take_error(&self) -> io::Result<Option<io::Error>> { |
| self.0.take_error() |
| } |
| |
| /// Creates a new independently owned handle to the underlying socket. |
| /// |
| /// The returned `UnixStream` is a reference to the same stream that this |
| /// object references. Both handles will read and write the same stream of |
| /// data, and options set on one stream will be propagated to the other |
| /// stream. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// let sock_copy = socket.try_clone().expect("Couldn't clone socket"); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn try_clone(&self) -> io::Result<UnixStream> { |
| self.0.duplicate().map(UnixStream) |
| } |
| |
| /// Returns the write timeout of this socket. |
| /// |
| /// # Examples |
| /// |
| /// ```no_run |
| /// #![feature(windows_unix_domain_sockets)] |
| /// use std::os::windows::net::UnixStream; |
| /// use std::time::Duration; |
| /// |
| /// fn main() -> std::io::Result<()> { |
| /// let socket = UnixStream::connect("/tmp/sock")?; |
| /// socket.set_write_timeout(Some(Duration::new(1, 0))) |
| /// .expect("Couldn't set write timeout"); |
| /// assert_eq!(socket.write_timeout()?, Some(Duration::new(1, 0))); |
| /// Ok(()) |
| /// } |
| /// ``` |
| pub fn write_timeout(&self) -> io::Result<Option<Duration>> { |
| self.0.timeout(SO_SNDTIMEO) |
| } |
| } |
| |
| impl io::Read for UnixStream { |
| fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> { |
| io::Read::read(&mut &*self, buf) |
| } |
| } |
| |
| impl<'a> io::Read for &'a UnixStream { |
| fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> { |
| self.0.read(buf) |
| } |
| } |
| |
| impl io::Write for UnixStream { |
| fn write(&mut self, buf: &[u8]) -> io::Result<usize> { |
| io::Write::write(&mut &*self, buf) |
| } |
| |
| fn flush(&mut self) -> io::Result<()> { |
| io::Write::flush(&mut &*self) |
| } |
| } |
| impl<'a> io::Write for &'a UnixStream { |
| fn write(&mut self, buf: &[u8]) -> io::Result<usize> { |
| self.write_vectored(&[io::IoSlice::new(buf)]) |
| } |
| #[inline] |
| fn flush(&mut self) -> io::Result<()> { |
| Ok(()) |
| } |
| fn write_vectored(&mut self, bufs: &[io::IoSlice<'_>]) -> io::Result<usize> { |
| self.0.write_vectored(bufs) |
| } |
| #[inline] |
| fn is_write_vectored(&self) -> bool { |
| self.0.is_write_vectored() |
| } |
| } |
| |
| impl AsSocket for UnixStream { |
| #[inline] |
| fn as_socket(&self) -> BorrowedSocket<'_> { |
| self.0.as_socket() |
| } |
| } |
| |
| impl AsRawSocket for UnixStream { |
| #[inline] |
| fn as_raw_socket(&self) -> RawSocket { |
| self.0.as_raw_socket() |
| } |
| } |
| |
| impl FromRawSocket for UnixStream { |
| #[inline] |
| unsafe fn from_raw_socket(sock: RawSocket) -> Self { |
| unsafe { UnixStream(Socket::from_raw_socket(sock)) } |
| } |
| } |
| |
| impl IntoRawSocket for UnixStream { |
| fn into_raw_socket(self) -> RawSocket { |
| self.0.into_raw_socket() |
| } |
| } |