mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 21:07:39 +03:00
242 lines
6.2 KiB
Rust
242 lines
6.2 KiB
Rust
use std::{cell::Cell, future, pin::Pin, rc::Rc, task, task::Context, task::Poll};
|
|
|
|
use crate::{ctx::ServiceCall, ctx::Waiters, Service, ServiceCtx, ServiceFactory};
|
|
|
|
#[derive(Debug)]
|
|
/// Container for a service.
|
|
///
|
|
/// Container allows to call enclosed service and adds support of shared readiness.
|
|
pub struct Pipeline<S> {
|
|
svc: Rc<S>,
|
|
pending: Cell<bool>,
|
|
pub(crate) waiters: Waiters,
|
|
}
|
|
|
|
impl<S> Pipeline<S> {
|
|
#[inline]
|
|
/// Construct new container instance.
|
|
pub fn new(svc: S) -> Self {
|
|
Pipeline {
|
|
svc: Rc::new(svc),
|
|
pending: Cell::new(false),
|
|
waiters: Waiters::new(),
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
/// Return reference to enclosed service
|
|
pub fn get_ref(&self) -> &S {
|
|
self.svc.as_ref()
|
|
}
|
|
|
|
#[inline]
|
|
/// Returns `Ready` when the service is able to process requests.
|
|
pub fn poll_ready<R>(&self, cx: &mut Context<'_>) -> Poll<Result<(), S::Error>>
|
|
where
|
|
S: Service<R>,
|
|
{
|
|
let res = self.svc.poll_ready(cx);
|
|
if res.is_pending() {
|
|
self.pending.set(true);
|
|
self.waiters.register(cx)
|
|
} else if self.pending.get() {
|
|
self.pending.set(false);
|
|
self.waiters.notify()
|
|
}
|
|
res
|
|
}
|
|
|
|
#[inline]
|
|
/// Shutdown enclosed service.
|
|
pub fn poll_shutdown<R>(&self, cx: &mut Context<'_>) -> Poll<()>
|
|
where
|
|
S: Service<R>,
|
|
{
|
|
self.svc.poll_shutdown(cx)
|
|
}
|
|
|
|
#[deprecated(since = "1.2.3", note = "Use Pipeline::call() instead")]
|
|
#[doc(hidden)]
|
|
#[inline]
|
|
/// Wait for service readiness and then create future object
|
|
/// that resolves to service result.
|
|
pub fn service_call<R>(&self, req: R) -> ServiceCall<'_, S, R>
|
|
where
|
|
S: Service<R>,
|
|
{
|
|
ServiceCall::call_pipeline(req, self)
|
|
}
|
|
|
|
#[inline]
|
|
/// Wait for service readiness and then create future object
|
|
/// that resolves to service result.
|
|
pub fn call<R>(&self, req: R) -> ServiceCall<'_, S, R>
|
|
where
|
|
S: Service<R>,
|
|
{
|
|
ServiceCall::call_pipeline(req, self)
|
|
}
|
|
|
|
#[inline]
|
|
/// Wait for service readiness and then create future object
|
|
/// that resolves to service result.
|
|
pub fn call_static<R>(&self, req: R) -> PipelineCall<S, R>
|
|
where
|
|
S: Service<R> + 'static,
|
|
{
|
|
PipelineCall {
|
|
state: PipelineCallState::Ready { req: Some(req) },
|
|
pipeline: self.clone(),
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
/// Call service and create future object that resolves to service result.
|
|
///
|
|
/// Note, this call does not check service readiness.
|
|
pub fn call_nowait<R>(&self, req: R) -> PipelineCall<S, R>
|
|
where
|
|
S: Service<R> + 'static,
|
|
{
|
|
PipelineCall {
|
|
state: PipelineCallState::new_call(self, req),
|
|
pipeline: self.clone(),
|
|
}
|
|
}
|
|
|
|
/// Extract service if container hadnt been cloned before.
|
|
pub fn into_service(self) -> Option<S> {
|
|
let svc = self.svc.clone();
|
|
drop(self);
|
|
Rc::try_unwrap(svc).ok()
|
|
}
|
|
}
|
|
|
|
impl<S> From<S> for Pipeline<S> {
|
|
#[inline]
|
|
fn from(svc: S) -> Self {
|
|
Pipeline::new(svc)
|
|
}
|
|
}
|
|
|
|
impl<S> Clone for Pipeline<S> {
|
|
#[inline]
|
|
fn clone(&self) -> Self {
|
|
Self {
|
|
svc: self.svc.clone(),
|
|
pending: Cell::new(false),
|
|
waiters: self.waiters.clone(),
|
|
}
|
|
}
|
|
}
|
|
|
|
pin_project_lite::pin_project! {
|
|
#[must_use = "futures do nothing unless polled"]
|
|
pub struct PipelineCall<S, R>
|
|
where
|
|
S: Service<R>,
|
|
S: 'static,
|
|
R: 'static,
|
|
{
|
|
#[pin]
|
|
state: PipelineCallState<S, R>,
|
|
pipeline: Pipeline<S>,
|
|
}
|
|
}
|
|
|
|
pin_project_lite::pin_project! {
|
|
#[project = PipelineCallStateProject]
|
|
enum PipelineCallState<S, Req>
|
|
where
|
|
S: Service<Req>,
|
|
S: 'static,
|
|
Req: 'static,
|
|
{
|
|
Ready { req: Option<Req> },
|
|
Call { #[pin] fut: S::Future<'static> },
|
|
Empty,
|
|
}
|
|
}
|
|
|
|
impl<S, R> PipelineCallState<S, R>
|
|
where
|
|
S: Service<R> + 'static,
|
|
R: 'static,
|
|
{
|
|
fn new_call(pl: &Pipeline<S>, req: R) -> Self {
|
|
let ctx = ServiceCtx::new(&pl.waiters);
|
|
let svc_call = pl.get_ref().call(req, ctx);
|
|
|
|
// SAFETY: `svc_call` has same lifetime same as lifetime of `pl.svc`
|
|
// Pipeline::svc is heap allocated(Rc<S>), we keep it alive until
|
|
// `svc_call` get resolved to result
|
|
let fut = unsafe { std::mem::transmute(svc_call) };
|
|
|
|
PipelineCallState::Call { fut }
|
|
}
|
|
}
|
|
|
|
impl<S, R> future::Future for PipelineCall<S, R>
|
|
where
|
|
S: Service<R>,
|
|
{
|
|
type Output = Result<S::Response, S::Error>;
|
|
|
|
#[inline]
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
let mut this = self.as_mut().project();
|
|
|
|
match this.state.as_mut().project() {
|
|
PipelineCallStateProject::Ready { req } => {
|
|
task::ready!(this.pipeline.poll_ready(cx))?;
|
|
|
|
let st = PipelineCallState::new_call(this.pipeline, req.take().unwrap());
|
|
this.state.set(st);
|
|
self.poll(cx)
|
|
}
|
|
PipelineCallStateProject::Call { fut, .. } => fut.poll(cx).map(|r| {
|
|
this.state.set(PipelineCallState::Empty);
|
|
r
|
|
}),
|
|
PipelineCallStateProject::Empty => {
|
|
panic!("future must not be polled after it returned `Poll::Ready`")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pin_project_lite::pin_project! {
|
|
#[must_use = "futures do nothing unless polled"]
|
|
pub struct CreatePipeline<'f, F, R, C>
|
|
where F: ServiceFactory<R, C>,
|
|
F: ?Sized,
|
|
F: 'f,
|
|
C: 'f,
|
|
{
|
|
#[pin]
|
|
fut: F::Future<'f>,
|
|
}
|
|
}
|
|
|
|
impl<'f, F, R, C> CreatePipeline<'f, F, R, C>
|
|
where
|
|
F: ServiceFactory<R, C> + 'f,
|
|
{
|
|
pub(crate) fn new(fut: F::Future<'f>) -> Self {
|
|
Self { fut }
|
|
}
|
|
}
|
|
|
|
impl<'f, F, R, C> future::Future for CreatePipeline<'f, F, R, C>
|
|
where
|
|
F: ServiceFactory<R, C> + 'f,
|
|
{
|
|
type Output = Result<Pipeline<F::Service>, F::InitError>;
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
Poll::Ready(Ok(Pipeline::new(std::task::ready!(self
|
|
.project()
|
|
.fut
|
|
.poll(cx))?)))
|
|
}
|
|
}
|