mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 21:37:58 +03:00
Add EitherService/EitherServiceFactory (#500)
This commit is contained in:
parent
cd56883197
commit
71ba4d28a3
4 changed files with 245 additions and 1 deletions
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.9.0] - 2025-01-15
|
||||||
|
|
||||||
|
* Add EitherService/EitherServiceFactory
|
||||||
|
|
||||||
## [2.8.0] - 2024-12-04
|
## [2.8.0] - 2024-12-04
|
||||||
|
|
||||||
* Use updated Service trait
|
* Use updated Service trait
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-util"
|
name = "ntex-util"
|
||||||
version = "2.8.0"
|
version = "2.9.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"]
|
||||||
|
|
239
ntex-util/src/services/either.rs
Normal file
239
ntex-util/src/services/either.rs
Normal file
|
@ -0,0 +1,239 @@
|
||||||
|
//! Either service allows to use different services for handling request
|
||||||
|
use std::{fmt, task::Context};
|
||||||
|
|
||||||
|
use ntex_service::{Service, ServiceCtx, ServiceFactory};
|
||||||
|
|
||||||
|
use crate::future::Either;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
/// Either service
|
||||||
|
///
|
||||||
|
/// Either service allows to use different services for handling requests
|
||||||
|
pub struct EitherService<SLeft, SRight> {
|
||||||
|
svc: Either<SLeft, SRight>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
/// Either service factory
|
||||||
|
///
|
||||||
|
/// Either service allows to use different services for handling requests
|
||||||
|
pub struct EitherServiceFactory<ChooseFn, SFLeft, SFRight> {
|
||||||
|
left: SFLeft,
|
||||||
|
right: SFRight,
|
||||||
|
choose_left_fn: ChooseFn,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<ChooseFn, SFLeft, SFRight> EitherServiceFactory<ChooseFn, SFLeft, SFRight> {
|
||||||
|
/// Create `Either` service factory
|
||||||
|
pub fn new(choose_left_fn: ChooseFn, sf_left: SFLeft, sf_right: SFRight) -> Self {
|
||||||
|
EitherServiceFactory {
|
||||||
|
choose_left_fn,
|
||||||
|
left: sf_left,
|
||||||
|
right: sf_right,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<ChooseFn, SFLeft, SFRight> fmt::Debug
|
||||||
|
for EitherServiceFactory<ChooseFn, SFLeft, SFRight>
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
f.debug_struct("EitherServiceFactory")
|
||||||
|
.field("left", &std::any::type_name::<SFLeft>())
|
||||||
|
.field("right", &std::any::type_name::<SFRight>())
|
||||||
|
.field("choose_fn", &std::any::type_name::<ChooseFn>())
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R, C, ChooseFn, SFLeft, SFRight> ServiceFactory<R, C>
|
||||||
|
for EitherServiceFactory<ChooseFn, SFLeft, SFRight>
|
||||||
|
where
|
||||||
|
ChooseFn: Fn(&C) -> bool,
|
||||||
|
SFLeft: ServiceFactory<R, C>,
|
||||||
|
SFRight: ServiceFactory<
|
||||||
|
R,
|
||||||
|
C,
|
||||||
|
Response = SFLeft::Response,
|
||||||
|
InitError = SFLeft::InitError,
|
||||||
|
Error = SFLeft::Error,
|
||||||
|
>,
|
||||||
|
{
|
||||||
|
type Response = SFLeft::Response;
|
||||||
|
type Error = SFLeft::Error;
|
||||||
|
type InitError = SFLeft::InitError;
|
||||||
|
type Service = EitherService<SFLeft::Service, SFRight::Service>;
|
||||||
|
|
||||||
|
async fn create(&self, cfg: C) -> Result<Self::Service, Self::InitError> {
|
||||||
|
let choose_left = (self.choose_left_fn)(&cfg);
|
||||||
|
|
||||||
|
if choose_left {
|
||||||
|
let svc = self.left.create(cfg).await?;
|
||||||
|
Ok(EitherService {
|
||||||
|
svc: Either::Left(svc),
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let svc = self.right.create(cfg).await?;
|
||||||
|
Ok(EitherService {
|
||||||
|
svc: Either::Right(svc),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<SLeft, SRight> EitherService<SLeft, SRight> {
|
||||||
|
/// Create `Either` service
|
||||||
|
pub fn left(svc: SLeft) -> Self {
|
||||||
|
EitherService {
|
||||||
|
svc: Either::Left(svc),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create `Either` service
|
||||||
|
pub fn right(svc: SRight) -> Self {
|
||||||
|
EitherService {
|
||||||
|
svc: Either::Right(svc),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<SLeft, SRight> fmt::Debug for EitherService<SLeft, SRight> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
f.debug_struct("EitherService")
|
||||||
|
.field("left", &std::any::type_name::<SLeft>())
|
||||||
|
.field("right", &std::any::type_name::<SRight>())
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Req, SLeft, SRight> Service<Req> for EitherService<SLeft, SRight>
|
||||||
|
where
|
||||||
|
SLeft: Service<Req>,
|
||||||
|
SRight: Service<Req, Response = SLeft::Response, Error = SLeft::Error>,
|
||||||
|
{
|
||||||
|
type Response = SLeft::Response;
|
||||||
|
type Error = SLeft::Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
async fn ready(&self, ctx: ServiceCtx<'_, Self>) -> Result<(), Self::Error> {
|
||||||
|
match self.svc {
|
||||||
|
Either::Left(ref svc) => ctx.ready(svc).await,
|
||||||
|
Either::Right(ref svc) => ctx.ready(svc).await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
async fn shutdown(&self) {
|
||||||
|
match self.svc {
|
||||||
|
Either::Left(ref svc) => svc.shutdown().await,
|
||||||
|
Either::Right(ref svc) => svc.shutdown().await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
async fn call(
|
||||||
|
&self,
|
||||||
|
req: Req,
|
||||||
|
ctx: ServiceCtx<'_, Self>,
|
||||||
|
) -> Result<Self::Response, Self::Error> {
|
||||||
|
match self.svc {
|
||||||
|
Either::Left(ref svc) => ctx.call(svc, req).await,
|
||||||
|
Either::Right(ref svc) => ctx.call(svc, req).await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn poll(&self, cx: &mut Context<'_>) -> Result<(), Self::Error> {
|
||||||
|
match self.svc {
|
||||||
|
Either::Left(ref svc) => svc.poll(cx),
|
||||||
|
Either::Right(ref svc) => svc.poll(cx),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use ntex_service::{Pipeline, ServiceFactory};
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, PartialEq)]
|
||||||
|
struct Svc1;
|
||||||
|
impl Service<()> for Svc1 {
|
||||||
|
type Response = &'static str;
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<&'static str, ()> {
|
||||||
|
Ok("svc1")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct Svc1Factory;
|
||||||
|
impl ServiceFactory<(), &'static str> for Svc1Factory {
|
||||||
|
type Response = &'static str;
|
||||||
|
type Error = ();
|
||||||
|
type InitError = ();
|
||||||
|
type Service = Svc1;
|
||||||
|
|
||||||
|
async fn create(&self, _: &'static str) -> Result<Self::Service, Self::InitError> {
|
||||||
|
Ok(Svc1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, PartialEq)]
|
||||||
|
struct Svc2;
|
||||||
|
impl Service<()> for Svc2 {
|
||||||
|
type Response = &'static str;
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
async fn call(&self, _: (), _: ServiceCtx<'_, Self>) -> Result<&'static str, ()> {
|
||||||
|
Ok("svc2")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct Svc2Factory;
|
||||||
|
impl ServiceFactory<(), &'static str> for Svc2Factory {
|
||||||
|
type Response = &'static str;
|
||||||
|
type Error = ();
|
||||||
|
type InitError = ();
|
||||||
|
type Service = Svc2;
|
||||||
|
|
||||||
|
async fn create(&self, _: &'static str) -> Result<Self::Service, Self::InitError> {
|
||||||
|
Ok(Svc2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type Either = EitherService<Svc1, Svc2>;
|
||||||
|
type EitherFactory<F> = EitherServiceFactory<F, Svc1Factory, Svc2Factory>;
|
||||||
|
|
||||||
|
#[ntex_macros::rt_test2]
|
||||||
|
async fn test_success() {
|
||||||
|
let svc = Pipeline::new(Either::left(Svc1).clone());
|
||||||
|
assert_eq!(svc.call(()).await, Ok("svc1"));
|
||||||
|
assert_eq!(svc.ready().await, Ok(()));
|
||||||
|
svc.shutdown().await;
|
||||||
|
|
||||||
|
let svc = Pipeline::new(Either::right(Svc2).clone());
|
||||||
|
assert_eq!(svc.call(()).await, Ok("svc2"));
|
||||||
|
assert_eq!(svc.ready().await, Ok(()));
|
||||||
|
svc.shutdown().await;
|
||||||
|
|
||||||
|
assert!(format!("{:?}", svc).contains("EitherService"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[ntex_macros::rt_test2]
|
||||||
|
async fn test_factory() {
|
||||||
|
let factory =
|
||||||
|
EitherFactory::new(|s: &&'static str| *s == "svc1", Svc1Factory, Svc2Factory)
|
||||||
|
.clone();
|
||||||
|
assert!(format!("{:?}", factory).contains("EitherServiceFactory"));
|
||||||
|
|
||||||
|
let svc = factory.pipeline("svc1").await.unwrap();
|
||||||
|
assert_eq!(svc.call(()).await, Ok("svc1"));
|
||||||
|
|
||||||
|
let svc = factory.pipeline("other").await.unwrap();
|
||||||
|
assert_eq!(svc.call(()).await, Ok("svc2"));
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
pub mod buffer;
|
pub mod buffer;
|
||||||
|
pub mod either;
|
||||||
mod extensions;
|
mod extensions;
|
||||||
pub mod inflight;
|
pub mod inflight;
|
||||||
pub mod keepalive;
|
pub mod keepalive;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue