Skip to content

Commit

Permalink
feat(timers): implement a drop-in replacement for futures_timer::Delay
Browse files Browse the repository at this point in the history
  • Loading branch information
oblique committed Aug 7, 2023
1 parent ed58d66 commit 8a0d7f5
Show file tree
Hide file tree
Showing 3 changed files with 95 additions and 0 deletions.
4 changes: 4 additions & 0 deletions src/ic-cdk-timers/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## [unreleased]

### Added

- Implemented a drop-in replacement for `futures_timer::Delay`.

## [0.4.0] - 2023-07-13

### Changed
Expand Down
87 changes: 87 additions & 0 deletions src/ic-cdk-timers/src/delay.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
use std::fmt;
use std::future::Future;
use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, Poll};
use std::time::Duration;

use futures_util::task::AtomicWaker;

use crate::{clear_timer, set_timer, TimerId};

/// A future representing the notification that an elapsed duration has
/// occurred.
///
/// This is created through the `Delay::new` method indicating when the future should fire.
/// Note that these futures are not intended for high resolution timers.
pub struct Delay {
timer_id: Option<TimerId>,
waker: Arc<AtomicWaker>,
at: Duration,
}

impl Delay {
/// Creates a new future which will fire at `dur` time into the future.
pub fn new(dur: Duration) -> Delay {
let now = duration_since_epoch();

Delay {
timer_id: None,
waker: Arc::new(AtomicWaker::new()),
at: now + dur,
}
}

/// Resets this timeout to an new timeout which will fire at the time
/// specified by `at`.
pub fn reset(&mut self, dur: Duration) {
let now = duration_since_epoch();
self.at = now + dur;

if let Some(id) = self.timer_id.take() {
clear_timer(id);
}
}
}

impl Future for Delay {
type Output = ();

fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let now = duration_since_epoch();

if now >= self.at {
Poll::Ready(())
} else {
// Register the latest waker
self.waker.register(cx.waker());

// Register to global timer
if self.timer_id.is_none() {
let waker = self.waker.clone();
let id = set_timer(self.at - now, move || waker.wake());
self.timer_id = Some(id);
}

Poll::Pending
}
}
}

impl Drop for Delay {
fn drop(&mut self) {
if let Some(id) = self.timer_id.take() {
clear_timer(id);
}
}
}

impl fmt::Debug for Delay {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
f.debug_struct("Delay").finish()
}
}

fn duration_since_epoch() -> Duration {
Duration::from_nanos(ic_cdk::api::time())
}
4 changes: 4 additions & 0 deletions src/ic-cdk-timers/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,10 @@ use slotmap::{new_key_type, KeyData, SlotMap};

use ic_cdk::api::call::RejectionCode;

mod delay;

pub use delay::Delay;

// To ensure that tasks are removable seamlessly, there are two separate concepts here: tasks, for the actual function being called,
// and timers, the scheduled execution of tasks. As this is an implementation detail, this does not affect the exported name TimerId,
// which is more accurately a task ID. (The obvious solution to this, `pub use`, invokes a very silly compiler error.)
Expand Down

0 comments on commit 8a0d7f5

Please sign in to comment.