use std::{sync::Arc, thread, time::Duration};
use nio::{sleep, spawn, spawn_local, test};
#[test]
async fn test_abort_without_panic_3157() {
let handle = spawn_local(async move { sleep(Duration::new(100, 0)).await });
sleep(Duration::from_millis(10)).await;
handle.abort_handle().abort();
let _ = handle.await;
}
#[test(worker_threads = 2)]
async fn test_abort_without_panic_3662() {
use std::sync::Arc;
use std::sync::atomic::{AtomicBool, Ordering};
struct DropCheck(Arc<AtomicBool>);
impl Drop for DropCheck {
fn drop(&mut self) {
self.0.store(true, Ordering::SeqCst);
}
}
let drop_flag = Arc::new(AtomicBool::new(false));
let drop_check = DropCheck(drop_flag.clone());
let j = spawn(async move {
let _drop_check = drop_check;
std::future::pending::<()>().await;
});
let drop_flag2 = drop_flag.clone();
let task = thread::spawn(move || {
assert!(!drop_flag2.load(Ordering::SeqCst));
j.abort_handle().abort();
j
})
.join()
.unwrap();
let result = task.await;
assert!(drop_flag.load(Ordering::SeqCst));
assert!(result.unwrap_err().is_cancelled());
let i = spawn(async move {
nio_future::yield_now().await;
});
i.await.unwrap();
}
#[test]
async fn remote_abort() {
struct DropCheck {
created_on: thread::ThreadId,
not_send: std::marker::PhantomData<*const ()>,
}
impl DropCheck {
fn new() -> Self {
Self {
created_on: thread::current().id(),
not_send: std::marker::PhantomData,
}
}
}
impl Drop for DropCheck {
fn drop(&mut self) {
if thread::current().id() != self.created_on {
panic!("non-Send value dropped in another thread!");
}
}
}
let check = DropCheck::new();
let jh = spawn_local(async move {
std::future::pending::<()>().await;
drop(check);
});
let jh_abort = jh.abort_handle();
let jh2 = thread::spawn(move || {
thread::sleep(Duration::from_millis(10));
jh_abort.abort();
});
assert!(jh.await.is_err());
jh2.join().unwrap();
}
#[test]
async fn test_abort_wakes_task_3964() {
let notify_dropped = Arc::new(());
let weak_notify_dropped = Arc::downgrade(¬ify_dropped);
let handle = spawn_local(async move {
let _notify_dropped = notify_dropped;
sleep(Duration::new(100, 0)).await
});
sleep(Duration::from_millis(10)).await;
handle.abort_handle().abort();
drop(handle);
sleep(Duration::from_millis(10)).await;
assert!(weak_notify_dropped.upgrade().is_none());
}
#[cfg(panic = "unwind")]
struct PanicOnDrop;
#[cfg(panic = "unwind")]
impl Drop for PanicOnDrop {
fn drop(&mut self) {
panic!("Well what did you expect would happen...");
}
}
#[test]
#[cfg(panic = "unwind")]
async fn test_abort_task_that_panics_on_drop_contained() {
let handle = spawn_local(async move {
let _panic_dropped = PanicOnDrop;
sleep(Duration::new(100, 0)).await
});
sleep(Duration::from_millis(10)).await;
handle.abort_handle().abort();
drop(handle);
sleep(Duration::from_millis(10)).await;
}
#[test]
#[cfg(panic = "unwind")]
async fn test_abort_task_that_panics_on_drop_returned() {
let handle = spawn_local(async move {
let _panic_dropped = PanicOnDrop;
sleep(Duration::new(100, 0)).await
});
sleep(Duration::from_millis(10)).await;
handle.abort_handle().abort();
assert!(handle.await.unwrap_err().is_panic());
}
#[test(worker_threads = 2)]
#[cfg(panic = "unwind")]
async fn test_join_error_display() {
let join_err = spawn(async move {
let value = 1234;
panic!("Format-args payload: {value}")
})
.await
.unwrap_err();
let join_err_str = join_err.to_string();
assert!(
join_err_str.starts_with("task ")
&& join_err_str.ends_with(" panicked with message \"Format-args payload: 1234\""),
"Unexpected join_err_str {join_err_str:?}"
);
let join_err = spawn(async move { panic!("Const payload") })
.await
.unwrap_err();
let join_err_str = join_err.to_string();
assert!(
join_err_str.starts_with("task ")
&& join_err_str.ends_with(" panicked with message \"Const payload\""),
"Unexpected join_err_str {join_err_str:?}"
);
let join_err = spawn(async move { std::panic::panic_any(1234i32) })
.await
.unwrap_err();
let join_err_str = join_err.to_string();
assert!(
join_err_str.starts_with("task ") && join_err_str.ends_with(" panicked"),
"Unexpected join_err_str {join_err_str:?}"
);
}