From a5f37b737c1ca55a532de6208965904fd16bd6c6 Mon Sep 17 00:00:00 2001 From: dswij Date: Sun, 17 Mar 2024 20:47:53 +0800 Subject: [PATCH 1/3] feat: add `{http1,http2}_only` for auto conn --- src/server/conn/auto.rs | 120 ++++++++++++++++++++++++++++++++++++---- 1 file changed, 110 insertions(+), 10 deletions(-) diff --git a/src/server/conn/auto.rs b/src/server/conn/auto.rs index 7e74d16..c32270b 100644 --- a/src/server/conn/auto.rs +++ b/src/server/conn/auto.rs @@ -58,6 +58,8 @@ pub struct Builder { http1: http1::Builder, #[cfg(feature = "http2")] http2: http2::Builder, + #[cfg(any(feature = "http1", feature = "http2"))] + version: Option, #[cfg(not(feature = "http2"))] _executor: E, } @@ -84,6 +86,8 @@ impl Builder { http1: http1::Builder::new(), #[cfg(feature = "http2")] http2: http2::Builder::new(executor), + #[cfg(any(feature = "http1", feature = "http2"))] + version: None, #[cfg(not(feature = "http2"))] _executor: executor, } @@ -101,6 +105,26 @@ impl Builder { Http2Builder { inner: self } } + /// Only accepts HTTP/2 + /// + /// Does not do anything if used with [`serve_connection_with_upgrades`] + #[cfg(feature = "http2")] + pub fn http2_only(mut self) -> Self { + assert!(self.version.is_none()); + self.version = Some(Version::H2); + self + } + + /// Only accepts HTTP/1 + /// + /// Does not do anything if used with [`serve_connection_with_upgrades`] + #[cfg(feature = "http1")] + pub fn http1_only(mut self) -> Self { + assert!(self.version.is_none()); + self.version = Some(Version::H1); + self + } + /// Bind a connection together with a [`Service`]. pub fn serve_connection(&self, io: I, service: S) -> Connection<'_, I, S, E> where @@ -112,13 +136,28 @@ impl Builder { I: Read + Write + Unpin + 'static, E: HttpServerConnExec, { - Connection { - state: ConnState::ReadVersion { + let state = match self.version { + #[cfg(feature = "http1")] + Some(Version::H1) => { + let io = Rewind::new_buffered(io, Bytes::new()); + let conn = self.http1.serve_connection(io, service); + ConnState::H1 { conn } + } + #[cfg(feature = "http2")] + Some(Version::H2) => { + let io = Rewind::new_buffered(io, Bytes::new()); + let conn = self.http2.serve_connection(io, service); + ConnState::H2 { conn } + } + #[cfg(any(feature = "http1", feature = "http2"))] + _ => ConnState::ReadVersion { read_version: read_version(io), builder: self, service: Some(service), }, - } + }; + + Connection { state } } /// Bind a connection together with a [`Service`], with the ability to @@ -148,7 +187,7 @@ impl Builder { } } -#[derive(Copy, Clone)] +#[derive(Copy, Clone, Debug)] enum Version { H1, H2, @@ -865,7 +904,7 @@ mod tests { #[cfg(not(miri))] #[tokio::test] async fn http1() { - let addr = start_server().await; + let addr = start_server(false, false).await; let mut sender = connect_h1(addr).await; let response = sender @@ -881,7 +920,23 @@ mod tests { #[cfg(not(miri))] #[tokio::test] async fn http2() { - let addr = start_server().await; + let addr = start_server(false, false).await; + let mut sender = connect_h2(addr).await; + + let response = sender + .send_request(Request::new(Empty::::new())) + .await + .unwrap(); + + let body = response.into_body().collect().await.unwrap().to_bytes(); + + assert_eq!(body, BODY); + } + + #[cfg(not(miri))] + #[tokio::test] + async fn http2_only() { + let addr = start_server(false, true).await; let mut sender = connect_h2(addr).await; let response = sender @@ -894,6 +949,46 @@ mod tests { assert_eq!(body, BODY); } + #[cfg(not(miri))] + #[tokio::test] + async fn http2_only_fail_if_client_is_http1() { + let addr = start_server(false, true).await; + let mut sender = connect_h1(addr).await; + + let _ = sender + .send_request(Request::new(Empty::::new())) + .await + .expect_err("should fail"); + } + + #[cfg(not(miri))] + #[tokio::test] + async fn http1_only() { + let addr = start_server(true, false).await; + let mut sender = connect_h1(addr).await; + + let response = sender + .send_request(Request::new(Empty::::new())) + .await + .unwrap(); + + let body = response.into_body().collect().await.unwrap().to_bytes(); + + assert_eq!(body, BODY); + } + + #[cfg(not(miri))] + #[tokio::test] + async fn http1_only_fail_if_client_is_http2() { + let addr = start_server(true, false).await; + let mut sender = connect_h2(addr).await; + + let _ = sender + .send_request(Request::new(Empty::::new())) + .await + .expect_err("should fail"); + } + #[cfg(not(miri))] #[tokio::test] async fn graceful_shutdown() { @@ -959,7 +1054,7 @@ mod tests { sender } - async fn start_server() -> SocketAddr { + async fn start_server(h1_only: bool, h2_only: bool) -> SocketAddr { let addr: SocketAddr = ([127, 0, 0, 1], 0).into(); let listener = TcpListener::bind(addr).await.unwrap(); @@ -970,9 +1065,14 @@ mod tests { let (stream, _) = listener.accept().await.unwrap(); let stream = TokioIo::new(stream); tokio::task::spawn(async move { - let _ = auto::Builder::new(TokioExecutor::new()) - .serve_connection(stream, service_fn(hello)) - .await; + let mut builder = auto::Builder::new(TokioExecutor::new()); + if h1_only { + builder = builder.http1_only(); + } else if h2_only { + builder = builder.http2_only(); + } + + builder.serve_connection(stream, service_fn(hello)).await; }); } }); From a434ae8fc322696ec077542ac0185b442560a13f Mon Sep 17 00:00:00 2001 From: tison Date: Fri, 24 May 2024 21:37:25 +0800 Subject: [PATCH 2/3] address comments Signed-off-by: tison --- src/server/conn/auto.rs | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/server/conn/auto.rs b/src/server/conn/auto.rs index 7366fe9..60138fa 100644 --- a/src/server/conn/auto.rs +++ b/src/server/conn/auto.rs @@ -1107,15 +1107,19 @@ mod tests { let stream = TokioIo::new(stream); tokio::task::spawn(async move { let mut builder = auto::Builder::new(TokioExecutor::new()); + + builder + .http2() + .max_header_list_size(4096); + if h1_only { builder = builder.http1_only(); } else if h2_only { builder = builder.http2_only(); } - + builder - .max_header_list_size(4096) - .serve_connection(stream, service_fn(hello)) + .serve_connection_with_upgrades(stream, service_fn(hello)) .await; }); } From 582cd9fcb926651f601e9d2423c9e54f568d0cf5 Mon Sep 17 00:00:00 2001 From: tison Date: Fri, 24 May 2024 21:51:57 +0800 Subject: [PATCH 3/3] fixup tests Signed-off-by: tison --- src/server/conn/auto.rs | 31 ++++++++++++++++++++++++------- 1 file changed, 24 insertions(+), 7 deletions(-) diff --git a/src/server/conn/auto.rs b/src/server/conn/auto.rs index 60138fa..5cd3694 100644 --- a/src/server/conn/auto.rs +++ b/src/server/conn/auto.rs @@ -178,10 +178,27 @@ impl Builder { E: HttpServerConnExec, { UpgradeableConnection { - state: UpgradeableConnState::ReadVersion { - read_version: read_version(io), - builder: self, - service: Some(service), + state: match self.version { + #[cfg(feature = "http1")] + Some(Version::H1) => { + let io = Rewind::new_buffered(io, Bytes::new()); + UpgradeableConnState::H1 { + conn: self.http1.serve_connection(io, service).with_upgrades(), + } + }, + #[cfg(feature = "http2")] + Some(Version::H2) => { + let io = Rewind::new_buffered(io, Bytes::new()); + UpgradeableConnState::H2 { + conn: self.http2.serve_connection(io, service), + } + }, + #[cfg(any(feature = "http1", feature = "http2"))] + _ => UpgradeableConnState::ReadVersion { + read_version: read_version(io), + builder: self, + service: Some(service), + } }, } } @@ -1107,17 +1124,17 @@ mod tests { let stream = TokioIo::new(stream); tokio::task::spawn(async move { let mut builder = auto::Builder::new(TokioExecutor::new()); - + builder .http2() .max_header_list_size(4096); - + if h1_only { builder = builder.http1_only(); } else if h2_only { builder = builder.http2_only(); } - + builder .serve_connection_with_upgrades(stream, service_fn(hello)) .await;