From a5d2590c8e23fdd8ff57f8e2addb7271898c3f2a Mon Sep 17 00:00:00 2001 From: Sam Clark <3758302+goatgoose@users.noreply.github.com> Date: Wed, 11 Dec 2024 18:33:04 -0500 Subject: [PATCH 1/2] refactor(s2n-tls-hyper): Add HttpsConnector builder --- bindings/rust/s2n-tls-hyper/src/connector.rs | 126 +++++++++++++++---- 1 file changed, 101 insertions(+), 25 deletions(-) diff --git a/bindings/rust/s2n-tls-hyper/src/connector.rs b/bindings/rust/s2n-tls-hyper/src/connector.rs index ea5670d62f0..338a5e4c07a 100644 --- a/bindings/rust/s2n-tls-hyper/src/connector.rs +++ b/bindings/rust/s2n-tls-hyper/src/connector.rs @@ -12,6 +12,7 @@ use s2n_tls::{config::Config, connection}; use s2n_tls_tokio::TlsConnector; use std::{ future::Future, + marker::PhantomData, pin::Pin, task::{Context, Poll}, }; @@ -24,49 +25,101 @@ use tower_service::Service; /// which sends and receives requests over TCP. The `HttpsConnector` struct wraps an HTTP connector, /// and uses it to negotiate TLS when the HTTPS scheme is in use. #[derive(Clone)] -pub struct HttpsConnector { +pub struct HttpsConnector { http: Http, - conn_builder: Builder, + conn_builder: ConnBuilder, } -impl HttpsConnector +impl HttpsConnector where - Builder: connection::Builder, - ::Output: Unpin, + ConnBuilder: connection::Builder, + ::Output: Unpin, { - /// Creates a new `HttpsConnector`. + /// Creates a new `HttpsConnector` with the default configuration. Use `HttpsConnector::builder` + /// instead to configure an `HttpsConnector`. + /// + /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, + /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. + /// + /// ``` + /// use s2n_tls_hyper::connector::HttpsConnector; + /// use s2n_tls::config::Config; + /// + /// // Create a new HttpsConnector. + /// let connector = HttpsConnector::new(Config::default()); + /// ``` + /// + /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values + /// configured on `conn_builder` with APIs like + /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. + pub fn new(conn_builder: ConnBuilder) -> HttpsConnector { + HttpsConnector::builder().build(conn_builder) + } +} + +impl HttpsConnector +where + ConnBuilder: connection::Builder, + ::Output: Unpin, +{ + /// Creates a `Builder` to configure a new `HttpsConnector`. + /// + /// ``` + /// use s2n_tls_hyper::connector::HttpsConnector; + /// use s2n_tls::config::Config; + /// + /// // Create and configure a builder. + /// let builder = HttpsConnector::builder(); + /// + /// // Build a new HttpsConnector. + /// let connector = builder.build(Config::default()); + /// ``` + pub fn builder() -> Builder { + Builder { + http: PhantomData, + conn_builder: PhantomData, + } + } +} + +/// Builder used to configure an `HttpsConnector`. Create a new Builder with +/// `HttpsConnector::builder`. +pub struct Builder { + http: PhantomData, + conn_builder: PhantomData, +} + +impl Builder { + /// Builds a new `HttpsConnector`. /// /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. /// /// This API creates an `HttpsConnector` using the default hyper `HttpConnector`. To use an - /// existing HTTP connector, use `HttpsConnector::new_with_http()`. + /// existing HTTP connector, use `Builder::build_with_http()`. /// /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values /// configured on `conn_builder` with APIs like /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. - pub fn new(conn_builder: Builder) -> HttpsConnector { + pub fn build(self, conn_builder: ConnBuilder) -> HttpsConnector { let mut http = HttpConnector::new(); // By default, the `HttpConnector` only allows the HTTP URI scheme to be used. To negotiate // HTTP over TLS via the HTTPS scheme, `enforce_http` must be disabled. http.enforce_http(false); - Self { http, conn_builder } + self.build_with_http(http, conn_builder) } } -impl HttpsConnector -where - Builder: connection::Builder, - ::Output: Unpin, -{ - /// Creates a new `HttpsConnector`. +impl Builder { + /// Builds a new `HttpsConnector`. /// /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. /// - /// This API allows an `HttpsConnector` to be constructed with an existing HTTP connector, as follows: + /// This API allows an `HttpsConnector` to be constructed with an existing HTTP connector, as + /// follows: /// ``` /// use s2n_tls_hyper::connector::HttpsConnector; /// use s2n_tls::config::Config; @@ -77,16 +130,21 @@ where /// // Ensure that the HTTP connector permits the HTTPS scheme. /// http.enforce_http(false); /// - /// let connector = HttpsConnector::new_with_http(http, Config::default()); + /// let connector = HttpsConnector::builder() + /// .build_with_http(http, Config::default()); /// ``` /// - /// `HttpsConnector::new()` can be used to create the HTTP connector automatically. + /// `Builder::build()` can be used to create the HTTP connector automatically. /// /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values /// configured on `conn_builder` with APIs like /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. - pub fn new_with_http(http: Http, conn_builder: Builder) -> HttpsConnector { - Self { http, conn_builder } + pub fn build_with_http( + self, + http: Http, + conn_builder: ConnBuilder, + ) -> HttpsConnector { + HttpsConnector { http, conn_builder } } } @@ -96,19 +154,22 @@ where // https://docs.rs/hyper-util/latest/hyper_util/client/legacy/connect/trait.Connect.html // // The hyper compatibility traits for `Service::Response` are implemented in `MaybeHttpsStream`. -impl Service for HttpsConnector +impl Service for HttpsConnector where Http: Service, Http::Response: Read + Write + Connection + Unpin + Send + 'static, Http::Future: Send + 'static, Http::Error: Into>, - Builder: connection::Builder + Send + Sync + 'static, - ::Output: Unpin + Send, + ConnBuilder: connection::Builder + Send + Sync + 'static, + ::Output: Unpin + Send, { - type Response = MaybeHttpsStream; + type Response = MaybeHttpsStream; type Error = Error; type Future = Pin< - Box, Error>> + Send>, + Box< + dyn Future, Error>> + + Send, + >, >; fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll> { @@ -180,6 +241,21 @@ mod tests { use hyper_util::{client::legacy::Client, rt::TokioExecutor}; use std::{error::Error as StdError, str::FromStr}; + #[tokio::test] + async fn connector_creation() { + let config = Config::default(); + let connector_from_new = HttpsConnector::new(config.clone()); + let _assert_type: HttpsConnector = connector_from_new; + + let connector_from_build = HttpsConnector::builder().build(config.clone()); + let _assert_type: HttpsConnector = connector_from_build; + + let http: u32 = 10; + let connect_from_build_with_http = + HttpsConnector::builder().build_with_http(http, config.clone()); + let _assert_type: HttpsConnector = connect_from_build_with_http; + } + #[tokio::test] async fn test_unsecure_http() -> Result<(), Box> { let connector = HttpsConnector::new(Config::default()); From 7c2f691ab3fb5716c0b6b766d3bb241f8e23e179 Mon Sep 17 00:00:00 2001 From: Sam Clark <3758302+goatgoose@users.noreply.github.com> Date: Wed, 11 Dec 2024 19:52:06 -0500 Subject: [PATCH 2/2] specify generic types upfront --- bindings/rust/s2n-tls-hyper/src/connector.rs | 110 ++++++++----------- 1 file changed, 44 insertions(+), 66 deletions(-) diff --git a/bindings/rust/s2n-tls-hyper/src/connector.rs b/bindings/rust/s2n-tls-hyper/src/connector.rs index 338a5e4c07a..8721bff70ee 100644 --- a/bindings/rust/s2n-tls-hyper/src/connector.rs +++ b/bindings/rust/s2n-tls-hyper/src/connector.rs @@ -12,7 +12,6 @@ use s2n_tls::{config::Config, connection}; use s2n_tls_tokio::TlsConnector; use std::{ future::Future, - marker::PhantomData, pin::Pin, task::{Context, Poll}, }; @@ -35,85 +34,48 @@ where ConnBuilder: connection::Builder, ::Output: Unpin, { - /// Creates a new `HttpsConnector` with the default configuration. Use `HttpsConnector::builder` - /// instead to configure an `HttpsConnector`. + /// Creates a new `HttpsConnector` with the default settings. + /// + /// Use `HttpsConnector::builder()` instead to configure an `HttpsConnector`. /// /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. /// - /// ``` - /// use s2n_tls_hyper::connector::HttpsConnector; - /// use s2n_tls::config::Config; - /// - /// // Create a new HttpsConnector. - /// let connector = HttpsConnector::new(Config::default()); - /// ``` - /// /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values /// configured on `conn_builder` with APIs like /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. pub fn new(conn_builder: ConnBuilder) -> HttpsConnector { - HttpsConnector::builder().build(conn_builder) + HttpsConnector::builder(conn_builder).build() } -} -impl HttpsConnector -where - ConnBuilder: connection::Builder, - ::Output: Unpin, -{ /// Creates a `Builder` to configure a new `HttpsConnector`. /// - /// ``` - /// use s2n_tls_hyper::connector::HttpsConnector; - /// use s2n_tls::config::Config; - /// - /// // Create and configure a builder. - /// let builder = HttpsConnector::builder(); - /// - /// // Build a new HttpsConnector. - /// let connector = builder.build(Config::default()); - /// ``` - pub fn builder() -> Builder { - Builder { - http: PhantomData, - conn_builder: PhantomData, - } - } -} - -/// Builder used to configure an `HttpsConnector`. Create a new Builder with -/// `HttpsConnector::builder`. -pub struct Builder { - http: PhantomData, - conn_builder: PhantomData, -} - -impl Builder { - /// Builds a new `HttpsConnector`. - /// /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. /// - /// This API creates an `HttpsConnector` using the default hyper `HttpConnector`. To use an - /// existing HTTP connector, use `Builder::build_with_http()`. + /// This API creates a `Builder` with the default hyper `HttpConnector`. To use an existing HTTP + /// connector, use `HttpsConnector::builder_with_http()`. /// /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values /// configured on `conn_builder` with APIs like /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. - pub fn build(self, conn_builder: ConnBuilder) -> HttpsConnector { + pub fn builder(conn_builder: ConnBuilder) -> Builder { let mut http = HttpConnector::new(); // By default, the `HttpConnector` only allows the HTTP URI scheme to be used. To negotiate // HTTP over TLS via the HTTPS scheme, `enforce_http` must be disabled. http.enforce_http(false); - self.build_with_http(http, conn_builder) + HttpsConnector::builder_with_http(http, conn_builder) } } -impl Builder { - /// Builds a new `HttpsConnector`. +impl HttpsConnector +where + ConnBuilder: connection::Builder, + ::Output: Unpin, +{ + /// Creates a `Builder` to configure a new `HttpsConnector`. /// /// `conn_builder` will be used to produce the s2n-tls Connections used for negotiating HTTPS, /// which can be an `s2n_tls::config::Config` or other `s2n_tls::connection::Builder`. @@ -130,21 +92,33 @@ impl Builder { /// // Ensure that the HTTP connector permits the HTTPS scheme. /// http.enforce_http(false); /// - /// let connector = HttpsConnector::builder() - /// .build_with_http(http, Config::default()); + /// let connector = HttpsConnector::builder_with_http(http, Config::default()).build(); /// ``` /// - /// `Builder::build()` can be used to create the HTTP connector automatically. + /// `HttpsConnector::builder()` can be used to create the HTTP connector automatically. /// /// Note that s2n-tls-hyper will override the ALPN extension to negotiate HTTP. Any ALPN values /// configured on `conn_builder` with APIs like /// `s2n_tls::config::Builder::set_application_protocol_preference()` will be ignored. - pub fn build_with_http( - self, - http: Http, - conn_builder: ConnBuilder, - ) -> HttpsConnector { - HttpsConnector { http, conn_builder } + pub fn builder_with_http(http: Http, conn_builder: ConnBuilder) -> Builder { + Builder { http, conn_builder } + } +} + +/// Builder used to configure an `HttpsConnector`. Create a new Builder with +/// `HttpsConnector::builder`. +pub struct Builder { + http: Http, + conn_builder: ConnBuilder, +} + +impl Builder { + /// Builds a new `HttpsConnector`. + pub fn build(self) -> HttpsConnector { + HttpsConnector { + http: self.http, + conn_builder: self.conn_builder, + } } } @@ -247,13 +221,17 @@ mod tests { let connector_from_new = HttpsConnector::new(config.clone()); let _assert_type: HttpsConnector = connector_from_new; - let connector_from_build = HttpsConnector::builder().build(config.clone()); - let _assert_type: HttpsConnector = connector_from_build; + let connector_from_builder = HttpsConnector::builder(config.clone()).build(); + let _assert_type: HttpsConnector = connector_from_builder; let http: u32 = 10; - let connect_from_build_with_http = - HttpsConnector::builder().build_with_http(http, config.clone()); - let _assert_type: HttpsConnector = connect_from_build_with_http; + let connector_from_builder_with_http = + HttpsConnector::builder_with_http(http, config.clone()).build(); + let _assert_type: HttpsConnector = connector_from_builder_with_http; + + let builder = HttpsConnector::builder(config.clone()); + let connector_from_builder = builder.build(); + let _assert_type: HttpsConnector = connector_from_builder; } #[tokio::test]