mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 04:47:39 +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
|
||||
|
||||
## [2.9.0] - 2025-01-15
|
||||
|
||||
* Add EitherService/EitherServiceFactory
|
||||
|
||||
## [2.8.0] - 2024-12-04
|
||||
|
||||
* Use updated Service trait
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
[package]
|
||||
name = "ntex-util"
|
||||
version = "2.8.0"
|
||||
version = "2.9.0"
|
||||
authors = ["ntex contributors <team@ntex.rs>"]
|
||||
description = "Utilities for ntex framework"
|
||||
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 either;
|
||||
mod extensions;
|
||||
pub mod inflight;
|
||||
pub mod keepalive;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue