Vendor dependencies
Let's see how I like this workflow.
This commit is contained in:
parent
34d1830413
commit
9c435dc440
7500 changed files with 1665121 additions and 99 deletions
448
vendor/tokio/tests/rt_basic.rs
vendored
Normal file
448
vendor/tokio/tests/rt_basic.rs
vendored
Normal file
|
|
@ -0,0 +1,448 @@
|
|||
#![warn(rust_2018_idioms)]
|
||||
#![cfg(feature = "full")]
|
||||
|
||||
use tokio::runtime::Runtime;
|
||||
use tokio::sync::oneshot;
|
||||
use tokio::time::{timeout, Duration};
|
||||
use tokio_test::{assert_err, assert_ok};
|
||||
|
||||
use std::future::Future;
|
||||
use std::pin::Pin;
|
||||
use std::sync::atomic::{AtomicBool, Ordering};
|
||||
use std::task::{Context, Poll};
|
||||
use std::thread;
|
||||
|
||||
mod support {
|
||||
pub(crate) mod mpsc_stream;
|
||||
}
|
||||
|
||||
macro_rules! cfg_metrics {
|
||||
($($t:tt)*) => {
|
||||
#[cfg(tokio_unstable)]
|
||||
{
|
||||
$( $t )*
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn spawned_task_does_not_progress_without_block_on() {
|
||||
let (tx, mut rx) = oneshot::channel();
|
||||
|
||||
let rt = rt();
|
||||
|
||||
rt.spawn(async move {
|
||||
assert_ok!(tx.send("hello"));
|
||||
});
|
||||
|
||||
thread::sleep(Duration::from_millis(50));
|
||||
|
||||
assert_err!(rx.try_recv());
|
||||
|
||||
let out = rt.block_on(async { assert_ok!(rx.await) });
|
||||
|
||||
assert_eq!(out, "hello");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn no_extra_poll() {
|
||||
use pin_project_lite::pin_project;
|
||||
use std::pin::Pin;
|
||||
use std::sync::{
|
||||
atomic::{AtomicUsize, Ordering::SeqCst},
|
||||
Arc,
|
||||
};
|
||||
use std::task::{Context, Poll};
|
||||
use tokio_stream::{Stream, StreamExt};
|
||||
|
||||
pin_project! {
|
||||
struct TrackPolls<S> {
|
||||
npolls: Arc<AtomicUsize>,
|
||||
#[pin]
|
||||
s: S,
|
||||
}
|
||||
}
|
||||
|
||||
impl<S> Stream for TrackPolls<S>
|
||||
where
|
||||
S: Stream,
|
||||
{
|
||||
type Item = S::Item;
|
||||
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
||||
let this = self.project();
|
||||
this.npolls.fetch_add(1, SeqCst);
|
||||
this.s.poll_next(cx)
|
||||
}
|
||||
}
|
||||
|
||||
let (tx, rx) = support::mpsc_stream::unbounded_channel_stream::<()>();
|
||||
let rx = TrackPolls {
|
||||
npolls: Arc::new(AtomicUsize::new(0)),
|
||||
s: rx,
|
||||
};
|
||||
let npolls = Arc::clone(&rx.npolls);
|
||||
|
||||
let rt = rt();
|
||||
|
||||
// TODO: could probably avoid this, but why not.
|
||||
let mut rx = Box::pin(rx);
|
||||
|
||||
rt.spawn(async move { while rx.next().await.is_some() {} });
|
||||
rt.block_on(async {
|
||||
tokio::task::yield_now().await;
|
||||
});
|
||||
|
||||
// should have been polled exactly once: the initial poll
|
||||
assert_eq!(npolls.load(SeqCst), 1);
|
||||
|
||||
tx.send(()).unwrap();
|
||||
rt.block_on(async {
|
||||
tokio::task::yield_now().await;
|
||||
});
|
||||
|
||||
// should have been polled twice more: once to yield Some(), then once to yield Pending
|
||||
assert_eq!(npolls.load(SeqCst), 1 + 2);
|
||||
|
||||
drop(tx);
|
||||
rt.block_on(async {
|
||||
tokio::task::yield_now().await;
|
||||
});
|
||||
|
||||
// should have been polled once more: to yield None
|
||||
assert_eq!(npolls.load(SeqCst), 1 + 2 + 1);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn acquire_mutex_in_drop() {
|
||||
use futures::future::pending;
|
||||
use tokio::task;
|
||||
|
||||
let (tx1, rx1) = oneshot::channel();
|
||||
let (tx2, rx2) = oneshot::channel();
|
||||
|
||||
let rt = rt();
|
||||
|
||||
rt.spawn(async move {
|
||||
let _ = rx2.await;
|
||||
unreachable!();
|
||||
});
|
||||
|
||||
rt.spawn(async move {
|
||||
let _ = rx1.await;
|
||||
tx2.send(()).unwrap();
|
||||
unreachable!();
|
||||
});
|
||||
|
||||
// Spawn a task that will never notify
|
||||
rt.spawn(async move {
|
||||
pending::<()>().await;
|
||||
tx1.send(()).unwrap();
|
||||
});
|
||||
|
||||
// Tick the loop
|
||||
rt.block_on(async {
|
||||
task::yield_now().await;
|
||||
});
|
||||
|
||||
// Drop the rt
|
||||
drop(rt);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn drop_tasks_in_context() {
|
||||
static SUCCESS: AtomicBool = AtomicBool::new(false);
|
||||
|
||||
struct ContextOnDrop;
|
||||
|
||||
impl Future for ContextOnDrop {
|
||||
type Output = ();
|
||||
|
||||
fn poll(self: Pin<&mut Self>, _cx: &mut Context<'_>) -> Poll<()> {
|
||||
Poll::Pending
|
||||
}
|
||||
}
|
||||
|
||||
impl Drop for ContextOnDrop {
|
||||
fn drop(&mut self) {
|
||||
if tokio::runtime::Handle::try_current().is_ok() {
|
||||
SUCCESS.store(true, Ordering::SeqCst);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let rt = rt();
|
||||
rt.spawn(ContextOnDrop);
|
||||
drop(rt);
|
||||
|
||||
assert!(SUCCESS.load(Ordering::SeqCst));
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg_attr(tokio_wasi, ignore = "Wasi does not support panic recovery")]
|
||||
#[should_panic(expected = "boom")]
|
||||
fn wake_in_drop_after_panic() {
|
||||
let (tx, rx) = oneshot::channel::<()>();
|
||||
|
||||
struct WakeOnDrop(Option<oneshot::Sender<()>>);
|
||||
|
||||
impl Drop for WakeOnDrop {
|
||||
fn drop(&mut self) {
|
||||
self.0.take().unwrap().send(()).unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
let rt = rt();
|
||||
|
||||
rt.spawn(async move {
|
||||
let _wake_on_drop = WakeOnDrop(Some(tx));
|
||||
// wait forever
|
||||
futures::future::pending::<()>().await;
|
||||
});
|
||||
|
||||
let _join = rt.spawn(async move { rx.await });
|
||||
|
||||
rt.block_on(async {
|
||||
tokio::task::yield_now().await;
|
||||
panic!("boom");
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn spawn_two() {
|
||||
let rt = rt();
|
||||
|
||||
let out = rt.block_on(async {
|
||||
let (tx, rx) = oneshot::channel();
|
||||
|
||||
tokio::spawn(async move {
|
||||
tokio::spawn(async move {
|
||||
tx.send("ZOMG").unwrap();
|
||||
});
|
||||
});
|
||||
|
||||
assert_ok!(rx.await)
|
||||
});
|
||||
|
||||
assert_eq!(out, "ZOMG");
|
||||
|
||||
cfg_metrics! {
|
||||
let metrics = rt.metrics();
|
||||
drop(rt);
|
||||
assert_eq!(0, metrics.remote_schedule_count());
|
||||
|
||||
let mut local = 0;
|
||||
for i in 0..metrics.num_workers() {
|
||||
local += metrics.worker_local_schedule_count(i);
|
||||
}
|
||||
|
||||
assert_eq!(2, local);
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg_attr(tokio_wasi, ignore = "WASI: std::thread::spawn not supported")]
|
||||
#[test]
|
||||
fn spawn_remote() {
|
||||
let rt = rt();
|
||||
|
||||
let out = rt.block_on(async {
|
||||
let (tx, rx) = oneshot::channel();
|
||||
|
||||
let handle = tokio::spawn(async move {
|
||||
std::thread::spawn(move || {
|
||||
std::thread::sleep(Duration::from_millis(10));
|
||||
tx.send("ZOMG").unwrap();
|
||||
});
|
||||
|
||||
rx.await.unwrap()
|
||||
});
|
||||
|
||||
handle.await.unwrap()
|
||||
});
|
||||
|
||||
assert_eq!(out, "ZOMG");
|
||||
|
||||
cfg_metrics! {
|
||||
let metrics = rt.metrics();
|
||||
drop(rt);
|
||||
assert_eq!(1, metrics.remote_schedule_count());
|
||||
|
||||
let mut local = 0;
|
||||
for i in 0..metrics.num_workers() {
|
||||
local += metrics.worker_local_schedule_count(i);
|
||||
}
|
||||
|
||||
assert_eq!(1, local);
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg_attr(tokio_wasi, ignore = "Wasi does not support panic recovery")]
|
||||
#[should_panic(
|
||||
expected = "A Tokio 1.x context was found, but timers are disabled. Call `enable_time` on the runtime builder to enable timers."
|
||||
)]
|
||||
fn timeout_panics_when_no_time_handle() {
|
||||
let rt = tokio::runtime::Builder::new_current_thread()
|
||||
.build()
|
||||
.unwrap();
|
||||
rt.block_on(async {
|
||||
let (_tx, rx) = oneshot::channel::<()>();
|
||||
let dur = Duration::from_millis(20);
|
||||
let _ = timeout(dur, rx).await;
|
||||
});
|
||||
}
|
||||
|
||||
#[cfg(tokio_unstable)]
|
||||
mod unstable {
|
||||
use tokio::runtime::{Builder, RngSeed, UnhandledPanic};
|
||||
|
||||
#[test]
|
||||
#[should_panic(
|
||||
expected = "a spawned task panicked and the runtime is configured to shut down on unhandled panic"
|
||||
)]
|
||||
fn shutdown_on_panic() {
|
||||
let rt = Builder::new_current_thread()
|
||||
.unhandled_panic(UnhandledPanic::ShutdownRuntime)
|
||||
.build()
|
||||
.unwrap();
|
||||
|
||||
rt.block_on(async {
|
||||
tokio::spawn(async {
|
||||
panic!("boom");
|
||||
});
|
||||
|
||||
futures::future::pending::<()>().await;
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg_attr(tokio_wasi, ignore = "Wasi does not support panic recovery")]
|
||||
fn spawns_do_nothing() {
|
||||
use std::sync::Arc;
|
||||
|
||||
let rt = Builder::new_current_thread()
|
||||
.unhandled_panic(UnhandledPanic::ShutdownRuntime)
|
||||
.build()
|
||||
.unwrap();
|
||||
|
||||
let rt1 = Arc::new(rt);
|
||||
let rt2 = rt1.clone();
|
||||
|
||||
let _ = std::thread::spawn(move || {
|
||||
rt2.block_on(async {
|
||||
tokio::spawn(async {
|
||||
panic!("boom");
|
||||
});
|
||||
|
||||
futures::future::pending::<()>().await;
|
||||
})
|
||||
})
|
||||
.join();
|
||||
|
||||
let task = rt1.spawn(async {});
|
||||
let res = futures::executor::block_on(task);
|
||||
assert!(res.is_err());
|
||||
}
|
||||
|
||||
#[test]
|
||||
#[cfg_attr(tokio_wasi, ignore = "Wasi does not support panic recovery")]
|
||||
fn shutdown_all_concurrent_block_on() {
|
||||
const N: usize = 2;
|
||||
use std::sync::{mpsc, Arc};
|
||||
|
||||
let rt = Builder::new_current_thread()
|
||||
.unhandled_panic(UnhandledPanic::ShutdownRuntime)
|
||||
.build()
|
||||
.unwrap();
|
||||
|
||||
let rt = Arc::new(rt);
|
||||
let mut ths = vec![];
|
||||
let (tx, rx) = mpsc::channel();
|
||||
|
||||
for _ in 0..N {
|
||||
let rt = rt.clone();
|
||||
let tx = tx.clone();
|
||||
ths.push(std::thread::spawn(move || {
|
||||
rt.block_on(async {
|
||||
tx.send(()).unwrap();
|
||||
futures::future::pending::<()>().await;
|
||||
});
|
||||
}));
|
||||
}
|
||||
|
||||
for _ in 0..N {
|
||||
rx.recv().unwrap();
|
||||
}
|
||||
|
||||
rt.spawn(async {
|
||||
panic!("boom");
|
||||
});
|
||||
|
||||
for th in ths {
|
||||
assert!(th.join().is_err());
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn rng_seed() {
|
||||
let seed = b"bytes used to generate seed";
|
||||
let rt1 = tokio::runtime::Builder::new_current_thread()
|
||||
.rng_seed(RngSeed::from_bytes(seed))
|
||||
.build()
|
||||
.unwrap();
|
||||
let rt1_values = rt1.block_on(async {
|
||||
let rand_1 = tokio::macros::support::thread_rng_n(100);
|
||||
let rand_2 = tokio::macros::support::thread_rng_n(100);
|
||||
|
||||
(rand_1, rand_2)
|
||||
});
|
||||
|
||||
let rt2 = tokio::runtime::Builder::new_current_thread()
|
||||
.rng_seed(RngSeed::from_bytes(seed))
|
||||
.build()
|
||||
.unwrap();
|
||||
let rt2_values = rt2.block_on(async {
|
||||
let rand_1 = tokio::macros::support::thread_rng_n(100);
|
||||
let rand_2 = tokio::macros::support::thread_rng_n(100);
|
||||
|
||||
(rand_1, rand_2)
|
||||
});
|
||||
|
||||
assert_eq!(rt1_values, rt2_values);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn rng_seed_multi_enter() {
|
||||
let seed = b"bytes used to generate seed";
|
||||
|
||||
fn two_rand_values() -> (u32, u32) {
|
||||
let rand_1 = tokio::macros::support::thread_rng_n(100);
|
||||
let rand_2 = tokio::macros::support::thread_rng_n(100);
|
||||
|
||||
(rand_1, rand_2)
|
||||
}
|
||||
|
||||
let rt1 = tokio::runtime::Builder::new_current_thread()
|
||||
.rng_seed(RngSeed::from_bytes(seed))
|
||||
.build()
|
||||
.unwrap();
|
||||
let rt1_values_1 = rt1.block_on(async { two_rand_values() });
|
||||
let rt1_values_2 = rt1.block_on(async { two_rand_values() });
|
||||
|
||||
let rt2 = tokio::runtime::Builder::new_current_thread()
|
||||
.rng_seed(RngSeed::from_bytes(seed))
|
||||
.build()
|
||||
.unwrap();
|
||||
let rt2_values_1 = rt2.block_on(async { two_rand_values() });
|
||||
let rt2_values_2 = rt2.block_on(async { two_rand_values() });
|
||||
|
||||
assert_eq!(rt1_values_1, rt2_values_1);
|
||||
assert_eq!(rt1_values_2, rt2_values_2);
|
||||
}
|
||||
}
|
||||
|
||||
fn rt() -> Runtime {
|
||||
tokio::runtime::Builder::new_current_thread()
|
||||
.enable_all()
|
||||
.build()
|
||||
.unwrap()
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue