mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-03 21:07:39 +03:00
Add EitherService/EitherServiceFactory
This commit is contained in:
parent
cd56883197
commit
1b28c7ff74
4 changed files with 157 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"]
|
||||
|
|
151
ntex-util/src/services/either.rs
Normal file
151
ntex-util/src/services/either.rs
Normal file
|
@ -0,0 +1,151 @@
|
|||
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),
|
||||
}
|
||||
}
|
||||
}
|
|
@ -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