You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
521 lines
16 KiB
Rust
521 lines
16 KiB
Rust
use std::io::{IoSlice, IoSliceMut};
|
|
use std::mem;
|
|
use std::net::{self, SocketAddr, ToSocketAddrs};
|
|
use std::pin::Pin;
|
|
|
|
use cfg_if::cfg_if;
|
|
use futures::future;
|
|
|
|
use crate::io;
|
|
use crate::net::driver::IoHandle;
|
|
use crate::task::{Context, Poll};
|
|
|
|
/// A TCP stream between a local and a remote socket.
|
|
///
|
|
/// A `TcpStream` can either be created by connecting to an endpoint, via the [`connect`] method,
|
|
/// or by [accepting] a connection from a [listener]. It can be read or written to using the
|
|
/// [`AsyncRead`], [`AsyncWrite`], and related extension traits in [`futures::io`].
|
|
///
|
|
/// The connection will be closed when the value is dropped. The reading and writing portions of
|
|
/// the connection can also be shut down individually with the [`shutdown`] method.
|
|
///
|
|
/// This type is an async version of [`std::net::TcpStream`].
|
|
///
|
|
/// [`connect`]: struct.TcpStream.html#method.connect
|
|
/// [accepting]: struct.TcpListener.html#method.accept
|
|
/// [listener]: struct.TcpListener.html
|
|
/// [`AsyncRead`]: https://docs/futures-preview/0.3.0-alpha.13/futures/io/trait.AsyncRead.html
|
|
/// [`AsyncWrite`]: https://docs/futures-preview/0.3.0-alpha.13/futures/io/trait.AsyncRead.html
|
|
/// [`futures::io`]: https://docs/futures-preview/0.3.0-alpha.13/futures/io
|
|
/// [`shutdown`]: struct.TcpStream.html#method.shutdown
|
|
/// [`std::net::TcpStream`]: https://doc.rust-lang.org/std/net/struct.TcpStream.html
|
|
///
|
|
/// ## Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::{net::TcpStream, prelude::*};
|
|
///
|
|
/// let mut stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
/// stream.write_all(b"hello world").await?;
|
|
///
|
|
/// let mut buf = vec![0u8; 1024];
|
|
/// let n = stream.read(&mut buf).await?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
#[derive(Debug)]
|
|
pub struct TcpStream {
|
|
pub(super) io_handle: IoHandle<mio::net::TcpStream>,
|
|
|
|
#[cfg(unix)]
|
|
pub(super) raw_fd: std::os::unix::io::RawFd,
|
|
// #[cfg(windows)]
|
|
// pub(super) raw_socket: std::os::windows::io::RawSocket,
|
|
}
|
|
|
|
impl TcpStream {
|
|
/// Creates a new TCP stream connected to the specified address.
|
|
///
|
|
/// This method will create a new TCP socket and attempt to connect it to the `addr`
|
|
/// provided. The [returned future] will be resolved once the stream has successfully
|
|
/// connected, or it will return an error if one occurs.
|
|
///
|
|
/// [returned future]: struct.Connect.html
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:0").await?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub async fn connect<A: ToSocketAddrs>(addrs: A) -> io::Result<TcpStream> {
|
|
enum State {
|
|
Waiting(TcpStream),
|
|
Error(io::Error),
|
|
Done,
|
|
}
|
|
|
|
let mut last_err = None;
|
|
|
|
for addr in addrs.to_socket_addrs()? {
|
|
let mut state = {
|
|
match mio::net::TcpStream::connect(&addr) {
|
|
Ok(mio_stream) => {
|
|
#[cfg(unix)]
|
|
let stream = TcpStream {
|
|
raw_fd: mio_stream.as_raw_fd(),
|
|
io_handle: IoHandle::new(mio_stream),
|
|
};
|
|
|
|
#[cfg(windows)]
|
|
let stream = TcpStream {
|
|
// raw_socket: mio_stream.as_raw_socket(),
|
|
io_handle: IoHandle::new(mio_stream),
|
|
};
|
|
|
|
State::Waiting(stream)
|
|
}
|
|
Err(err) => State::Error(err),
|
|
}
|
|
};
|
|
|
|
let res = future::poll_fn(|cx| {
|
|
match mem::replace(&mut state, State::Done) {
|
|
State::Waiting(stream) => {
|
|
// Once we've connected, wait for the stream to be writable as that's when
|
|
// the actual connection has been initiated. Once we're writable we check
|
|
// for `take_socket_error` to see if the connect actually hit an error or
|
|
// not.
|
|
//
|
|
// If all that succeeded then we ship everything on up.
|
|
if let Poll::Pending = stream.io_handle.poll_writable(cx)? {
|
|
state = State::Waiting(stream);
|
|
return Poll::Pending;
|
|
}
|
|
|
|
if let Some(err) = stream.io_handle.get_ref().take_error()? {
|
|
return Poll::Ready(Err(err));
|
|
}
|
|
|
|
Poll::Ready(Ok(stream))
|
|
}
|
|
State::Error(err) => Poll::Ready(Err(err)),
|
|
State::Done => panic!("`TcpStream::connect()` future polled after completion"),
|
|
}
|
|
})
|
|
.await;
|
|
|
|
match res {
|
|
Ok(stream) => return Ok(stream),
|
|
Err(err) => last_err = Some(err),
|
|
}
|
|
}
|
|
|
|
Err(last_err.unwrap_or_else(|| {
|
|
io::Error::new(
|
|
io::ErrorKind::InvalidInput,
|
|
"could not resolve to any addresses",
|
|
)
|
|
}))
|
|
}
|
|
|
|
/// Returns the local address that this stream is connected to.
|
|
///
|
|
/// ## Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
/// let addr = stream.local_addr()?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn local_addr(&self) -> io::Result<SocketAddr> {
|
|
self.io_handle.get_ref().local_addr()
|
|
}
|
|
|
|
/// Returns the remote address that this stream is connected to.
|
|
///
|
|
/// ## Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
/// let peer = stream.peer_addr()?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn peer_addr(&self) -> io::Result<SocketAddr> {
|
|
self.io_handle.get_ref().peer_addr()
|
|
}
|
|
|
|
/// Gets the value of the `IP_TTL` option for this socket.
|
|
///
|
|
/// For more information about this option, see [`set_ttl`].
|
|
///
|
|
/// [`set_ttl`]: #method.set_ttl
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
///
|
|
/// stream.set_ttl(100)?;
|
|
/// assert_eq!(stream.ttl()?, 100);
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn ttl(&self) -> io::Result<u32> {
|
|
self.io_handle.get_ref().ttl()
|
|
}
|
|
|
|
/// Sets the value for the `IP_TTL` option on this socket.
|
|
///
|
|
/// This value sets the time-to-live field that is used in every packet sent
|
|
/// from this socket.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
///
|
|
/// stream.set_ttl(100)?;
|
|
/// assert_eq!(stream.ttl()?, 100);
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn set_ttl(&self, ttl: u32) -> io::Result<()> {
|
|
self.io_handle.get_ref().set_ttl(ttl)
|
|
}
|
|
|
|
/// Receives data on the socket from the remote address to which it is connected, without
|
|
/// removing that data from the queue.
|
|
///
|
|
/// On success, returns the number of bytes peeked.
|
|
///
|
|
/// Successive calls return the same data. This is accomplished by passing `MSG_PEEK` as a flag
|
|
/// to the underlying `recv` system call.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8000").await?;
|
|
///
|
|
/// let mut buf = vec![0; 1024];
|
|
/// let n = stream.peek(&mut buf).await?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub async fn peek(&self, buf: &mut [u8]) -> io::Result<usize> {
|
|
let res = future::poll_fn(|cx| {
|
|
futures::ready!(self.io_handle.poll_readable(cx)?);
|
|
|
|
match self.io_handle.get_ref().peek(buf) {
|
|
Ok(len) => Poll::Ready(Ok(len)),
|
|
Err(ref err) if err.kind() == io::ErrorKind::WouldBlock => {
|
|
self.io_handle.clear_readable(cx)?;
|
|
Poll::Pending
|
|
}
|
|
Err(e) => Poll::Ready(Err(e)),
|
|
}
|
|
})
|
|
.await?;
|
|
Ok(res)
|
|
}
|
|
|
|
/// Gets the value of the `TCP_NODELAY` option on this socket.
|
|
///
|
|
/// For more information about this option, see [`set_nodelay`].
|
|
///
|
|
/// [`set_nodelay`]: #method.set_nodelay
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
///
|
|
/// stream.set_nodelay(true)?;
|
|
/// assert_eq!(stream.nodelay()?, true);
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn nodelay(&self) -> io::Result<bool> {
|
|
self.io_handle.get_ref().nodelay()
|
|
}
|
|
|
|
/// Sets the value of the `TCP_NODELAY` option on this socket.
|
|
///
|
|
/// If set, this option disables the Nagle algorithm. This means that
|
|
/// segments are always sent as soon as possible, even if there is only a
|
|
/// small amount of data. When not set, data is buffered until there is a
|
|
/// sufficient amount to send out, thereby avoiding the frequent sending of
|
|
/// small packets.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
///
|
|
/// stream.set_nodelay(true)?;
|
|
/// assert_eq!(stream.nodelay()?, true);
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn set_nodelay(&self, nodelay: bool) -> io::Result<()> {
|
|
self.io_handle.get_ref().set_nodelay(nodelay)
|
|
}
|
|
|
|
/// Shuts down the read, write, or both halves of this connection.
|
|
///
|
|
/// This method will cause all pending and future I/O on the specified portions to return
|
|
/// immediately with an appropriate value (see the documentation of [`Shutdown`]).
|
|
///
|
|
/// [`Shutdown`]: https://doc.rust-lang.org/std/net/enum.Shutdown.html
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```no_run
|
|
/// # #![feature(async_await)]
|
|
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
|
|
/// #
|
|
/// use async_std::net::TcpStream;
|
|
/// use std::net::Shutdown;
|
|
///
|
|
/// let stream = TcpStream::connect("127.0.0.1:8080").await?;
|
|
/// stream.shutdown(Shutdown::Both)?;
|
|
/// #
|
|
/// # Ok(()) }) }
|
|
/// ```
|
|
pub fn shutdown(&self, how: std::net::Shutdown) -> std::io::Result<()> {
|
|
self.io_handle.get_ref().shutdown(how)
|
|
}
|
|
}
|
|
|
|
impl futures::io::AsyncRead for TcpStream {
|
|
fn poll_read(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
buf: &mut [u8],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &*self).poll_read(cx, buf)
|
|
}
|
|
|
|
fn poll_read_vectored(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
bufs: &mut [IoSliceMut<'_>],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &*self).poll_read_vectored(cx, bufs)
|
|
}
|
|
}
|
|
|
|
impl futures::io::AsyncRead for &TcpStream {
|
|
fn poll_read(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
buf: &mut [u8],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &self.io_handle).poll_read(cx, buf)
|
|
}
|
|
|
|
fn poll_read_vectored(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
bufs: &mut [IoSliceMut<'_>],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &self.io_handle).poll_read_vectored(cx, bufs)
|
|
}
|
|
}
|
|
|
|
impl futures::io::AsyncWrite for TcpStream {
|
|
fn poll_write(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
buf: &[u8],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &*self).poll_write(cx, buf)
|
|
}
|
|
|
|
fn poll_write_vectored(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
bufs: &[IoSlice<'_>],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &*self).poll_write_vectored(cx, bufs)
|
|
}
|
|
|
|
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
|
Pin::new(&mut &*self).poll_flush(cx)
|
|
}
|
|
|
|
fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
|
Pin::new(&mut &*self).poll_close(cx)
|
|
}
|
|
}
|
|
|
|
impl futures::io::AsyncWrite for &TcpStream {
|
|
fn poll_write(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
buf: &[u8],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &self.io_handle).poll_write(cx, buf)
|
|
}
|
|
|
|
fn poll_write_vectored(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
bufs: &[IoSlice<'_>],
|
|
) -> Poll<io::Result<usize>> {
|
|
Pin::new(&mut &self.io_handle).poll_write_vectored(cx, bufs)
|
|
}
|
|
|
|
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
|
Pin::new(&mut &self.io_handle).poll_flush(cx)
|
|
}
|
|
|
|
fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
|
Pin::new(&mut &self.io_handle).poll_close(cx)
|
|
}
|
|
}
|
|
|
|
impl From<net::TcpStream> for TcpStream {
|
|
/// Converts a `std::net::TcpStream` into its asynchronous equivalent.
|
|
fn from(stream: net::TcpStream) -> TcpStream {
|
|
let mio_stream = mio::net::TcpStream::from_stream(stream).unwrap();
|
|
|
|
#[cfg(unix)]
|
|
let stream = TcpStream {
|
|
raw_fd: mio_stream.as_raw_fd(),
|
|
io_handle: IoHandle::new(mio_stream),
|
|
};
|
|
|
|
#[cfg(windows)]
|
|
let stream = TcpStream {
|
|
// raw_socket: mio_stream.as_raw_socket(),
|
|
io_handle: IoHandle::new(mio_stream),
|
|
};
|
|
|
|
stream
|
|
}
|
|
}
|
|
|
|
cfg_if! {
|
|
if #[cfg(feature = "docs")] {
|
|
use crate::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd};
|
|
// use crate::os::windows::io::{AsRawHandle, FromRawHandle, IntoRawHandle, RawHandle};
|
|
} else if #[cfg(unix)] {
|
|
use std::os::unix::io::{AsRawFd, FromRawFd, IntoRawFd, RawFd};
|
|
} else if #[cfg(windows)] {
|
|
// use std::os::windows::io::{AsRawHandle, FromRawHandle, IntoRawHandle, RawHandle};
|
|
}
|
|
}
|
|
|
|
#[cfg_attr(feature = "docs", doc(cfg(unix)))]
|
|
cfg_if! {
|
|
if #[cfg(any(unix, feature = "docs"))] {
|
|
impl AsRawFd for TcpStream {
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
self.raw_fd
|
|
}
|
|
}
|
|
|
|
impl FromRawFd for TcpStream {
|
|
unsafe fn from_raw_fd(fd: RawFd) -> TcpStream {
|
|
net::TcpStream::from_raw_fd(fd).into()
|
|
}
|
|
}
|
|
|
|
impl IntoRawFd for TcpStream {
|
|
fn into_raw_fd(self) -> RawFd {
|
|
self.raw_fd
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg_attr(feature = "docs", doc(cfg(windows)))]
|
|
cfg_if! {
|
|
if #[cfg(any(windows, feature = "docs"))] {
|
|
// impl AsRawSocket for TcpStream {
|
|
// fn as_raw_socket(&self) -> RawSocket {
|
|
// self.raw_socket
|
|
// }
|
|
// }
|
|
//
|
|
// impl FromRawSocket for TcpStream {
|
|
// unsafe fn from_raw_socket(handle: RawSocket) -> TcpStream {
|
|
// net::TcpStream::from_raw_socket(handle).try_into().unwrap()
|
|
// }
|
|
// }
|
|
//
|
|
// impl IntoRawSocket for TcpListener {
|
|
// fn into_raw_socket(self) -> RawSocket {
|
|
// self.raw_socket
|
|
// }
|
|
// }
|
|
}
|
|
}
|