diff --git a/src/async_impl/client.rs b/src/async_impl/client.rs index ea8fde01c..33b6813e2 100644 --- a/src/async_impl/client.rs +++ b/src/async_impl/client.rs @@ -37,7 +37,7 @@ use crate::cookie; use crate::dns::trust_dns::TrustDnsResolver; use crate::dns::{gai::GaiResolver, DnsResolverWithOverrides, DynResolver, Resolve}; use crate::error; -use crate::into_url::{expect_uri, try_uri}; +use crate::into_url::try_uri; use crate::redirect::{self, remove_sensitive_headers}; #[cfg(feature = "__tls")] use crate::tls::{self, TlsBackend}; @@ -1820,7 +1820,10 @@ impl Client { } } - let uri = expect_uri(&url); + let uri = match try_uri(&url) { + Ok(uri) => uri, + _ => return Pending::new_err(error::url_invalid_uri(url)), + }; let (reusable, body) = match body { Some(body) => { @@ -2193,7 +2196,8 @@ impl PendingRequest { } self.retry_count += 1; - let uri = expect_uri(&self.url); + // If it parsed once, it should parse again + let uri = try_uri(&self.url).expect("URL was already validated as URI"); *self.as_mut().in_flight().get_mut() = match *self.as_mut().in_flight().as_ref() { #[cfg(feature = "http3")] @@ -2373,7 +2377,7 @@ impl Future for PendingRequest { // // If not, just log it and skip the redirect. let loc = loc.and_then(|url| { - if try_uri(&url).is_some() { + if try_uri(&url).is_ok() { Some(url) } else { None @@ -2418,7 +2422,7 @@ impl Future for PendingRequest { std::mem::replace(self.as_mut().headers(), HeaderMap::new()); remove_sensitive_headers(&mut headers, &self.url, &self.urls); - let uri = expect_uri(&self.url); + let uri = try_uri(&self.url)?; let body = match self.body { Some(Some(ref body)) => Body::reusable(body.clone()), _ => Body::empty(), @@ -2523,7 +2527,7 @@ fn add_cookie_header(headers: &mut HeaderMap, cookie_store: &dyn cookie::CookieS #[cfg(test)] mod tests { #[tokio::test] - async fn execute_request_rejects_invald_urls() { + async fn execute_request_rejects_invalid_urls() { let url_str = "hxxps://www.rust-lang.org/"; let url = url::Url::parse(url_str).unwrap(); let result = crate::get(url.clone()).await; @@ -2533,4 +2537,17 @@ mod tests { assert!(err.is_builder()); assert_eq!(url_str, err.url().unwrap().as_str()); } + + /// https://github.com/seanmonstar/reqwest/issues/668 + #[tokio::test] + async fn execute_request_rejects_invalid_hostname() { + let url_str = "https://{{hostname}}/"; + let url = url::Url::parse(url_str).unwrap(); + let result = crate::get(url.clone()).await; + + assert!(result.is_err()); + let err = result.err().unwrap(); + assert!(err.is_builder()); + assert_eq!(url_str, err.url().unwrap().as_str()); + } } diff --git a/src/error.rs b/src/error.rs index 9453fcd92..0d405a929 100644 --- a/src/error.rs +++ b/src/error.rs @@ -275,6 +275,10 @@ pub(crate) fn url_bad_scheme(url: Url) -> Error { Error::new(Kind::Builder, Some(BadScheme)).with_url(url) } +pub(crate) fn url_invalid_uri(url: Url) -> Error { + Error::new(Kind::Builder, Some("Parsed Url is not a valid Uri")).with_url(url) +} + if_wasm! { pub(crate) fn wasm(js_val: wasm_bindgen::JsValue) -> BoxError { format!("{:?}", js_val).into() diff --git a/src/into_url.rs b/src/into_url.rs index d60f5c952..a46ae5702 100644 --- a/src/into_url.rs +++ b/src/into_url.rs @@ -74,14 +74,10 @@ impl<'a> IntoUrlSealed for String { } if_hyper! { - pub(crate) fn expect_uri(url: &Url) -> http::Uri { + pub(crate) fn try_uri(url: &Url) -> crate::Result { url.as_str() .parse() - .expect("a parsed Url should always be a valid Uri") - } - - pub(crate) fn try_uri(url: &Url) -> Option { - url.as_str().parse().ok() + .map_err(|_| crate::error::url_invalid_uri(url.clone())) } }