wasmtimer 0.4.3

Time utils from std::time, tokio::time and tokio_util::time on WASM targets
Documentation
use std::{task::Poll, time::Duration};

use futures::Future;
use pin_utils::unsafe_pinned;

use crate::std::Instant;

use super::{error::Elapsed, Sleep};

pub struct Timeout<T> {
    delay: Sleep,
    future: T,
}

impl<F> Timeout<F>
where
    F: Future,
{
    unsafe_pinned!(future: F);
    unsafe_pinned!(delay: Sleep);

    pub(crate) fn new(dur: Duration, fut: F) -> Timeout<F> {
        Timeout {
            delay: Sleep::new(dur),
            future: fut,
        }
    }

    pub(crate) fn new_at(at: Instant, fut: F) -> Timeout<F> {
        Timeout {
            delay: Sleep::new_at(at),
            future: fut,
        }
    }

    pub fn get_ref(&self) -> &F {
        &self.future
    }

    pub fn get_mut(&mut self) -> &mut F {
        &mut self.future
    }

    pub fn into_inner(self) -> F {
        self.future
    }
}

impl<T> Future for Timeout<T>
where
    T: Future,
{
    type Output = Result<T::Output, Elapsed>;

    fn poll(
        mut self: std::pin::Pin<&mut Self>,
        cx: &mut std::task::Context<'_>,
    ) -> Poll<Self::Output> {
        match self.as_mut().future().poll(cx) {
            Poll::Pending => {}
            Poll::Ready(other) => return Poll::Ready(Ok(other)),
        }

        if self.delay().poll(cx).is_ready() {
            Poll::Ready(Err(Elapsed::new()))
        } else {
            Poll::Pending
        }
    }
}

pub fn timeout<F>(duration: Duration, future: F) -> Timeout<F>
where
    F: Future,
{
    Timeout::new(duration, future)
}

pub fn timeout_at<F>(deadline: Instant, future: F) -> Timeout<F>
where
    F: Future,
{
    Timeout::new_at(deadline, future)
}