1
0
mirror of https://github.com/fafhrd91/actix-net synced 2024-11-24 12:52:58 +01:00
actix-net/src/timer.rs
2018-09-20 11:16:12 -07:00

106 lines
2.5 KiB
Rust

use std::cell::RefCell;
use std::rc::Rc;
use std::time::{Duration, Instant};
use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll};
use tokio_current_thread::spawn;
use tokio_timer::sleep;
use super::service::{NewService, Service};
use super::Never;
#[derive(Clone, Debug)]
pub struct LowResTimer(Rc<RefCell<Inner>>);
#[derive(Debug)]
struct Inner {
interval: Duration,
current: Option<Instant>,
}
impl Inner {
fn new(interval: Duration) -> Self {
Inner {
interval,
current: None,
}
}
}
impl LowResTimer {
pub fn with_interval(interval: Duration) -> LowResTimer {
LowResTimer(Rc::new(RefCell::new(Inner::new(interval))))
}
pub fn timer(&self) -> LowResTimerService {
LowResTimerService(self.0.clone())
}
}
impl Default for LowResTimer {
fn default() -> Self {
LowResTimer(Rc::new(RefCell::new(Inner::new(Duration::from_secs(1)))))
}
}
impl NewService for LowResTimer {
type Request = ();
type Response = Instant;
type Error = Never;
type InitError = Never;
type Service = LowResTimerService;
type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future {
ok(self.timer())
}
}
#[derive(Clone, Debug)]
pub struct LowResTimerService(Rc<RefCell<Inner>>);
impl LowResTimerService {
pub fn with_resolution(resolution: Duration) -> LowResTimerService {
LowResTimerService(Rc::new(RefCell::new(Inner::new(resolution))))
}
/// Get current time. This function has to be called from
/// future's poll method, otherwise it panics.
pub fn now(&mut self) -> Instant {
let cur = self.0.borrow().current.clone();
if let Some(cur) = cur {
cur
} else {
let now = Instant::now();
let inner = self.0.clone();
let interval = {
let mut b = inner.borrow_mut();
b.current = Some(now);
b.interval
};
spawn(sleep(interval).map_err(|_| panic!()).and_then(move |_| {
inner.borrow_mut().current.take();
Ok(())
}));
now
}
}
}
impl Service for LowResTimerService {
type Request = ();
type Response = Instant;
type Error = Never;
type Future = FutureResult<Self::Response, Self::Error>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, _: ()) -> Self::Future {
ok(self.now())
}
}