mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 04:47:39 +03:00
Synchronized buffer (#216)
* added a new function to servicecall to allow only advancing to the service call and then returning the service response future * buffer can maintain order/backpressure by implementing strict readiness and synchronous calling * buffer can flush in order or cancel pending buffered futures on shutdown
This commit is contained in:
parent
4380b3a155
commit
eea5b3b539
7 changed files with 260 additions and 40 deletions
|
@ -1,5 +1,9 @@
|
|||
# Changes
|
||||
|
||||
## [1.2.2] - 2023-06-24
|
||||
|
||||
* Added `ServiceCall::advance_to_call`
|
||||
|
||||
## [1.2.1] - 2023-06-23
|
||||
|
||||
* Make `PipelineCall` static
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
[package]
|
||||
name = "ntex-service"
|
||||
version = "1.2.1"
|
||||
version = "1.2.2"
|
||||
authors = ["ntex contributors <team@ntex.rs>"]
|
||||
description = "ntex service"
|
||||
keywords = ["network", "framework", "async", "futures"]
|
||||
|
|
|
@ -166,6 +166,26 @@ pin_project_lite::pin_project! {
|
|||
}
|
||||
}
|
||||
|
||||
impl<'a, S, Req> ServiceCall<'a, S, Req>
|
||||
where
|
||||
S: Service<Req>,
|
||||
S: 'a,
|
||||
S: ?Sized,
|
||||
Req: 'a,
|
||||
{
|
||||
pub fn advance_to_call(self) -> ServiceCallToCall<'a, S, Req> {
|
||||
match self.state {
|
||||
ServiceCallState::Ready { .. } => {}
|
||||
ServiceCallState::Call { .. } | ServiceCallState::Empty => {
|
||||
panic!(
|
||||
"`ServiceCall::advance_to_call` must be called before `ServiceCall::poll`"
|
||||
)
|
||||
}
|
||||
}
|
||||
ServiceCallToCall { state: self.state }
|
||||
}
|
||||
}
|
||||
|
||||
pin_project_lite::pin_project! {
|
||||
#[project = ServiceCallStateProject]
|
||||
enum ServiceCallState<'a, S, Req>
|
||||
|
@ -234,6 +254,68 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
pin_project_lite::pin_project! {
|
||||
#[must_use = "futures do nothing unless polled"]
|
||||
pub struct ServiceCallToCall<'a, S, Req>
|
||||
where
|
||||
S: Service<Req>,
|
||||
S: 'a,
|
||||
S: ?Sized,
|
||||
Req: 'a,
|
||||
{
|
||||
#[pin]
|
||||
state: ServiceCallState<'a, S, Req>,
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, S, Req> Future for ServiceCallToCall<'a, S, Req>
|
||||
where
|
||||
S: Service<Req> + ?Sized,
|
||||
{
|
||||
type Output = Result<S::Future<'a>, S::Error>;
|
||||
|
||||
fn poll(
|
||||
mut self: Pin<&mut Self>,
|
||||
cx: &mut task::Context<'_>,
|
||||
) -> task::Poll<Self::Output> {
|
||||
let mut this = self.as_mut().project();
|
||||
|
||||
match this.state.as_mut().project() {
|
||||
ServiceCallStateProject::Ready {
|
||||
req,
|
||||
svc,
|
||||
idx,
|
||||
waiters,
|
||||
} => match svc.poll_ready(cx)? {
|
||||
task::Poll::Ready(()) => {
|
||||
waiters.notify();
|
||||
|
||||
let fut = svc.call(
|
||||
req.take().unwrap(),
|
||||
ServiceCtx {
|
||||
idx: *idx,
|
||||
waiters,
|
||||
_t: marker::PhantomData,
|
||||
},
|
||||
);
|
||||
this.state.set(ServiceCallState::Empty);
|
||||
task::Poll::Ready(Ok(fut))
|
||||
}
|
||||
task::Poll::Pending => {
|
||||
waiters.register(*idx, cx);
|
||||
task::Poll::Pending
|
||||
}
|
||||
},
|
||||
ServiceCallStateProject::Call { .. } => {
|
||||
unreachable!("`ServiceCallToCall` can only be constructed in `Ready` state")
|
||||
}
|
||||
ServiceCallStateProject::Empty => {
|
||||
panic!("future must not be polled after it returned `Poll::Ready`")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use ntex_util::future::{lazy, poll_fn, Ready};
|
||||
|
|
|
@ -24,7 +24,7 @@ mod then;
|
|||
|
||||
pub use self::apply::{apply_fn, apply_fn_factory};
|
||||
pub use self::chain::{chain, chain_factory};
|
||||
pub use self::ctx::{ServiceCall, ServiceCtx};
|
||||
pub use self::ctx::{ServiceCall, ServiceCallToCall, ServiceCtx};
|
||||
pub use self::fn_service::{fn_factory, fn_factory_with_config, fn_service};
|
||||
pub use self::fn_shutdown::fn_shutdown;
|
||||
pub use self::map_config::{map_config, unit_config};
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue