#![warn(rust_2018_idioms)]
use std::time::{Duration, Instant};
use tokio_test::task::MockTask;
use tokio_test::{assert_pending, assert_ready, clock};
use tokio_timer::delay;
use tokio_timer::timer::Handle;
#[test]
fn immediate_delay() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now());
assert_ready!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(1000));
});
}
#[test]
fn delayed_delay_level_0() {
let mut task = MockTask::new();
for &i in &[1, 10, 60] {
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(i));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(i));
assert_ready!(task.poll(&mut fut));
});
}
}
#[test]
fn sub_ms_delayed_delay() {
let mut task = MockTask::new();
clock::mock(|clock| {
for _ in 0..5 {
let deadline = clock.now() + Duration::from_millis(1) + Duration::new(0, 1);
let mut fut = delay(deadline);
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_ready!(task.poll(&mut fut));
assert!(clock.now() >= deadline);
clock.advance(Duration::new(0, 1));
}
});
}
#[test]
fn delayed_delay_wrapping_level_0() {
let mut task = MockTask::new();
clock::mock(|clock| {
clock.turn_for(ms(5));
assert_eq!(clock.advanced(), ms(5));
let mut fut = delay(clock.now() + ms(60));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(64));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(65));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn timer_wrapping_with_higher_levels() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut s1 = delay(clock.now() + ms(64));
assert_pending!(task.poll(&mut s1));
clock.turn_for(ms(5));
let mut s2 = delay(clock.now() + ms(60));
assert_pending!(task.poll(&mut s2));
clock.turn();
assert_eq!(clock.advanced(), ms(64));
assert_ready!(task.poll(&mut s1));
assert_pending!(task.poll(&mut s2));
clock.turn();
assert_eq!(clock.advanced(), ms(65));
assert_ready!(task.poll(&mut s2));
});
}
#[test]
fn delay_with_deadline_in_past() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() - ms(100));
assert_ready!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(1000));
});
}
#[test]
fn delayed_delay_level_1() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(234));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(192));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(234));
assert_ready!(task.poll(&mut fut));
});
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(234));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(100));
assert_eq!(clock.advanced(), ms(100));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(192));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(234));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn creating_delay_outside_of_context() {
let now = Instant::now();
let mut fut = delay(now + ms(500));
let mut task = MockTask::new();
clock::mock_at(now, |clock| {
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(448));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(500));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn concurrently_set_two_timers_second_one_shorter() {
let mut t1 = MockTask::new();
let mut t2 = MockTask::new();
clock::mock(|clock| {
let mut fut1 = delay(clock.now() + ms(500));
let mut fut2 = delay(clock.now() + ms(200));
assert_pending!(t1.poll(&mut fut1));
assert_pending!(t2.poll(&mut fut2));
clock.turn();
assert_eq!(clock.advanced(), ms(192));
clock.turn();
assert_eq!(clock.advanced(), ms(200));
assert_ready!(t2.poll(&mut fut2));
assert_pending!(t1.poll(&mut fut1));
clock.turn();
assert_eq!(clock.advanced(), ms(448));
assert_pending!(t1.poll(&mut fut1));
clock.turn();
assert_eq!(clock.advanced(), ms(500));
assert_ready!(t1.poll(&mut fut1));
})
}
#[test]
fn short_delay() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(1));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_ready!(task.poll(&mut fut));
assert_eq!(clock.advanced(), ms(1));
})
}
#[test]
fn sorta_long_delay() {
const MIN_5: u64 = 5 * 60 * 1000;
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(MIN_5));
assert_pending!(task.poll(&mut fut));
let cascades = &[262_144, 262_144 + 9 * 4096, 262_144 + 9 * 4096 + 15 * 64];
for &elapsed in cascades {
clock.turn();
assert_eq!(clock.advanced(), ms(elapsed));
assert_pending!(task.poll(&mut fut));
}
clock.turn();
assert_eq!(clock.advanced(), ms(MIN_5));
assert_ready!(task.poll(&mut fut));
})
}
#[test]
fn very_long_delay() {
const MO_5: u64 = 5 * 30 * 24 * 60 * 60 * 1000;
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(MO_5));
assert_pending!(task.poll(&mut fut));
let cascades = &[
12_884_901_888,
12_952_010_752,
12_959_875_072,
12_959_997_952,
];
for &elapsed in cascades {
clock.turn();
assert_eq!(clock.advanced(), ms(elapsed));
assert_pending!(task.poll(&mut fut));
}
clock.turn();
assert_eq!(clock.advanced(), ms(MO_5));
assert_ready!(task.poll(&mut fut));
})
}
#[test]
#[should_panic]
fn greater_than_max() {
const YR_5: u64 = 5 * 365 * 24 * 60 * 60 * 1000;
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(YR_5));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(0));
let _ = task.poll(&mut fut);
})
}
#[test]
fn unpark_is_delayed() {
let mut t1 = MockTask::new();
let mut t2 = MockTask::new();
let mut t3 = MockTask::new();
clock::mock(|clock| {
let mut fut1 = delay(clock.now() + ms(100));
let mut fut2 = delay(clock.now() + ms(101));
let mut fut3 = delay(clock.now() + ms(200));
assert_pending!(t1.poll(&mut fut1));
assert_pending!(t2.poll(&mut fut2));
assert_pending!(t3.poll(&mut fut3));
clock.park_for(ms(500));
assert_eq!(clock.advanced(), ms(500));
assert_ready!(t1.poll(&mut fut1));
assert_ready!(t2.poll(&mut fut2));
assert_ready!(t3.poll(&mut fut3));
})
}
#[test]
fn set_timeout_at_deadline_greater_than_max_timer() {
const YR_1: u64 = 365 * 24 * 60 * 60 * 1000;
const YR_5: u64 = 5 * YR_1;
let mut task = MockTask::new();
clock::mock(|clock| {
for _ in 0..5 {
clock.turn_for(ms(YR_1));
}
let mut fut = delay(clock.now() + ms(1));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(YR_5) + ms(1));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn reset_future_delay_before_fire() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(100));
assert_pending!(task.poll(&mut fut));
fut.reset(clock.now() + ms(200));
clock.turn();
assert_eq!(clock.advanced(), ms(192));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(200));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn reset_past_delay_before_turn() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(100));
assert_pending!(task.poll(&mut fut));
fut.reset(clock.now() + ms(80));
clock.turn();
assert_eq!(clock.advanced(), ms(64));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(80));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn reset_past_delay_before_fire() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(100));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(10));
assert_pending!(task.poll(&mut fut));
fut.reset(clock.now() + ms(80));
clock.turn();
assert_eq!(clock.advanced(), ms(64));
assert_pending!(task.poll(&mut fut));
clock.turn();
assert_eq!(clock.advanced(), ms(90));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn reset_future_delay_after_fire() {
let mut task = MockTask::new();
clock::mock(|clock| {
let mut fut = delay(clock.now() + ms(100));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(64));
clock.turn();
assert_eq!(clock.advanced(), ms(100));
assert_ready!(task.poll(&mut fut));
fut.reset(clock.now() + ms(10));
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1000));
assert_eq!(clock.advanced(), ms(110));
assert_ready!(task.poll(&mut fut));
});
}
#[test]
fn delay_with_default_handle() {
let handle = Handle::default();
let now = Instant::now();
let mut task = MockTask::new();
let mut fut = handle.delay(now + ms(1));
clock::mock_at(now, |clock| {
assert_pending!(task.poll(&mut fut));
clock.turn_for(ms(1));
assert_ready!(task.poll(&mut fut));
});
}
fn ms(n: u64) -> Duration {
Duration::from_millis(n)
}