2016-03-16 20:50:45 -07:00
|
|
|
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
#![unstable(feature = "unix_socket", reason = "newly added", issue = "32312")]
|
|
|
|
|
|
|
|
//! Unix-specific networking functionality
|
|
|
|
|
|
|
|
use libc;
|
|
|
|
|
|
|
|
use prelude::v1::*;
|
|
|
|
use ascii;
|
|
|
|
use ffi::OsStr;
|
|
|
|
use fmt;
|
|
|
|
use io;
|
|
|
|
use mem;
|
|
|
|
use net::Shutdown;
|
|
|
|
use os::unix::ffi::OsStrExt;
|
|
|
|
use os::unix::io::{RawFd, AsRawFd, FromRawFd, IntoRawFd};
|
|
|
|
use path::Path;
|
|
|
|
use time::Duration;
|
|
|
|
use sys::cvt;
|
|
|
|
use sys::net::Socket;
|
|
|
|
use sys_common::{AsInner, FromInner, IntoInner};
|
|
|
|
|
|
|
|
fn sun_path_offset() -> usize {
|
|
|
|
unsafe {
|
|
|
|
// Work with an actual instance of the type since using a null pointer is UB
|
|
|
|
let addr: libc::sockaddr_un = mem::uninitialized();
|
|
|
|
let base = &addr as *const _ as usize;
|
|
|
|
let path = &addr.sun_path as *const _ as usize;
|
|
|
|
path - base
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn sockaddr_un(path: &Path) -> io::Result<(libc::sockaddr_un, libc::socklen_t)> {
|
|
|
|
let mut addr: libc::sockaddr_un = mem::zeroed();
|
|
|
|
addr.sun_family = libc::AF_UNIX as libc::sa_family_t;
|
|
|
|
|
|
|
|
let bytes = path.as_os_str().as_bytes();
|
|
|
|
|
|
|
|
if bytes.contains(&0) {
|
|
|
|
return Err(io::Error::new(io::ErrorKind::InvalidInput,
|
|
|
|
"paths may not contain interior null bytes"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if bytes.len() >= addr.sun_path.len() {
|
|
|
|
return Err(io::Error::new(io::ErrorKind::InvalidInput,
|
|
|
|
"path must be shorter than SUN_LEN"));
|
|
|
|
}
|
|
|
|
for (dst, src) in addr.sun_path.iter_mut().zip(bytes.iter()) {
|
|
|
|
*dst = *src as libc::c_char;
|
|
|
|
}
|
|
|
|
// null byte for pathname addresses is already there because we zeroed the
|
|
|
|
// struct
|
|
|
|
|
|
|
|
let mut len = sun_path_offset() + bytes.len();
|
|
|
|
match bytes.get(0) {
|
|
|
|
Some(&0) | None => {}
|
|
|
|
Some(_) => len += 1,
|
|
|
|
}
|
|
|
|
Ok((addr, len as libc::socklen_t))
|
|
|
|
}
|
|
|
|
|
|
|
|
enum AddressKind<'a> {
|
|
|
|
Unnamed,
|
|
|
|
Pathname(&'a Path),
|
|
|
|
Abstract(&'a [u8]),
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An address associated with a Unix socket.
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct SocketAddr {
|
|
|
|
addr: libc::sockaddr_un,
|
|
|
|
len: libc::socklen_t,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SocketAddr {
|
|
|
|
fn new<F>(f: F) -> io::Result<SocketAddr>
|
|
|
|
where F: FnOnce(*mut libc::sockaddr, *mut libc::socklen_t) -> libc::c_int
|
|
|
|
{
|
|
|
|
unsafe {
|
|
|
|
let mut addr: libc::sockaddr_un = mem::zeroed();
|
|
|
|
let mut len = mem::size_of::<libc::sockaddr_un>() as libc::socklen_t;
|
2016-03-22 22:01:37 -05:00
|
|
|
cvt(f(&mut addr as *mut _ as *mut _, &mut len))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
SocketAddr::from_parts(addr, len)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn from_parts(addr: libc::sockaddr_un, mut len: libc::socklen_t) -> io::Result<SocketAddr> {
|
|
|
|
if len == 0 {
|
|
|
|
// When there is a datagram from unnamed unix socket
|
|
|
|
// linux returns zero bytes of address
|
|
|
|
len = sun_path_offset() as libc::socklen_t; // i.e. zero-length address
|
|
|
|
} else if addr.sun_family != libc::AF_UNIX as libc::sa_family_t {
|
|
|
|
return Err(io::Error::new(io::ErrorKind::InvalidInput,
|
|
|
|
"file descriptor did not correspond to a Unix socket"));
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(SocketAddr {
|
|
|
|
addr: addr,
|
|
|
|
len: len,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true iff the address is unnamed.
|
|
|
|
pub fn is_unnamed(&self) -> bool {
|
|
|
|
if let AddressKind::Unnamed = self.address() {
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the contents of this address if it is a `pathname` address.
|
|
|
|
pub fn as_pathname(&self) -> Option<&Path> {
|
|
|
|
if let AddressKind::Pathname(path) = self.address() {
|
|
|
|
Some(path)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn address<'a>(&'a self) -> AddressKind<'a> {
|
|
|
|
let len = self.len as usize - sun_path_offset();
|
|
|
|
let path = unsafe { mem::transmute::<&[libc::c_char], &[u8]>(&self.addr.sun_path) };
|
|
|
|
|
|
|
|
// OSX seems to return a len of 16 and a zeroed sun_path for unnamed addresses
|
|
|
|
if len == 0 || (cfg!(not(target_os = "linux")) && self.addr.sun_path[0] == 0) {
|
|
|
|
AddressKind::Unnamed
|
|
|
|
} else if self.addr.sun_path[0] == 0 {
|
|
|
|
AddressKind::Abstract(&path[1..len])
|
|
|
|
} else {
|
|
|
|
AddressKind::Pathname(OsStr::from_bytes(&path[..len - 1]).as_ref())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for SocketAddr {
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match self.address() {
|
|
|
|
AddressKind::Unnamed => write!(fmt, "(unnamed)"),
|
|
|
|
AddressKind::Abstract(name) => write!(fmt, "{} (abstract)", AsciiEscaped(name)),
|
|
|
|
AddressKind::Pathname(path) => write!(fmt, "{:?} (pathname)", path),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct AsciiEscaped<'a>(&'a [u8]);
|
|
|
|
|
|
|
|
impl<'a> fmt::Display for AsciiEscaped<'a> {
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "\"")?;
|
2016-03-16 20:50:45 -07:00
|
|
|
for byte in self.0.iter().cloned().flat_map(ascii::escape_default) {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "{}", byte as char)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
}
|
|
|
|
write!(fmt, "\"")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A Unix stream socket.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```rust,no_run
|
|
|
|
/// #![feature(unix_socket)]
|
|
|
|
///
|
|
|
|
/// use std::os::unix::net::UnixStream;
|
|
|
|
/// use std::io::prelude::*;
|
|
|
|
///
|
|
|
|
/// let mut stream = UnixStream::connect("/path/to/my/socket").unwrap();
|
|
|
|
/// stream.write_all(b"hello world").unwrap();
|
|
|
|
/// let mut response = String::new();
|
|
|
|
/// stream.read_to_string(&mut response).unwrap();
|
|
|
|
/// println!("{}", response);
|
|
|
|
/// ```
|
|
|
|
pub struct UnixStream(Socket);
|
|
|
|
|
|
|
|
impl fmt::Debug for UnixStream {
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let mut builder = fmt.debug_struct("UnixStream");
|
|
|
|
builder.field("fd", 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`.
|
|
|
|
pub fn connect<P: AsRef<Path>>(path: P) -> io::Result<UnixStream> {
|
|
|
|
fn inner(path: &Path) -> io::Result<UnixStream> {
|
|
|
|
unsafe {
|
2016-03-22 22:01:37 -05:00
|
|
|
let inner = Socket::new_raw(libc::AF_UNIX, libc::SOCK_STREAM)?;
|
|
|
|
let (addr, len) = sockaddr_un(path)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
cvt(libc::connect(*inner.as_inner(), &addr as *const _ as *const _, len))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok(UnixStream(inner))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inner(path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates an unnamed pair of connected sockets.
|
|
|
|
///
|
|
|
|
/// Returns two `UnixStream`s which are connected to each other.
|
|
|
|
pub fn pair() -> io::Result<(UnixStream, UnixStream)> {
|
2016-03-22 22:01:37 -05:00
|
|
|
let (i1, i2) = Socket::new_pair(libc::AF_UNIX, libc::SOCK_STREAM)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok((UnixStream(i1), UnixStream(i2)))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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 propogated to the other
|
|
|
|
/// stream.
|
|
|
|
pub fn try_clone(&self) -> io::Result<UnixStream> {
|
|
|
|
self.0.duplicate().map(UnixStream)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the socket address of the local half of this connection.
|
|
|
|
pub fn local_addr(&self) -> io::Result<SocketAddr> {
|
|
|
|
SocketAddr::new(|addr, len| unsafe { libc::getsockname(*self.0.as_inner(), addr, len) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the socket address of the remote half of this connection.
|
|
|
|
pub fn peer_addr(&self) -> io::Result<SocketAddr> {
|
|
|
|
SocketAddr::new(|addr, len| unsafe { libc::getpeername(*self.0.as_inner(), addr, len) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the read timeout for the socket.
|
|
|
|
///
|
|
|
|
/// If the provided value is `None`, then `read` calls will block
|
|
|
|
/// indefinitely. It is an error to pass the zero `Duration` to this
|
|
|
|
/// method.
|
|
|
|
pub fn set_read_timeout(&self, timeout: Option<Duration>) -> io::Result<()> {
|
|
|
|
self.0.set_timeout(timeout, libc::SO_RCVTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the write timeout for the socket.
|
|
|
|
///
|
|
|
|
/// If the provided value is `None`, then `write` calls will block
|
|
|
|
/// indefinitely. It is an error to pass the zero `Duration` to this
|
|
|
|
/// method.
|
|
|
|
pub fn set_write_timeout(&self, timeout: Option<Duration>) -> io::Result<()> {
|
|
|
|
self.0.set_timeout(timeout, libc::SO_SNDTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the read timeout of this socket.
|
|
|
|
pub fn read_timeout(&self) -> io::Result<Option<Duration>> {
|
|
|
|
self.0.timeout(libc::SO_RCVTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the write timeout of this socket.
|
|
|
|
pub fn write_timeout(&self) -> io::Result<Option<Duration>> {
|
|
|
|
self.0.timeout(libc::SO_SNDTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Moves the socket into or out of nonblocking mode.
|
|
|
|
pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
|
|
|
|
self.0.set_nonblocking(nonblocking)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the value of the `SO_ERROR` option.
|
|
|
|
pub fn take_error(&self) -> io::Result<Option<io::Error>> {
|
|
|
|
self.0.take_error()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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`).
|
|
|
|
pub fn shutdown(&self, how: Shutdown) -> io::Result<()> {
|
|
|
|
self.0.shutdown(how)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl io::Read for UnixStream {
|
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
|
|
|
io::Read::read(&mut &*self, buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn read_to_end(&mut self, buf: &mut Vec<u8>) -> io::Result<usize> {
|
|
|
|
io::Read::read_to_end(&mut &*self, buf)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> io::Read for &'a UnixStream {
|
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
|
|
|
self.0.read(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn read_to_end(&mut self, buf: &mut Vec<u8>) -> io::Result<usize> {
|
|
|
|
self.0.read_to_end(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.0.write(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> io::Result<()> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRawFd for UnixStream {
|
|
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
|
|
*self.0.as_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromRawFd for UnixStream {
|
|
|
|
unsafe fn from_raw_fd(fd: RawFd) -> UnixStream {
|
|
|
|
UnixStream(Socket::from_inner(fd))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoRawFd for UnixStream {
|
|
|
|
fn into_raw_fd(self) -> RawFd {
|
|
|
|
self.0.into_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A structure representing a Unix domain socket server.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```rust,no_run
|
|
|
|
/// #![feature(unix_socket)]
|
|
|
|
///
|
|
|
|
/// use std::thread;
|
|
|
|
/// use std::os::unix::net::{UnixStream, UnixListener};
|
|
|
|
///
|
|
|
|
/// fn handle_client(stream: UnixStream) {
|
|
|
|
/// // ...
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// let listener = UnixListener::bind("/path/to/the/socket").unwrap();
|
|
|
|
///
|
|
|
|
/// // accept connections and process them, spawning a new thread for each one
|
|
|
|
/// for stream in listener.incoming() {
|
|
|
|
/// match stream {
|
|
|
|
/// Ok(stream) => {
|
|
|
|
/// /* connection succeeded */
|
|
|
|
/// thread::spawn(|| handle_client(stream));
|
|
|
|
/// }
|
|
|
|
/// Err(err) => {
|
|
|
|
/// /* connection failed */
|
|
|
|
/// break;
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// // close the listener socket
|
|
|
|
/// drop(listener);
|
|
|
|
/// ```
|
|
|
|
pub struct UnixListener(Socket);
|
|
|
|
|
|
|
|
impl fmt::Debug for UnixListener {
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let mut builder = fmt.debug_struct("UnixListener");
|
|
|
|
builder.field("fd", self.0.as_inner());
|
|
|
|
if let Ok(addr) = self.local_addr() {
|
|
|
|
builder.field("local", &addr);
|
|
|
|
}
|
|
|
|
builder.finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl UnixListener {
|
|
|
|
/// Creates a new `UnixListener` bound to the specified socket.
|
|
|
|
pub fn bind<P: AsRef<Path>>(path: P) -> io::Result<UnixListener> {
|
|
|
|
fn inner(path: &Path) -> io::Result<UnixListener> {
|
|
|
|
unsafe {
|
2016-03-22 22:01:37 -05:00
|
|
|
let inner = Socket::new_raw(libc::AF_UNIX, libc::SOCK_STREAM)?;
|
|
|
|
let (addr, len) = sockaddr_un(path)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
cvt(libc::bind(*inner.as_inner(), &addr as *const _ as *const _, len))?;
|
|
|
|
cvt(libc::listen(*inner.as_inner(), 128))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
|
|
|
Ok(UnixListener(inner))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inner(path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Accepts a new incoming connection to this listener.
|
|
|
|
///
|
|
|
|
/// This function will block the calling thread until a new Unix connection
|
|
|
|
/// is established. When established, the corersponding `UnixStream` and
|
|
|
|
/// the remote peer's address will be returned.
|
|
|
|
pub fn accept(&self) -> io::Result<(UnixStream, SocketAddr)> {
|
|
|
|
let mut storage: libc::sockaddr_un = unsafe { mem::zeroed() };
|
|
|
|
let mut len = mem::size_of_val(&storage) as libc::socklen_t;
|
2016-03-22 22:01:37 -05:00
|
|
|
let sock = self.0.accept(&mut storage as *mut _ as *mut _, &mut len)?;
|
|
|
|
let addr = SocketAddr::from_parts(storage, len)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok((UnixStream(sock), addr))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new independently owned handle to the underlying socket.
|
|
|
|
///
|
|
|
|
/// The returned `UnixListener` is a reference to the same socket that this
|
|
|
|
/// object references. Both handles can be used to accept incoming
|
|
|
|
/// connections and options set on one listener will affect the other.
|
|
|
|
pub fn try_clone(&self) -> io::Result<UnixListener> {
|
|
|
|
self.0.duplicate().map(UnixListener)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the local socket address of this listener.
|
|
|
|
pub fn local_addr(&self) -> io::Result<SocketAddr> {
|
|
|
|
SocketAddr::new(|addr, len| unsafe { libc::getsockname(*self.0.as_inner(), addr, len) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Moves the socket into or out of nonblocking mode.
|
|
|
|
pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
|
|
|
|
self.0.set_nonblocking(nonblocking)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the value of the `SO_ERROR` option.
|
|
|
|
pub fn take_error(&self) -> io::Result<Option<io::Error>> {
|
|
|
|
self.0.take_error()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over incoming connections.
|
|
|
|
///
|
|
|
|
/// The iterator will never return `None` and will also not yield the
|
|
|
|
/// peer's `SocketAddr` structure.
|
|
|
|
pub fn incoming<'a>(&'a self) -> Incoming<'a> {
|
|
|
|
Incoming { listener: self }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRawFd for UnixListener {
|
|
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
|
|
*self.0.as_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromRawFd for UnixListener {
|
|
|
|
unsafe fn from_raw_fd(fd: RawFd) -> UnixListener {
|
|
|
|
UnixListener(Socket::from_inner(fd))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoRawFd for UnixListener {
|
|
|
|
fn into_raw_fd(self) -> RawFd {
|
|
|
|
self.0.into_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> IntoIterator for &'a UnixListener {
|
|
|
|
type Item = io::Result<UnixStream>;
|
|
|
|
type IntoIter = Incoming<'a>;
|
|
|
|
|
|
|
|
fn into_iter(self) -> Incoming<'a> {
|
|
|
|
self.incoming()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An iterator over incoming connections to a `UnixListener`.
|
|
|
|
///
|
|
|
|
/// It will never return `None`.
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Incoming<'a> {
|
|
|
|
listener: &'a UnixListener,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Iterator for Incoming<'a> {
|
|
|
|
type Item = io::Result<UnixStream>;
|
|
|
|
|
|
|
|
fn next(&mut self) -> Option<io::Result<UnixStream>> {
|
|
|
|
Some(self.listener.accept().map(|s| s.0))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
|
|
(usize::max_value(), None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A Unix datagram socket.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```rust,no_run
|
|
|
|
/// #![feature(unix_socket)]
|
|
|
|
///
|
|
|
|
/// use std::os::unix::net::UnixDatagram;
|
|
|
|
///
|
|
|
|
/// let socket = UnixDatagram::bind("/path/to/my/socket").unwrap();
|
|
|
|
/// socket.send_to(b"hello world", "/path/to/other/socket").unwrap();
|
|
|
|
/// let mut buf = [0; 100];
|
|
|
|
/// let (count, address) = socket.recv_from(&mut buf).unwrap();
|
|
|
|
/// println!("socket {:?} sent {:?}", address, &buf[..count]);
|
|
|
|
/// ```
|
|
|
|
pub struct UnixDatagram(Socket);
|
|
|
|
|
|
|
|
impl fmt::Debug for UnixDatagram {
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let mut builder = fmt.debug_struct("UnixDatagram");
|
|
|
|
builder.field("fd", 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 UnixDatagram {
|
|
|
|
/// Creates a Unix datagram socket bound to the given path.
|
|
|
|
pub fn bind<P: AsRef<Path>>(path: P) -> io::Result<UnixDatagram> {
|
|
|
|
fn inner(path: &Path) -> io::Result<UnixDatagram> {
|
|
|
|
unsafe {
|
2016-03-22 22:01:37 -05:00
|
|
|
let socket = UnixDatagram::unbound()?;
|
|
|
|
let (addr, len) = sockaddr_un(path)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
cvt(libc::bind(*socket.0.as_inner(), &addr as *const _ as *const _, len))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
|
|
|
Ok(socket)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inner(path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a Unix Datagram socket which is not bound to any address.
|
|
|
|
pub fn unbound() -> io::Result<UnixDatagram> {
|
2016-03-22 22:01:37 -05:00
|
|
|
let inner = Socket::new_raw(libc::AF_UNIX, libc::SOCK_DGRAM)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok(UnixDatagram(inner))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create an unnamed pair of connected sockets.
|
|
|
|
///
|
|
|
|
/// Returns two `UnixDatagrams`s which are connected to each other.
|
|
|
|
pub fn pair() -> io::Result<(UnixDatagram, UnixDatagram)> {
|
2016-03-22 22:01:37 -05:00
|
|
|
let (i1, i2) = Socket::new_pair(libc::AF_UNIX, libc::SOCK_DGRAM)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok((UnixDatagram(i1), UnixDatagram(i2)))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Connects the socket to the specified address.
|
|
|
|
///
|
|
|
|
/// The `send` method may be used to send data to the specified address.
|
|
|
|
/// `recv` and `recv_from` will only receive data from that address.
|
|
|
|
pub fn connect<P: AsRef<Path>>(&self, path: P) -> io::Result<()> {
|
|
|
|
fn inner(d: &UnixDatagram, path: &Path) -> io::Result<()> {
|
|
|
|
unsafe {
|
2016-03-22 22:01:37 -05:00
|
|
|
let (addr, len) = sockaddr_un(path)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
cvt(libc::connect(*d.0.as_inner(), &addr as *const _ as *const _, len))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inner(self, path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new independently owned handle to the underlying socket.
|
|
|
|
///
|
|
|
|
/// The returned `UnixListener` is a reference to the same socket that this
|
|
|
|
/// object references. Both handles can be used to accept incoming
|
|
|
|
/// connections and options set on one listener will affect the other.
|
|
|
|
pub fn try_clone(&self) -> io::Result<UnixDatagram> {
|
|
|
|
self.0.duplicate().map(UnixDatagram)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the address of this socket.
|
|
|
|
pub fn local_addr(&self) -> io::Result<SocketAddr> {
|
|
|
|
SocketAddr::new(|addr, len| unsafe { libc::getsockname(*self.0.as_inner(), addr, len) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the address of this socket's peer.
|
|
|
|
///
|
|
|
|
/// The `connect` method will connect the socket to a peer.
|
|
|
|
pub fn peer_addr(&self) -> io::Result<SocketAddr> {
|
|
|
|
SocketAddr::new(|addr, len| unsafe { libc::getpeername(*self.0.as_inner(), addr, len) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Receives data from the socket.
|
|
|
|
///
|
|
|
|
/// On success, returns the number of bytes read and the address from
|
|
|
|
/// whence the data came.
|
|
|
|
pub fn recv_from(&self, buf: &mut [u8]) -> io::Result<(usize, SocketAddr)> {
|
|
|
|
let mut count = 0;
|
2016-03-22 22:01:37 -05:00
|
|
|
let addr = SocketAddr::new(|addr, len| {
|
2016-03-16 20:50:45 -07:00
|
|
|
unsafe {
|
|
|
|
count = libc::recvfrom(*self.0.as_inner(),
|
|
|
|
buf.as_mut_ptr() as *mut _,
|
|
|
|
buf.len(),
|
|
|
|
0,
|
|
|
|
addr,
|
|
|
|
len);
|
|
|
|
if count > 0 {
|
|
|
|
1
|
|
|
|
} else if count == 0 {
|
|
|
|
0
|
|
|
|
} else {
|
|
|
|
-1
|
|
|
|
}
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
})?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
|
|
|
Ok((count as usize, addr))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Receives data from the socket.
|
|
|
|
///
|
|
|
|
/// On success, returns the number of bytes read.
|
|
|
|
pub fn recv(&self, buf: &mut [u8]) -> io::Result<usize> {
|
|
|
|
self.0.read(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sends data on the socket to the specified address.
|
|
|
|
///
|
|
|
|
/// On success, returns the number of bytes written.
|
|
|
|
pub fn send_to<P: AsRef<Path>>(&self, buf: &[u8], path: P) -> io::Result<usize> {
|
|
|
|
fn inner(d: &UnixDatagram, buf: &[u8], path: &Path) -> io::Result<usize> {
|
|
|
|
unsafe {
|
2016-03-22 22:01:37 -05:00
|
|
|
let (addr, len) = sockaddr_un(path)?;
|
2016-03-16 20:50:45 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
let count = cvt(libc::sendto(*d.0.as_inner(),
|
2016-03-16 20:50:45 -07:00
|
|
|
buf.as_ptr() as *const _,
|
|
|
|
buf.len(),
|
|
|
|
0,
|
|
|
|
&addr as *const _ as *const _,
|
2016-03-22 22:01:37 -05:00
|
|
|
len))?;
|
2016-03-16 20:50:45 -07:00
|
|
|
Ok(count as usize)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inner(self, buf, path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sends data on the socket to the socket's peer.
|
|
|
|
///
|
|
|
|
/// The peer address may be set by the `connect` method, and this method
|
|
|
|
/// will return an error if the socket has not already been connected.
|
|
|
|
///
|
|
|
|
/// On success, returns the number of bytes written.
|
|
|
|
pub fn send(&self, buf: &[u8]) -> io::Result<usize> {
|
|
|
|
self.0.write(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the read timeout for the socket.
|
|
|
|
///
|
|
|
|
/// If the provided value is `None`, then `recv` and `recv_from` calls will
|
|
|
|
/// block indefinitely. It is an error to pass the zero `Duration` to this
|
|
|
|
/// method.
|
|
|
|
pub fn set_read_timeout(&self, timeout: Option<Duration>) -> io::Result<()> {
|
|
|
|
self.0.set_timeout(timeout, libc::SO_RCVTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the write timeout for the socket.
|
|
|
|
///
|
|
|
|
/// If the provided value is `None`, then `send` and `send_to` calls will
|
|
|
|
/// block indefinitely. It is an error to pass the zero `Duration` to this
|
|
|
|
/// method.
|
|
|
|
pub fn set_write_timeout(&self, timeout: Option<Duration>) -> io::Result<()> {
|
|
|
|
self.0.set_timeout(timeout, libc::SO_SNDTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the read timeout of this socket.
|
|
|
|
pub fn read_timeout(&self) -> io::Result<Option<Duration>> {
|
|
|
|
self.0.timeout(libc::SO_RCVTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the write timeout of this socket.
|
|
|
|
pub fn write_timeout(&self) -> io::Result<Option<Duration>> {
|
|
|
|
self.0.timeout(libc::SO_SNDTIMEO)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Moves the socket into or out of nonblocking mode.
|
|
|
|
pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
|
|
|
|
self.0.set_nonblocking(nonblocking)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the value of the `SO_ERROR` option.
|
|
|
|
pub fn take_error(&self) -> io::Result<Option<io::Error>> {
|
|
|
|
self.0.take_error()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Shut 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`).
|
|
|
|
pub fn shutdown(&self, how: Shutdown) -> io::Result<()> {
|
|
|
|
self.0.shutdown(how)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRawFd for UnixDatagram {
|
|
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
|
|
*self.0.as_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromRawFd for UnixDatagram {
|
|
|
|
unsafe fn from_raw_fd(fd: RawFd) -> UnixDatagram {
|
|
|
|
UnixDatagram(Socket::from_inner(fd))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoRawFd for UnixDatagram {
|
|
|
|
fn into_raw_fd(self) -> RawFd {
|
|
|
|
self.0.into_inner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use prelude::v1::*;
|
|
|
|
use thread;
|
|
|
|
use io;
|
|
|
|
use io::prelude::*;
|
|
|
|
use time::Duration;
|
|
|
|
use sys_common::io::test::tmpdir;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
macro_rules! or_panic {
|
|
|
|
($e:expr) => {
|
|
|
|
match $e {
|
|
|
|
Ok(e) => e,
|
|
|
|
Err(e) => panic!("{}", e),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn basic() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
let msg1 = b"hello";
|
|
|
|
let msg2 = b"world!";
|
|
|
|
|
|
|
|
let listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
let thread = thread::spawn(move || {
|
|
|
|
let mut stream = or_panic!(listener.accept()).0;
|
|
|
|
let mut buf = [0; 5];
|
|
|
|
or_panic!(stream.read(&mut buf));
|
|
|
|
assert_eq!(&msg1[..], &buf[..]);
|
|
|
|
or_panic!(stream.write_all(msg2));
|
|
|
|
});
|
|
|
|
|
|
|
|
let mut stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
assert_eq!(Some(&*socket_path),
|
|
|
|
stream.peer_addr().unwrap().as_pathname());
|
|
|
|
or_panic!(stream.write_all(msg1));
|
|
|
|
let mut buf = vec![];
|
|
|
|
or_panic!(stream.read_to_end(&mut buf));
|
|
|
|
assert_eq!(&msg2[..], &buf[..]);
|
|
|
|
drop(stream);
|
|
|
|
|
|
|
|
thread.join().unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pair() {
|
|
|
|
let msg1 = b"hello";
|
|
|
|
let msg2 = b"world!";
|
|
|
|
|
|
|
|
let (mut s1, mut s2) = or_panic!(UnixStream::pair());
|
|
|
|
let thread = thread::spawn(move || {
|
|
|
|
// s1 must be moved in or the test will hang!
|
|
|
|
let mut buf = [0; 5];
|
|
|
|
or_panic!(s1.read(&mut buf));
|
|
|
|
assert_eq!(&msg1[..], &buf[..]);
|
|
|
|
or_panic!(s1.write_all(msg2));
|
|
|
|
});
|
|
|
|
|
|
|
|
or_panic!(s2.write_all(msg1));
|
|
|
|
let mut buf = vec![];
|
|
|
|
or_panic!(s2.read_to_end(&mut buf));
|
|
|
|
assert_eq!(&msg2[..], &buf[..]);
|
|
|
|
drop(s2);
|
|
|
|
|
|
|
|
thread.join().unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn try_clone() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
let msg1 = b"hello";
|
|
|
|
let msg2 = b"world";
|
|
|
|
|
|
|
|
let listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
let thread = thread::spawn(move || {
|
|
|
|
let mut stream = or_panic!(listener.accept()).0;
|
|
|
|
or_panic!(stream.write_all(msg1));
|
|
|
|
or_panic!(stream.write_all(msg2));
|
|
|
|
});
|
|
|
|
|
|
|
|
let mut stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
let mut stream2 = or_panic!(stream.try_clone());
|
|
|
|
|
|
|
|
let mut buf = [0; 5];
|
|
|
|
or_panic!(stream.read(&mut buf));
|
|
|
|
assert_eq!(&msg1[..], &buf[..]);
|
|
|
|
or_panic!(stream2.read(&mut buf));
|
|
|
|
assert_eq!(&msg2[..], &buf[..]);
|
|
|
|
|
|
|
|
thread.join().unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn iter() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
|
|
|
|
let listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
let thread = thread::spawn(move || {
|
|
|
|
for stream in listener.incoming().take(2) {
|
|
|
|
let mut stream = or_panic!(stream);
|
|
|
|
let mut buf = [0];
|
|
|
|
or_panic!(stream.read(&mut buf));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
for _ in 0..2 {
|
|
|
|
let mut stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
or_panic!(stream.write_all(&[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
thread.join().unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn long_path() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path()
|
|
|
|
.join("asdfasdfasdfasdfasdfasdfasdfasdfasdfasdfasdfasdfasdfasdfasdfa\
|
|
|
|
sasdfasdfasdasdfasdfasdfadfasdfasdfasdfasdfasdf");
|
|
|
|
match UnixStream::connect(&socket_path) {
|
|
|
|
Err(ref e) if e.kind() == io::ErrorKind::InvalidInput => {}
|
|
|
|
Err(e) => panic!("unexpected error {}", e),
|
|
|
|
Ok(_) => panic!("unexpected success"),
|
|
|
|
}
|
|
|
|
|
|
|
|
match UnixListener::bind(&socket_path) {
|
|
|
|
Err(ref e) if e.kind() == io::ErrorKind::InvalidInput => {}
|
|
|
|
Err(e) => panic!("unexpected error {}", e),
|
|
|
|
Ok(_) => panic!("unexpected success"),
|
|
|
|
}
|
|
|
|
|
|
|
|
match UnixDatagram::bind(&socket_path) {
|
|
|
|
Err(ref e) if e.kind() == io::ErrorKind::InvalidInput => {}
|
|
|
|
Err(e) => panic!("unexpected error {}", e),
|
|
|
|
Ok(_) => panic!("unexpected success"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn timeouts() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
|
|
|
|
let _listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
|
|
|
|
let stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
let dur = Duration::new(15410, 0);
|
|
|
|
|
|
|
|
assert_eq!(None, or_panic!(stream.read_timeout()));
|
|
|
|
|
|
|
|
or_panic!(stream.set_read_timeout(Some(dur)));
|
|
|
|
assert_eq!(Some(dur), or_panic!(stream.read_timeout()));
|
|
|
|
|
|
|
|
assert_eq!(None, or_panic!(stream.write_timeout()));
|
|
|
|
|
|
|
|
or_panic!(stream.set_write_timeout(Some(dur)));
|
|
|
|
assert_eq!(Some(dur), or_panic!(stream.write_timeout()));
|
|
|
|
|
|
|
|
or_panic!(stream.set_read_timeout(None));
|
|
|
|
assert_eq!(None, or_panic!(stream.read_timeout()));
|
|
|
|
|
|
|
|
or_panic!(stream.set_write_timeout(None));
|
|
|
|
assert_eq!(None, or_panic!(stream.write_timeout()));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_read_timeout() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
|
|
|
|
let _listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
|
|
|
|
let mut stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
or_panic!(stream.set_read_timeout(Some(Duration::from_millis(1000))));
|
|
|
|
|
|
|
|
let mut buf = [0; 10];
|
|
|
|
let kind = stream.read(&mut buf).err().expect("expected error").kind();
|
|
|
|
assert!(kind == io::ErrorKind::WouldBlock || kind == io::ErrorKind::TimedOut);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_read_with_timeout() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let socket_path = dir.path().join("sock");
|
|
|
|
|
|
|
|
let listener = or_panic!(UnixListener::bind(&socket_path));
|
|
|
|
|
|
|
|
let mut stream = or_panic!(UnixStream::connect(&socket_path));
|
|
|
|
or_panic!(stream.set_read_timeout(Some(Duration::from_millis(1000))));
|
|
|
|
|
|
|
|
let mut other_end = or_panic!(listener.accept()).0;
|
|
|
|
or_panic!(other_end.write_all(b"hello world"));
|
|
|
|
|
|
|
|
let mut buf = [0; 11];
|
|
|
|
or_panic!(stream.read(&mut buf));
|
|
|
|
assert_eq!(b"hello world", &buf[..]);
|
|
|
|
|
|
|
|
let kind = stream.read(&mut buf).err().expect("expected error").kind();
|
|
|
|
assert!(kind == io::ErrorKind::WouldBlock || kind == io::ErrorKind::TimedOut);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unix_datagram() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let path1 = dir.path().join("sock1");
|
|
|
|
let path2 = dir.path().join("sock2");
|
|
|
|
|
|
|
|
let sock1 = or_panic!(UnixDatagram::bind(&path1));
|
|
|
|
let sock2 = or_panic!(UnixDatagram::bind(&path2));
|
|
|
|
|
|
|
|
let msg = b"hello world";
|
|
|
|
or_panic!(sock1.send_to(msg, &path2));
|
|
|
|
let mut buf = [0; 11];
|
|
|
|
or_panic!(sock2.recv_from(&mut buf));
|
|
|
|
assert_eq!(msg, &buf[..]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unnamed_unix_datagram() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let path1 = dir.path().join("sock1");
|
|
|
|
|
|
|
|
let sock1 = or_panic!(UnixDatagram::bind(&path1));
|
|
|
|
let sock2 = or_panic!(UnixDatagram::unbound());
|
|
|
|
|
|
|
|
let msg = b"hello world";
|
|
|
|
or_panic!(sock2.send_to(msg, &path1));
|
|
|
|
let mut buf = [0; 11];
|
|
|
|
let (usize, addr) = or_panic!(sock1.recv_from(&mut buf));
|
|
|
|
assert_eq!(usize, 11);
|
|
|
|
assert!(addr.is_unnamed());
|
|
|
|
assert_eq!(msg, &buf[..]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_connect_unix_datagram() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let path1 = dir.path().join("sock1");
|
|
|
|
let path2 = dir.path().join("sock2");
|
|
|
|
|
|
|
|
let bsock1 = or_panic!(UnixDatagram::bind(&path1));
|
|
|
|
let bsock2 = or_panic!(UnixDatagram::bind(&path2));
|
|
|
|
let sock = or_panic!(UnixDatagram::unbound());
|
|
|
|
or_panic!(sock.connect(&path1));
|
|
|
|
|
|
|
|
// Check send()
|
|
|
|
let msg = b"hello there";
|
|
|
|
or_panic!(sock.send(msg));
|
|
|
|
let mut buf = [0; 11];
|
|
|
|
let (usize, addr) = or_panic!(bsock1.recv_from(&mut buf));
|
|
|
|
assert_eq!(usize, 11);
|
|
|
|
assert!(addr.is_unnamed());
|
|
|
|
assert_eq!(msg, &buf[..]);
|
|
|
|
|
|
|
|
// Changing default socket works too
|
|
|
|
or_panic!(sock.connect(&path2));
|
|
|
|
or_panic!(sock.send(msg));
|
|
|
|
or_panic!(bsock2.recv_from(&mut buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unix_datagram_recv() {
|
|
|
|
let dir = tmpdir();
|
|
|
|
let path1 = dir.path().join("sock1");
|
|
|
|
|
|
|
|
let sock1 = or_panic!(UnixDatagram::bind(&path1));
|
|
|
|
let sock2 = or_panic!(UnixDatagram::unbound());
|
|
|
|
or_panic!(sock2.connect(&path1));
|
|
|
|
|
|
|
|
let msg = b"hello world";
|
|
|
|
or_panic!(sock2.send(msg));
|
|
|
|
let mut buf = [0; 11];
|
|
|
|
let size = or_panic!(sock1.recv(&mut buf));
|
|
|
|
assert_eq!(size, 11);
|
|
|
|
assert_eq!(msg, &buf[..]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn datagram_pair() {
|
|
|
|
let msg1 = b"hello";
|
|
|
|
let msg2 = b"world!";
|
|
|
|
|
|
|
|
let (s1, s2) = or_panic!(UnixDatagram::pair());
|
|
|
|
let thread = thread::spawn(move || {
|
|
|
|
// s1 must be moved in or the test will hang!
|
|
|
|
let mut buf = [0; 5];
|
|
|
|
or_panic!(s1.recv(&mut buf));
|
|
|
|
assert_eq!(&msg1[..], &buf[..]);
|
|
|
|
or_panic!(s1.send(msg2));
|
|
|
|
});
|
|
|
|
|
|
|
|
or_panic!(s2.send(msg1));
|
|
|
|
let mut buf = [0; 6];
|
|
|
|
or_panic!(s2.recv(&mut buf));
|
|
|
|
assert_eq!(&msg2[..], &buf[..]);
|
|
|
|
drop(s2);
|
|
|
|
|
|
|
|
thread.join().unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn abstract_namespace_not_allowed() {
|
|
|
|
assert!(UnixStream::connect("\0asdf").is_err());
|
|
|
|
}
|
|
|
|
}
|