diff --git a/ntex-service/src/middleware.rs b/ntex-service/src/middleware.rs index b3656523..39592c38 100644 --- a/ntex-service/src/middleware.rs +++ b/ntex-service/src/middleware.rs @@ -27,23 +27,21 @@ where /// timeout: Duration, /// } /// -/// impl Service for Timeout +/// impl Service for Timeout /// where -/// S: Service, +/// S: Service, /// { -/// type Request = S::Request; /// type Response = S::Response; /// type Error = TimeoutError; -/// type Future = TimeoutResponse; /// /// fn poll_ready(&self, cx: &mut Context<'_>) -> Poll> { -/// ready!(self.service.poll_ready(cx)).map_err(TimeoutError::Service) +/// self.service.poll_ready(cx).map_err(TimeoutError::Service) /// } /// -/// fn call(&self, req: S::Request) -> Self::Future { -/// TimeoutServiceResponse { -/// fut: self.service.call(req), -/// sleep: Delay::new(clock::now() + self.timeout), +/// async fn call(&self, req: S::Request) -> Result { +/// match select(sleep(self.timeout), ctx.call(&self.service, req)).await { +/// Either::Left(_) => Err(TimeoutError::Timeout), +/// Either::Right(res) => res.map_err(TimeoutError::Service), /// } /// } /// } @@ -65,8 +63,6 @@ where /// } /// /// impl Middleware for TimeoutMiddleware -/// where -/// S: Service, /// { /// type Service = Timeout; /// diff --git a/ntex/CHANGES.md b/ntex/CHANGES.md index ca321576..2961ebfa 100644 --- a/ntex/CHANGES.md +++ b/ntex/CHANGES.md @@ -1,5 +1,9 @@ # Changes +## [1.0.0] - 2024-01-09 + +* web: Use async fn for Responder and Handler traits + ## [1.0.0-b.1] - 2024-01-08 * web: Refactor FromRequest trait, use async fn diff --git a/ntex/src/web/handler.rs b/ntex/src/web/handler.rs index c191f83f..d6950dba 100644 --- a/ntex/src/web/handler.rs +++ b/ntex/src/web/handler.rs @@ -14,11 +14,8 @@ where Err: ErrorRenderer, { type Output: Responder; - type Future<'f>: Future - where - Self: 'f; - fn call(&self, param: T) -> Self::Future<'_>; + fn call(&self, param: T) -> impl Future; } impl Handler<(), Err> for F @@ -28,11 +25,10 @@ where R::Output: Responder, Err: ErrorRenderer, { - type Future<'f> = R where Self: 'f; type Output = R::Output; - fn call(&self, _: ()) -> R { - (self)() + async fn call(&self, _: ()) -> R::Output { + (self)().await } } @@ -96,11 +92,10 @@ macro_rules! factory_tuple ({ $(($n:tt, $T:ident)),+} => { Res::Output: Responder, Err: ErrorRenderer, { - type Future<'f> = Res where Self: 'f; type Output = Res::Output; - fn call(&self, param: ($($T,)+)) -> Res { - (self)($(param.$n,)+) + async fn call(&self, param: ($($T,)+)) -> Self::Output { + (self)($(param.$n,)+).await } } }); diff --git a/ntex/src/web/mod.rs b/ntex/src/web/mod.rs index 4d8d75b3..a2f891fc 100644 --- a/ntex/src/web/mod.rs +++ b/ntex/src/web/mod.rs @@ -155,11 +155,11 @@ pub mod dev { #[inline(always)] pub fn __assert_handler( f: Fun, - ) -> impl for<'r> Handler<(), Err, Future<'r> = Fut, Output = Fut::Output> + ) -> impl Handler<(), Err, Output = Fut::Output> where Err: super::ErrorRenderer, Fun: Fn() -> Fut + 'static, - Fut: std::future::Future + 'static, + Fut: std::future::Future, Fut::Output: super::Responder, { f @@ -170,12 +170,12 @@ pub mod dev { #[inline(always)] pub fn $name( f: Fun, - ) -> impl for<'r> Handler<($($T,)+), Err, Future<'r> = Fut, Output = Fut::Output> + ) -> impl Handler<($($T,)+), Err, Output = Fut::Output> where Err: $crate::web::ErrorRenderer, Fun: Fn($($T,)+) -> Fut + 'static, Fut: std::future::Future + 'static, - Fut::Output: $crate::web::Responder, + Fut::Output: super::Responder, $($T: $crate::web::FromRequest),+, { f diff --git a/ntex/src/web/responder.rs b/ntex/src/web/responder.rs index 359a0adc..4e31fdce 100644 --- a/ntex/src/web/responder.rs +++ b/ntex/src/web/responder.rs @@ -1,5 +1,4 @@ -use std::task::{Context, Poll}; -use std::{future::Future, marker::PhantomData, pin::Pin}; +use std::{future::Future, marker::PhantomData}; use crate::http::error::HttpError; use crate::http::header::{HeaderMap, HeaderName, HeaderValue}; @@ -11,34 +10,12 @@ use super::error::{ }; use super::httprequest::HttpRequest; -pub struct Ready(Option); - -impl Unpin for Ready {} - -impl From for Ready { - fn from(t: T) -> Self { - Ready(Some(t)) - } -} - -impl Future for Ready { - type Output = T; - - #[inline] - fn poll(mut self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll { - Poll::Ready(self.0.take().expect("Ready polled after completion")) - } -} - /// Trait implemented by types that can be converted to a http response. /// /// Types that implement this trait can be used as the return type of a handler. pub trait Responder { - /// The future response value. - type Future: Future; - - /// Convert itself to `AsyncResult` or `Error`. - fn respond_to(self, req: &HttpRequest) -> Self::Future; + /// Convert itself to http response. + fn respond_to(self, req: &HttpRequest) -> impl Future; /// Override a status code for a Responder. /// @@ -90,20 +67,16 @@ pub trait Responder { } impl Responder for Response { - type Future = Ready; - #[inline] - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some(self)) + async fn respond_to(self, _: &HttpRequest) -> Response { + self } } impl Responder for ResponseBuilder { - type Future = Ready; - #[inline] - fn respond_to(mut self, _: &HttpRequest) -> Self::Future { - Ready(Some(self.finish())) + async fn respond_to(mut self, _: &HttpRequest) -> Response { + self.finish() } } @@ -112,14 +85,10 @@ where T: Responder, Err: ErrorRenderer, { - type Future = Either>; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { + async fn respond_to(self, req: &HttpRequest) -> Response { match self { - Some(t) => Either::Left(t.respond_to(req)), - None => { - Either::Right(Ready(Some(Response::build(StatusCode::NOT_FOUND).finish()))) - } + Some(t) => t.respond_to(req).await, + None => Response::build(StatusCode::NOT_FOUND).finish(), } } } @@ -130,12 +99,10 @@ where E: Into, Err: ErrorRenderer, { - type Future = Either>; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { + async fn respond_to(self, req: &HttpRequest) -> Response { match self { - Ok(val) => Either::Left(val.respond_to(req)), - Err(e) => Either::Right(Ready(Some(e.into().error_response(req)))), + Ok(val) => val.respond_to(req).await, + Err(e) => e.into().error_response(req), } } } @@ -145,86 +112,58 @@ where T: Responder, Err: ErrorRenderer, { - type Future = CustomResponderFut; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { - CustomResponderFut { - fut: self.0.respond_to(req), - status: Some(self.1), - headers: None, - } + async fn respond_to(self, req: &HttpRequest) -> Response { + let mut res = self.0.respond_to(req).await; + *res.status_mut() = self.1; + res } } impl Responder for &'static str { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("text/plain; charset=utf-8") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("text/plain; charset=utf-8") + .body(self) } } impl Responder for &'static [u8] { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("application/octet-stream") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("application/octet-stream") + .body(self) } } impl Responder for String { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("text/plain; charset=utf-8") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("text/plain; charset=utf-8") + .body(self) } } impl<'a, Err: ErrorRenderer> Responder for &'a String { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("text/plain; charset=utf-8") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("text/plain; charset=utf-8") + .body(self) } } impl Responder for Bytes { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("application/octet-stream") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("application/octet-stream") + .body(self) } } impl Responder for BytesMut { - type Future = Ready; - - fn respond_to(self, _: &HttpRequest) -> Self::Future { - Ready(Some( - Response::build(StatusCode::OK) - .content_type("application/octet-stream") - .body(self), - )) + async fn respond_to(self, _: &HttpRequest) -> Response { + Response::build(StatusCode::OK) + .content_type("application/octet-stream") + .body(self) } } @@ -308,47 +247,18 @@ impl, Err> CustomResponder { } impl, Err: ErrorRenderer> Responder for CustomResponder { - type Future = CustomResponderFut; + async fn respond_to(self, req: &HttpRequest) -> Response { + let mut res = self.responder.respond_to(req).await; - fn respond_to(self, req: &HttpRequest) -> Self::Future { - CustomResponderFut { - fut: self.responder.respond_to(req), - status: self.status, - headers: self.headers, - } - } -} - -pin_project_lite::pin_project! { - pub struct CustomResponderFut, Err: ErrorRenderer> { - #[pin] - fut: T::Future, - status: Option, - headers: Option, - } -} - -impl, Err: ErrorRenderer> Future for CustomResponderFut { - type Output = Response; - - fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { - let this = self.project(); - - let mut res = if let Poll::Ready(res) = this.fut.poll(cx) { - res - } else { - return Poll::Pending; - }; - - if let Some(status) = this.status.take() { + if let Some(status) = self.status { *res.status_mut() = status; } - if let Some(ref headers) = this.headers { + if let Some(ref headers) = self.headers { for (k, v) in headers { res.headers_mut().insert(k.clone(), v.clone()); } } - Poll::Ready(res) + res } } @@ -375,12 +285,10 @@ where B: Responder, Err: ErrorRenderer, { - type Future = Either; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { + async fn respond_to(self, req: &HttpRequest) -> Response { match self { - Either::Left(a) => Either::Left(a.respond_to(req)), - Either::Right(b) => Either::Right(b.respond_to(req)), + Either::Left(a) => a.respond_to(req).await, + Either::Right(b) => b.respond_to(req).await, } } } @@ -390,10 +298,8 @@ where T: std::fmt::Debug + std::fmt::Display + 'static, Err: ErrorRenderer, { - type Future = Ready; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { - Ready(Some(self.error_response(req))) + async fn respond_to(self, req: &HttpRequest) -> Response { + self.error_response(req) } } diff --git a/ntex/src/web/types/form.rs b/ntex/src/web/types/form.rs index d2d2df44..dba40d62 100644 --- a/ntex/src/web/types/form.rs +++ b/ntex/src/web/types/form.rs @@ -10,8 +10,7 @@ use crate::http::header::{CONTENT_LENGTH, CONTENT_TYPE}; use crate::http::{HttpMessage, Payload, Response, StatusCode}; use crate::util::{stream_recv, BoxFuture, BytesMut}; use crate::web::error::{ErrorRenderer, UrlencodedError, WebResponseError}; -use crate::web::responder::{Ready, Responder}; -use crate::web::{FromRequest, HttpRequest}; +use crate::web::{FromRequest, HttpRequest, Responder}; /// Form data helper (`application/x-www-form-urlencoded`) /// @@ -134,18 +133,15 @@ impl Responder for Form where Err::Container: From, { - type Future = Ready; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { + async fn respond_to(self, req: &HttpRequest) -> Response { let body = match serde_urlencoded::to_string(&self.0) { Ok(body) => body, - Err(e) => return e.error_response(req).into(), + Err(e) => return e.error_response(req), }; Response::build(StatusCode::OK) .header(CONTENT_TYPE, "application/x-www-form-urlencoded") .body(body) - .into() } } diff --git a/ntex/src/web/types/json.rs b/ntex/src/web/types/json.rs index 6eba6795..9e9ca8e9 100644 --- a/ntex/src/web/types/json.rs +++ b/ntex/src/web/types/json.rs @@ -9,8 +9,7 @@ use crate::http::header::CONTENT_LENGTH; use crate::http::{HttpMessage, Payload, Response, StatusCode}; use crate::util::{stream_recv, BoxFuture, BytesMut}; use crate::web::error::{ErrorRenderer, JsonError, JsonPayloadError, WebResponseError}; -use crate::web::responder::{Ready, Responder}; -use crate::web::{FromRequest, HttpRequest}; +use crate::web::{FromRequest, HttpRequest, Responder}; /// Json helper /// @@ -112,18 +111,15 @@ impl Responder for Json where Err::Container: From, { - type Future = Ready; - - fn respond_to(self, req: &HttpRequest) -> Self::Future { + async fn respond_to(self, req: &HttpRequest) -> Response { let body = match serde_json::to_string(&self.0) { Ok(body) => body, - Err(e) => return e.error_response(req).into(), + Err(e) => return e.error_response(req), }; Response::build(StatusCode::OK) .content_type("application/json") .body(body) - .into() } }