1 #![cfg(feature = "macros")]
2 #![allow(clippy::disallowed_names)]
3
4 use std::sync::Arc;
5
6 use tokio::sync::{oneshot, Semaphore};
7 use tokio_test::{assert_pending, assert_ready, task};
8
9 #[cfg(all(target_family = "wasm", not(target_os = "wasi")))]
10 use wasm_bindgen_test::wasm_bindgen_test as maybe_tokio_test;
11
12 #[cfg(not(all(target_family = "wasm", not(target_os = "wasi"))))]
13 use tokio::test as maybe_tokio_test;
14
15 #[maybe_tokio_test]
sync_one_lit_expr_comma()16 async fn sync_one_lit_expr_comma() {
17 let foo = tokio::try_join!(async { ok(1) },);
18
19 assert_eq!(foo, Ok((1,)));
20 }
21
22 #[maybe_tokio_test]
sync_one_lit_expr_no_comma()23 async fn sync_one_lit_expr_no_comma() {
24 let foo = tokio::try_join!(async { ok(1) });
25
26 assert_eq!(foo, Ok((1,)));
27 }
28
29 #[maybe_tokio_test]
sync_two_lit_expr_comma()30 async fn sync_two_lit_expr_comma() {
31 let foo = tokio::try_join!(async { ok(1) }, async { ok(2) },);
32
33 assert_eq!(foo, Ok((1, 2)));
34 }
35
36 #[maybe_tokio_test]
sync_two_lit_expr_no_comma()37 async fn sync_two_lit_expr_no_comma() {
38 let foo = tokio::try_join!(async { ok(1) }, async { ok(2) });
39
40 assert_eq!(foo, Ok((1, 2)));
41 }
42
43 #[maybe_tokio_test]
two_await()44 async fn two_await() {
45 let (tx1, rx1) = oneshot::channel::<&str>();
46 let (tx2, rx2) = oneshot::channel::<u32>();
47
48 let mut join = task::spawn(async { tokio::try_join!(rx1, rx2) });
49
50 assert_pending!(join.poll());
51
52 tx2.send(123).unwrap();
53 assert!(join.is_woken());
54 assert_pending!(join.poll());
55
56 tx1.send("hello").unwrap();
57 assert!(join.is_woken());
58 let res: Result<(&str, u32), _> = assert_ready!(join.poll());
59
60 assert_eq!(Ok(("hello", 123)), res);
61 }
62
63 #[maybe_tokio_test]
err_abort_early()64 async fn err_abort_early() {
65 let (tx1, rx1) = oneshot::channel::<&str>();
66 let (tx2, rx2) = oneshot::channel::<u32>();
67 let (_tx3, rx3) = oneshot::channel::<u32>();
68
69 let mut join = task::spawn(async { tokio::try_join!(rx1, rx2, rx3) });
70
71 assert_pending!(join.poll());
72
73 tx2.send(123).unwrap();
74 assert!(join.is_woken());
75 assert_pending!(join.poll());
76
77 drop(tx1);
78 assert!(join.is_woken());
79
80 let res = assert_ready!(join.poll());
81
82 assert!(res.is_err());
83 }
84
85 #[test]
86 #[cfg(target_pointer_width = "64")]
join_size()87 fn join_size() {
88 use futures::future;
89 use std::mem;
90
91 let fut = async {
92 let ready = future::ready(ok(0i32));
93 tokio::try_join!(ready)
94 };
95 assert_eq!(mem::size_of_val(&fut), 32);
96
97 let fut = async {
98 let ready1 = future::ready(ok(0i32));
99 let ready2 = future::ready(ok(0i32));
100 tokio::try_join!(ready1, ready2)
101 };
102 assert_eq!(mem::size_of_val(&fut), 48);
103 }
104
ok<T>(val: T) -> Result<T, ()>105 fn ok<T>(val: T) -> Result<T, ()> {
106 Ok(val)
107 }
108
non_cooperative_task(permits: Arc<Semaphore>) -> Result<usize, String>109 async fn non_cooperative_task(permits: Arc<Semaphore>) -> Result<usize, String> {
110 let mut exceeded_budget = 0;
111
112 for _ in 0..5 {
113 // Another task should run after this task uses its whole budget
114 for _ in 0..128 {
115 let _permit = permits.clone().acquire_owned().await.unwrap();
116 }
117
118 exceeded_budget += 1;
119 }
120
121 Ok(exceeded_budget)
122 }
123
poor_little_task(permits: Arc<Semaphore>) -> Result<usize, String>124 async fn poor_little_task(permits: Arc<Semaphore>) -> Result<usize, String> {
125 let mut how_many_times_i_got_to_run = 0;
126
127 for _ in 0..5 {
128 let _permit = permits.clone().acquire_owned().await.unwrap();
129
130 how_many_times_i_got_to_run += 1;
131 }
132
133 Ok(how_many_times_i_got_to_run)
134 }
135
136 #[tokio::test]
try_join_does_not_allow_tasks_to_starve()137 async fn try_join_does_not_allow_tasks_to_starve() {
138 let permits = Arc::new(Semaphore::new(10));
139
140 // non_cooperative_task should yield after its budget is exceeded and then poor_little_task should run.
141 let result = tokio::try_join!(
142 non_cooperative_task(Arc::clone(&permits)),
143 poor_little_task(permits)
144 );
145
146 let (non_cooperative_result, little_task_result) = result.unwrap();
147
148 assert_eq!(5, non_cooperative_result);
149 assert_eq!(5, little_task_result);
150 }
151
152 #[tokio::test]
a_different_future_is_polled_first_every_time_poll_fn_is_polled()153 async fn a_different_future_is_polled_first_every_time_poll_fn_is_polled() {
154 let poll_order = Arc::new(std::sync::Mutex::new(vec![]));
155
156 let fut = |x, poll_order: Arc<std::sync::Mutex<Vec<i32>>>| async move {
157 for _ in 0..4 {
158 {
159 let mut guard = poll_order.lock().unwrap();
160
161 guard.push(x);
162 }
163
164 tokio::task::yield_now().await;
165 }
166 };
167
168 tokio::join!(
169 fut(1, Arc::clone(&poll_order)),
170 fut(2, Arc::clone(&poll_order)),
171 fut(3, Arc::clone(&poll_order)),
172 );
173
174 // Each time the future created by join! is polled, it should start
175 // by polling a different future first.
176 assert_eq!(
177 vec![1, 2, 3, 2, 3, 1, 3, 1, 2, 1, 2, 3],
178 *poll_order.lock().unwrap()
179 );
180 }
181
182 #[tokio::test]
empty_try_join()183 async fn empty_try_join() {
184 assert_eq!(tokio::try_join!() as Result<_, ()>, Ok(()));
185 }
186