1 #![cfg(feature = "full")]
2
3 use tokio::time::{self, sleep, Duration};
4 use tokio_stream::{self, StreamExt};
5 use tokio_test::*;
6
7 use futures::StreamExt as _;
8
maybe_sleep(idx: i32) -> i329 async fn maybe_sleep(idx: i32) -> i32 {
10 if idx % 2 == 0 {
11 sleep(ms(200)).await;
12 }
13 idx
14 }
15
ms(n: u64) -> Duration16 fn ms(n: u64) -> Duration {
17 Duration::from_millis(n)
18 }
19
20 #[tokio::test]
basic_usage()21 async fn basic_usage() {
22 time::pause();
23
24 // Items 2 and 4 time out. If we run the stream until it completes,
25 // we end up with the following items:
26 //
27 // [Ok(1), Err(Elapsed), Ok(2), Ok(3), Err(Elapsed), Ok(4)]
28
29 let stream = stream::iter(1..=4).then(maybe_sleep).timeout(ms(100));
30 let mut stream = task::spawn(stream);
31
32 // First item completes immediately
33 assert_ready_eq!(stream.poll_next(), Some(Ok(1)));
34
35 // Second item is delayed 200ms, times out after 100ms
36 assert_pending!(stream.poll_next());
37
38 time::advance(ms(150)).await;
39 let v = assert_ready!(stream.poll_next());
40 assert!(v.unwrap().is_err());
41
42 assert_pending!(stream.poll_next());
43
44 time::advance(ms(100)).await;
45 assert_ready_eq!(stream.poll_next(), Some(Ok(2)));
46
47 // Third item is ready immediately
48 assert_ready_eq!(stream.poll_next(), Some(Ok(3)));
49
50 // Fourth item is delayed 200ms, times out after 100ms
51 assert_pending!(stream.poll_next());
52
53 time::advance(ms(60)).await;
54 assert_pending!(stream.poll_next()); // nothing ready yet
55
56 time::advance(ms(60)).await;
57 let v = assert_ready!(stream.poll_next());
58 assert!(v.unwrap().is_err()); // timeout!
59
60 time::advance(ms(120)).await;
61 assert_ready_eq!(stream.poll_next(), Some(Ok(4)));
62
63 // Done.
64 assert_ready_eq!(stream.poll_next(), None);
65 }
66
67 #[tokio::test]
return_elapsed_errors_only_once()68 async fn return_elapsed_errors_only_once() {
69 time::pause();
70
71 let stream = stream::iter(1..=3).then(maybe_sleep).timeout(ms(50));
72 let mut stream = task::spawn(stream);
73
74 // First item completes immediately
75 assert_ready_eq!(stream.poll_next(), Some(Ok(1)));
76
77 // Second item is delayed 200ms, times out after 50ms. Only one `Elapsed`
78 // error is returned.
79 assert_pending!(stream.poll_next());
80 //
81 time::advance(ms(51)).await;
82 let v = assert_ready!(stream.poll_next());
83 assert!(v.unwrap().is_err()); // timeout!
84
85 // deadline elapses again, but no error is returned
86 time::advance(ms(50)).await;
87 assert_pending!(stream.poll_next());
88
89 time::advance(ms(100)).await;
90 assert_ready_eq!(stream.poll_next(), Some(Ok(2)));
91 assert_ready_eq!(stream.poll_next(), Some(Ok(3)));
92
93 // Done
94 assert_ready_eq!(stream.poll_next(), None);
95 }
96
97 #[tokio::test]
no_timeouts()98 async fn no_timeouts() {
99 let stream = stream::iter(vec![1, 3, 5])
100 .then(maybe_sleep)
101 .timeout(ms(100));
102
103 let mut stream = task::spawn(stream);
104
105 assert_ready_eq!(stream.poll_next(), Some(Ok(1)));
106 assert_ready_eq!(stream.poll_next(), Some(Ok(3)));
107 assert_ready_eq!(stream.poll_next(), Some(Ok(5)));
108 assert_ready_eq!(stream.poll_next(), None);
109 }
110