From dcb865a6dbc4848112975a741e0bb0e53314a99f Mon Sep 17 00:00:00 2001 From: tottoto Date: Tue, 3 Dec 2024 06:44:26 +0900 Subject: [PATCH] chore(web): Refactor constructing immediate response future --- tonic-web/src/service.rs | 46 +++++++++++++++++++++------------------- 1 file changed, 24 insertions(+), 22 deletions(-) diff --git a/tonic-web/src/service.rs b/tonic-web/src/service.rs index ca2a43d5b..e43223a36 100644 --- a/tonic-web/src/service.rs +++ b/tonic-web/src/service.rs @@ -43,24 +43,6 @@ impl GrpcWebService { } } -impl GrpcWebService -where - S: Service, Response = Response>, -{ - fn response(&self, status: StatusCode) -> ResponseFuture { - ResponseFuture { - case: Case::ImmediateResponse { - res: Some( - Response::builder() - .status(status) - .body(Body::default()) - .unwrap(), - ), - }, - } - } -} - impl Service> for GrpcWebService where S: Service, Response = Response>, @@ -106,7 +88,10 @@ where // This is not a valid grpc-web request, return HTTP 405. RequestKind::GrpcWeb { .. } => { debug!(kind = "simple", error="method not allowed", method = ?req.method()); - self.response(StatusCode::METHOD_NOT_ALLOWED) + + ResponseFuture { + case: Case::immediate(StatusCode::METHOD_NOT_ALLOWED), + } } // All http/2 requests that are not grpc-web are passed through to the inner service, @@ -123,7 +108,10 @@ where // Return HTTP 400 for all other requests. RequestKind::Other(_) => { debug!(kind = "other h1", content_type = ?req.headers().get(header::CONTENT_TYPE)); - self.response(StatusCode::BAD_REQUEST) + + ResponseFuture { + case: Case::immediate(StatusCode::BAD_REQUEST), + } } } } @@ -149,10 +137,21 @@ enum Case { future: F, }, ImmediateResponse { - res: Option>, + res: Option, }, } +impl Case { + fn immediate(status: StatusCode) -> Self { + let (res, ()) = Response::builder() + .status(status) + .body(()) + .unwrap() + .into_parts(); + Self::ImmediateResponse { res: Some(res) } + } +} + impl Future for ResponseFuture where F: Future, E>>, @@ -169,7 +168,10 @@ where Poll::Ready(Ok(coerce_response(res, *accept))) } CaseProj::Other { future } => future.poll(cx), - CaseProj::ImmediateResponse { res } => Poll::Ready(Ok(res.take().unwrap())), + CaseProj::ImmediateResponse { res } => { + let res = Response::from_parts(res.take().unwrap(), Body::empty()); + Poll::Ready(Ok(res)) + } } } }