mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 05:17:39 +03:00
restore time util mod
This commit is contained in:
parent
5a7e3ed589
commit
9511caf1db
3 changed files with 257 additions and 45 deletions
|
@ -65,7 +65,7 @@ const LOWRES_RESOLUTION: time::Duration = time::Duration::from_millis(5);
|
|||
/// Resolution is ~5ms
|
||||
#[inline]
|
||||
pub fn now() -> time::Instant {
|
||||
TIMER.with(|t| t.0.borrow_mut().now(&t.0))
|
||||
TIMER.with(|t| t.borrow_mut().now(t))
|
||||
}
|
||||
|
||||
/// Returns the system time corresponding to “now”.
|
||||
|
@ -73,30 +73,34 @@ pub fn now() -> time::Instant {
|
|||
/// Resolution is ~5ms
|
||||
#[inline]
|
||||
pub fn system_time() -> time::SystemTime {
|
||||
TIMER.with(|t| t.0.borrow_mut().system_time(&t.0))
|
||||
TIMER.with(|t| t.borrow_mut().system_time(t))
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct TimerHandle(usize);
|
||||
|
||||
impl TimerHandle {
|
||||
/// Createt new timer and return handle
|
||||
pub fn new(millis: u64) -> Self {
|
||||
Timer::add_timer(millis)
|
||||
TIMER.with(|t| Timer::add_timer(t, millis))
|
||||
}
|
||||
|
||||
/// Resets the `TimerHandle` instance to a new deadline.
|
||||
pub fn reset(&self, millis: u64) {
|
||||
Timer::update_timer(self.0, millis);
|
||||
TIMER.with(|t| Timer::update_timer(t, self.0, millis))
|
||||
}
|
||||
|
||||
pub fn is_elapsed(&self) -> bool {
|
||||
Timer::with_entry(self.0, |entry| {
|
||||
entry.flags.contains(TimerEntryFlags::ELAPSED)
|
||||
TIMER.with(|t| {
|
||||
t.borrow_mut().timers[self.0]
|
||||
.flags
|
||||
.contains(TimerEntryFlags::ELAPSED)
|
||||
})
|
||||
}
|
||||
|
||||
pub fn poll_elapsed(&self, cx: &mut task::Context<'_>) -> Poll<()> {
|
||||
Timer::with_entry(self.0, |entry| {
|
||||
TIMER.with(|t| {
|
||||
let entry = &t.borrow_mut().timers[self.0];
|
||||
if entry.flags.contains(TimerEntryFlags::ELAPSED) {
|
||||
Poll::Ready(())
|
||||
} else {
|
||||
|
@ -109,7 +113,7 @@ impl TimerHandle {
|
|||
|
||||
impl Drop for TimerHandle {
|
||||
fn drop(&mut self) {
|
||||
Timer::remove_timer(self.0);
|
||||
TIMER.with(|t| t.borrow_mut().remove_timer(self.0));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -123,13 +127,11 @@ bitflags::bitflags! {
|
|||
}
|
||||
}
|
||||
|
||||
struct Timer(Rc<RefCell<TimerInner>>);
|
||||
|
||||
thread_local! {
|
||||
static TIMER: Timer = Timer::new();
|
||||
static TIMER: Rc<RefCell<Timer>>= Rc::new(RefCell::new(Timer::new()));
|
||||
}
|
||||
|
||||
struct TimerInner {
|
||||
struct Timer {
|
||||
timers: Slab<TimerEntry>,
|
||||
elapsed: u64,
|
||||
elapsed_instant: time::Instant,
|
||||
|
@ -146,33 +148,7 @@ struct TimerInner {
|
|||
|
||||
impl Timer {
|
||||
fn new() -> Self {
|
||||
Timer(Rc::new(RefCell::new(TimerInner::new())))
|
||||
}
|
||||
|
||||
fn with_entry<F, R>(no: usize, f: F) -> R
|
||||
where
|
||||
F: Fn(&mut TimerEntry) -> R,
|
||||
{
|
||||
TIMER.with(|t| f(&mut t.0.borrow_mut().timers[no]))
|
||||
}
|
||||
|
||||
// Add the timer into the hash bucket
|
||||
fn add_timer(expires: u64) -> TimerHandle {
|
||||
TIMER.with(|t| TimerInner::add_timer(&t.0, expires))
|
||||
}
|
||||
|
||||
fn update_timer(handle: usize, expires: u64) {
|
||||
TIMER.with(|t| TimerInner::update_timer(&t.0, handle, expires));
|
||||
}
|
||||
|
||||
fn remove_timer(handle: usize) {
|
||||
TIMER.with(|t| t.0.borrow_mut().remove_timer(handle));
|
||||
}
|
||||
}
|
||||
|
||||
impl TimerInner {
|
||||
fn new() -> Self {
|
||||
TimerInner {
|
||||
Timer {
|
||||
buckets: Self::create_buckets(),
|
||||
timers: Slab::default(),
|
||||
elapsed: 0,
|
||||
|
@ -197,7 +173,7 @@ impl TimerInner {
|
|||
buckets
|
||||
}
|
||||
|
||||
fn now(&mut self, inner: &Rc<RefCell<TimerInner>>) -> time::Instant {
|
||||
fn now(&mut self, inner: &Rc<RefCell<Timer>>) -> time::Instant {
|
||||
let cur = self.lowres_time.get();
|
||||
if let Some(cur) = cur {
|
||||
cur
|
||||
|
@ -214,7 +190,7 @@ impl TimerInner {
|
|||
}
|
||||
}
|
||||
|
||||
fn system_time(&mut self, inner: &Rc<RefCell<TimerInner>>) -> time::SystemTime {
|
||||
fn system_time(&mut self, inner: &Rc<RefCell<Timer>>) -> time::SystemTime {
|
||||
let cur = self.lowres_stime.get();
|
||||
if let Some(cur) = cur {
|
||||
cur
|
||||
|
@ -555,12 +531,12 @@ impl TimerEntry {
|
|||
}
|
||||
|
||||
struct TimerDriver {
|
||||
inner: Rc<RefCell<TimerInner>>,
|
||||
inner: Rc<RefCell<Timer>>,
|
||||
sleep: Pin<Box<Sleep>>,
|
||||
}
|
||||
|
||||
impl TimerDriver {
|
||||
fn start(cell: &Rc<RefCell<TimerInner>>) {
|
||||
fn start(cell: &Rc<RefCell<Timer>>) {
|
||||
let mut inner = cell.borrow_mut();
|
||||
inner.flags.insert(Flags::TIMER_ACTIVE);
|
||||
|
||||
|
@ -617,12 +593,12 @@ impl Future for TimerDriver {
|
|||
}
|
||||
|
||||
struct LowresTimerDriver {
|
||||
inner: Rc<RefCell<TimerInner>>,
|
||||
inner: Rc<RefCell<Timer>>,
|
||||
sleep: Pin<Box<Sleep>>,
|
||||
}
|
||||
|
||||
impl LowresTimerDriver {
|
||||
fn start(slf: &mut TimerInner, cell: &Rc<RefCell<TimerInner>>) {
|
||||
fn start(slf: &mut Timer, cell: &Rc<RefCell<Timer>>) {
|
||||
slf.flags.insert(Flags::LOWRES_DRIVER | Flags::LOWRES_TIMER);
|
||||
|
||||
crate::rt::spawn(LowresTimerDriver {
|
||||
|
|
|
@ -5,6 +5,7 @@ pub mod inflight;
|
|||
pub mod keepalive;
|
||||
pub mod sink;
|
||||
pub mod stream;
|
||||
pub mod time;
|
||||
pub mod timeout;
|
||||
pub mod variant;
|
||||
|
||||
|
|
235
ntex/src/util/time.rs
Normal file
235
ntex/src/util/time.rs
Normal file
|
@ -0,0 +1,235 @@
|
|||
use std::task::{Context, Poll};
|
||||
use std::time::{self, Duration, Instant};
|
||||
use std::{cell::RefCell, convert::Infallible, rc::Rc};
|
||||
|
||||
use crate::rt::time::sleep;
|
||||
use crate::service::{Service, ServiceFactory};
|
||||
use crate::util::Ready;
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct LowResTime(Rc<RefCell<Inner>>);
|
||||
|
||||
#[derive(Debug)]
|
||||
struct Inner {
|
||||
resolution: Duration,
|
||||
current: Option<Instant>,
|
||||
}
|
||||
|
||||
impl Inner {
|
||||
fn new(resolution: Duration) -> Self {
|
||||
Inner {
|
||||
resolution,
|
||||
current: None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl LowResTime {
|
||||
pub fn with(resolution: Duration) -> LowResTime {
|
||||
LowResTime(Rc::new(RefCell::new(Inner::new(resolution))))
|
||||
}
|
||||
|
||||
pub fn timer(&self) -> LowResTimeService {
|
||||
LowResTimeService(self.0.clone())
|
||||
}
|
||||
}
|
||||
|
||||
impl Default for LowResTime {
|
||||
fn default() -> Self {
|
||||
LowResTime(Rc::new(RefCell::new(Inner::new(Duration::from_secs(1)))))
|
||||
}
|
||||
}
|
||||
|
||||
impl ServiceFactory for LowResTime {
|
||||
type Request = ();
|
||||
type Response = Instant;
|
||||
type Error = Infallible;
|
||||
type InitError = Infallible;
|
||||
type Config = ();
|
||||
type Service = LowResTimeService;
|
||||
type Future = Ready<Self::Service, Self::InitError>;
|
||||
|
||||
#[inline]
|
||||
fn new_service(&self, _: ()) -> Self::Future {
|
||||
Ready::Ok(self.timer())
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct LowResTimeService(Rc<RefCell<Inner>>);
|
||||
|
||||
impl LowResTimeService {
|
||||
pub fn with(resolution: Duration) -> LowResTimeService {
|
||||
LowResTimeService(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(&self) -> Instant {
|
||||
let cur = self.0.borrow().current;
|
||||
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.resolution
|
||||
};
|
||||
|
||||
crate::rt::spawn(async move {
|
||||
sleep(interval).await;
|
||||
inner.borrow_mut().current.take();
|
||||
});
|
||||
now
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Service for LowResTimeService {
|
||||
type Request = ();
|
||||
type Response = Instant;
|
||||
type Error = Infallible;
|
||||
type Future = Ready<Self::Response, Self::Error>;
|
||||
|
||||
#[inline]
|
||||
fn poll_ready(&self, _: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
||||
Poll::Ready(Ok(()))
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn call(&self, _: ()) -> Self::Future {
|
||||
Ready::Ok(self.now())
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct SystemTime(Rc<RefCell<SystemTimeInner>>);
|
||||
|
||||
#[derive(Debug)]
|
||||
struct SystemTimeInner {
|
||||
resolution: Duration,
|
||||
current: Option<time::SystemTime>,
|
||||
}
|
||||
|
||||
impl SystemTimeInner {
|
||||
fn new(resolution: Duration) -> Self {
|
||||
SystemTimeInner {
|
||||
resolution,
|
||||
current: None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct SystemTimeService(Rc<RefCell<SystemTimeInner>>);
|
||||
|
||||
impl SystemTimeService {
|
||||
pub fn with(resolution: Duration) -> SystemTimeService {
|
||||
SystemTimeService(Rc::new(RefCell::new(SystemTimeInner::new(resolution))))
|
||||
}
|
||||
|
||||
/// Get current time. This function has to be called from
|
||||
/// future's poll method, otherwise it panics.
|
||||
pub fn now(&self) -> time::SystemTime {
|
||||
let cur = self.0.borrow().current;
|
||||
if let Some(cur) = cur {
|
||||
cur
|
||||
} else {
|
||||
let now = time::SystemTime::now();
|
||||
let inner = self.0.clone();
|
||||
let interval = {
|
||||
let mut b = inner.borrow_mut();
|
||||
b.current = Some(now);
|
||||
b.resolution
|
||||
};
|
||||
|
||||
crate::rt::spawn(async move {
|
||||
sleep(interval).await;
|
||||
inner.borrow_mut().current.take();
|
||||
});
|
||||
now
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use crate::util::lazy;
|
||||
use std::time::{Duration, SystemTime};
|
||||
|
||||
#[crate::rt_test]
|
||||
async fn low_res_timee() {
|
||||
let f = LowResTime::default();
|
||||
let srv = f.new_service(()).await.unwrap();
|
||||
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
||||
srv.call(()).await.unwrap();
|
||||
}
|
||||
|
||||
/// State Under Test: Two calls of `SystemTimeService::now()` return the same value if they are done within resolution interval of `SystemTimeService`.
|
||||
///
|
||||
/// Expected Behavior: Two back-to-back calls of `SystemTimeService::now()` return the same value.
|
||||
#[crate::rt_test]
|
||||
async fn system_time_service_time_does_not_immediately_change() {
|
||||
let resolution = Duration::from_millis(50);
|
||||
|
||||
let time_service = SystemTimeService::with(resolution);
|
||||
assert_eq!(time_service.now(), time_service.now());
|
||||
}
|
||||
|
||||
/// State Under Test: Two calls of `LowResTimeService::now()` return the same value if they are done within resolution interval of `SystemTimeService`.
|
||||
///
|
||||
/// Expected Behavior: Two back-to-back calls of `LowResTimeService::now()` return the same value.
|
||||
#[crate::rt_test]
|
||||
async fn lowres_time_service_time_does_not_immediately_change() {
|
||||
let resolution = Duration::from_millis(50);
|
||||
let time_service = LowResTimeService::with(resolution);
|
||||
assert_eq!(time_service.now(), time_service.now());
|
||||
}
|
||||
|
||||
/// State Under Test: `SystemTimeService::now()` updates returned value every resolution period.
|
||||
///
|
||||
/// Expected Behavior: Two calls of `LowResTimeService::now()` made in subsequent resolution interval return different values
|
||||
/// and second value is greater than the first one at least by a resolution interval.
|
||||
#[crate::rt_test]
|
||||
async fn system_time_service_time_updates_after_resolution_interval() {
|
||||
let resolution = Duration::from_millis(100);
|
||||
let wait_time = Duration::from_millis(300);
|
||||
|
||||
let time_service = SystemTimeService::with(resolution);
|
||||
|
||||
let first_time = time_service
|
||||
.now()
|
||||
.duration_since(SystemTime::UNIX_EPOCH)
|
||||
.unwrap();
|
||||
|
||||
sleep(wait_time).await;
|
||||
|
||||
let second_time = time_service
|
||||
.now()
|
||||
.duration_since(SystemTime::UNIX_EPOCH)
|
||||
.unwrap();
|
||||
|
||||
assert!(second_time - first_time >= wait_time);
|
||||
}
|
||||
|
||||
/// State Under Test: `LowResTimeService::now()` updates returned value every resolution period.
|
||||
///
|
||||
/// Expected Behavior: Two calls of `LowResTimeService::now()` made in subsequent resolution interval return different values
|
||||
/// and second value is greater than the first one at least by a resolution interval.
|
||||
#[crate::rt_test]
|
||||
async fn lowres_time_service_time_updates_after_resolution_interval() {
|
||||
let resolution = Duration::from_millis(100);
|
||||
let wait_time = Duration::from_millis(300);
|
||||
let time_service = LowResTimeService::with(resolution);
|
||||
|
||||
let first_time = time_service.now();
|
||||
|
||||
sleep(wait_time).await;
|
||||
|
||||
let second_time = time_service.now();
|
||||
assert!(second_time - first_time >= wait_time);
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue