mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 05:17:39 +03:00
Use async fn for Service::ready() and Service::shutdown() (#363)
This commit is contained in:
parent
dec6ab083a
commit
b04bdf41f6
33 changed files with 285 additions and 299 deletions
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-async-std"
|
name = "ntex-async-std"
|
||||||
version = "0.4.0"
|
version = "0.5.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "async-std intergration for ntex framework"
|
description = "async-std intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -16,9 +16,9 @@ name = "ntex_async_std"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.21"
|
ntex-bytes = "0.1"
|
||||||
ntex-io = "1.0.0"
|
ntex-io = "2.0"
|
||||||
ntex-util = "1.0.0"
|
ntex-util = "2.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
async-std = { version = "1", features = ["unstable"] }
|
async-std = { version = "1", features = ["unstable"] }
|
||||||
oneshot = { version = "0.1", default-features = false, features = ["async"] }
|
oneshot = { version = "0.1", default-features = false, features = ["async"] }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-glommio"
|
name = "ntex-glommio"
|
||||||
version = "0.4.0"
|
version = "0.5.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "glommio intergration for ntex framework"
|
description = "glommio intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -16,9 +16,9 @@ name = "ntex_glommio"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.24"
|
ntex-bytes = "0.1"
|
||||||
ntex-io = "1.0.0"
|
ntex-io = "2.0"
|
||||||
ntex-util = "1.0.0"
|
ntex-util = "2.0"
|
||||||
futures-lite = "2.2"
|
futures-lite = "2.2"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
oneshot = { version = "0.1", default-features = false, features = ["async"] }
|
oneshot = { version = "0.1", default-features = false, features = ["async"] }
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.0.0] - 2024-05-28
|
||||||
|
|
||||||
|
* Use async fn for Service::ready() and Service::shutdown()
|
||||||
|
|
||||||
## [1.2.0] - 2024-05-12
|
## [1.2.0] - 2024-05-12
|
||||||
|
|
||||||
* Better write back-pressure handling
|
* Better write back-pressure handling
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-io"
|
name = "ntex-io"
|
||||||
version = "1.2.0"
|
version = "2.0.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Utilities for encoding and decoding frames"
|
description = "Utilities for encoding and decoding frames"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -18,8 +18,8 @@ path = "src/lib.rs"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-codec = "0.6.2"
|
ntex-codec = "0.6.2"
|
||||||
ntex-bytes = "0.1.24"
|
ntex-bytes = "0.1.24"
|
||||||
ntex-util = "1.0"
|
ntex-util = "2.0"
|
||||||
ntex-service = "2.0"
|
ntex-service = "3.0"
|
||||||
|
|
||||||
bitflags = "2"
|
bitflags = "2"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
|
@ -2,9 +2,8 @@
|
||||||
#![allow(clippy::let_underscore_future)]
|
#![allow(clippy::let_underscore_future)]
|
||||||
use std::{cell::Cell, future::Future, pin::Pin, rc::Rc, task::Context, task::Poll};
|
use std::{cell::Cell, future::Future, pin::Pin, rc::Rc, task::Context, task::Poll};
|
||||||
|
|
||||||
use ntex_bytes::Pool;
|
|
||||||
use ntex_codec::{Decoder, Encoder};
|
use ntex_codec::{Decoder, Encoder};
|
||||||
use ntex_service::{IntoService, Pipeline, PipelineCall, Service};
|
use ntex_service::{IntoService, Pipeline, PipelineBinding, PipelineCall, Service};
|
||||||
use ntex_util::{future::Either, ready, spawn, time::Seconds};
|
use ntex_util::{future::Either, ready, spawn, time::Seconds};
|
||||||
|
|
||||||
use crate::{Decoded, DispatchItem, IoBoxed, IoStatusUpdate, RecvError};
|
use crate::{Decoded, DispatchItem, IoBoxed, IoStatusUpdate, RecvError};
|
||||||
|
@ -144,7 +143,6 @@ where
|
||||||
flags: Flags,
|
flags: Flags,
|
||||||
shared: Rc<DispatcherShared<S, U>>,
|
shared: Rc<DispatcherShared<S, U>>,
|
||||||
response: Option<PipelineCall<S, DispatchItem<U>>>,
|
response: Option<PipelineCall<S, DispatchItem<U>>>,
|
||||||
pool: Pool,
|
|
||||||
cfg: DispatcherConfig,
|
cfg: DispatcherConfig,
|
||||||
read_remains: u32,
|
read_remains: u32,
|
||||||
read_remains_prev: u32,
|
read_remains_prev: u32,
|
||||||
|
@ -158,7 +156,7 @@ where
|
||||||
{
|
{
|
||||||
io: IoBoxed,
|
io: IoBoxed,
|
||||||
codec: U,
|
codec: U,
|
||||||
service: Pipeline<S>,
|
service: PipelineBinding<S, DispatchItem<U>>,
|
||||||
error: Cell<Option<DispatcherError<S::Error, <U as Encoder>::Error>>>,
|
error: Cell<Option<DispatcherError<S::Error, <U as Encoder>::Error>>>,
|
||||||
inflight: Cell<usize>,
|
inflight: Cell<usize>,
|
||||||
}
|
}
|
||||||
|
@ -194,7 +192,7 @@ impl<S, U> From<Either<S, U>> for DispatcherError<S, U> {
|
||||||
|
|
||||||
impl<S, U> Dispatcher<S, U>
|
impl<S, U> Dispatcher<S, U>
|
||||||
where
|
where
|
||||||
S: Service<DispatchItem<U>, Response = Option<Response<U>>>,
|
S: Service<DispatchItem<U>, Response = Option<Response<U>>> + 'static,
|
||||||
U: Decoder + Encoder + 'static,
|
U: Decoder + Encoder + 'static,
|
||||||
{
|
{
|
||||||
/// Construct new `Dispatcher` instance.
|
/// Construct new `Dispatcher` instance.
|
||||||
|
@ -217,18 +215,16 @@ where
|
||||||
Flags::KA_ENABLED
|
Flags::KA_ENABLED
|
||||||
};
|
};
|
||||||
|
|
||||||
let pool = io.memory_pool().pool();
|
|
||||||
let shared = Rc::new(DispatcherShared {
|
let shared = Rc::new(DispatcherShared {
|
||||||
io,
|
io,
|
||||||
codec,
|
codec,
|
||||||
error: Cell::new(None),
|
error: Cell::new(None),
|
||||||
inflight: Cell::new(0),
|
inflight: Cell::new(0),
|
||||||
service: Pipeline::new(service.into_service()),
|
service: Pipeline::new(service.into_service()).bind(),
|
||||||
});
|
});
|
||||||
|
|
||||||
Dispatcher {
|
Dispatcher {
|
||||||
inner: DispatcherInner {
|
inner: DispatcherInner {
|
||||||
pool,
|
|
||||||
shared,
|
shared,
|
||||||
flags,
|
flags,
|
||||||
cfg: cfg.clone(),
|
cfg: cfg.clone(),
|
||||||
|
@ -284,14 +280,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// handle memory pool pressure
|
|
||||||
if slf.pool.poll_ready(cx).is_pending() {
|
|
||||||
slf.flags.remove(Flags::KA_TIMEOUT | Flags::READ_TIMEOUT);
|
|
||||||
slf.shared.io.stop_timer();
|
|
||||||
slf.shared.io.pause();
|
|
||||||
return Poll::Pending;
|
|
||||||
}
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match slf.st {
|
match slf.st {
|
||||||
DispatcherState::Processing => {
|
DispatcherState::Processing => {
|
||||||
|
@ -434,7 +422,7 @@ where
|
||||||
U: Decoder + Encoder + 'static,
|
U: Decoder + Encoder + 'static,
|
||||||
{
|
{
|
||||||
fn call_service(&mut self, cx: &mut Context<'_>, item: DispatchItem<U>) {
|
fn call_service(&mut self, cx: &mut Context<'_>, item: DispatchItem<U>) {
|
||||||
let mut fut = self.shared.service.call_static(item);
|
let mut fut = self.shared.service.call(item);
|
||||||
self.shared.inflight.set(self.shared.inflight.get() + 1);
|
self.shared.inflight.set(self.shared.inflight.get() + 1);
|
||||||
|
|
||||||
// optimize first call
|
// optimize first call
|
||||||
|
@ -682,11 +670,7 @@ mod tests {
|
||||||
U: Decoder + Encoder + 'static,
|
U: Decoder + Encoder + 'static,
|
||||||
{
|
{
|
||||||
/// Construct new `Dispatcher` instance
|
/// Construct new `Dispatcher` instance
|
||||||
pub(crate) fn debug<T: IoStream, F: IntoService<S, DispatchItem<U>>>(
|
pub(crate) fn debug<T: IoStream>(io: T, codec: U, service: S) -> (Self, State) {
|
||||||
io: T,
|
|
||||||
codec: U,
|
|
||||||
service: F,
|
|
||||||
) -> (Self, State) {
|
|
||||||
let cfg = DispatcherConfig::default()
|
let cfg = DispatcherConfig::default()
|
||||||
.set_keepalive_timeout(Seconds(1))
|
.set_keepalive_timeout(Seconds(1))
|
||||||
.clone();
|
.clone();
|
||||||
|
@ -694,14 +678,13 @@ mod tests {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Construct new `Dispatcher` instance
|
/// Construct new `Dispatcher` instance
|
||||||
pub(crate) fn debug_cfg<T: IoStream, F: IntoService<S, DispatchItem<U>>>(
|
pub(crate) fn debug_cfg<T: IoStream>(
|
||||||
io: T,
|
io: T,
|
||||||
codec: U,
|
codec: U,
|
||||||
service: F,
|
service: S,
|
||||||
cfg: DispatcherConfig,
|
cfg: DispatcherConfig,
|
||||||
) -> (Self, State) {
|
) -> (Self, State) {
|
||||||
let state = Io::new(io);
|
let state = Io::new(io);
|
||||||
let pool = state.memory_pool().pool();
|
|
||||||
state.set_disconnect_timeout(cfg.disconnect_timeout());
|
state.set_disconnect_timeout(cfg.disconnect_timeout());
|
||||||
state.set_tag("DBG");
|
state.set_tag("DBG");
|
||||||
|
|
||||||
|
@ -719,7 +702,7 @@ mod tests {
|
||||||
io: state.into(),
|
io: state.into(),
|
||||||
error: Cell::new(None),
|
error: Cell::new(None),
|
||||||
inflight: Cell::new(0),
|
inflight: Cell::new(0),
|
||||||
service: Pipeline::new(service.into_service()),
|
service: Pipeline::new(service).bind(),
|
||||||
});
|
});
|
||||||
|
|
||||||
(
|
(
|
||||||
|
@ -731,7 +714,6 @@ mod tests {
|
||||||
read_remains: 0,
|
read_remains: 0,
|
||||||
read_remains_prev: 0,
|
read_remains_prev: 0,
|
||||||
read_max_timeout: Seconds::ZERO,
|
read_max_timeout: Seconds::ZERO,
|
||||||
pool,
|
|
||||||
shared,
|
shared,
|
||||||
cfg,
|
cfg,
|
||||||
flags,
|
flags,
|
||||||
|
@ -864,9 +846,9 @@ mod tests {
|
||||||
type Response = Option<Response<BytesCodec>>;
|
type Response = Option<Response<BytesCodec>>;
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
fn poll_ready(&self, _: &mut Context<'_>) -> Poll<Result<(), ()>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
||||||
self.0.set(self.0.get() + 1);
|
self.0.set(self.0.get() + 1);
|
||||||
Poll::Ready(Err(()))
|
Err(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(
|
async fn call(
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.0.0] - 2024-05-28
|
||||||
|
|
||||||
|
* Use async fn for Service::ready() and Service::shutdown()
|
||||||
|
|
||||||
## [1.0.2] - 2024-03-30
|
## [1.0.2] - 2024-03-30
|
||||||
|
|
||||||
* Fix glommio compat feature #327
|
* Fix glommio compat feature #327
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-net"
|
name = "ntex-net"
|
||||||
version = "1.0.2"
|
version = "2.0.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "ntexwork utils for ntex framework"
|
description = "ntexwork utils for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -28,16 +28,16 @@ glommio = ["ntex-rt/glommio", "ntex-glommio"]
|
||||||
async-std = ["ntex-rt/async-std", "ntex-async-std"]
|
async-std = ["ntex-rt/async-std", "ntex-async-std"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-service = "2.0"
|
ntex-service = "3.0"
|
||||||
ntex-bytes = "0.1.24"
|
ntex-bytes = "0.1"
|
||||||
ntex-http = "0.1"
|
ntex-http = "0.1"
|
||||||
ntex-io = "1.0"
|
ntex-io = "2.0"
|
||||||
ntex-rt = "0.4.11"
|
ntex-rt = "0.4.11"
|
||||||
ntex-util = "1.0"
|
ntex-util = "2.0"
|
||||||
|
|
||||||
ntex-tokio = { version = "0.4.0", optional = true }
|
ntex-tokio = { version = "0.5.0", optional = true }
|
||||||
ntex-glommio = { version = "0.4.0", optional = true }
|
ntex-glommio = { version = "0.5.0", optional = true }
|
||||||
ntex-async-std = { version = "0.4.0", optional = true }
|
ntex-async-std = { version = "0.5.0", optional = true }
|
||||||
|
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
|
|
|
@ -6,7 +6,6 @@ use ntex_util::future::Either;
|
||||||
|
|
||||||
use super::{Address, Connect, ConnectError};
|
use super::{Address, Connect, ConnectError};
|
||||||
|
|
||||||
#[derive(Copy)]
|
|
||||||
/// DNS Resolver Service
|
/// DNS Resolver Service
|
||||||
pub struct Resolver<T>(marker::PhantomData<T>);
|
pub struct Resolver<T>(marker::PhantomData<T>);
|
||||||
|
|
||||||
|
@ -17,6 +16,8 @@ impl<T> Resolver<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T> Copy for Resolver<T> {}
|
||||||
|
|
||||||
impl<T: Address> Resolver<T> {
|
impl<T: Address> Resolver<T> {
|
||||||
/// Lookup ip addresses for provided host
|
/// Lookup ip addresses for provided host
|
||||||
pub async fn lookup(&self, req: Connect<T>) -> Result<Connect<T>, ConnectError> {
|
pub async fn lookup(&self, req: Connect<T>) -> Result<Connect<T>, ConnectError> {
|
||||||
|
@ -100,7 +101,7 @@ impl<T> Default for Resolver<T> {
|
||||||
|
|
||||||
impl<T> Clone for Resolver<T> {
|
impl<T> Clone for Resolver<T> {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Resolver(marker::PhantomData)
|
*self
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -117,7 +118,7 @@ impl<T: Address, C> ServiceFactory<Connect<T>, C> for Resolver<T> {
|
||||||
type InitError = ();
|
type InitError = ();
|
||||||
|
|
||||||
async fn create(&self, _: C) -> Result<Self::Service, Self::InitError> {
|
async fn create(&self, _: C) -> Result<Self::Service, Self::InitError> {
|
||||||
Ok(self.clone())
|
Ok(*self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -144,7 +145,7 @@ mod tests {
|
||||||
async fn resolver() {
|
async fn resolver() {
|
||||||
let resolver = Resolver::default().clone();
|
let resolver = Resolver::default().clone();
|
||||||
assert!(format!("{:?}", resolver).contains("Resolver"));
|
assert!(format!("{:?}", resolver).contains("Resolver"));
|
||||||
let srv = resolver.pipeline(()).await.unwrap();
|
let srv = resolver.pipeline(()).await.unwrap().bind();
|
||||||
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
||||||
|
|
||||||
let res = srv.call(Connect::new("www.rust-lang.org")).await;
|
let res = srv.call(Connect::new("www.rust-lang.org")).await;
|
||||||
|
|
|
@ -9,13 +9,15 @@ use ntex_util::future::{BoxFuture, Either};
|
||||||
use super::{Address, Connect, ConnectError, Resolver};
|
use super::{Address, Connect, ConnectError, Resolver};
|
||||||
use crate::tcp_connect_in;
|
use crate::tcp_connect_in;
|
||||||
|
|
||||||
#[derive(Copy)]
|
/// Basic tcp stream connector
|
||||||
pub struct Connector<T> {
|
pub struct Connector<T> {
|
||||||
resolver: Resolver<T>,
|
resolver: Resolver<T>,
|
||||||
pool: PoolRef,
|
pool: PoolRef,
|
||||||
tag: &'static str,
|
tag: &'static str,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T> Copy for Connector<T> {}
|
||||||
|
|
||||||
impl<T> Connector<T> {
|
impl<T> Connector<T> {
|
||||||
/// Construct new connect service with default dns resolver
|
/// Construct new connect service with default dns resolver
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
|
@ -85,11 +87,7 @@ impl<T> Default for Connector<T> {
|
||||||
|
|
||||||
impl<T> Clone for Connector<T> {
|
impl<T> Clone for Connector<T> {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Connector {
|
*self
|
||||||
resolver: self.resolver.clone(),
|
|
||||||
tag: self.tag,
|
|
||||||
pool: self.pool,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -110,7 +108,7 @@ impl<T: Address, C> ServiceFactory<Connect<T>, C> for Connector<T> {
|
||||||
type InitError = ();
|
type InitError = ();
|
||||||
|
|
||||||
async fn create(&self, _: C) -> Result<Self::Service, Self::InitError> {
|
async fn create(&self, _: C) -> Result<Self::Service, Self::InitError> {
|
||||||
Ok(self.clone())
|
Ok(*self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.0.0] - 2024-05-28
|
||||||
|
|
||||||
|
* Use async fn for Service::ready() and Service::shutdown()
|
||||||
|
|
||||||
## [1.0.5] - 2024-04-02
|
## [1.0.5] - 2024-04-02
|
||||||
|
|
||||||
* Fix external configuration handling
|
* Fix external configuration handling
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-server"
|
name = "ntex-server"
|
||||||
version = "1.0.5"
|
version = "2.0.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Server for ntex framework"
|
description = "Server for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -16,11 +16,11 @@ name = "ntex_server"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.24"
|
ntex-bytes = "0.1"
|
||||||
ntex-net = "1.0"
|
ntex-net = "2.0"
|
||||||
ntex-service = "2.0"
|
ntex-service = "3.0"
|
||||||
ntex-rt = "0.4.13"
|
ntex-rt = "0.4"
|
||||||
ntex-util = "1.0"
|
ntex-util = "2.0"
|
||||||
|
|
||||||
async-channel = "2"
|
async-channel = "2"
|
||||||
async-broadcast = "0.7"
|
async-broadcast = "0.7"
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{cell::Cell, rc::Rc, task};
|
use std::{cell::Cell, future::poll_fn, rc::Rc, task, task::Poll};
|
||||||
|
|
||||||
use ntex_util::task::LocalWaker;
|
use ntex_util::task::LocalWaker;
|
||||||
|
|
||||||
|
@ -30,8 +30,15 @@ impl Counter {
|
||||||
|
|
||||||
/// Check if counter is not at capacity. If counter at capacity
|
/// Check if counter is not at capacity. If counter at capacity
|
||||||
/// it registers notification for current task.
|
/// it registers notification for current task.
|
||||||
pub(super) fn available(&self, cx: &mut task::Context<'_>) -> bool {
|
pub(super) async fn available(&self) {
|
||||||
self.0.available(cx)
|
poll_fn(|cx| {
|
||||||
|
if self.0.available(cx) {
|
||||||
|
Poll::Ready(())
|
||||||
|
} else {
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get total number of acquired counts
|
/// Get total number of acquired counts
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
use std::task::{Context, Poll};
|
|
||||||
use std::{fmt, future::Future, marker::PhantomData};
|
use std::{fmt, future::Future, marker::PhantomData};
|
||||||
|
|
||||||
use ntex_bytes::PoolId;
|
use ntex_bytes::PoolId;
|
||||||
|
@ -144,10 +143,10 @@ where
|
||||||
type Response = ();
|
type Response = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
ntex_service::forward_poll_shutdown!(inner);
|
ntex_service::forward_shutdown!(inner);
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), ()>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
||||||
self.inner.poll_ready(cx).map_err(|_| ())
|
ctx.ready(&self.inner).await.map_err(|_| ())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(&self, req: Io, ctx: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
async fn call(&self, req: Io, ctx: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
use std::{task::Context, task::Poll};
|
|
||||||
|
|
||||||
use ntex_bytes::{Pool, PoolRef};
|
use ntex_bytes::{Pool, PoolRef};
|
||||||
use ntex_net::Io;
|
use ntex_net::Io;
|
||||||
use ntex_service::{boxed, Service, ServiceCtx, ServiceFactory};
|
use ntex_service::{boxed, Service, ServiceCtx, ServiceFactory};
|
||||||
|
@ -142,53 +140,34 @@ impl Service<ServerMessage> for StreamServiceImpl {
|
||||||
type Response = ();
|
type Response = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
let mut ready = self.conns.available(cx);
|
self.conns.available().await;
|
||||||
for (idx, svc) in self.services.iter().enumerate() {
|
for (idx, svc) in self.services.iter().enumerate() {
|
||||||
match svc.poll_ready(cx) {
|
match ctx.ready(svc).await {
|
||||||
Poll::Pending => ready = false,
|
Ok(()) => (),
|
||||||
Poll::Ready(Ok(())) => (),
|
Err(_) => {
|
||||||
Poll::Ready(Err(_)) => {
|
|
||||||
for (idx_, tag, _, _) in self.tokens.values() {
|
for (idx_, tag, _, _) in self.tokens.values() {
|
||||||
if idx == *idx_ {
|
if idx == *idx_ {
|
||||||
log::error!("{}: Service readiness has failed", tag);
|
log::error!("{}: Service readiness has failed", tag);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Poll::Ready(Err(()));
|
return Err(());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// check memory pools
|
Ok(())
|
||||||
for (_, _, pool, _) in self.tokens.values() {
|
|
||||||
ready = pool.poll_ready(cx).is_ready() && ready;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ready {
|
|
||||||
Poll::Ready(Ok(()))
|
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_shutdown(&self, cx: &mut Context<'_>) -> Poll<()> {
|
async fn shutdown(&self) {
|
||||||
let mut ready = true;
|
|
||||||
for svc in &self.services {
|
for svc in &self.services {
|
||||||
match svc.poll_shutdown(cx) {
|
svc.shutdown().await;
|
||||||
Poll::Pending => ready = false,
|
|
||||||
Poll::Ready(_) => (),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ready {
|
|
||||||
log::info!(
|
|
||||||
"Worker service shutdown, {} connections",
|
|
||||||
super::num_connections()
|
|
||||||
);
|
|
||||||
Poll::Ready(())
|
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
}
|
||||||
|
log::info!(
|
||||||
|
"Worker service shutdown, {} connections",
|
||||||
|
super::num_connections()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(&self, req: ServerMessage, ctx: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
async fn call(&self, req: ServerMessage, ctx: ServiceCtx<'_, Self>) -> Result<(), ()> {
|
||||||
|
|
|
@ -6,7 +6,7 @@ use async_broadcast::{self as bus, broadcast};
|
||||||
use async_channel::{unbounded, Receiver, Sender};
|
use async_channel::{unbounded, Receiver, Sender};
|
||||||
|
|
||||||
use ntex_rt::{spawn, Arbiter};
|
use ntex_rt::{spawn, Arbiter};
|
||||||
use ntex_service::{Pipeline, ServiceFactory};
|
use ntex_service::{Pipeline, PipelineBinding, ServiceFactory};
|
||||||
use ntex_util::future::{select, stream_recv, Either, Stream};
|
use ntex_util::future::{select, stream_recv, Either, Stream};
|
||||||
use ntex_util::time::{sleep, timeout_checked, Millis};
|
use ntex_util::time::{sleep, timeout_checked, Millis};
|
||||||
|
|
||||||
|
@ -240,8 +240,10 @@ struct WorkerSt<T, F: ServiceFactory<WorkerMessage<T>>> {
|
||||||
availability: WorkerAvailabilityTx,
|
availability: WorkerAvailabilityTx,
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn run_worker<T, F>(mut svc: Pipeline<F::Service>, mut wrk: WorkerSt<T, F>)
|
async fn run_worker<T, F>(
|
||||||
where
|
mut svc: PipelineBinding<F::Service, WorkerMessage<T>>,
|
||||||
|
mut wrk: WorkerSt<T, F>,
|
||||||
|
) where
|
||||||
T: Send + 'static,
|
T: Send + 'static,
|
||||||
F: ServiceFactory<WorkerMessage<T>> + 'static,
|
F: ServiceFactory<WorkerMessage<T>> + 'static,
|
||||||
{
|
{
|
||||||
|
@ -250,7 +252,7 @@ where
|
||||||
ready!(svc.poll_ready(cx)?);
|
ready!(svc.poll_ready(cx)?);
|
||||||
|
|
||||||
if let Some(item) = ready!(Pin::new(&mut wrk.rx).poll_next(cx)) {
|
if let Some(item) = ready!(Pin::new(&mut wrk.rx).poll_next(cx)) {
|
||||||
let fut = svc.call_static(WorkerMessage::New(item));
|
let fut = svc.call(WorkerMessage::New(item));
|
||||||
let _ = spawn(async move {
|
let _ = spawn(async move {
|
||||||
let _ = fut.await;
|
let _ = fut.await;
|
||||||
});
|
});
|
||||||
|
@ -267,17 +269,17 @@ where
|
||||||
wrk.availability.set(false);
|
wrk.availability.set(false);
|
||||||
|
|
||||||
if timeout.is_zero() {
|
if timeout.is_zero() {
|
||||||
let fut = svc.call_static(WorkerMessage::ForceShutdown);
|
let fut = svc.call(WorkerMessage::ForceShutdown);
|
||||||
let _ = spawn(async move {
|
let _ = spawn(async move {
|
||||||
let _ = fut.await;
|
let _ = fut.await;
|
||||||
});
|
});
|
||||||
sleep(STOP_TIMEOUT).await;
|
sleep(STOP_TIMEOUT).await;
|
||||||
} else {
|
} else {
|
||||||
let fut = svc.call_static(WorkerMessage::Shutdown(timeout));
|
let fut = svc.call(WorkerMessage::Shutdown(timeout));
|
||||||
let res = timeout_checked(timeout, fut).await;
|
let res = timeout_checked(timeout, fut).await;
|
||||||
let _ = result.send(res.is_ok());
|
let _ = result.send(res.is_ok());
|
||||||
};
|
};
|
||||||
poll_fn(|cx| svc.poll_shutdown(cx)).await;
|
svc.shutdown().await;
|
||||||
|
|
||||||
log::info!("Stopping worker {:?}", wrk.id);
|
log::info!("Stopping worker {:?}", wrk.id);
|
||||||
return;
|
return;
|
||||||
|
@ -289,7 +291,7 @@ where
|
||||||
match select(wrk.factory.create(()), stream_recv(&mut wrk.stop)).await {
|
match select(wrk.factory.create(()), stream_recv(&mut wrk.stop)).await {
|
||||||
Either::Left(Ok(service)) => {
|
Either::Left(Ok(service)) => {
|
||||||
wrk.availability.set(true);
|
wrk.availability.set(true);
|
||||||
svc = Pipeline::new(service);
|
svc = Pipeline::new(service).bind();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Either::Left(Err(_)) => sleep(STOP_TIMEOUT).await,
|
Either::Left(Err(_)) => sleep(STOP_TIMEOUT).await,
|
||||||
|
@ -305,7 +307,13 @@ async fn create<T, F>(
|
||||||
stop: Receiver<Shutdown>,
|
stop: Receiver<Shutdown>,
|
||||||
factory: Result<F, ()>,
|
factory: Result<F, ()>,
|
||||||
availability: WorkerAvailabilityTx,
|
availability: WorkerAvailabilityTx,
|
||||||
) -> Result<(Pipeline<F::Service>, WorkerSt<T, F>), ()>
|
) -> Result<
|
||||||
|
(
|
||||||
|
PipelineBinding<F::Service, WorkerMessage<T>>,
|
||||||
|
WorkerSt<T, F>,
|
||||||
|
),
|
||||||
|
(),
|
||||||
|
>
|
||||||
where
|
where
|
||||||
T: Send + 'static,
|
T: Send + 'static,
|
||||||
F: ServiceFactory<WorkerMessage<T>> + 'static,
|
F: ServiceFactory<WorkerMessage<T>> + 'static,
|
||||||
|
@ -317,7 +325,7 @@ where
|
||||||
let mut stop = Box::pin(stop);
|
let mut stop = Box::pin(stop);
|
||||||
|
|
||||||
let svc = match select(factory.create(()), stream_recv(&mut stop)).await {
|
let svc = match select(factory.create(()), stream_recv(&mut stop)).await {
|
||||||
Either::Left(Ok(svc)) => Pipeline::new(svc),
|
Either::Left(Ok(svc)) => Pipeline::new(svc).bind(),
|
||||||
Either::Left(Err(_)) => return Err(()),
|
Either::Left(Err(_)) => return Err(()),
|
||||||
Either::Right(Some(Shutdown { result, .. })) => {
|
Either::Right(Some(Shutdown { result, .. })) => {
|
||||||
log::trace!("Shutdown uninitialized worker");
|
log::trace!("Shutdown uninitialized worker");
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.0.0] - 2024-05-28
|
||||||
|
|
||||||
|
* Use async fn for Service::ready() and Service::shutdown()
|
||||||
|
|
||||||
## [1.1.0] - 2024-03-24
|
## [1.1.0] - 2024-03-24
|
||||||
|
|
||||||
* Move tls connectors from ntex-connect
|
* Move tls connectors from ntex-connect
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tls"
|
name = "ntex-tls"
|
||||||
version = "1.1.0"
|
version = "2.0.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "An implementation of SSL streams for ntex backed by OpenSSL"
|
description = "An implementation of SSL streams for ntex backed by OpenSSL"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -25,11 +25,11 @@ openssl = ["tls_openssl"]
|
||||||
rustls = ["tls_rust"]
|
rustls = ["tls_rust"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.21"
|
ntex-bytes = "0.1"
|
||||||
ntex-io = "1.0"
|
ntex-io = "2.0"
|
||||||
ntex-util = "1.0"
|
ntex-util = "2.0"
|
||||||
ntex-service = "2.0"
|
ntex-service = "3.0"
|
||||||
ntex-net = "1.0"
|
ntex-net = "2.0"
|
||||||
|
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
#![allow(dead_code)]
|
use std::{cell::Cell, future::poll_fn, rc::Rc, task, task::Poll};
|
||||||
use std::{cell::Cell, rc::Rc, task};
|
|
||||||
|
|
||||||
use ntex_util::task::LocalWaker;
|
use ntex_util::task::LocalWaker;
|
||||||
|
|
||||||
|
@ -33,8 +32,15 @@ impl Counter {
|
||||||
|
|
||||||
/// Check if counter is not at capacity. If counter at capacity
|
/// Check if counter is not at capacity. If counter at capacity
|
||||||
/// it registers notification for current task.
|
/// it registers notification for current task.
|
||||||
pub(super) fn available(&self, cx: &mut task::Context<'_>) -> bool {
|
pub(super) async fn available(&self) {
|
||||||
self.0.available(cx)
|
poll_fn(|cx| {
|
||||||
|
if self.0.available(cx) {
|
||||||
|
Poll::Ready(())
|
||||||
|
} else {
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{cell::RefCell, error::Error, fmt, io, task::Context, task::Poll};
|
use std::{cell::RefCell, error::Error, fmt, io};
|
||||||
|
|
||||||
use ntex_io::{Filter, Io, Layer};
|
use ntex_io::{Filter, Io, Layer};
|
||||||
use ntex_service::{Service, ServiceCtx, ServiceFactory};
|
use ntex_service::{Service, ServiceCtx, ServiceFactory};
|
||||||
|
@ -97,12 +97,9 @@ impl<F: Filter> Service<Io<F>> for SslAcceptorService {
|
||||||
type Response = Io<Layer<SslFilter, F>>;
|
type Response = Io<Layer<SslFilter, F>>;
|
||||||
type Error = Box<dyn Error>;
|
type Error = Box<dyn Error>;
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
if self.conns.available(cx) {
|
self.conns.available().await;
|
||||||
Poll::Ready(Ok(()))
|
Ok(())
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(
|
async fn call(
|
||||||
|
|
|
@ -27,12 +27,7 @@ impl<T: Address> SslConnector<T> {
|
||||||
/// Use specified memory pool for memory allocations. By default P0
|
/// Use specified memory pool for memory allocations. By default P0
|
||||||
/// memory pool is used.
|
/// memory pool is used.
|
||||||
pub fn memory_pool(self, id: PoolId) -> Self {
|
pub fn memory_pool(self, id: PoolId) -> Self {
|
||||||
let connector = self
|
let connector = self.connector.get_ref().memory_pool(id).into();
|
||||||
.connector
|
|
||||||
.into_service()
|
|
||||||
.expect("Connector has been cloned")
|
|
||||||
.memory_pool(id)
|
|
||||||
.into();
|
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
connector,
|
connector,
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
use std::task::{Context, Poll};
|
|
||||||
use std::{io, sync::Arc};
|
use std::{io, sync::Arc};
|
||||||
|
|
||||||
use tls_rust::ServerConfig;
|
use tls_rust::ServerConfig;
|
||||||
|
@ -81,12 +80,9 @@ impl<F: Filter> Service<Io<F>> for TlsAcceptorService {
|
||||||
type Response = Io<Layer<TlsServerFilter, F>>;
|
type Response = Io<Layer<TlsServerFilter, F>>;
|
||||||
type Error = io::Error;
|
type Error = io::Error;
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
if self.conns.available(cx) {
|
self.conns.available().await;
|
||||||
Poll::Ready(Ok(()))
|
Ok(())
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(
|
async fn call(
|
||||||
|
|
|
@ -36,12 +36,7 @@ impl<T: Address> TlsConnector<T> {
|
||||||
/// Use specified memory pool for memory allocations. By default P0
|
/// Use specified memory pool for memory allocations. By default P0
|
||||||
/// memory pool is used.
|
/// memory pool is used.
|
||||||
pub fn memory_pool(self, id: PoolId) -> Self {
|
pub fn memory_pool(self, id: PoolId) -> Self {
|
||||||
let connector = self
|
let connector = self.connector.get_ref().memory_pool(id).into();
|
||||||
.connector
|
|
||||||
.into_service()
|
|
||||||
.unwrap()
|
|
||||||
.memory_pool(id)
|
|
||||||
.into();
|
|
||||||
Self {
|
Self {
|
||||||
connector,
|
connector,
|
||||||
config: self.config,
|
config: self.config,
|
||||||
|
@ -146,7 +141,7 @@ mod tests {
|
||||||
.memory_pool(PoolId::P5)
|
.memory_pool(PoolId::P5)
|
||||||
.clone();
|
.clone();
|
||||||
|
|
||||||
let srv = factory.pipeline(&()).await.unwrap();
|
let srv = factory.pipeline(&()).await.unwrap().bind();
|
||||||
// always ready
|
// always ready
|
||||||
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
assert!(lazy(|cx| srv.poll_ready(cx)).await.is_ready());
|
||||||
let result = srv
|
let result = srv
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-tokio"
|
name = "ntex-tokio"
|
||||||
version = "0.4.0"
|
version = "0.5.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "tokio intergration for ntex framework"
|
description = "tokio intergration for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -16,8 +16,8 @@ name = "ntex_tokio"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-bytes = "0.1.21"
|
ntex-bytes = "0.1"
|
||||||
ntex-io = "1.0.0"
|
ntex-io = "2.0"
|
||||||
ntex-util = "1.0.0"
|
ntex-util = "2.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
tokio = { version = "1", default-features = false, features = ["rt", "net", "sync", "signal"] }
|
tokio = { version = "1", default-features = false, features = ["rt", "net", "sync", "signal"] }
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.0.0] - 2024-05-28
|
||||||
|
|
||||||
|
* Use async fn for Service::ready() and Service::shutdown()
|
||||||
|
|
||||||
## [1.1.0] - 2024-04-xx
|
## [1.1.0] - 2024-04-xx
|
||||||
|
|
||||||
* Change Extensions::insert() method according doc #345
|
* Change Extensions::insert() method according doc #345
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-util"
|
name = "ntex-util"
|
||||||
version = "1.1.0"
|
version = "2.0.0"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Utilities for ntex framework"
|
description = "Utilities for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -16,7 +16,7 @@ name = "ntex_util"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
ntex-service = "2.0"
|
ntex-service = "3.0"
|
||||||
ntex-rt = "0.4"
|
ntex-rt = "0.4"
|
||||||
bitflags = "2.4"
|
bitflags = "2.4"
|
||||||
fxhash = "0.2.1"
|
fxhash = "0.2.1"
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
//! Service that buffers incomming requests.
|
//! Service that buffers incomming requests.
|
||||||
use std::cell::{Cell, RefCell};
|
use std::cell::{Cell, RefCell};
|
||||||
use std::task::{ready, Context, Poll};
|
use std::task::{ready, Poll};
|
||||||
use std::{collections::VecDeque, fmt, marker::PhantomData};
|
use std::{collections::VecDeque, fmt, future::poll_fn, marker::PhantomData};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service, ServiceCtx};
|
use ntex_service::{Middleware, Service, ServiceCtx};
|
||||||
|
|
||||||
use crate::channel::oneshot;
|
use crate::channel::oneshot;
|
||||||
|
|
||||||
|
@ -121,15 +121,12 @@ impl<R, S> BufferService<R, S>
|
||||||
where
|
where
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
pub fn new<U>(size: usize, service: U) -> Self
|
pub fn new(size: usize, service: S) -> Self {
|
||||||
where
|
|
||||||
U: IntoService<S, R>,
|
|
||||||
{
|
|
||||||
Self {
|
Self {
|
||||||
size,
|
size,
|
||||||
|
service,
|
||||||
cancel_on_shutdown: false,
|
cancel_on_shutdown: false,
|
||||||
ready: Cell::new(false),
|
ready: Cell::new(false),
|
||||||
service: service.into_service(),
|
|
||||||
buf: RefCell::new(VecDeque::with_capacity(size)),
|
buf: RefCell::new(VecDeque::with_capacity(size)),
|
||||||
next_call: RefCell::default(),
|
next_call: RefCell::default(),
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
|
@ -185,7 +182,7 @@ where
|
||||||
type Error = BufferServiceError<S::Error>;
|
type Error = BufferServiceError<S::Error>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
let mut buffer = self.buf.borrow_mut();
|
let mut buffer = self.buf.borrow_mut();
|
||||||
let mut next_call = self.next_call.borrow_mut();
|
let mut next_call = self.next_call.borrow_mut();
|
||||||
if let Some(next_call) = &*next_call {
|
if let Some(next_call) = &*next_call {
|
||||||
|
@ -220,42 +217,45 @@ where
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_shutdown(&self, cx: &mut std::task::Context<'_>) -> Poll<()> {
|
async fn shutdown(&self) {
|
||||||
let mut buffer = self.buf.borrow_mut();
|
poll_fn(|cx| {
|
||||||
if self.cancel_on_shutdown {
|
let mut buffer = self.buf.borrow_mut();
|
||||||
buffer.clear();
|
if self.cancel_on_shutdown {
|
||||||
} else if !buffer.is_empty() {
|
buffer.clear();
|
||||||
let mut next_call = self.next_call.borrow_mut();
|
} else if !buffer.is_empty() {
|
||||||
if let Some(next_call) = &*next_call {
|
let mut next_call = self.next_call.borrow_mut();
|
||||||
// hold advancement until the last released task either makes a call or is dropped
|
if let Some(next_call) = &*next_call {
|
||||||
let _ = ready!(next_call.poll_recv(cx));
|
// hold advancement until the last released task either makes a call or is dropped
|
||||||
}
|
let _ = ready!(next_call.poll_recv(cx));
|
||||||
next_call.take();
|
|
||||||
|
|
||||||
if ready!(self.service.poll_ready(cx)).is_err() {
|
|
||||||
log::error!(
|
|
||||||
"Buffered inner service failed while buffer flushing on shutdown"
|
|
||||||
);
|
|
||||||
return Poll::Ready(());
|
|
||||||
}
|
|
||||||
|
|
||||||
while let Some(sender) = buffer.pop_front() {
|
|
||||||
let (next_call_tx, next_call_rx) = oneshot::channel();
|
|
||||||
if sender.send(next_call_tx).is_err()
|
|
||||||
|| next_call_rx.poll_recv(cx).is_ready()
|
|
||||||
{
|
|
||||||
// the task is gone
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
next_call.replace(next_call_rx);
|
next_call.take();
|
||||||
if buffer.is_empty() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return Poll::Pending;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.service.poll_shutdown(cx)
|
if ready!(self.service.poll_ready(cx)).is_err() {
|
||||||
|
log::error!(
|
||||||
|
"Buffered inner service failed while buffer flushing on shutdown"
|
||||||
|
);
|
||||||
|
return Poll::Ready(());
|
||||||
|
}
|
||||||
|
|
||||||
|
while let Some(sender) = buffer.pop_front() {
|
||||||
|
let (next_call_tx, next_call_rx) = oneshot::channel();
|
||||||
|
if sender.send(next_call_tx).is_err()
|
||||||
|
|| next_call_rx.poll_recv(cx).is_ready()
|
||||||
|
{
|
||||||
|
// the task is gone
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
next_call.replace(next_call_rx);
|
||||||
|
if buffer.is_empty() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return Poll::Pending;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
self.service.shutdown().await;
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(
|
async fn call(
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{cell::Cell, rc::Rc, task};
|
use std::{cell::Cell, future::poll_fn, rc::Rc, task};
|
||||||
|
|
||||||
use crate::task::LocalWaker;
|
use crate::task::LocalWaker;
|
||||||
|
|
||||||
|
@ -32,7 +32,20 @@ impl Counter {
|
||||||
|
|
||||||
/// Check if counter is not at capacity. If counter at capacity
|
/// Check if counter is not at capacity. If counter at capacity
|
||||||
/// it registers notification for current task.
|
/// it registers notification for current task.
|
||||||
pub fn available(&self, cx: &mut task::Context<'_>) -> bool {
|
pub async fn available(&self) {
|
||||||
|
poll_fn(|cx| {
|
||||||
|
if self.poll_available(cx) {
|
||||||
|
task::Poll::Ready(())
|
||||||
|
} else {
|
||||||
|
task::Poll::Pending
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if counter is not at capacity. If counter at capacity
|
||||||
|
/// it registers notification for current task.
|
||||||
|
pub fn poll_available(&self, cx: &mut task::Context<'_>) -> bool {
|
||||||
self.0.available(cx)
|
self.0.available(cx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
//! Service that limits number of in-flight async requests.
|
//! Service that limits number of in-flight async requests.
|
||||||
use std::{task::Context, task::Poll};
|
use ntex_service::{Middleware, Service, ServiceCtx};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service, ServiceCtx};
|
|
||||||
|
|
||||||
use super::counter::Counter;
|
use super::counter::Counter;
|
||||||
|
|
||||||
|
@ -44,14 +42,13 @@ pub struct InFlightService<S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> InFlightService<S> {
|
impl<S> InFlightService<S> {
|
||||||
pub fn new<U, R>(max: usize, service: U) -> Self
|
pub fn new<R>(max: usize, service: S) -> Self
|
||||||
where
|
where
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
U: IntoService<S, R>,
|
|
||||||
{
|
{
|
||||||
Self {
|
Self {
|
||||||
|
service,
|
||||||
count: Counter::new(max),
|
count: Counter::new(max),
|
||||||
service: service.into_service(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -64,15 +61,9 @@ where
|
||||||
type Error = T::Error;
|
type Error = T::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
if self.service.poll_ready(cx)?.is_pending() {
|
self.count.available().await;
|
||||||
Poll::Pending
|
ctx.ready(&self.service).await
|
||||||
} else if !self.count.available(cx) {
|
|
||||||
log::trace!("InFlight limit exceeded");
|
|
||||||
Poll::Pending
|
|
||||||
} else {
|
|
||||||
Poll::Ready(Ok(()))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -85,13 +76,14 @@ where
|
||||||
ctx.call(&self.service, req).await
|
ctx.call(&self.service, req).await
|
||||||
}
|
}
|
||||||
|
|
||||||
ntex_service::forward_poll_shutdown!(service);
|
ntex_service::forward_shutdown!(service);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
|
use std::{cell::RefCell, task::Poll, time::Duration};
|
||||||
|
|
||||||
use ntex_service::{apply, fn_factory, Pipeline, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Pipeline, ServiceFactory};
|
||||||
use std::{cell::RefCell, time::Duration};
|
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{channel::oneshot, future::lazy};
|
use crate::{channel::oneshot, future::lazy};
|
||||||
|
@ -112,7 +104,7 @@ mod tests {
|
||||||
async fn test_service() {
|
async fn test_service() {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
|
|
||||||
let srv = Pipeline::new(InFlightService::new(1, SleepService(rx)));
|
let srv = Pipeline::new(InFlightService::new(1, SleepService(rx))).bind();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let srv2 = srv.clone();
|
let srv2 = srv.clone();
|
||||||
|
@ -125,7 +117,7 @@ mod tests {
|
||||||
let _ = tx.send(());
|
let _ = tx.send(());
|
||||||
crate::time::sleep(Duration::from_millis(25)).await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
assert_eq!(srv.shutdown().await, ());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
|
@ -146,7 +138,7 @@ mod tests {
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
let srv = srv.pipeline(&()).await.unwrap();
|
let srv = srv.pipeline(&()).await.unwrap().bind();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let srv2 = srv.clone();
|
let srv2 = srv.clone();
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
use std::task::{Context, Poll};
|
use std::{cell::Cell, convert::Infallible, fmt, future::poll_fn, marker, task, time};
|
||||||
use std::{cell::Cell, convert::Infallible, fmt, marker, time::Duration, time::Instant};
|
|
||||||
|
|
||||||
use ntex_service::{Service, ServiceCtx, ServiceFactory};
|
use ntex_service::{Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
|
@ -73,7 +72,7 @@ pub struct KeepAliveService<R, E, F> {
|
||||||
f: F,
|
f: F,
|
||||||
dur: Millis,
|
dur: Millis,
|
||||||
sleep: Sleep,
|
sleep: Sleep,
|
||||||
expire: Cell<Instant>,
|
expire: Cell<time::Instant>,
|
||||||
_t: marker::PhantomData<(R, E)>,
|
_t: marker::PhantomData<(R, E)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -111,23 +110,24 @@ where
|
||||||
type Response = R;
|
type Response = R;
|
||||||
type Error = E;
|
type Error = E;
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
match self.sleep.poll_elapsed(cx) {
|
poll_fn(|cx| match self.sleep.poll_elapsed(cx) {
|
||||||
Poll::Ready(_) => {
|
task::Poll::Ready(_) => {
|
||||||
let now = now();
|
let now = now();
|
||||||
let expire = self.expire.get() + Duration::from(self.dur);
|
let expire = self.expire.get() + time::Duration::from(self.dur);
|
||||||
if expire <= now {
|
if expire <= now {
|
||||||
Poll::Ready(Err((self.f)()))
|
task::Poll::Ready(Err((self.f)()))
|
||||||
} else {
|
} else {
|
||||||
let expire = expire - now;
|
let expire = expire - now;
|
||||||
self.sleep
|
self.sleep
|
||||||
.reset(Millis(expire.as_millis().try_into().unwrap_or(u32::MAX)));
|
.reset(Millis(expire.as_millis().try_into().unwrap_or(u32::MAX)));
|
||||||
let _ = self.sleep.poll_elapsed(cx);
|
let _ = self.sleep.poll_elapsed(cx);
|
||||||
Poll::Ready(Ok(()))
|
task::Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Poll::Pending => Poll::Ready(Ok(())),
|
task::Poll::Pending => task::Poll::Ready(Ok(())),
|
||||||
}
|
})
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(&self, req: R, _: ServiceCtx<'_, Self>) -> Result<R, E> {
|
async fn call(&self, req: R, _: ServiceCtx<'_, Self>) -> Result<R, E> {
|
||||||
|
@ -138,6 +138,8 @@ where
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
|
use std::task::Poll;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::lazy;
|
use crate::future::lazy;
|
||||||
|
|
||||||
|
@ -150,7 +152,7 @@ mod tests {
|
||||||
assert!(format!("{:?}", factory).contains("KeepAlive"));
|
assert!(format!("{:?}", factory).contains("KeepAlive"));
|
||||||
let _ = factory.clone();
|
let _ = factory.clone();
|
||||||
|
|
||||||
let service = factory.pipeline(&()).await.unwrap();
|
let service = factory.pipeline(&()).await.unwrap().bind();
|
||||||
assert!(format!("{:?}", service).contains("KeepAliveService"));
|
assert!(format!("{:?}", service).contains("KeepAliveService"));
|
||||||
|
|
||||||
assert_eq!(service.call(1usize).await, Ok(1usize));
|
assert_eq!(service.call(1usize).await, Ok(1usize));
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
pub mod buffer;
|
// pub mod buffer;
|
||||||
pub mod counter;
|
pub mod counter;
|
||||||
mod extensions;
|
mod extensions;
|
||||||
pub mod inflight;
|
pub mod inflight;
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! Service that limits number of in-flight async requests to 1.
|
//! Service that limits number of in-flight async requests to 1.
|
||||||
use std::{cell::Cell, task::Context, task::Poll};
|
use std::{cell::Cell, future::poll_fn, task::Poll};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service, ServiceCtx};
|
use ntex_service::{Middleware, Service, ServiceCtx};
|
||||||
|
|
||||||
use crate::task::LocalWaker;
|
use crate::task::LocalWaker;
|
||||||
|
|
||||||
|
@ -30,13 +30,12 @@ pub struct OneRequestService<S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> OneRequestService<S> {
|
impl<S> OneRequestService<S> {
|
||||||
pub fn new<U, R>(service: U) -> Self
|
pub fn new<R>(service: S) -> Self
|
||||||
where
|
where
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
U: IntoService<S, R>,
|
|
||||||
{
|
{
|
||||||
Self {
|
Self {
|
||||||
service: service.into_service(),
|
service,
|
||||||
ready: Cell::new(true),
|
ready: Cell::new(true),
|
||||||
waker: LocalWaker::new(),
|
waker: LocalWaker::new(),
|
||||||
}
|
}
|
||||||
|
@ -51,15 +50,18 @@ where
|
||||||
type Error = T::Error;
|
type Error = T::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
self.waker.register(cx.waker());
|
poll_fn(|cx| {
|
||||||
if self.service.poll_ready(cx)?.is_pending() {
|
self.waker.register(cx.waker());
|
||||||
Poll::Pending
|
if self.ready.get() {
|
||||||
} else if self.ready.get() {
|
Poll::Ready(())
|
||||||
Poll::Ready(Ok(()))
|
} else {
|
||||||
} else {
|
Poll::Pending
|
||||||
Poll::Pending
|
}
|
||||||
}
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
ctx.ready(&self.service).await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -76,7 +78,7 @@ where
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
ntex_service::forward_poll_shutdown!(service);
|
ntex_service::forward_shutdown!(service);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -103,7 +105,7 @@ mod tests {
|
||||||
async fn test_oneshot() {
|
async fn test_oneshot() {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
|
|
||||||
let srv = Pipeline::new(OneRequestService::new(SleepService(rx)));
|
let srv = Pipeline::new(OneRequestService::new(SleepService(rx))).bind();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let srv2 = srv.clone();
|
let srv2 = srv.clone();
|
||||||
|
@ -116,7 +118,7 @@ mod tests {
|
||||||
let _ = tx.send(());
|
let _ = tx.send(());
|
||||||
crate::time::sleep(Duration::from_millis(25)).await;
|
crate::time::sleep(Duration::from_millis(25)).await;
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
assert!(lazy(|cx| srv.poll_shutdown(cx)).await.is_ready());
|
assert_eq!(srv.shutdown().await, ());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
|
@ -133,7 +135,7 @@ mod tests {
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
let srv = srv.pipeline(&()).await.unwrap();
|
let srv = srv.pipeline(&()).await.unwrap().bind();
|
||||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||||
|
|
||||||
let srv2 = srv.clone();
|
let srv2 = srv.clone();
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
//! will be aborted.
|
//! will be aborted.
|
||||||
use std::{fmt, marker};
|
use std::{fmt, marker};
|
||||||
|
|
||||||
use ntex_service::{IntoService, Middleware, Service, ServiceCtx};
|
use ntex_service::{Middleware, Service, ServiceCtx};
|
||||||
|
|
||||||
use crate::future::{select, Either};
|
use crate::future::{select, Either};
|
||||||
use crate::time::{sleep, Millis};
|
use crate::time::{sleep, Millis};
|
||||||
|
@ -104,15 +104,14 @@ pub struct TimeoutService<S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> TimeoutService<S> {
|
impl<S> TimeoutService<S> {
|
||||||
pub fn new<T, U, R>(timeout: T, service: U) -> Self
|
pub fn new<T, R>(timeout: T, service: S) -> Self
|
||||||
where
|
where
|
||||||
T: Into<Millis>,
|
T: Into<Millis>,
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
U: IntoService<S, R>,
|
|
||||||
{
|
{
|
||||||
TimeoutService {
|
TimeoutService {
|
||||||
|
service,
|
||||||
timeout: timeout.into(),
|
timeout: timeout.into(),
|
||||||
service: service.into_service(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -141,8 +140,8 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ntex_service::forward_poll_ready!(service, TimeoutError::Service);
|
ntex_service::forward_ready!(service, TimeoutError::Service);
|
||||||
ntex_service::forward_poll_shutdown!(service);
|
ntex_service::forward_shutdown!(service);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -152,7 +151,6 @@ mod tests {
|
||||||
use ntex_service::{apply, fn_factory, Pipeline, ServiceFactory};
|
use ntex_service::{apply, fn_factory, Pipeline, ServiceFactory};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::lazy;
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
struct SleepService(Duration);
|
struct SleepService(Duration);
|
||||||
|
@ -184,8 +182,8 @@ mod tests {
|
||||||
let timeout =
|
let timeout =
|
||||||
Pipeline::new(TimeoutService::new(resolution, SleepService(wait_time)).clone());
|
Pipeline::new(TimeoutService::new(resolution, SleepService(wait_time)).clone());
|
||||||
assert_eq!(timeout.call(()).await, Ok(()));
|
assert_eq!(timeout.call(()).await, Ok(()));
|
||||||
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
assert_eq!(timeout.ready().await, Ok(()));
|
||||||
assert!(lazy(|cx| timeout.poll_shutdown(cx)).await.is_ready());
|
assert_eq!(timeout.shutdown().await, ());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
|
@ -196,7 +194,7 @@ mod tests {
|
||||||
let timeout =
|
let timeout =
|
||||||
Pipeline::new(TimeoutService::new(resolution, SleepService(wait_time)));
|
Pipeline::new(TimeoutService::new(resolution, SleepService(wait_time)));
|
||||||
assert_eq!(timeout.call(()).await, Ok(()));
|
assert_eq!(timeout.call(()).await, Ok(()));
|
||||||
assert!(lazy(|cx| timeout.poll_ready(cx)).await.is_ready());
|
assert_eq!(timeout.ready().await, Ok(()));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[ntex_macros::rt_test2]
|
#[ntex_macros::rt_test2]
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
//! Contains `Variant` service and related types and functions.
|
//! Contains `Variant` service and related types and functions.
|
||||||
use std::{fmt, marker::PhantomData, task::Context, task::Poll};
|
#![allow(non_snake_case)]
|
||||||
|
use std::{fmt, marker::PhantomData, task::Poll};
|
||||||
|
|
||||||
use ntex_service::{IntoServiceFactory, Service, ServiceCtx, ServiceFactory};
|
use ntex_service::{IntoServiceFactory, Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
|
@ -70,7 +71,7 @@ macro_rules! variant_impl_and ({$fac1_type:ident, $fac2_type:ident, $name:ident,
|
||||||
Response = V1::Response,
|
Response = V1::Response,
|
||||||
Error = V1::Error,
|
Error = V1::Error,
|
||||||
InitError = V1::InitError>,
|
InitError = V1::InitError>,
|
||||||
F: IntoServiceFactory<$name, $r_name, V1C>,
|
F: IntoServiceFactory<$name, $r_name, V1C>,
|
||||||
{
|
{
|
||||||
$fac2_type {
|
$fac2_type {
|
||||||
V1: self.V1,
|
V1: self.V1,
|
||||||
|
@ -124,30 +125,30 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident,
|
||||||
type Response = V1::Response;
|
type Response = V1::Response;
|
||||||
type Error = V1::Error;
|
type Error = V1::Error;
|
||||||
|
|
||||||
fn poll_ready(&self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
let mut ready = self.V1.poll_ready(cx)?.is_ready();
|
use std::{future::Future, pin::Pin};
|
||||||
$(ready = self.$T.poll_ready(cx)?.is_ready() && ready;)+
|
|
||||||
|
|
||||||
if ready {
|
let mut fut1 = ::std::pin::pin!(ctx.ready(&self.V1));
|
||||||
Poll::Ready(Ok(()))
|
$(let mut $T = ::std::pin::pin!(ctx.ready(&self.$T));)+
|
||||||
} else {
|
|
||||||
Poll::Pending
|
::std::future::poll_fn(|cx| {
|
||||||
}
|
let mut ready = Pin::new(&mut fut1).poll(cx)?.is_ready();
|
||||||
|
$(ready = Pin::new(&mut $T).poll(cx)?.is_ready() && ready;)+
|
||||||
|
|
||||||
|
if ready {
|
||||||
|
Poll::Ready(Ok(()))
|
||||||
|
} else {
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
}).await
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_shutdown(&self, cx: &mut Context<'_>) -> Poll<()> {
|
async fn shutdown(&self) {
|
||||||
let mut ready = self.V1.poll_shutdown(cx).is_ready();
|
self.V1.shutdown().await;
|
||||||
$(ready = self.$T.poll_shutdown(cx).is_ready() && ready;)+
|
$(self.$T.shutdown().await;)+
|
||||||
|
|
||||||
if ready {
|
|
||||||
Poll::Ready(())
|
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn call(&self, req: $enum_type<V1R, $($R,)+>, ctx: ServiceCtx<'_, Self>) -> Result<Self::Response, Self::Error>
|
async fn call(&self, req: $enum_type<V1R, $($R,)+>, ctx: ServiceCtx<'_, Self>) -> Result<Self::Response, Self::Error> {
|
||||||
{
|
|
||||||
match req {
|
match req {
|
||||||
$enum_type::V1(req) => ctx.call(&self.V1, req).await,
|
$enum_type::V1(req) => ctx.call(&self.V1, req).await,
|
||||||
$($enum_type::$T(req) => ctx.call(&self.$T, req).await,)+
|
$($enum_type::$T(req) => ctx.call(&self.$T, req).await,)+
|
||||||
|
@ -235,7 +236,6 @@ mod tests {
|
||||||
use ntex_service::fn_factory;
|
use ntex_service::fn_factory;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::future::lazy;
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct Srv1;
|
struct Srv1;
|
||||||
|
@ -244,13 +244,11 @@ mod tests {
|
||||||
type Response = usize;
|
type Response = usize;
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
fn poll_ready(&self, _: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
Poll::Ready(Ok(()))
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_shutdown(&self, _: &mut Context<'_>) -> Poll<()> {
|
async fn shutdown(&self) {}
|
||||||
Poll::Ready(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<usize, ()> {
|
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<usize, ()> {
|
||||||
Ok(1)
|
Ok(1)
|
||||||
|
@ -264,13 +262,11 @@ mod tests {
|
||||||
type Response = usize;
|
type Response = usize;
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
fn poll_ready(&self, _: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
Poll::Ready(Ok(()))
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_shutdown(&self, _: &mut Context<'_>) -> Poll<()> {
|
async fn shutdown(&self) {}
|
||||||
Poll::Ready(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<usize, ()> {
|
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<usize, ()> {
|
||||||
Ok(2)
|
Ok(2)
|
||||||
|
@ -286,8 +282,8 @@ mod tests {
|
||||||
.clone();
|
.clone();
|
||||||
let service = factory.pipeline(&()).await.unwrap().clone();
|
let service = factory.pipeline(&()).await.unwrap().clone();
|
||||||
|
|
||||||
assert!(lazy(|cx| service.poll_ready(cx)).await.is_ready());
|
assert!(service.ready().await.is_ok());
|
||||||
assert!(lazy(|cx| service.poll_shutdown(cx)).await.is_ready());
|
assert_eq!(service.shutdown().await, ());
|
||||||
|
|
||||||
assert_eq!(service.call(Variant3::V1(())).await, Ok(1));
|
assert_eq!(service.call(Variant3::V1(())).await, Ok(1));
|
||||||
assert_eq!(service.call(Variant3::V2(())).await, Ok(2));
|
assert_eq!(service.call(Variant3::V2(())).await, Ok(2));
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue