mirror of
https://github.com/tokio-rs/tokio.git
synced 2025-09-28 12:10:37 +00:00
142 lines
4.3 KiB
Rust
142 lines
4.3 KiB
Rust
#![feature(async_await)]
|
|
|
|
#[macro_use]
|
|
extern crate log;
|
|
extern crate tokio_io;
|
|
extern crate tokio_process;
|
|
|
|
use std::future::Future;
|
|
use std::io;
|
|
use std::pin::Pin;
|
|
use std::process::{Command, ExitStatus, Stdio};
|
|
|
|
use futures_util::future;
|
|
use futures_util::future::FutureExt;
|
|
use futures_util::io::AsyncBufReadExt;
|
|
use futures_util::io::AsyncReadExt;
|
|
use futures_util::io::AsyncWriteExt;
|
|
use futures_util::io::BufReader;
|
|
use futures_util::stream::{self, StreamExt};
|
|
use tokio_process::{Child, CommandExt};
|
|
|
|
mod support;
|
|
|
|
fn cat() -> Command {
|
|
let mut cmd = support::cmd("cat");
|
|
cmd.stdin(Stdio::piped()).stdout(Stdio::piped());
|
|
cmd
|
|
}
|
|
|
|
fn feed_cat(mut cat: Child, n: usize) -> Pin<Box<dyn Future<Output = io::Result<ExitStatus>>>> {
|
|
let stdin = cat.stdin().take().unwrap();
|
|
let stdout = cat.stdout().take().unwrap();
|
|
|
|
debug!("starting to feed");
|
|
// Produce n lines on the child's stdout.
|
|
let numbers = stream::iter(0..n);
|
|
let write = numbers
|
|
.fold(stdin, move |mut stdin, i| {
|
|
let fut = async move {
|
|
debug!("sending line {} to child", i);
|
|
let bytes = format!("line {}\n", i).into_bytes();
|
|
AsyncWriteExt::write_all(&mut stdin, &bytes).await.unwrap();
|
|
stdin
|
|
};
|
|
fut
|
|
})
|
|
.map(|_| ());
|
|
|
|
// Try to read `n + 1` lines, ensuring the last one is empty
|
|
// (i.e. EOF is reached after `n` lines.
|
|
let reader = BufReader::new(stdout);
|
|
let expected_numbers = stream::iter(0..=n);
|
|
let read = expected_numbers.fold((reader, 0), move |(mut reader, i), _| {
|
|
let fut = async move {
|
|
let done = i >= n;
|
|
debug!("starting read from child");
|
|
let mut vec = Vec::new();
|
|
AsyncBufReadExt::read_until(&mut reader, b'\n', &mut vec)
|
|
.await
|
|
.unwrap();
|
|
debug!(
|
|
"read line {} from child ({} bytes, done: {})",
|
|
i,
|
|
vec.len(),
|
|
done
|
|
);
|
|
match (done, vec.len()) {
|
|
(false, 0) => {
|
|
panic!("broken pipe");
|
|
}
|
|
(true, n) if n != 0 => {
|
|
panic!("extraneous data");
|
|
}
|
|
_ => {
|
|
let s = std::str::from_utf8(&vec).unwrap();
|
|
let expected = format!("line {}\n", i);
|
|
if done || s == expected {
|
|
(reader, i + 1)
|
|
} else {
|
|
panic!("unexpected data");
|
|
}
|
|
}
|
|
}
|
|
};
|
|
fut
|
|
});
|
|
|
|
// Compose reading and writing concurrently.
|
|
future::join(write, read).then(|_| cat).boxed()
|
|
}
|
|
|
|
/// Check for the following properties when feeding stdin and
|
|
/// consuming stdout of a cat-like process:
|
|
///
|
|
/// - A number of lines that amounts to a number of bytes exceeding a
|
|
/// typical OS buffer size can be fed to the child without
|
|
/// deadlock. This tests that we also consume the stdout
|
|
/// concurrently; otherwise this would deadlock.
|
|
///
|
|
/// - We read the same lines from the child that we fed it.
|
|
///
|
|
/// - The child does produce EOF on stdout after the last line.
|
|
#[test]
|
|
fn feed_a_lot() {
|
|
let child = cat().spawn_async().unwrap();
|
|
let status = support::run_with_timeout(feed_cat(child, 10000)).unwrap();
|
|
assert_eq!(status.code(), Some(0));
|
|
}
|
|
|
|
#[test]
|
|
fn wait_with_output_captures() {
|
|
let mut child = cat().spawn_async().unwrap();
|
|
let mut stdin = child.stdin().take().unwrap();
|
|
|
|
let write_bytes = b"1234";
|
|
|
|
let future = async {
|
|
AsyncWriteExt::write_all(&mut stdin, write_bytes).await?;
|
|
drop(stdin);
|
|
let out = child.wait_with_output();
|
|
out.await
|
|
};
|
|
|
|
let ret = support::run_with_timeout(future).unwrap();
|
|
let output = ret;
|
|
|
|
assert!(output.status.success());
|
|
assert_eq!(output.stdout, write_bytes);
|
|
assert_eq!(output.stderr.len(), 0);
|
|
}
|
|
|
|
#[test]
|
|
fn status_closes_any_pipes() {
|
|
// Cat will open a pipe between the parent and child.
|
|
// If `status_async` doesn't ensure the handles are closed,
|
|
// we would end up blocking forever (and time out).
|
|
let child = cat().status_async().expect("failed to spawn child");
|
|
|
|
support::run_with_timeout(child)
|
|
.expect("time out exceeded! did we get stuck waiting on the child?");
|
|
}
|