#![cfg(feature = "macros")] #![allow(clippy::disallowed_names)] use std::sync::Arc; use tokio::sync::{oneshot, Semaphore}; #[cfg(not(tokio_wasm_not_wasi))] use tokio::test as maybe_tokio_test; use tokio_test::{assert_pending, assert_ready, task}; #[cfg(tokio_wasm_not_wasi)] use wasm_bindgen_test::wasm_bindgen_test as maybe_tokio_test; #[maybe_tokio_test] async fn sync_one_lit_expr_comma() { let foo = cancel_safe_futures::join_then_try!(async { ok(1) },); assert_eq!(foo, Ok((1,))); } #[maybe_tokio_test] async fn sync_one_lit_expr_no_comma() { let foo = cancel_safe_futures::join_then_try!(async { ok(1) }); assert_eq!(foo, Ok((1,))); } #[maybe_tokio_test] async fn sync_two_lit_expr_comma() { let foo = cancel_safe_futures::join_then_try!(async { ok(1) }, async { ok(2) },); assert_eq!(foo, Ok((1, 2))); } #[maybe_tokio_test] async fn sync_two_lit_expr_no_comma() { let foo = cancel_safe_futures::join_then_try!(async { ok(1) }, async { ok(2) }); assert_eq!(foo, Ok((1, 2))); } #[maybe_tokio_test] async fn two_await() { let (tx1, rx1) = oneshot::channel::<&str>(); let (tx2, rx2) = oneshot::channel::(); let mut join = task::spawn(async { cancel_safe_futures::join_then_try!(rx1, rx2) }); assert_pending!(join.poll()); tx2.send(123).unwrap(); assert!(join.is_woken()); assert_pending!(join.poll()); tx1.send("hello").unwrap(); assert!(join.is_woken()); let res: Result<(&str, u32), _> = assert_ready!(join.poll()); assert_eq!(Ok(("hello", 123)), res); } #[maybe_tokio_test] async fn err_no_abort_early() { let (tx1, rx1) = oneshot::channel::<&str>(); let (tx2, rx2) = oneshot::channel::(); let (tx3, rx3) = oneshot::channel::<&str>(); let (tx4, rx4) = oneshot::channel::(); let mut join = task::spawn(async { cancel_safe_futures::join_then_try!( async { rx1.await.map_err(|_| "rx1 failed") }, async { rx2.await.map_err(|_| "rx2 failed") }, async { rx3.await.map_err(|_| "rx3 failed") }, async { rx4.await.map_err(|_| "rx4 failed") }, ) }); assert_pending!(join.poll()); tx2.send(123).unwrap(); assert!(join.is_woken()); assert_pending!(join.poll()); drop(tx3); assert!(join.is_woken()); assert_pending!(join.poll()); drop(tx1); assert!(join.is_woken()); // The join should still be pending since tx3 and tx4 haven't completed. // This is a difference from tokio::try_join. assert_pending!(join.poll()); tx4.send(456).unwrap(); assert!(join.is_woken()); // All futures have completed. let res = assert_ready!(join.poll()); // The first future listed in the join_then_try macro to produce an error is rx1. Ensure that rx1 // (and not rx3, which is the first future to actually cause an error) is returned here. assert_eq!(res.unwrap_err(), "rx1 failed"); } #[test] #[cfg(target_pointer_width = "64")] fn join_size() { use std::mem; let fut = async { let ready = std::future::ready(ok(0i32)); cancel_safe_futures::join_then_try!(ready) }; assert_eq!(mem::size_of_val(&fut), 32); let fut = async { let ready1 = std::future::ready(ok(0i32)); let ready2 = std::future::ready(ok(0i32)); cancel_safe_futures::join_then_try!(ready1, ready2) }; assert_eq!(mem::size_of_val(&fut), 48); } fn ok(val: T) -> Result { Ok(val) } async fn non_cooperative_task(permits: Arc) -> Result { let mut exceeded_budget = 0; for _ in 0..5 { // Another task should run after this task uses its whole budget for _ in 0..128 { let _permit = permits.clone().acquire_owned().await.unwrap(); } exceeded_budget += 1; } Ok(exceeded_budget) } async fn poor_little_task(permits: Arc) -> Result { let mut how_many_times_i_got_to_run = 0; for _ in 0..5 { let _permit = permits.clone().acquire_owned().await.unwrap(); how_many_times_i_got_to_run += 1; } Ok(how_many_times_i_got_to_run) } #[tokio::test] async fn try_join_does_not_allow_tasks_to_starve() { let permits = Arc::new(Semaphore::new(10)); // non_cooperative_task should yield after its budget is exceeded and then poor_little_task should run. let result = cancel_safe_futures::join_then_try!( non_cooperative_task(Arc::clone(&permits)), poor_little_task(permits) ); let (non_cooperative_result, little_task_result) = result.unwrap(); assert_eq!(5, non_cooperative_result); assert_eq!(5, little_task_result); } #[tokio::test] async fn a_different_future_is_polled_first_every_time_poll_fn_is_polled() { let poll_order = Arc::new(std::sync::Mutex::new(vec![])); let fut = |x, poll_order: Arc>>| async move { for _ in 0..4 { { let mut guard = poll_order.lock().unwrap(); guard.push(x); } tokio::task::yield_now().await; } }; tokio::join!( fut(1, Arc::clone(&poll_order)), fut(2, Arc::clone(&poll_order)), fut(3, Arc::clone(&poll_order)), ); // Each time the future created by join! is polled, it should start // by polling a different future first. assert_eq!( vec![1, 2, 3, 2, 3, 1, 3, 1, 2, 1, 2, 3], *poll_order.lock().unwrap() ); } #[tokio::test] async fn empty_try_join() { assert_eq!( cancel_safe_futures::join_then_try!() as Result<_, ()>, Ok(()) ); }