feat: client (wip), still unstable for http2 due to alpn issues
This commit is contained in:
		
					parent
					
						
							
								f020ece60d
							
						
					
				
			
			
				commit
				
					
						a9f5e0ede5
					
				
			
		
					 15 changed files with 199 additions and 70 deletions
				
			
		|  | @ -78,6 +78,10 @@ pub enum RpxyError { | |||
|   #[error("Failed to copy bidirectional for upgraded connections: {0}")] | ||||
|   FailedToCopyBidirectional(String), | ||||
| 
 | ||||
|   // Forwarder errors
 | ||||
|   #[error("Failed to fetch from upstream: {0}")] | ||||
|   FailedToFetchFromUpstream(String), | ||||
| 
 | ||||
|   // Upstream connection setting errors
 | ||||
|   #[error("Unsupported upstream option")] | ||||
|   UnsupportedUpstreamOption, | ||||
|  |  | |||
							
								
								
									
										117
									
								
								rpxy-lib/src/forwarder/client.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								rpxy-lib/src/forwarder/client.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,117 @@ | |||
| use crate::{ | ||||
|   error::{RpxyError, RpxyResult}, | ||||
|   globals::Globals, | ||||
|   hyper_ext::{ | ||||
|     body::{wrap_incoming_body_response, IncomingOr}, | ||||
|     rt::LocalExecutor, | ||||
|   }, | ||||
| }; | ||||
| use async_trait::async_trait; | ||||
| use http::{Request, Response, Version}; | ||||
| use hyper::body::Body; | ||||
| use hyper_tls::HttpsConnector; | ||||
| use hyper_util::client::legacy::{ | ||||
|   connect::{Connect, HttpConnector}, | ||||
|   Client, | ||||
| }; | ||||
| use std::sync::Arc; | ||||
| 
 | ||||
| #[async_trait] | ||||
| /// Definition of the forwarder that simply forward requests from downstream client to upstream app servers.
 | ||||
| pub trait ForwardRequest<B1, B2> { | ||||
|   type Error; | ||||
|   async fn request(&self, req: Request<B1>) -> Result<Response<B2>, Self::Error>; | ||||
| } | ||||
| 
 | ||||
| /// Forwarder http client struct responsible to cache handling
 | ||||
| pub struct Forwarder<C, B> { | ||||
|   // #[cfg(feature = "cache")]
 | ||||
|   // cache: Option<RpxyCache>,
 | ||||
|   inner: Client<C, B>, | ||||
|   inner_h2: Client<C, B>, // `h2c` or http/2-only client is defined separately
 | ||||
| } | ||||
| 
 | ||||
| #[async_trait] | ||||
| impl<C, B1, B2> ForwardRequest<B1, IncomingOr<B2>> for Forwarder<C, B1> | ||||
| where | ||||
|   C: Send + Sync + Connect + Clone + 'static, | ||||
|   B1: Body + Send + Unpin + 'static, | ||||
|   <B1 as Body>::Data: Send, | ||||
|   <B1 as Body>::Error: Into<Box<(dyn std::error::Error + Send + Sync + 'static)>>, | ||||
|   B2: Body, | ||||
| { | ||||
|   type Error = RpxyError; | ||||
| 
 | ||||
|   async fn request(&self, req: Request<B1>) -> Result<Response<IncomingOr<B2>>, Self::Error> { | ||||
|     self.request_directly(req).await | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| impl<C, B1> Forwarder<C, B1> | ||||
| where | ||||
|   C: Send + Sync + Connect + Clone + 'static, | ||||
|   B1: Body + Send + Unpin + 'static, | ||||
|   <B1 as Body>::Data: Send, | ||||
|   <B1 as Body>::Error: Into<Box<(dyn std::error::Error + Send + Sync + 'static)>>, | ||||
| { | ||||
|   async fn request_directly<B2: Body>(&self, req: Request<B1>) -> RpxyResult<Response<IncomingOr<B2>>> { | ||||
|     match req.version() { | ||||
|       Version::HTTP_2 => self.inner_h2.request(req).await, // handles `h2c` requests
 | ||||
|       _ => self.inner.request(req).await, | ||||
|     } | ||||
|     .map_err(|e| RpxyError::FailedToFetchFromUpstream(e.to_string())) | ||||
|     .map(wrap_incoming_body_response::<B2>) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| impl<B1> Forwarder<HttpsConnector<HttpConnector>, B1> | ||||
| where | ||||
|   B1: Body + Send + Unpin + 'static, | ||||
|   <B1 as Body>::Data: Send, | ||||
|   <B1 as Body>::Error: Into<Box<(dyn std::error::Error + Send + Sync + 'static)>>, | ||||
| { | ||||
|   /// Build forwarder
 | ||||
|   pub async fn new(_globals: &Arc<Globals>) -> Self { | ||||
|     // build hyper client with hyper-tls
 | ||||
|     // TODO: Frame size errorが取れない > H2 どうしようもない。。。。 hyper_rustlsのリリース待ち?
 | ||||
|     let connector = HttpsConnector::new(); | ||||
|     let executor = LocalExecutor::new(_globals.runtime_handle.clone().clone()); | ||||
|     let inner = Client::builder(executor.clone()).build::<_, B1>(connector); | ||||
| 
 | ||||
|     let connector = HttpsConnector::new(); | ||||
|     let executor = LocalExecutor::new(_globals.runtime_handle.clone()); | ||||
|     let inner_h2 = Client::builder(executor) | ||||
|       .http2_adaptive_window(true) | ||||
|       .http2_only(true) | ||||
|       .set_host(true) | ||||
|       .build::<_, B1>(connector); | ||||
| 
 | ||||
|     // #[cfg(feature = "native-roots")]
 | ||||
|     // let builder = hyper_rustls::HttpsConnectorBuilder::new().with_native_roots();
 | ||||
|     // #[cfg(feature = "native-roots")]
 | ||||
|     // let builder_h2 = hyper_rustls::HttpsConnectorBuilder::new().with_native_roots();
 | ||||
|     // #[cfg(feature = "native-roots")]
 | ||||
|     // info!("Native cert store is used for the connection to backend applications");
 | ||||
| 
 | ||||
|     // #[cfg(not(feature = "native-roots"))]
 | ||||
|     // let builder = hyper_rustls::HttpsConnectorBuilder::new().with_webpki_roots();
 | ||||
|     // #[cfg(not(feature = "native-roots"))]
 | ||||
|     // let builder_h2 = hyper_rustls::HttpsConnectorBuilder::new().with_webpki_roots();
 | ||||
|     // #[cfg(not(feature = "native-roots"))]
 | ||||
|     // info!("Mozilla WebPKI root certs is used for the connection to backend applications");
 | ||||
| 
 | ||||
|     // let connector = builder.https_or_http().enable_http1().enable_http2().build();
 | ||||
|     // let connector_h2 = builder_h2.https_or_http().enable_http2().build();
 | ||||
| 
 | ||||
|     // let inner = Client::builder().build::<_, Body>(connector);
 | ||||
|     // let inner_h2 = Client::builder().http2_only(true).build::<_, Body>(connector_h2);
 | ||||
| 
 | ||||
|     // #[cfg(feature = "cache")]
 | ||||
|     // {
 | ||||
|     //   let cache = RpxyCache::new(_globals).await;
 | ||||
|     //   Self { inner, inner_h2, cache }
 | ||||
|     // }
 | ||||
|     // #[cfg(not(feature = "cache"))]
 | ||||
|     Self { inner, inner_h2 } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										8
									
								
								rpxy-lib/src/forwarder/mod.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								rpxy-lib/src/forwarder/mod.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | |||
| mod client; | ||||
| 
 | ||||
| use crate::hyper_ext::body::{IncomingLike, IncomingOr}; | ||||
| use hyper_tls::HttpsConnector; | ||||
| use hyper_util::client::legacy::connect::HttpConnector; | ||||
| pub type Forwarder = client::Forwarder<HttpsConnector<HttpConnector>, IncomingOr<IncomingLike>>; | ||||
| 
 | ||||
| pub use client::ForwardRequest; | ||||
|  | @ -1,3 +1,4 @@ | |||
| use http::Response; | ||||
| use http_body_util::{combinators, BodyExt, Either, Empty, Full}; | ||||
| use hyper::body::{Bytes, Incoming}; | ||||
| 
 | ||||
|  | @ -6,6 +7,19 @@ pub(crate) type BoxBody = combinators::BoxBody<Bytes, hyper::Error>; | |||
| /// Type for either passthrough body or given body type, specifically synthetic boxed body
 | ||||
| pub(crate) type IncomingOr<B> = Either<Incoming, B>; | ||||
| 
 | ||||
| /// helper function to build http response with passthrough body
 | ||||
| pub(crate) fn wrap_incoming_body_response<B>(response: Response<Incoming>) -> Response<IncomingOr<B>> | ||||
| where | ||||
|   B: hyper::body::Body, | ||||
| { | ||||
|   response.map(IncomingOr::Left) | ||||
| } | ||||
| 
 | ||||
| /// helper function to build http response with synthetic body
 | ||||
| pub(crate) fn wrap_synthetic_body_response<B>(response: Response<B>) -> Response<IncomingOr<B>> { | ||||
|   response.map(IncomingOr::Right) | ||||
| } | ||||
| 
 | ||||
| /// helper function to build a empty body
 | ||||
| pub(crate) fn empty() -> BoxBody { | ||||
|   Empty::<Bytes>::new().map_err(|never| match never {}).boxed() | ||||
|  |  | |||
|  | @ -8,5 +8,7 @@ pub(crate) mod rt { | |||
| } | ||||
| pub(crate) mod body { | ||||
|   pub(crate) use super::body_incoming_like::IncomingLike; | ||||
|   pub(crate) use super::body_type::{empty, full, BoxBody, IncomingOr}; | ||||
|   pub(crate) use super::body_type::{ | ||||
|     empty, full, wrap_incoming_body_response, wrap_synthetic_body_response, BoxBody, IncomingOr, | ||||
|   }; | ||||
| } | ||||
|  |  | |||
|  | @ -3,6 +3,7 @@ mod constants; | |||
| mod count; | ||||
| mod crypto; | ||||
| mod error; | ||||
| mod forwarder; | ||||
| mod globals; | ||||
| mod hyper_ext; | ||||
| mod log; | ||||
|  | @ -11,8 +12,8 @@ mod name_exp; | |||
| mod proxy; | ||||
| 
 | ||||
| use crate::{ | ||||
|   crypto::build_cert_reloader, error::*, globals::Globals, log::*, message_handler::HttpMessageHandlerBuilder, | ||||
|   proxy::Proxy, | ||||
|   crypto::build_cert_reloader, error::*, forwarder::Forwarder, globals::Globals, log::*, | ||||
|   message_handler::HttpMessageHandlerBuilder, proxy::Proxy, | ||||
| }; | ||||
| use futures::future::select_all; | ||||
| use std::sync::Arc; | ||||
|  | @ -90,10 +91,12 @@ where | |||
|   }); | ||||
| 
 | ||||
|   // 4. build message handler containing Arc-ed http_client and backends, and make it contained in Arc as well
 | ||||
|   let forwarder = Arc::new(Forwarder::new(&globals).await); | ||||
|   let message_handler = Arc::new( | ||||
|     HttpMessageHandlerBuilder::default() | ||||
|       .globals(globals.clone()) | ||||
|       .app_manager(app_manager.clone()) | ||||
|       .forwarder(forwarder) | ||||
|       .build()?, | ||||
|   ); | ||||
| 
 | ||||
|  |  | |||
|  | @ -9,6 +9,7 @@ use crate::{ | |||
|   backend::{BackendAppManager, LoadBalanceContext}, | ||||
|   crypto::CryptoSource, | ||||
|   error::*, | ||||
|   forwarder::{ForwardRequest, Forwarder}, | ||||
|   globals::Globals, | ||||
|   hyper_ext::body::{BoxBody, IncomingLike, IncomingOr}, | ||||
|   log::*, | ||||
|  | @ -18,7 +19,7 @@ use derive_builder::Builder; | |||
| use http::{Request, Response, StatusCode}; | ||||
| use hyper_util::rt::TokioIo; | ||||
| use std::{net::SocketAddr, sync::Arc}; | ||||
| use tokio::io::copy_bidirectional; | ||||
| use tokio::{io::copy_bidirectional, time::timeout}; | ||||
| 
 | ||||
| #[allow(dead_code)] | ||||
| #[derive(Debug)] | ||||
|  | @ -36,10 +37,9 @@ pub(super) struct HandlerContext { | |||
| // pub struct HttpMessageHandler<T, U>
 | ||||
| pub struct HttpMessageHandler<U> | ||||
| where | ||||
|   // T: Connect + Clone + Sync + Send + 'static,
 | ||||
|   U: CryptoSource + Clone, | ||||
| { | ||||
|   // forwarder: Arc<Forwarder<T>>,
 | ||||
|   forwarder: Arc<Forwarder>, | ||||
|   pub(super) globals: Arc<Globals>, | ||||
|   app_manager: Arc<BackendAppManager<U>>, | ||||
| } | ||||
|  | @ -81,7 +81,7 @@ where | |||
|         Ok(v) | ||||
|       } | ||||
|       Err(e) => { | ||||
|         debug!("{e}"); | ||||
|         error!("{e}"); | ||||
|         let code = StatusCode::from(e); | ||||
|         log_data.status_code(&code).output(); | ||||
|         synthetic_error_response(code) | ||||
|  | @ -155,14 +155,14 @@ where | |||
|       tls_enabled, | ||||
|     ) { | ||||
|       Err(e) => { | ||||
|         error!("Failed to generate upstream request for backend application: {}", e); | ||||
|         return Err(HttpError::FailedToGenerateUpstreamRequest(e.to_string())); | ||||
|       } | ||||
|       Ok(v) => v, | ||||
|     }; | ||||
|     debug!( | ||||
|       "Request to be forwarded: uri {}, version {:?}, headers {:?}", | ||||
|       "Request to be forwarded: [uri {}, method: {}, version {:?}, headers {:?}]", | ||||
|       req.uri(), | ||||
|       req.method(), | ||||
|       req.version(), | ||||
|       req.headers() | ||||
|     ); | ||||
|  | @ -171,37 +171,31 @@ where | |||
|     //////
 | ||||
| 
 | ||||
|     //////////////
 | ||||
|     // // TODO: remove later
 | ||||
|     let body = crate::hyper_ext::body::full(hyper::body::Bytes::from("not yet implemented")); | ||||
|     let mut res_backend = super::synthetic_response::synthetic_response(Response::builder().body(body).unwrap()); | ||||
|     // // Forward request to a chosen backend
 | ||||
|     // let mut res_backend = {
 | ||||
|     //   let Ok(result) = timeout(self.globals.proxy_config.upstream_timeout, self.forwarder.request(req)).await else {
 | ||||
|     //     return self.return_with_error_log(StatusCode::GATEWAY_TIMEOUT, &mut log_data);
 | ||||
|     //   };
 | ||||
|     //   match result {
 | ||||
|     //     Ok(res) => res,
 | ||||
|     //     Err(e) => {
 | ||||
|     //       error!("Failed to get response from backend: {}", e);
 | ||||
|     //       return self.return_with_error_log(StatusCode::SERVICE_UNAVAILABLE, &mut log_data);
 | ||||
|     //     }
 | ||||
|     //   }
 | ||||
|     // };
 | ||||
|     // Forward request to a chosen backend
 | ||||
|     let mut res_backend = { | ||||
|       let Ok(result) = timeout(self.globals.proxy_config.upstream_timeout, self.forwarder.request(req)).await else { | ||||
|         return Err(HttpError::TimeoutUpstreamRequest); | ||||
|       }; | ||||
|       match result { | ||||
|         Ok(res) => res, | ||||
|         Err(e) => { | ||||
|           return Err(HttpError::FailedToGetResponseFromBackend(e.to_string())); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
|     //////////////
 | ||||
|     // Process reverse proxy context generated during the forwarding request generation.
 | ||||
|     #[cfg(feature = "sticky-cookie")] | ||||
|     if let Some(context_from_lb) = _context.context_lb { | ||||
|       let res_headers = res_backend.headers_mut(); | ||||
|       if let Err(e) = set_sticky_cookie_lb_context(res_headers, &context_from_lb) { | ||||
|         error!("Failed to append context to the response given from backend: {}", e); | ||||
|         return Err(HttpError::FailedToAddSetCookeInResponse); | ||||
|         return Err(HttpError::FailedToAddSetCookeInResponse(e.to_string())); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if res_backend.status() != StatusCode::SWITCHING_PROTOCOLS { | ||||
|       // Generate response to client
 | ||||
|       if let Err(e) = self.generate_response_forwarded(&mut res_backend, backend_app) { | ||||
|         error!("Failed to generate downstream response for clients: {}", e); | ||||
|         return Err(HttpError::FailedToGenerateDownstreamResponse(e.to_string())); | ||||
|       } | ||||
|       return Ok(res_backend); | ||||
|  | @ -215,18 +209,15 @@ where | |||
|     }; | ||||
| 
 | ||||
|     if !should_upgrade { | ||||
|       error!( | ||||
|       return Err(HttpError::FailedToUpgrade(format!( | ||||
|         "Backend tried to switch to protocol {:?} when {:?} was requested", | ||||
|         upgrade_in_response, upgrade_in_request | ||||
|       ); | ||||
|       return Err(HttpError::FailedToUpgrade); | ||||
|       ))); | ||||
|     } | ||||
|     let Some(request_upgraded) = request_upgraded else { | ||||
|       error!("Request does not have an upgrade extension"); | ||||
|       return Err(HttpError::NoUpgradeExtensionInRequest); | ||||
|     }; | ||||
|     let Some(onupgrade) = res_backend.extensions_mut().remove::<hyper::upgrade::OnUpgrade>() else { | ||||
|       error!("Response does not have an upgrade extension"); | ||||
|       return Err(HttpError::NoUpgradeExtensionInResponse); | ||||
|     }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -20,16 +20,20 @@ pub enum HttpError { | |||
|   FailedToRedirect(String), | ||||
|   #[error("No upstream candidates")] | ||||
|   NoUpstreamCandidates, | ||||
|   #[error("Failed to generate upstream request: {0}")] | ||||
|   #[error("Failed to generate upstream request for backend application: {0}")] | ||||
|   FailedToGenerateUpstreamRequest(String), | ||||
|   #[error("Timeout in upstream request")] | ||||
|   TimeoutUpstreamRequest, | ||||
|   #[error("Failed to get response from backend: {0}")] | ||||
|   FailedToGetResponseFromBackend(String), | ||||
| 
 | ||||
|   #[error("Failed to add set-cookie header in response")] | ||||
|   FailedToAddSetCookeInResponse, | ||||
|   #[error("Failed to generated downstream response: {0}")] | ||||
|   #[error("Failed to add set-cookie header in response {0}")] | ||||
|   FailedToAddSetCookeInResponse(String), | ||||
|   #[error("Failed to generated downstream response for clients: {0}")] | ||||
|   FailedToGenerateDownstreamResponse(String), | ||||
| 
 | ||||
|   #[error("Failed to upgrade connection")] | ||||
|   FailedToUpgrade, | ||||
|   #[error("Failed to upgrade connection: {0}")] | ||||
|   FailedToUpgrade(String), | ||||
|   #[error("Request does not have an upgrade extension")] | ||||
|   NoUpgradeExtensionInRequest, | ||||
|   #[error("Response does not have an upgrade extension")] | ||||
|  | @ -49,9 +53,10 @@ impl From<HttpError> for StatusCode { | |||
|       HttpError::FailedToRedirect(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::NoUpstreamCandidates => StatusCode::NOT_FOUND, | ||||
|       HttpError::FailedToGenerateUpstreamRequest(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::FailedToAddSetCookeInResponse => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::TimeoutUpstreamRequest => StatusCode::GATEWAY_TIMEOUT, | ||||
|       HttpError::FailedToAddSetCookeInResponse(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::FailedToGenerateDownstreamResponse(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::FailedToUpgrade => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::FailedToUpgrade(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|       HttpError::NoUpgradeExtensionInRequest => StatusCode::BAD_REQUEST, | ||||
|       HttpError::NoUpgradeExtensionInResponse => StatusCode::BAD_GATEWAY, | ||||
|       _ => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|  |  | |||
|  | @ -1,25 +1,10 @@ | |||
| use super::http_result::{HttpError, HttpResult}; | ||||
| use crate::{ | ||||
|   error::*, | ||||
|   hyper_ext::body::{empty, BoxBody, IncomingOr}, | ||||
|   name_exp::ServerName, | ||||
| }; | ||||
| use http::{Request, Response, StatusCode, Uri}; | ||||
| use hyper::body::Incoming; | ||||
| 
 | ||||
| use super::http_result::{HttpError, HttpResult}; | ||||
| 
 | ||||
| /// helper function to build http response with passthrough body
 | ||||
| pub(crate) fn passthrough_response<B>(response: Response<Incoming>) -> Response<IncomingOr<B>> | ||||
| where | ||||
|   B: hyper::body::Body, | ||||
| { | ||||
|   response.map(IncomingOr::Left) | ||||
| } | ||||
| 
 | ||||
| /// helper function to build http response with synthetic body
 | ||||
| pub(crate) fn synthetic_response<B>(response: Response<B>) -> Response<IncomingOr<B>> { | ||||
|   response.map(IncomingOr::Right) | ||||
| } | ||||
| 
 | ||||
| /// build http response with status code of 4xx and 5xx
 | ||||
| pub(crate) fn synthetic_error_response(status_code: StatusCode) -> RpxyResult<Response<IncomingOr<BoxBody>>> { | ||||
|  |  | |||
|  | @ -57,11 +57,11 @@ pub(super) fn apply_upstream_options_to_request_line<B>( | |||
| ) -> anyhow::Result<()> { | ||||
|   for opt in upstream.options.iter() { | ||||
|     match opt { | ||||
|       UpstreamOption::ForceHttp11Upstream => *req.version_mut() = hyper::Version::HTTP_11, | ||||
|       UpstreamOption::ForceHttp11Upstream => *req.version_mut() = http::Version::HTTP_11, | ||||
|       UpstreamOption::ForceHttp2Upstream => { | ||||
|         // case: h2c -> https://www.rfc-editor.org/rfc/rfc9113.txt
 | ||||
|         // Upgrade from HTTP/1.1 to HTTP/2 is deprecated. So, http-2 prior knowledge is required.
 | ||||
|         *req.version_mut() = hyper::Version::HTTP_2; | ||||
|         *req.version_mut() = http::Version::HTTP_2; | ||||
|       } | ||||
|       _ => (), | ||||
|     } | ||||
|  |  | |||
|  | @ -92,13 +92,9 @@ where | |||
|   } | ||||
| 
 | ||||
|   /// Serves a request stream from a client
 | ||||
|   /// TODO: TODO: TODO: TODO:
 | ||||
|   /// TODO: Body in hyper-0.14 was changed to Incoming in hyper-1.0, and it is not accessible from outside.
 | ||||
|   /// Thus, we need to implement IncomingLike trait using channel. Also, the backend handler must feed the body in the form of
 | ||||
|   /// Body in hyper-0.14 was changed to Incoming in hyper-1.0, and it is not accessible from outside.
 | ||||
|   /// Thus, we needed to implement IncomingLike trait using channel. Also, the backend handler must feed the body in the form of
 | ||||
|   /// Either<Incoming, IncomingLike> as body.
 | ||||
|   /// Also, the downstream from the backend handler could be Incoming, but will be wrapped as Either<Incoming, ()/Empty> as well due to H3.
 | ||||
|   /// Result<Either<_,_>, E> type includes E as HttpError to generate the status code and related Response<BoxBody>.
 | ||||
|   /// Thus to handle synthetic error messages in BoxBody, the serve() function outputs Response<Either<Either<Incoming, ()/Empty>, BoxBody>>>.
 | ||||
|   async fn h3_serve_stream<S>( | ||||
|     &self, | ||||
|     req: Request<()>, | ||||
|  | @ -146,7 +142,7 @@ where | |||
|       Ok(()) as RpxyResult<()> | ||||
|     }); | ||||
| 
 | ||||
|     let mut new_req: Request<IncomingOr<IncomingLike>> = Request::from_parts(req_parts, IncomingOr::Right(req_body)); | ||||
|     let new_req: Request<IncomingOr<IncomingLike>> = Request::from_parts(req_parts, IncomingOr::Right(req_body)); | ||||
|     // Response<IncomingOr<BoxBody>> wrapped by RpxyResult
 | ||||
|     let res = self | ||||
|       .message_handler | ||||
|  |  | |||
|  | @ -26,7 +26,7 @@ use tokio::time::timeout; | |||
| /// Wrapper function to handle request for HTTP/1.1 and HTTP/2
 | ||||
| /// HTTP/3 is handled in proxy_h3.rs which directly calls the message handler
 | ||||
| async fn serve_request<U>( | ||||
|   mut req: Request<Incoming>, | ||||
|   req: Request<Incoming>, | ||||
|   // handler: Arc<HttpMessageHandler<T, U>>,
 | ||||
|   handler: Arc<HttpMessageHandler<U>>, | ||||
|   client_addr: SocketAddr, | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Jun Kurihara
				Jun Kurihara