mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 04:47:39 +03:00
Add .apply_fn() for chain and chain factory (#220)
* Add .apply_fn() for chain and chain factory
This commit is contained in:
parent
594bf0a8e2
commit
17ffaf105e
4 changed files with 102 additions and 9 deletions
|
@ -1,5 +1,11 @@
|
|||
# Changes
|
||||
|
||||
## [1.2.4] - 2023-08-12
|
||||
|
||||
* Forward readiness check for Apply service
|
||||
|
||||
* Add .apply_fn() for chain and chanin factory
|
||||
|
||||
## [1.2.3] - 2023-08-10
|
||||
|
||||
* Check readiness for pipeline calls
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
[package]
|
||||
name = "ntex-service"
|
||||
version = "1.2.3"
|
||||
version = "1.2.4"
|
||||
authors = ["ntex contributors <team@ntex.rs>"]
|
||||
description = "ntex service"
|
||||
keywords = ["network", "framework", "async", "futures"]
|
||||
|
|
|
@ -15,11 +15,7 @@ where
|
|||
R: Future<Output = Result<Out, Err>>,
|
||||
U: IntoService<T, Req>,
|
||||
{
|
||||
Apply {
|
||||
f,
|
||||
service: Pipeline::new(service.into_service()),
|
||||
r: marker::PhantomData,
|
||||
}
|
||||
Apply::new(service.into_service(), f)
|
||||
}
|
||||
|
||||
/// Service factory that produces `apply_fn` service.
|
||||
|
@ -60,6 +56,21 @@ impl<S> ApplyService<S> {
|
|||
}
|
||||
}
|
||||
|
||||
impl<T, Req, F, R, In, Out, Err> Apply<T, Req, F, R, In, Out, Err>
|
||||
where
|
||||
T: Service<Req, Error = Err>,
|
||||
F: Fn(In, ApplyService<T>) -> R,
|
||||
R: Future<Output = Result<Out, Err>>,
|
||||
{
|
||||
pub(crate) fn new(service: T, f: F) -> Self {
|
||||
Apply {
|
||||
f,
|
||||
service: Pipeline::new(service),
|
||||
r: marker::PhantomData,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T, Req, F, R, In, Out, Err> Clone for Apply<T, Req, F, R, In, Out, Err>
|
||||
where
|
||||
T: Service<Req, Error = Err> + Clone,
|
||||
|
@ -85,6 +96,7 @@ where
|
|||
type Error = Err;
|
||||
type Future<'f> = R where Self: 'f, In: 'f, R: 'f;
|
||||
|
||||
crate::forward_poll_ready!(service);
|
||||
crate::forward_poll_shutdown!(service);
|
||||
|
||||
#[inline]
|
||||
|
@ -115,7 +127,7 @@ where
|
|||
R: Future<Output = Result<Out, Err>>,
|
||||
{
|
||||
/// Create new `ApplyNewService` new service instance
|
||||
fn new(service: T, f: F) -> Self {
|
||||
pub(crate) fn new(service: T, f: F) -> Self {
|
||||
Self {
|
||||
f,
|
||||
service,
|
||||
|
@ -246,6 +258,25 @@ mod tests {
|
|||
assert_eq!(res.unwrap(), ("srv", ()));
|
||||
}
|
||||
|
||||
#[ntex::test]
|
||||
async fn test_call_chain() {
|
||||
let srv = chain(Srv)
|
||||
.apply_fn(|req: &'static str, svc| async move {
|
||||
svc.call(()).await.unwrap();
|
||||
Ok((req, ()))
|
||||
})
|
||||
.clone()
|
||||
.pipeline();
|
||||
|
||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||
let res = lazy(|cx| srv.poll_shutdown(cx)).await;
|
||||
assert_eq!(res, Poll::Ready(()));
|
||||
|
||||
let res = srv.call("srv").await;
|
||||
assert!(res.is_ok());
|
||||
assert_eq!(res.unwrap(), ("srv", ()));
|
||||
}
|
||||
|
||||
#[ntex::test]
|
||||
async fn test_create() {
|
||||
let new_srv = chain_factory(
|
||||
|
@ -267,4 +298,22 @@ mod tests {
|
|||
assert!(res.is_ok());
|
||||
assert_eq!(res.unwrap(), ("srv", ()));
|
||||
}
|
||||
|
||||
#[ntex::test]
|
||||
async fn test_create_chain() {
|
||||
let new_srv = chain_factory(|| Ready::<_, ()>::Ok(Srv))
|
||||
.apply_fn(|req: &'static str, srv| async move {
|
||||
srv.call(()).await.unwrap();
|
||||
Ok((req, ()))
|
||||
})
|
||||
.clone();
|
||||
|
||||
let srv = new_srv.pipeline(&()).await.unwrap();
|
||||
|
||||
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
||||
|
||||
let res = srv.call("srv").await;
|
||||
assert!(res.is_ok());
|
||||
assert_eq!(res.unwrap(), ("srv", ()));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,6 +1,8 @@
|
|||
use std::marker::PhantomData;
|
||||
#![allow(clippy::type_complexity)]
|
||||
use std::{future::Future, marker::PhantomData};
|
||||
|
||||
use crate::and_then::{AndThen, AndThenFactory};
|
||||
use crate::apply::{Apply, ApplyFactory, ApplyService};
|
||||
use crate::ctx::{ServiceCall, ServiceCtx};
|
||||
use crate::map::{Map, MapFactory};
|
||||
use crate::map_err::{MapErr, MapErrFactory};
|
||||
|
@ -118,6 +120,24 @@ impl<Svc: Service<Req>, Req> ServiceChain<Svc, Req> {
|
|||
}
|
||||
}
|
||||
|
||||
/// Use function as middleware for current service.
|
||||
///
|
||||
/// Short version of `apply_fn(chain(...), fn)`
|
||||
pub fn apply_fn<F, R, In, Out, Err>(
|
||||
self,
|
||||
f: F,
|
||||
) -> ServiceChain<Apply<Svc, Req, F, R, In, Out, Err>, In>
|
||||
where
|
||||
F: Fn(In, ApplyService<Svc>) -> R,
|
||||
R: Future<Output = Result<Out, Err>>,
|
||||
Svc: Service<Req, Error = Err>,
|
||||
{
|
||||
ServiceChain {
|
||||
service: Apply::new(self.service, f),
|
||||
_t: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
/// Create service pipeline
|
||||
pub fn pipeline(self) -> Pipeline<Svc> {
|
||||
Pipeline::new(self.service)
|
||||
|
@ -175,7 +195,7 @@ impl<T: ServiceFactory<Req, C>, Req, C> ServiceChainFactory<T, Req, C> {
|
|||
|
||||
/// Apply middleware to current service factory.
|
||||
///
|
||||
/// Short version of `apply(middleware, pipeline_factory(...))`
|
||||
/// Short version of `apply(middleware, chain_factory(...))`
|
||||
pub fn apply<U>(self, tr: U) -> ServiceChainFactory<ApplyMiddleware<U, T, C>, Req, C>
|
||||
where
|
||||
U: Middleware<T::Service>,
|
||||
|
@ -186,6 +206,24 @@ impl<T: ServiceFactory<Req, C>, Req, C> ServiceChainFactory<T, Req, C> {
|
|||
}
|
||||
}
|
||||
|
||||
/// Apply function middleware to current service factory.
|
||||
///
|
||||
/// Short version of `apply_fn_factory(chain_factory(...), fn)`
|
||||
pub fn apply_fn<F, R, In, Out, Err>(
|
||||
self,
|
||||
f: F,
|
||||
) -> ServiceChainFactory<ApplyFactory<T, Req, C, F, R, In, Out, Err>, In, C>
|
||||
where
|
||||
F: Fn(In, ApplyService<T::Service>) -> R + Clone,
|
||||
R: Future<Output = Result<Out, Err>>,
|
||||
T: ServiceFactory<Req, C, Error = Err>,
|
||||
{
|
||||
ServiceChainFactory {
|
||||
factory: ApplyFactory::new(self.factory, f),
|
||||
_t: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
/// Create `NewService` to chain on a computation for when a call to the
|
||||
/// service finished, passing the result of the call to the next
|
||||
/// service `U`.
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue