mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
refactor
This commit is contained in:
parent
595c7332e6
commit
5697be0f81
15 changed files with 222 additions and 254 deletions
|
@ -18,6 +18,7 @@ path = "src/lib.rs"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
slab = "0.4"
|
slab = "0.4"
|
||||||
bitflags = "2"
|
bitflags = "2"
|
||||||
|
pin-project-lite = "0.2"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
ntex = "2"
|
ntex = "2"
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use std::future::Future;
|
||||||
|
|
||||||
use super::{util, Service, ServiceCtx, ServiceFactory};
|
use super::{util, Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
@ -26,13 +28,8 @@ where
|
||||||
type Error = A::Error;
|
type Error = A::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
util::ready(&self.svc1, &self.svc2, ctx).await
|
util::ready(&self.svc1, &self.svc2).await
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
util::unready(&self.svc1, &self.svc2).await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -100,13 +100,11 @@ where
|
||||||
type Error = Err;
|
type Error = Err;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, _: ServiceCtx<'_, Self>) -> Result<(), Err> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Err>>> {
|
||||||
self.service.ready().await.map_err(From::from)
|
self.service
|
||||||
}
|
.ready()
|
||||||
|
.await
|
||||||
#[inline]
|
.map(|fut| async move { fut.await.map_err(From::from) })
|
||||||
async fn unready(&self) -> Result<(), Err> {
|
|
||||||
self.service.unready().await.map_err(From::from)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -3,6 +3,7 @@ use std::{fmt, future::Future, pin::Pin};
|
||||||
use crate::ctx::{ServiceCtx, WaitersRef};
|
use crate::ctx::{ServiceCtx, WaitersRef};
|
||||||
|
|
||||||
type BoxFuture<'a, I, E> = Pin<Box<dyn Future<Output = Result<I, E>> + 'a>>;
|
type BoxFuture<'a, I, E> = Pin<Box<dyn Future<Output = Result<I, E>> + 'a>>;
|
||||||
|
type BoxFutureOne<'a, T> = Pin<Box<dyn Future<Output = T> + 'a>>;
|
||||||
pub struct BoxService<Req, Res, Err>(Box<dyn ServiceObj<Req, Response = Res, Error = Err>>);
|
pub struct BoxService<Req, Res, Err>(Box<dyn ServiceObj<Req, Response = Res, Error = Err>>);
|
||||||
pub struct BoxServiceFactory<Cfg, Req, Res, Err, InitErr>(
|
pub struct BoxServiceFactory<Cfg, Req, Res, Err, InitErr>(
|
||||||
Box<dyn ServiceFactoryObj<Req, Cfg, Response = Res, Error = Err, InitError = InitErr>>,
|
Box<dyn ServiceFactoryObj<Req, Cfg, Response = Res, Error = Err, InitError = InitErr>>,
|
||||||
|
@ -48,14 +49,7 @@ trait ServiceObj<Req> {
|
||||||
type Response;
|
type Response;
|
||||||
type Error;
|
type Error;
|
||||||
|
|
||||||
fn ready<'a>(
|
fn ready<'a>(&'a self) -> BoxFutureOne<'a, Option<BoxFuture<'a, (), Self::Error>>>;
|
||||||
&'a self,
|
|
||||||
idx: u32,
|
|
||||||
bound: bool,
|
|
||||||
waiters: &'a WaitersRef,
|
|
||||||
) -> BoxFuture<'a, (), Self::Error>;
|
|
||||||
|
|
||||||
fn unready(&self) -> BoxFuture<'_, (), Self::Error>;
|
|
||||||
|
|
||||||
fn call<'a>(
|
fn call<'a>(
|
||||||
&'a self,
|
&'a self,
|
||||||
|
@ -76,24 +70,17 @@ where
|
||||||
type Error = S::Error;
|
type Error = S::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn ready<'a>(
|
fn ready<'a>(&'a self) -> BoxFutureOne<'a, Option<BoxFuture<'a, (), Self::Error>>> {
|
||||||
&'a self,
|
|
||||||
idx: u32,
|
|
||||||
bound: bool,
|
|
||||||
waiters: &'a WaitersRef,
|
|
||||||
) -> BoxFuture<'a, (), Self::Error> {
|
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
ServiceCtx::<'a, S>::from_ref(idx, bound, waiters)
|
if let Some(fut) = self.ready().await {
|
||||||
.ready(self)
|
let r: BoxFuture<'a, (), Self::Error> = Box::pin(fut);
|
||||||
.await
|
Some(r)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn unready(&self) -> BoxFuture<'_, (), Self::Error> {
|
|
||||||
Box::pin(crate::Service::unready(self))
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn shutdown<'a>(&'a self) -> Pin<Box<dyn Future<Output = ()> + 'a>> {
|
fn shutdown<'a>(&'a self) -> Pin<Box<dyn Future<Output = ()> + 'a>> {
|
||||||
Box::pin(crate::Service::shutdown(self))
|
Box::pin(crate::Service::shutdown(self))
|
||||||
|
@ -107,7 +94,7 @@ where
|
||||||
waiters: &'a WaitersRef,
|
waiters: &'a WaitersRef,
|
||||||
) -> BoxFuture<'a, Self::Response, Self::Error> {
|
) -> BoxFuture<'a, Self::Response, Self::Error> {
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
ServiceCtx::<'a, S>::from_ref(idx, false, waiters)
|
ServiceCtx::<'a, S>::from_ref(idx, waiters)
|
||||||
.call_nowait(self, req)
|
.call_nowait(self, req)
|
||||||
.await
|
.await
|
||||||
})
|
})
|
||||||
|
@ -159,14 +146,8 @@ where
|
||||||
type Error = Err;
|
type Error = Err;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
let (idx, bound, waiters) = ctx.inner();
|
self.0.ready().await
|
||||||
self.0.ready(idx, bound, waiters).await
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
self.0.unready().await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -176,7 +157,7 @@ where
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn call(&self, req: Req, ctx: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
async fn call(&self, req: Req, ctx: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
||||||
let (idx, _, waiters) = ctx.inner();
|
let (idx, waiters) = ctx.inner();
|
||||||
self.0.call(req, idx, waiters).await
|
self.0.call(req, idx, waiters).await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -172,7 +172,6 @@ impl<Svc: Service<Req>, Req> Service<Req> for ServiceChain<Svc, Req> {
|
||||||
type Error = Svc::Error;
|
type Error = Svc::Error;
|
||||||
|
|
||||||
crate::forward_ready!(service);
|
crate::forward_ready!(service);
|
||||||
crate::forward_unready!(service);
|
|
||||||
crate::forward_shutdown!(service);
|
crate::forward_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -5,7 +5,6 @@ use crate::Service;
|
||||||
|
|
||||||
pub struct ServiceCtx<'a, S: ?Sized> {
|
pub struct ServiceCtx<'a, S: ?Sized> {
|
||||||
idx: u32,
|
idx: u32,
|
||||||
bound: bool,
|
|
||||||
waiters: &'a WaitersRef,
|
waiters: &'a WaitersRef,
|
||||||
_t: marker::PhantomData<Rc<S>>,
|
_t: marker::PhantomData<Rc<S>>,
|
||||||
}
|
}
|
||||||
|
@ -78,6 +77,10 @@ impl WaitersRef {
|
||||||
self.state.get().contains(Flags::READY)
|
self.state.get().contains(Flags::READY)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn is_bound(&self) -> bool {
|
||||||
|
self.state.get().contains(Flags::BOUND)
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn bind(&self) {
|
pub(crate) fn bind(&self) {
|
||||||
self.state.set(Flags::BOUND);
|
self.state.set(Flags::BOUND);
|
||||||
self.notify();
|
self.notify();
|
||||||
|
@ -168,32 +171,21 @@ impl<'a, S> ServiceCtx<'a, S> {
|
||||||
pub(crate) fn new(waiters: &'a Waiters) -> Self {
|
pub(crate) fn new(waiters: &'a Waiters) -> Self {
|
||||||
Self {
|
Self {
|
||||||
idx: waiters.index,
|
idx: waiters.index,
|
||||||
bound: false,
|
|
||||||
waiters: waiters.get_ref(),
|
waiters: waiters.get_ref(),
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn new_bound(waiters: &'a Waiters) -> Self {
|
pub(crate) fn from_ref(idx: u32, waiters: &'a WaitersRef) -> Self {
|
||||||
Self {
|
|
||||||
idx: waiters.index,
|
|
||||||
bound: true,
|
|
||||||
waiters: waiters.get_ref(),
|
|
||||||
_t: marker::PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn from_ref(idx: u32, bound: bool, waiters: &'a WaitersRef) -> Self {
|
|
||||||
Self {
|
Self {
|
||||||
idx,
|
idx,
|
||||||
bound,
|
|
||||||
waiters,
|
waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn inner(self) -> (u32, bool, &'a WaitersRef) {
|
pub(crate) fn inner(self) -> (u32, &'a WaitersRef) {
|
||||||
(self.idx, self.bound, self.waiters)
|
(self.idx, self.waiters)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Mark service as un-ready, force readiness re-evaluation for pipeline
|
/// Mark service as un-ready, force readiness re-evaluation for pipeline
|
||||||
|
@ -205,36 +197,23 @@ impl<'a, S> ServiceCtx<'a, S> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns when the service is able to process requests.
|
/// Returns when the service is able to process requests.
|
||||||
pub async fn ready<T, R>(&self, svc: &'a T) -> Result<(), T::Error>
|
pub async fn check_readiness<T, R>(&self, svc: &'a T) -> Result<(), T::Error>
|
||||||
where
|
where
|
||||||
T: Service<R>,
|
T: Service<R>,
|
||||||
{
|
{
|
||||||
if self.bound {
|
// active readiness
|
||||||
svc.ready(ServiceCtx {
|
if self.waiters.is_ready() {
|
||||||
idx: self.idx,
|
Ok(())
|
||||||
bound: true,
|
|
||||||
waiters: self.waiters,
|
|
||||||
_t: marker::PhantomData,
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
} else {
|
} else {
|
||||||
// active readiness
|
// check readiness and notify waiters
|
||||||
if self.waiters.is_ready() {
|
ReadyCall {
|
||||||
Ok(())
|
fut: Some(svc.ready()),
|
||||||
} else {
|
fut1: None,
|
||||||
// check readiness and notify waiters
|
ctx: *self,
|
||||||
ReadyCall {
|
completed: false,
|
||||||
completed: false,
|
_t: marker::PhantomData,
|
||||||
fut: svc.ready(ServiceCtx {
|
|
||||||
idx: self.idx,
|
|
||||||
bound: false,
|
|
||||||
waiters: self.waiters,
|
|
||||||
_t: marker::PhantomData,
|
|
||||||
}),
|
|
||||||
ctx: *self,
|
|
||||||
}
|
|
||||||
.await
|
|
||||||
}
|
}
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -245,13 +224,12 @@ impl<'a, S> ServiceCtx<'a, S> {
|
||||||
T: Service<R>,
|
T: Service<R>,
|
||||||
R: 'a,
|
R: 'a,
|
||||||
{
|
{
|
||||||
self.ready(svc).await?;
|
self.check_readiness(svc).await?;
|
||||||
|
|
||||||
svc.call(
|
svc.call(
|
||||||
req,
|
req,
|
||||||
ServiceCtx {
|
ServiceCtx {
|
||||||
idx: self.idx,
|
idx: self.idx,
|
||||||
bound: false,
|
|
||||||
waiters: self.waiters,
|
waiters: self.waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
},
|
},
|
||||||
|
@ -274,7 +252,6 @@ impl<'a, S> ServiceCtx<'a, S> {
|
||||||
req,
|
req,
|
||||||
ServiceCtx {
|
ServiceCtx {
|
||||||
idx: self.idx,
|
idx: self.idx,
|
||||||
bound: false,
|
|
||||||
waiters: self.waiters,
|
waiters: self.waiters,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
},
|
},
|
||||||
|
@ -301,26 +278,35 @@ impl<'a, S> fmt::Debug for ServiceCtx<'a, S> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ReadyCall<'a, S: ?Sized, F: Future<Output = Result<(), E>>, E> {
|
struct ReadyCall<'a, S: ?Sized, F: Future<Output = Option<R>>, E, R> {
|
||||||
completed: bool,
|
completed: bool,
|
||||||
fut: F,
|
fut: Option<F>,
|
||||||
|
fut1: Option<R>,
|
||||||
ctx: ServiceCtx<'a, S>,
|
ctx: ServiceCtx<'a, S>,
|
||||||
|
_t: marker::PhantomData<(R, E)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, S: ?Sized, F: Future<Output = Result<(), E>>, E> Drop for ReadyCall<'a, S, F, E> {
|
impl<'a, S: ?Sized, F: Future<Output = Option<R>>, E, R> Drop
|
||||||
|
for ReadyCall<'a, S, F, E, R>
|
||||||
|
{
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if !self.completed && !self.ctx.waiters.state.get().contains(Flags::BOUND) {
|
if !self.completed && !self.ctx.waiters.is_bound() {
|
||||||
self.ctx.waiters.notify();
|
self.ctx.waiters.notify();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, S: ?Sized, F: Future<Output = Result<(), E>>, E> Unpin for ReadyCall<'a, S, F, E> {}
|
impl<'a, S: ?Sized, F: Future<Output = Option<R>>, E, R> Unpin
|
||||||
|
for ReadyCall<'a, S, F, E, R>
|
||||||
impl<'a, S: ?Sized, F: Future<Output = Result<(), E>>, E> Future
|
|
||||||
for ReadyCall<'a, S, F, E>
|
|
||||||
{
|
{
|
||||||
type Output = F::Output;
|
}
|
||||||
|
|
||||||
|
impl<'a, S: ?Sized, F: Future<Output = Option<R>>, E, R> Future
|
||||||
|
for ReadyCall<'a, S, F, E, R>
|
||||||
|
where
|
||||||
|
R: Future<Output = Result<(), E>>,
|
||||||
|
{
|
||||||
|
type Output = Result<(), E>;
|
||||||
|
|
||||||
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
||||||
let st = self.ctx.waiters.state.get();
|
let st = self.ctx.waiters.state.get();
|
||||||
|
@ -333,17 +319,36 @@ impl<'a, S: ?Sized, F: Future<Output = Result<(), E>>, E> Future
|
||||||
}
|
}
|
||||||
} else if self.ctx.waiters.can_check(self.ctx.idx, cx) {
|
} else if self.ctx.waiters.can_check(self.ctx.idx, cx) {
|
||||||
// SAFETY: `fut` never moves
|
// SAFETY: `fut` never moves
|
||||||
let result = unsafe { Pin::new_unchecked(&mut self.as_mut().fut).poll(cx) };
|
if let Some(ref mut fut) = self.as_mut().fut {
|
||||||
match result {
|
let result = unsafe { Pin::new_unchecked(fut).poll(cx) };
|
||||||
Poll::Pending => {
|
match result {
|
||||||
self.ctx.waiters.register(self.ctx.idx, cx);
|
Poll::Pending => {
|
||||||
Poll::Pending
|
self.ctx.waiters.register(self.ctx.idx, cx);
|
||||||
|
return Poll::Pending;
|
||||||
|
}
|
||||||
|
Poll::Ready(res) => {
|
||||||
|
self.fut1 = res;
|
||||||
|
let _ = self.fut.take();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Poll::Ready(res) => {
|
}
|
||||||
self.completed = true;
|
|
||||||
self.ctx.waiters.notify();
|
if let Some(ref mut fut) = self.as_mut().fut1 {
|
||||||
Poll::Ready(res)
|
match unsafe { Pin::new_unchecked(fut).poll(cx) } {
|
||||||
|
Poll::Pending => {
|
||||||
|
self.ctx.waiters.register(self.ctx.idx, cx);
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
Poll::Ready(res) => {
|
||||||
|
self.completed = true;
|
||||||
|
self.ctx.waiters.notify();
|
||||||
|
Poll::Ready(res)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
self.completed = true;
|
||||||
|
self.ctx.waiters.notify();
|
||||||
|
Poll::Ready(Ok(()))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Poll::Pending
|
Poll::Pending
|
||||||
|
@ -373,11 +378,6 @@ mod tests {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
self.1.ready().await;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn call(
|
async fn call(
|
||||||
&self,
|
&self,
|
||||||
req: &'static str,
|
req: &'static str,
|
||||||
|
|
|
@ -134,11 +134,6 @@ where
|
||||||
type Response = Res;
|
type Response = Res;
|
||||||
type Error = Err;
|
type Error = Err;
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
std::future::pending().await
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
||||||
(self.f)(req).await
|
(self.f)(req).await
|
||||||
|
@ -212,11 +207,6 @@ where
|
||||||
type Response = Res;
|
type Response = Res;
|
||||||
type Error = Err;
|
type Error = Err;
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
std::future::pending().await
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Res, Err> {
|
||||||
(self.f)(req).await
|
(self.f)(req).await
|
||||||
|
|
|
@ -62,11 +62,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
std::future::pending().await
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Req, Err> {
|
async fn call(&self, req: Req, _: ServiceCtx<'_, Self>) -> Result<Req, Err> {
|
||||||
Ok(req)
|
Ok(req)
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
//! See [`Service`] docs for information on this crate's foundational trait.
|
//! See [`Service`] docs for information on this crate's foundational trait.
|
||||||
#![allow(async_fn_in_trait)]
|
#![allow(async_fn_in_trait)]
|
||||||
#![deny(
|
// #![deny(
|
||||||
rust_2018_idioms,
|
// rust_2018_idioms,
|
||||||
warnings,
|
// warnings,
|
||||||
unreachable_pub,
|
// unreachable_pub,
|
||||||
missing_debug_implementations
|
// missing_debug_implementations
|
||||||
)]
|
// )]
|
||||||
use std::rc::Rc;
|
use std::{future::Future, rc::Rc};
|
||||||
|
|
||||||
mod and_then;
|
mod and_then;
|
||||||
mod apply;
|
mod apply;
|
||||||
|
@ -114,14 +114,8 @@ pub trait Service<Req> {
|
||||||
/// This is a **best effort** implementation. False positives are permitted. It is permitted for
|
/// This is a **best effort** implementation. False positives are permitted. It is permitted for
|
||||||
/// the service to returns from a `ready` call and the next invocation of `call`
|
/// the service to returns from a `ready` call and the next invocation of `call`
|
||||||
/// results in an error.
|
/// results in an error.
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
Ok(())
|
None::<util::Ready<Self::Error>>
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns when the service becomes un-ready and not able to process requests.
|
|
||||||
///
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
std::future::pending().await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -248,13 +242,8 @@ where
|
||||||
type Error = S::Error;
|
type Error = S::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), S::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
ctx.ready(&**self).await
|
(&**self).ready().await
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), S::Error> {
|
|
||||||
(**self).unready().await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -280,13 +269,8 @@ where
|
||||||
type Error = S::Error;
|
type Error = S::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), S::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
ctx.ready(&**self).await
|
(&**self).ready().await
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), S::Error> {
|
|
||||||
(**self).unready().await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -5,34 +5,8 @@ macro_rules! forward_ready {
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(
|
async fn ready(
|
||||||
&self,
|
&self,
|
||||||
ctx: $crate::ServiceCtx<'_, Self>,
|
) -> Option<impl ::std::future::Future<Output = Result<(), Self::Error>>> {
|
||||||
) -> Result<(), Self::Error> {
|
self.$field.ready().await
|
||||||
ctx.ready(&self.$field)
|
|
||||||
.await
|
|
||||||
.map_err(::core::convert::Into::into)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
($field:ident, $err:expr) => {
|
|
||||||
#[inline]
|
|
||||||
async fn ready(
|
|
||||||
&self,
|
|
||||||
ctx: $crate::ServiceCtx<'_, Self>,
|
|
||||||
) -> Result<(), Self::Error> {
|
|
||||||
ctx.ready(&self.$field).await.map_err($err)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/// An implementation of [`unready`] that forwards unready checks to a field.
|
|
||||||
#[macro_export]
|
|
||||||
macro_rules! forward_unready {
|
|
||||||
($field:ident) => {
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
self.$field
|
|
||||||
.unready()
|
|
||||||
.await
|
|
||||||
.map_err(::core::convert::Into::into)
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,7 +62,6 @@ where
|
||||||
type Error = A::Error;
|
type Error = A::Error;
|
||||||
|
|
||||||
crate::forward_ready!(service);
|
crate::forward_ready!(service);
|
||||||
crate::forward_unready!(service);
|
|
||||||
crate::forward_shutdown!(service);
|
crate::forward_shutdown!(service);
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{fmt, marker::PhantomData};
|
use std::{fmt, future::Future, marker::PhantomData};
|
||||||
|
|
||||||
use super::{Service, ServiceCtx, ServiceFactory};
|
use super::{Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
|
@ -63,13 +63,11 @@ where
|
||||||
type Error = E;
|
type Error = E;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), E>>> {
|
||||||
ctx.ready(&self.service).await.map_err(&self.f)
|
self.service
|
||||||
}
|
.ready()
|
||||||
|
.await
|
||||||
#[inline]
|
.map(|fut| async move { fut.await.map_err(&self.f) })
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
self.service.unready().await.map_err(&self.f)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -30,25 +30,13 @@ impl<S> Pipeline<S> {
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
/// Returns when the service is able to process requests.
|
/// Returns when the service is able to process requests.
|
||||||
pub async fn ready<R>(&self) -> Result<(), S::Error>
|
pub async fn ready<R>(
|
||||||
|
&self,
|
||||||
|
) -> Option<impl Future<Output = Result<(), S::Error>> + use<'_, S, R>>
|
||||||
where
|
where
|
||||||
S: Service<R>,
|
S: Service<R>,
|
||||||
{
|
{
|
||||||
ServiceCtx::<'_, S>::new(&self.waiters)
|
self.svc.as_ref().ready().await
|
||||||
.ready(self.svc.as_ref())
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
/// Returns when the service is able to process requests.
|
|
||||||
pub async fn unready<R>(&self) -> Result<(), S::Error>
|
|
||||||
where
|
|
||||||
S: Service<R>,
|
|
||||||
{
|
|
||||||
self.waiters.set_ready();
|
|
||||||
let result = self.svc.unready().await;
|
|
||||||
self.waiters.set_notready();
|
|
||||||
result
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -61,7 +49,7 @@ impl<S> Pipeline<S> {
|
||||||
let ctx = ServiceCtx::<'_, S>::new(&self.waiters);
|
let ctx = ServiceCtx::<'_, S>::new(&self.waiters);
|
||||||
|
|
||||||
// check service readiness
|
// check service readiness
|
||||||
ctx.ready(self.svc.as_ref()).await?;
|
ctx.check_readiness(self.svc.as_ref()).await?;
|
||||||
|
|
||||||
// call service
|
// call service
|
||||||
self.svc.as_ref().call(req, ctx).await
|
self.svc.as_ref().call(req, ctx).await
|
||||||
|
@ -168,10 +156,15 @@ where
|
||||||
|
|
||||||
let fut = Box::pin(async move {
|
let fut = Box::pin(async move {
|
||||||
loop {
|
loop {
|
||||||
pl.svc
|
if let Some(fut) = pl.svc.ready().await {
|
||||||
.ready(ServiceCtx::<'_, S>::new_bound(&pl.waiters))
|
pl.waiters.set_ready();
|
||||||
.await?;
|
let result = fut.await;
|
||||||
pl.unready().await?;
|
pl.waiters.set_notready();
|
||||||
|
result?
|
||||||
|
} else {
|
||||||
|
pl.waiters.set_ready();
|
||||||
|
std::future::pending().await
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use std::future::Future;
|
||||||
|
|
||||||
use super::{util, Service, ServiceCtx, ServiceFactory};
|
use super::{util, Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -26,13 +28,8 @@ where
|
||||||
type Error = B::Error;
|
type Error = B::Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
async fn ready(&self) -> Option<impl Future<Output = Result<(), Self::Error>>> {
|
||||||
util::ready(&self.svc1, &self.svc2, ctx).await
|
util::ready(&self.svc1, &self.svc2).await
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
async fn unready(&self) -> Result<(), Self::Error> {
|
|
||||||
util::unready(&self.svc1, &self.svc2).await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
@ -1,6 +1,16 @@
|
||||||
use std::{future::poll_fn, future::Future, pin, task::Poll};
|
use std::{future::poll_fn, future::Future, marker, pin, task::Context, task::Poll};
|
||||||
|
|
||||||
use crate::{Service, ServiceCtx};
|
use crate::Service;
|
||||||
|
|
||||||
|
pub(crate) struct Ready<E>(marker::PhantomData<E>);
|
||||||
|
|
||||||
|
impl<E> Future for Ready<E> {
|
||||||
|
type Output = Result<(), E>;
|
||||||
|
|
||||||
|
fn poll(self: pin::Pin<&mut Self>, _: &mut Context<'_>) -> Poll<Self::Output> {
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) async fn shutdown<A, AR, B, BR>(svc1: &A, svc2: &B)
|
pub(crate) async fn shutdown<A, AR, B, BR>(svc1: &A, svc2: &B)
|
||||||
where
|
where
|
||||||
|
@ -29,53 +39,105 @@ where
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn ready<S, A, AR, B, BR>(
|
pub(crate) async fn ready<'a, A, AR, B, BR>(
|
||||||
svc1: &A,
|
svc1: &'a A,
|
||||||
svc2: &B,
|
svc2: &'a B,
|
||||||
ctx: ServiceCtx<'_, S>,
|
) -> Option<impl Future<Output = Result<(), A::Error>> + use<'a, A, AR, B, BR>>
|
||||||
) -> Result<(), A::Error>
|
|
||||||
where
|
where
|
||||||
A: Service<AR>,
|
A: Service<AR>,
|
||||||
B: Service<BR, Error = A::Error>,
|
B: Service<BR, Error = A::Error>,
|
||||||
{
|
{
|
||||||
let mut fut1 = pin::pin!(ctx.ready(svc1));
|
let mut fut1 = pin::pin!(svc1.ready());
|
||||||
let mut fut2 = pin::pin!(ctx.ready(svc2));
|
let mut fut2 = pin::pin!(svc2.ready());
|
||||||
|
|
||||||
let mut ready1 = false;
|
let mut ready1 = false;
|
||||||
let mut ready2 = false;
|
let mut ready2 = false;
|
||||||
|
|
||||||
poll_fn(move |cx| {
|
let mut r_fut1 = None;
|
||||||
if !ready1 && pin::Pin::new(&mut fut1).poll(cx)?.is_ready() {
|
let mut r_fut2 = None;
|
||||||
ready1 = true;
|
|
||||||
|
poll_fn(|cx| {
|
||||||
|
if !ready1 {
|
||||||
|
if let Poll::Ready(res) = pin::Pin::new(&mut fut1).poll(cx) {
|
||||||
|
r_fut1 = res;
|
||||||
|
ready1 = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if !ready2 && pin::Pin::new(&mut fut2).poll(cx)?.is_ready() {
|
if !ready2 {
|
||||||
ready2 = true;
|
if let Poll::Ready(res) = pin::Pin::new(&mut fut2).poll(cx) {
|
||||||
|
r_fut2 = res;
|
||||||
|
ready2 = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if ready1 && ready2 {
|
if ready1 && ready2 {
|
||||||
Poll::Ready(Ok(()))
|
Poll::Ready(())
|
||||||
} else {
|
} else {
|
||||||
Poll::Pending
|
Poll::Pending
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.await
|
.await;
|
||||||
|
|
||||||
|
if r_fut1.is_none() && r_fut2.is_none() {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(async move {
|
||||||
|
match (r_fut1, r_fut2) {
|
||||||
|
(Some(fut), None) => fut.await?,
|
||||||
|
(None, Some(fut)) => fut.await?,
|
||||||
|
(Some(fut1), Some(fut2)) => match select(fut1, fut2).await {
|
||||||
|
Either::Left(res) => res?,
|
||||||
|
Either::Right(res) => res?,
|
||||||
|
},
|
||||||
|
(None, None) => (),
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn unready<A, AR, B, BR>(svc1: &A, svc2: &B) -> Result<(), A::Error>
|
pub(crate) enum Either<A, B> {
|
||||||
|
/// First branch of the type
|
||||||
|
Left(A),
|
||||||
|
/// Second branch of the type
|
||||||
|
Right(B),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Waits for either one of two differently-typed futures to complete.
|
||||||
|
pub(crate) async fn select<A, B>(fut_a: A, fut_b: B) -> Either<A::Output, B::Output>
|
||||||
where
|
where
|
||||||
A: Service<AR>,
|
A: Future,
|
||||||
B: Service<BR, Error = A::Error>,
|
B: Future,
|
||||||
{
|
{
|
||||||
let mut fut1 = pin::pin!(svc1.unready());
|
Select { fut_a, fut_b }.await
|
||||||
let mut fut2 = pin::pin!(svc2.unready());
|
}
|
||||||
|
|
||||||
poll_fn(move |cx| {
|
pin_project_lite::pin_project! {
|
||||||
if pin::Pin::new(&mut fut1).poll(cx)?.is_ready()
|
pub(crate) struct Select<A, B> {
|
||||||
|| pin::Pin::new(&mut fut2).poll(cx)?.is_ready()
|
#[pin]
|
||||||
{
|
fut_a: A,
|
||||||
Poll::Ready(Ok(()))
|
#[pin]
|
||||||
} else {
|
fut_b: B,
|
||||||
Poll::Pending
|
}
|
||||||
}
|
}
|
||||||
})
|
|
||||||
.await
|
impl<A, B> Future for Select<A, B>
|
||||||
|
where
|
||||||
|
A: Future,
|
||||||
|
B: Future,
|
||||||
|
{
|
||||||
|
type Output = Either<A::Output, B::Output>;
|
||||||
|
|
||||||
|
fn poll(self: pin::Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
||||||
|
let this = self.project();
|
||||||
|
|
||||||
|
if let Poll::Ready(item) = this.fut_a.poll(cx) {
|
||||||
|
return Poll::Ready(Either::Left(item));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Poll::Ready(item) = this.fut_b.poll(cx) {
|
||||||
|
return Poll::Ready(Either::Right(item));
|
||||||
|
}
|
||||||
|
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue