use crate::Error;
use crate::regex_generator::generate_exact_match_regex;
use crate::types::RequestInfo;
use http_body_util::Full;
use hyper::Response;
use hyper::body::Bytes;
use regex::Regex;
use std::fmt::{self, Debug, Formatter};
use std::future::Future;
use std::pin::Pin;
type HandlerWithoutInfo<E> = Box<dyn Fn(Response<Full<Bytes>>) -> HandlerWithoutInfoReturn<E> + Send + Sync + 'static>;
type HandlerWithoutInfoReturn<E> = Box<dyn Future<Output = Result<Response<Full<Bytes>>, E>> + Send + 'static>;
type HandlerWithInfo<E> =
Box<dyn Fn(Response<Full<Bytes>>, RequestInfo) -> HandlerWithInfoReturn<E> + Send + Sync + 'static>;
type HandlerWithInfoReturn<E> = Box<dyn Future<Output = Result<Response<Full<Bytes>>, E>> + Send + 'static>;
pub struct PostMiddleware<E> {
pub(crate) path: String,
pub(crate) regex: Regex,
pub(crate) handler: Option<Handler<E>>,
pub(crate) scope_depth: u32,
}
pub(crate) enum Handler<E> {
WithoutInfo(HandlerWithoutInfo<E>),
WithInfo(HandlerWithInfo<E>),
}
impl<E: Into<Box<dyn std::error::Error + Send + Sync>> + 'static> PostMiddleware<E> {
pub(crate) fn new_with_boxed_handler<P: Into<String>>(
path: P,
handler: Handler<E>,
scope_depth: u32,
) -> crate::Result<PostMiddleware<E>> {
let path = path.into();
let (re, _) = generate_exact_match_regex(path.as_str()).map_err(|e| {
Error::new(format!(
"Could not create an exact match regex for the post middleware path: {}",
e
))
})?;
Ok(PostMiddleware {
path,
regex: re,
handler: Some(handler),
scope_depth,
})
}
pub fn new<P, H, R>(path: P, handler: H) -> crate::Result<PostMiddleware<E>>
where
P: Into<String>,
H: Fn(Response<Full<Bytes>>) -> R + Send + Sync + 'static,
R: Future<Output = Result<Response<Full<Bytes>>, E>> + Send + 'static,
{
let handler: HandlerWithoutInfo<E> = Box::new(move |res: Response<Full<Bytes>>| Box::new(handler(res)));
PostMiddleware::new_with_boxed_handler(path, Handler::WithoutInfo(handler), 1)
}
pub fn new_with_info<P, H, R>(path: P, handler: H) -> crate::Result<PostMiddleware<E>>
where
P: Into<String>,
H: Fn(Response<Full<Bytes>>, RequestInfo) -> R + Send + Sync + 'static,
R: Future<Output = Result<Response<Full<Bytes>>, E>> + Send + 'static,
{
let handler: HandlerWithInfo<E> =
Box::new(move |res: Response<Full<Bytes>>, req_info: RequestInfo| Box::new(handler(res, req_info)));
PostMiddleware::new_with_boxed_handler(path, Handler::WithInfo(handler), 1)
}
pub(crate) fn should_require_req_meta(&self) -> bool {
if let Some(ref handler) = self.handler {
match handler {
Handler::WithInfo(_) => true,
Handler::WithoutInfo(_) => false,
}
} else {
false
}
}
pub(crate) async fn process(
&self,
res: Response<Full<Bytes>>,
req_info: Option<RequestInfo>,
) -> crate::Result<Response<Full<Bytes>>> {
let handler = self
.handler
.as_ref()
.expect("A router can not be used after mounting into another router");
match handler {
Handler::WithoutInfo(handler) => Pin::from(handler(res)).await.map_err(Into::into),
Handler::WithInfo(handler) => Pin::from(handler(res, req_info.expect("No RequestInfo is provided")))
.await
.map_err(Into::into),
}
}
}
impl<E> Debug for PostMiddleware<E> {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{{ path: {:?}, regex: {:?} }}", self.path, self.regex)
}
}