Add time::timeout_checked, if duration is zero then timeout is disabled

This commit is contained in:
Nikolay Kim 2022-01-04 16:04:04 +06:00
parent 3724c51c0c
commit 9cb52bd37b
4 changed files with 80 additions and 3 deletions

View file

@ -1,5 +1,9 @@
# Changes
## [0.1.7] - 2022-01-04
* Add time::timeout_checked, if duration is zero then timeout is disabled
## [0.1.6] - 2022-01-03
* Use ntex-rt::spawn

View file

@ -1,6 +1,6 @@
[package]
name = "ntex-util"
version = "0.1.6"
version = "0.1.7"
authors = ["ntex contributors <team@ntex.rs>"]
description = "Utilities for ntex framework"
keywords = ["network", "framework", "async", "futures"]
@ -28,7 +28,7 @@ futures-sink = { version = "0.3", default-features = false, features = ["alloc"]
pin-project-lite = "0.2.6"
[dev-dependencies]
ntex = "0.5.4"
ntex = { version = "0.5.6", features = ["tokio"] }
ntex-bytes = "0.1.8"
ntex-macros = "0.1.3"
derive_more = "0.99"

View file

@ -40,6 +40,20 @@ where
Timeout::new_with_delay(future, Sleep::new(dur.into()))
}
/// Require a `Future` to complete before the specified duration has elapsed.
///
/// If the future completes before the duration has elapsed, then the completed
/// value is returned. Otherwise, an error is returned and the future is
/// canceled. If duration value is zero then timeout is disabled.
#[inline]
pub fn timeout_checked<T, U>(dur: U, future: T) -> TimeoutChecked<T>
where
T: Future,
U: Into<Millis>,
{
TimeoutChecked::new_with_delay(future, dur.into())
}
/// Future returned by [`sleep`](sleep).
///
/// # Examples
@ -141,6 +155,53 @@ where
}
}
pin_project_lite::pin_project! {
/// Future returned by [`timeout_checked`](timeout_checked).
#[must_use = "futures do nothing unless you `.await` or poll them"]
pub struct TimeoutChecked<T> {
#[pin]
state: TimeoutCheckedState<T>,
}
}
pin_project_lite::pin_project! {
#[project = TimeoutCheckedStateProject]
enum TimeoutCheckedState<T> {
Timeout{ #[pin] fut: Timeout<T> },
NoTimeout{ #[pin] fut: T },
}
}
impl<T> TimeoutChecked<T> {
pub(crate) fn new_with_delay(value: T, delay: Millis) -> TimeoutChecked<T> {
if delay.is_zero() {
TimeoutChecked {
state: TimeoutCheckedState::NoTimeout { fut: value },
}
} else {
TimeoutChecked {
state: TimeoutCheckedState::Timeout {
fut: Timeout::new_with_delay(value, sleep(delay)),
},
}
}
}
}
impl<T> Future for TimeoutChecked<T>
where
T: Future,
{
type Output = Result<T::Output, ()>;
fn poll(self: Pin<&mut Self>, cx: &mut task::Context<'_>) -> Poll<Self::Output> {
match self.project().state.as_mut().project() {
TimeoutCheckedStateProject::Timeout { fut } => fut.poll(cx),
TimeoutCheckedStateProject::NoTimeout { fut } => fut.poll(cx).map(Result::Ok),
}
}
}
/// Interval returned by [`interval`]
///
/// This type allows you to wait on a sequence of instants with a certain
@ -288,4 +349,16 @@ mod tests {
);
}
}
#[ntex_macros::rt_test2]
async fn test_timeout_checked() {
let result = timeout_checked(Millis(200), sleep(Millis(100))).await;
assert!(result.is_ok());
let result = timeout_checked(Millis(5), sleep(Millis(100))).await;
assert!(result.is_err());
let result = timeout_checked(Millis(0), sleep(Millis(100))).await;
assert!(result.is_ok());
}
}

View file

@ -49,7 +49,7 @@ ntex-codec = "0.6.0"
ntex-router = "0.5.1"
ntex-service = "0.3.1"
ntex-macros = "0.1.3"
ntex-util = "0.1.6"
ntex-util = "0.1.7"
ntex-bytes = "0.1.8"
ntex-tls = "0.1.0"
ntex-rt = "0.4.1"