use std::convert::Infallible; use std::future::Future; use std::marker::PhantomData; use std::pin::Pin; use std::task::{Context, Poll}; use actix_http::{Error, Response}; use actix_service::{Service, ServiceFactory}; use futures_util::future::{ok, Ready}; use futures_util::ready; use pin_project::pin_project; use crate::extract::FromRequest; use crate::request::HttpRequest; use crate::responder::Responder; use crate::service::{ServiceRequest, ServiceResponse}; /// Async handler converter factory pub trait Factory: Clone + 'static where R: Future, O: Responder, { fn call(&self, param: T) -> R; } impl Factory<(), R, O> for F where F: Fn() -> R + Clone + 'static, R: Future, O: Responder, { fn call(&self, _: ()) -> R { (self)() } } #[doc(hidden)] pub struct Handler where F: Factory, R: Future, O: Responder, { hnd: F, _t: PhantomData<(T, R, O)>, } impl Handler where F: Factory, R: Future, O: Responder, { pub fn new(hnd: F) -> Self { Handler { hnd, _t: PhantomData, } } } impl Clone for Handler where F: Factory, R: Future, O: Responder, { fn clone(&self) -> Self { Handler { hnd: self.hnd.clone(), _t: PhantomData, } } } impl Service for Handler where F: Factory, R: Future, O: Responder, { type Request = (T, HttpRequest); type Response = ServiceResponse; type Error = Infallible; type Future = HandlerServiceResponse; fn poll_ready(&mut self, _: &mut Context<'_>) -> Poll> { Poll::Ready(Ok(())) } fn call(&mut self, (param, req): (T, HttpRequest)) -> Self::Future { HandlerServiceResponse { fut: self.hnd.call(param), fut2: None, req: Some(req), } } } #[doc(hidden)] #[pin_project] pub struct HandlerServiceResponse where T: Future, R: Responder, { #[pin] fut: T, #[pin] fut2: Option, req: Option, } impl Future for HandlerServiceResponse where T: Future, R: Responder, { type Output = Result; fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { let this = self.as_mut().project(); if let Some(fut) = this.fut2.as_pin_mut() { return match fut.poll(cx) { Poll::Ready(Ok(res)) => { Poll::Ready(Ok(ServiceResponse::new(this.req.take().unwrap(), res))) } Poll::Pending => Poll::Pending, Poll::Ready(Err(e)) => { let res: Response = e.into().into(); Poll::Ready(Ok(ServiceResponse::new(this.req.take().unwrap(), res))) } }; } match this.fut.poll(cx) { Poll::Ready(res) => { let fut = res.respond_to(this.req.as_ref().unwrap()); self.as_mut().project().fut2.set(Some(fut)); self.poll(cx) } Poll::Pending => Poll::Pending, } } } /// Extract arguments from request pub struct Extract { service: S, _t: PhantomData, } impl Extract { pub fn new(service: S) -> Self { Extract { service, _t: PhantomData, } } } impl ServiceFactory for Extract where S: Service< Request = (T, HttpRequest), Response = ServiceResponse, Error = Infallible, > + Clone, { type Config = (); type Request = ServiceRequest; type Response = ServiceResponse; type Error = (Error, ServiceRequest); type InitError = (); type Service = ExtractService; type Future = Ready>; fn new_service(&self, _: ()) -> Self::Future { ok(ExtractService { _t: PhantomData, service: self.service.clone(), }) } } pub struct ExtractService { service: S, _t: PhantomData, } impl Service for ExtractService where S: Service< Request = (T, HttpRequest), Response = ServiceResponse, Error = Infallible, > + Clone, { type Request = ServiceRequest; type Response = ServiceResponse; type Error = (Error, ServiceRequest); type Future = ExtractResponse; fn poll_ready(&mut self, _: &mut Context<'_>) -> Poll> { Poll::Ready(Ok(())) } fn call(&mut self, req: ServiceRequest) -> Self::Future { let (req, mut payload) = req.into_parts(); let fut = T::from_request(&req, &mut payload); ExtractResponse { fut, req, fut_s: None, service: self.service.clone(), } } } #[pin_project] pub struct ExtractResponse { req: HttpRequest, service: S, #[pin] fut: T::Future, #[pin] fut_s: Option, } impl Future for ExtractResponse where S: Service< Request = (T, HttpRequest), Response = ServiceResponse, Error = Infallible, >, { type Output = Result; fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { let this = self.as_mut().project(); if let Some(fut) = this.fut_s.as_pin_mut() { return fut.poll(cx).map_err(|_| panic!()); } match ready!(this.fut.poll(cx)) { Err(e) => { let req = ServiceRequest::new(this.req.clone()); Poll::Ready(Err((e.into(), req))) } Ok(item) => { let fut = Some(this.service.call((item, this.req.clone()))); self.as_mut().project().fut_s.set(fut); self.poll(cx) } } } } /// FromRequest trait impl for tuples macro_rules! factory_tuple ({ $(($n:tt, $T:ident)),+} => { impl Factory<($($T,)+), Res, O> for Func where Func: Fn($($T,)+) -> Res + Clone + 'static, Res: Future, O: Responder, { fn call(&self, param: ($($T,)+)) -> Res { (self)($(param.$n,)+) } } }); #[rustfmt::skip] mod m { use super::*; factory_tuple!((0, A)); factory_tuple!((0, A), (1, B)); factory_tuple!((0, A), (1, B), (2, C)); factory_tuple!((0, A), (1, B), (2, C), (3, D)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E), (5, F)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E), (5, F), (6, G)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E), (5, F), (6, G), (7, H)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E), (5, F), (6, G), (7, H), (8, I)); factory_tuple!((0, A), (1, B), (2, C), (3, D), (4, E), (5, F), (6, G), (7, H), (8, I), (9, J)); }