diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 402c8eb1..8168e194 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -59,7 +59,9 @@ jobs: - native-certs - gzip - brotli + - http - http-interop + - http http-interop env: RUST_BACKTRACE: "1" RUSTFLAGS: "-D dead_code -D unused-variables -D unused" diff --git a/Cargo.lock b/Cargo.lock index 53d95f79..d94b6473 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -259,6 +259,17 @@ version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d77f7ec81a6d05a3abb01ab6eb7590f6083d08449fe5a1c8b1e620283546ccb7" +[[package]] +name = "http" +version = "0.2.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8947b1a6fad4393052c7ba1f4cd97bed3e953a95c79c92ad9b051a04611d9fbb" +dependencies = [ + "bytes", + "fnv", + "itoa", +] + [[package]] name = "http" version = "1.0.0" @@ -778,7 +789,8 @@ dependencies = [ "encoding_rs", "env_logger", "flate2", - "http", + "http 0.2.11", + "http 1.0.0", "log", "native-tls", "once_cell", diff --git a/Cargo.toml b/Cargo.toml index fc602848..1f1309b2 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -25,7 +25,8 @@ cookies = ["dep:cookie", "dep:cookie_store"] socks-proxy = ["dep:socks"] gzip = ["dep:flate2"] brotli = ["dep:brotli-decompressor"] -http-interop = ["dep:http"] +http-interop = ["dep:http-02"] +http = ["dep:http"] proxy-from-env = [] [dependencies] @@ -46,6 +47,7 @@ rustls-native-certs = { version = "0.6", optional = true } native-tls = { version = "0.2", optional = true } flate2 = { version = "1.0.22", optional = true } brotli-decompressor = { version = "2.3.2", optional = true } +http-02 = { package = "http", version = "0.2", optional = true } http = { version = "1.0", optional = true } [dev-dependencies] diff --git a/src/header.rs b/src/header.rs index ce3b87e8..bc9f962c 100644 --- a/src/header.rs +++ b/src/header.rs @@ -158,7 +158,7 @@ pub fn get_header<'h>(headers: &'h [Header], name: &str) -> Option<&'h str> { .and_then(|h| h.value()) } -#[cfg(any(doc, all(test, feature = "http-interop")))] +#[cfg(any(doc, all(test, any(feature = "http-interop", feature = "http"))))] pub fn get_header_raw<'h>(headers: &'h [Header], name: &str) -> Option<&'h [u8]> { headers .iter() diff --git a/src/http.rs b/src/http.rs new file mode 100644 index 00000000..ad37d49e --- /dev/null +++ b/src/http.rs @@ -0,0 +1,455 @@ +use std::{ + io::{Cursor, Read}, + net::{IpAddr, Ipv4Addr, SocketAddr}, +}; + +use crate::{header::HeaderLine, response::ResponseStatusIndex, Request, Response}; + +/// Converts an [`http::Response`] into a [`Response`]. +/// +/// As an [`http::Response`] does not contain a URL, `"https://example.com/"` is used as a +/// placeholder. Additionally, if the response has a header which cannot be converted into a valid +/// [`Header`](crate::Header), it will be skipped rather than having the conversion fail. The remote +/// address property will also always be `127.0.0.1:80` for similar reasons to the URL. +/// +/// ``` +/// # fn main() -> Result<(), http::Error> { +/// # ureq::is_test(true); +/// let http_response = http::Response::builder().status(200).body("")?; +/// let response: ureq::Response = http_response.into(); +/// # Ok(()) +/// # } +/// ``` +impl + Send + Sync + 'static> From> for Response { + fn from(value: http::Response) -> Self { + let version_str = format!("{:?}", value.version()); + let status_line = format!("{} {}", version_str, value.status()); + let status_num = u16::from(value.status()); + Response { + url: "https://example.com/".parse().unwrap(), + status_line, + index: ResponseStatusIndex { + http_version: version_str.len(), + response_code: version_str.len() + status_num.to_string().len(), + }, + status: status_num, + headers: value + .headers() + .iter() + .map(|(name, value)| { + let mut raw_header: Vec = name.to_string().into_bytes(); + raw_header.extend(b": "); + raw_header.extend(value.as_bytes()); + + HeaderLine::from(raw_header).into_header().unwrap() + }) + .collect::>(), + reader: Box::new(Cursor::new(value.into_body())), + remote_addr: SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 80), + local_addr: SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 0), + history: vec![], + } + } +} + +fn create_builder(response: &Response) -> http::response::Builder { + let http_version = match response.http_version() { + "HTTP/0.9" => http::Version::HTTP_09, + "HTTP/1.0" => http::Version::HTTP_10, + "HTTP/1.1" => http::Version::HTTP_11, + "HTTP/2.0" => http::Version::HTTP_2, + "HTTP/3.0" => http::Version::HTTP_3, + _ => unreachable!(), + }; + + let response_builder = response + .headers + .iter() + .fold(http::Response::builder(), |builder, header| { + builder.header(header.name(), header.value_raw()) + }) + .status(response.status()) + .version(http_version); + + response_builder +} + +/// Converts a [`Response`] into an [`http::Response`], where the body is a reader containing the +/// body of the response. +/// +/// ``` +/// # fn main() -> Result<(), ureq::Error> { +/// # ureq::is_test(true); +/// use std::io::Read; +/// let response = ureq::get("http://example.com").call()?; +/// let http_response: http::Response> = response.into(); +/// # Ok(()) +/// # } +/// ``` +impl From for http::Response> { + fn from(value: Response) -> Self { + create_builder(&value).body(value.into_reader()).unwrap() + } +} + +/// Converts a [`Response`] into an [`http::Response`], where the body is a String. +/// +/// ``` +/// # fn main() -> Result<(), ureq::Error> { +/// # ureq::is_test(true); +/// let response = ureq::get("http://example.com").call()?; +/// let http_response: http::Response = response.into(); +/// # Ok(()) +/// # } +/// ``` +impl From for http::Response { + fn from(value: Response) -> Self { + create_builder(&value) + .body(value.into_string().unwrap()) + .unwrap() + } +} + +/// Converts a [`Response`] into an [`http::Response`], where the body is a [`Vec`]. +/// +/// ``` +/// # fn main() -> Result<(), ureq::Error> { +/// # ureq::is_test(true); +/// let response = ureq::get("http://example.com").call()?; +/// let http_response: http::Response> = response.into(); +/// # Ok(()) +/// # } +/// ``` +impl From for http::Response> { + fn from(value: Response) -> Self { + create_builder(&value) + .body(value.into_string().unwrap().into_bytes()) + .unwrap() + } +} + +/// Converts an [`http::request::Builder`] into a [`Request`]. +/// +/// ``` +/// # fn main() -> Result<(), Box> { +/// # ureq::is_test(true); +/// use std::convert::TryInto; +/// +/// let http_request_builder = http::Request::builder().method("GET").uri("http://example.com"); +/// let request: ureq::Request = http_request_builder.try_into()?; +/// request.call()?; +/// # Ok(()) +/// # } +/// ``` +/// +/// +/// # Converting from [`http::Request`] +/// +/// Notably `ureq` does _not_ implement the conversion from [`http::Request`] because it contains +/// the body of a request together with the actual request data. However, [`http`] provides +/// [`http::Request::into_parts()`] to split out a request into [`http::request::Parts`] and a +/// `body`, for which the conversion _is_ implemented and can be used as follows: +/// +/// ``` +/// # fn main() -> Result<(), ureq::Error> { +/// # ureq::is_test(true); +/// let http_request = http::Request::builder().method("GET").uri("http://example.com").body(vec![0u8]).unwrap(); +/// let (http_parts, body) = http_request.into_parts(); +/// let request: ureq::Request = http_parts.into(); +/// request.send_bytes(&body)?; +/// # Ok(()) +/// # } +/// ``` +impl core::convert::TryFrom for Request { + type Error = http::Error; + + fn try_from(value: http::request::Builder) -> Result { + let (parts, ()) = value.body(())?.into_parts(); + Ok(parts.into()) + } +} + +/// Converts [`http::request::Parts`] into a [`Request`]. +/// +/// An [`http::Request`] can be split out into its [`http::request::Parts`] and body as follows: +/// +/// ``` +/// # fn main() -> Result<(), ureq::Error> { +/// # ureq::is_test(true); +/// let http_request = http::Request::builder().method("GET").uri("http://example.com").body(vec![0u8]).unwrap(); +/// let (http_parts, body) = http_request.into_parts(); +/// let request: ureq::Request = http_parts.into(); +/// request.send_bytes(&body)?; +/// # Ok(()) +/// # } +/// ``` +impl From for Request { + fn from(value: http::request::Parts) -> Self { + let mut new_request = crate::agent().request(value.method.as_str(), &value.uri.to_string()); + + for (name, value) in &value.headers { + // TODO: Aren't complete header values available as raw byte slices? + let mut raw_header: Vec = name.to_string().into_bytes(); + raw_header.extend(b": "); + raw_header.extend(value.as_bytes()); + + let header = HeaderLine::from(raw_header) + .into_header() + .expect("Unreachable"); + + crate::header::add_header(&mut new_request.headers, header) + } + + new_request + } +} + +/// Converts a [`Request`] into an [`http::request::Builder`]. +/// +/// The method and URI are preserved. The HTTP version will always be set to `HTTP/1.1`. +/// +/// ``` +/// # fn main() -> Result<(), http::Error> { +/// # ureq::is_test(true); +/// let request = ureq::get("https://my-website.com"); +/// let http_request_builder: http::request::Builder = request.into(); +/// +/// http_request_builder.body(())?; +/// # Ok(()) +/// # } +/// ``` +impl From for http::request::Builder { + fn from(value: Request) -> Self { + value + .headers + .iter() + .fold(http::Request::builder(), |builder, header| { + builder.header(header.name(), header.value_raw()) + }) + .method(value.method()) + .version(http::Version::HTTP_11) + .uri(value.url()) + } +} + +#[cfg(test)] +mod tests { + use crate::header::{add_header, get_header_raw, HeaderLine}; + use std::convert::TryInto; + + #[test] + fn convert_http_response() { + use http::{Response, StatusCode, Version}; + + let http_response_body = vec![0xaa; 10240]; + let http_response = Response::builder() + .version(Version::HTTP_2) + .header("Custom-Header", "custom value") + .header("Content-Type", "application/octet-stream") + .status(StatusCode::IM_A_TEAPOT) + .body(http_response_body.clone()) + .unwrap(); + + let response: super::Response = http_response.into(); + assert_eq!(response.get_url(), "https://example.com/"); + assert_eq!(response.http_version(), "HTTP/2.0"); + assert_eq!(response.status(), u16::from(StatusCode::IM_A_TEAPOT)); + assert_eq!(response.status_text(), "I'm a teapot"); + assert_eq!(response.remote_addr().to_string().as_str(), "127.0.0.1:80"); + assert_eq!(response.header("Custom-Header"), Some("custom value")); + assert_eq!(response.content_type(), "application/octet-stream"); + + let mut body_buf: Vec = vec![]; + response.into_reader().read_to_end(&mut body_buf).unwrap(); + assert_eq!(body_buf, http_response_body); + } + + #[test] + fn convert_http_response_string() { + use http::{Response, StatusCode, Version}; + + let http_response_body = "Some body string".to_string(); + let http_response = Response::builder() + .version(Version::HTTP_11) + .status(StatusCode::OK) + .body(http_response_body.clone()) + .unwrap(); + + let response: super::Response = http_response.into(); + assert_eq!(response.get_url(), "https://example.com/"); + assert_eq!(response.content_type(), "text/plain"); + assert_eq!(response.into_string().unwrap(), http_response_body); + } + + #[test] + fn convert_http_response_bad_header() { + use http::{Response, StatusCode, Version}; + + let http_response = Response::builder() + .version(Version::HTTP_11) + .status(StatusCode::OK) + .header("Some-Invalid-Header", vec![0xde, 0xad, 0xbe, 0xef]) + .header("Some-Valid-Header", vec![0x48, 0x45, 0x4c, 0x4c, 0x4f]) + .body(vec![]) + .unwrap(); + + let response: super::Response = http_response.into(); + assert_eq!(response.header("Some-Invalid-Header"), None); + assert_eq!(response.header("Some-Valid-Header"), Some("HELLO")); + } + + #[test] + fn convert_to_http_response_string() { + use http::Response; + + let mut response = super::Response::new(418, "I'm a teapot", "some body text").unwrap(); + response.headers.push( + HeaderLine::from("Content-Type: text/plain".as_bytes().to_vec()) + .into_header() + .unwrap(), + ); + let http_response: Response = response.into(); + + assert_eq!(http_response.body(), "some body text"); + assert_eq!(http_response.status().as_u16(), 418); + assert_eq!( + http_response.status().canonical_reason(), + Some("I'm a teapot") + ); + assert_eq!( + http_response + .headers() + .get("content-type") + .map(|f| f.to_str().unwrap()), + Some("text/plain") + ); + } + + #[test] + fn convert_to_http_response_bytes() { + use http::Response; + use std::io::{Cursor, Read}; + + let mut response = super::Response::new(200, "OK", "tbr").unwrap(); + response.reader = Box::new(Cursor::new(vec![0xde, 0xad, 0xbe, 0xef])); + let http_response: Response> = response.into(); + + let mut buf = vec![]; + http_response.into_body().read_to_end(&mut buf).unwrap(); + assert_eq!(buf, vec![0xde, 0xad, 0xbe, 0xef]); + } + + #[test] + fn convert_http_response_builder_with_invalid_utf8_header() { + use http::Response; + + let http_response = Response::builder() + .header("Some-Key", b"some\xff\xffvalue".as_slice()) + .body(b"hello") + .unwrap(); + let response: super::Response = http_response.into(); + + assert_eq!( + get_header_raw(&response.headers, "some-key"), + Some(b"some\xff\xffvalue".as_slice()) + ); + } + + #[test] + fn convert_to_http_response_builder_with_invalid_utf8_header() { + let mut response = super::Response::new(200, "OK", "tbr").unwrap(); + add_header( + &mut response.headers, + HeaderLine::from(b"Some-Key: some\xff\xffvalue".to_vec()) + .into_header() + .unwrap(), + ); + dbg!(&response); + let http_response: http::Response> = response.into(); + + assert_eq!( + http_response + .headers() + .get("some-key") + .map(|h| h.as_bytes()), + Some(b"some\xff\xffvalue".as_slice()) + ); + } + + #[test] + fn convert_http_request_builder() { + use http::Request; + + let http_request = Request::builder() + .method("PUT") + .header("Some-Key", "some value") + .uri("https://google.com/?some=query"); + let request: super::Request = http_request.try_into().unwrap(); + + assert_eq!(request.header("some-key"), Some("some value")); + assert_eq!(request.method(), "PUT"); + assert_eq!(request.url(), "https://google.com/?some=query"); + } + + #[test] + fn convert_to_http_request_builder() { + use http::request::Builder; + + let request = crate::agent() + .head("http://some-website.com") + .set("Some-Key", "some value"); + let http_request_builder: Builder = request.try_into().unwrap(); + let http_request = http_request_builder.body(()).unwrap(); + + assert_eq!( + http_request + .headers() + .get("some-key") + .map(|v| v.to_str().unwrap()), + Some("some value") + ); + assert_eq!(http_request.uri(), "http://some-website.com"); + assert_eq!(http_request.version(), http::Version::HTTP_11); + } + + #[test] + fn convert_http_request_builder_with_invalid_utf8_header() { + use http::Request; + + let http_request = Request::builder() + .method("PUT") + .header("Some-Key", b"some\xff\xffvalue".as_slice()) + .uri("https://google.com/?some=query"); + let request: super::Request = http_request.try_into().unwrap(); + + assert_eq!( + get_header_raw(&request.headers, "some-key"), + Some(b"some\xff\xffvalue".as_slice()) + ); + assert_eq!(request.method(), "PUT"); + assert_eq!(request.url(), "https://google.com/?some=query"); + } + + #[test] + fn convert_to_http_request_builder_with_invalid_utf8_header() { + use http::request::Builder; + + let mut request = crate::agent().head("http://some-website.com"); + add_header( + &mut request.headers, + HeaderLine::from(b"Some-Key: some\xff\xffvalue".to_vec()) + .into_header() + .unwrap(), + ); + dbg!(&request); + let http_request_builder: Builder = request.try_into().unwrap(); + let http_request = http_request_builder.body(()).unwrap(); + + assert_eq!( + http_request.headers().get("some-key").map(|h| h.as_bytes()), + Some(b"some\xff\xffvalue".as_slice()) + ); + assert_eq!(http_request.uri(), "http://some-website.com"); + assert_eq!(http_request.version(), http::Version::HTTP_11); + } +} diff --git a/src/http_interop.rs b/src/http_interop.rs index c3cd8eac..05a0eaa9 100644 --- a/src/http_interop.rs +++ b/src/http_interop.rs @@ -1,3 +1,5 @@ +use http_02 as http; + use std::{ io::{Cursor, Read}, net::{IpAddr, Ipv4Addr, SocketAddr}, @@ -13,6 +15,7 @@ use crate::{header::HeaderLine, response::ResponseStatusIndex, Request, Response /// address property will also always be `127.0.0.1:80` for similar reasons to the URL. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), http::Error> { /// # ureq::is_test(true); /// let http_response = http::Response::builder().status(200).body("")?; @@ -78,6 +81,7 @@ fn create_builder(response: &Response) -> http::response::Builder { /// body of the response. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// use std::io::Read; @@ -95,6 +99,7 @@ impl From for http::Response> { /// Converts a [`Response`] into an [`http::Response`], where the body is a String. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// let response = ureq::get("http://example.com").call()?; @@ -113,6 +118,7 @@ impl From for http::Response { /// Converts a [`Response`] into an [`http::Response`], where the body is a [`Vec`]. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// let response = ureq::get("http://example.com").call()?; @@ -131,6 +137,7 @@ impl From for http::Response> { /// Converts an [`http::request::Builder`] into a [`Request`]. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// let http_request_builder = http::Request::builder().method("GET").uri("http://example.com"); @@ -149,6 +156,7 @@ impl From for http::Response> { /// [`http::request::Parts`]: /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// ureq::is_test(true); /// let http_builder = http::Request::builder().method("GET").uri("http://example.com"); @@ -168,6 +176,7 @@ impl From for http::Response> { /// `body`, for which the conversion _is_ implemented and can be used as follows: /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// let http_request = http::Request::builder().method("GET").uri("http://example.com").body(vec![0u8]).unwrap(); @@ -206,6 +215,7 @@ impl From for Request { /// An [`http::Request`] can be split out into its [`http::request::Parts`] and body as follows: /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), ureq::Error> { /// # ureq::is_test(true); /// let http_request = http::Request::builder().method("GET").uri("http://example.com").body(vec![0u8]).unwrap(); @@ -241,6 +251,7 @@ impl From for Request { /// The method and URI are preserved. The HTTP version will always be set to `HTTP/1.1`. /// /// ``` +/// # use http_02 as http; /// # fn main() -> Result<(), http::Error> { /// # ureq::is_test(true); /// let request = ureq::get("https://my-website.com"); @@ -266,6 +277,7 @@ impl From for http::request::Builder { #[cfg(test)] mod tests { + use http_02 as http; use crate::header::{add_header, get_header_raw, HeaderLine}; #[test] diff --git a/src/lib.rs b/src/lib.rs index 8639c653..fa97ae10 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -163,7 +163,8 @@ //! does nothing for `native-tls`. //! * `gzip` enables requests of gzip-compressed responses and decompresses them. This is enabled by default. //! * `brotli` enables requests brotli-compressed responses and decompresses them. -//! * `http-interop` enables conversion methods to and from `http::Response` and `http::request::Builder`. +//! * `http-interop` enables conversion methods to and from `http::Response` and `http::request::Builder` (v0.2). +//! * `http` enables conversion methods to and from `http::Response` and `http::request::Builder` (v1.0). //! //! # Plain requests //! @@ -420,6 +421,9 @@ mod testserver; #[cfg(feature = "http-interop")] mod http_interop; +#[cfg(feature = "http")] +mod http; + pub use crate::agent::Agent; pub use crate::agent::AgentBuilder; pub use crate::agent::RedirectAuthHeaders;