1use std::convert::TryFrom;
2use std::fmt;
3use std::future::Future;
4use std::time::Duration;
5
6use serde::Serialize;
7#[cfg(feature = "json")]
8use serde_json;
9
10use super::body::Body;
11use super::client::{Client, Pending};
12#[cfg(feature = "multipart")]
13use super::multipart;
14use super::response::Response;
15#[cfg(feature = "multipart")]
16use crate::header::CONTENT_LENGTH;
17use crate::header::{HeaderMap, HeaderName, HeaderValue, CONTENT_TYPE};
18use crate::{Method, Url};
19use http::{request::Parts, Request as HttpRequest, Version};
20
21pub struct Request {
23    method: Method,
24    url: Url,
25    headers: HeaderMap,
26    body: Option<Body>,
27    timeout: Option<Duration>,
28    version: Version,
29}
30
31#[must_use = "RequestBuilder does nothing until you 'send' it"]
35pub struct RequestBuilder {
36    client: Client,
37    request: crate::Result<Request>,
38}
39
40impl Request {
41    #[inline]
43    pub fn new(method: Method, url: Url) -> Self {
44        Request {
45            method,
46            url,
47            headers: HeaderMap::new(),
48            body: None,
49            timeout: None,
50            version: Version::default(),
51        }
52    }
53
54    #[inline]
56    pub fn method(&self) -> &Method {
57        &self.method
58    }
59
60    #[inline]
62    pub fn method_mut(&mut self) -> &mut Method {
63        &mut self.method
64    }
65
66    #[inline]
68    pub fn url(&self) -> &Url {
69        &self.url
70    }
71
72    #[inline]
74    pub fn url_mut(&mut self) -> &mut Url {
75        &mut self.url
76    }
77
78    #[inline]
80    pub fn headers(&self) -> &HeaderMap {
81        &self.headers
82    }
83
84    #[inline]
86    pub fn headers_mut(&mut self) -> &mut HeaderMap {
87        &mut self.headers
88    }
89
90    #[inline]
92    pub fn body(&self) -> Option<&Body> {
93        self.body.as_ref()
94    }
95
96    #[inline]
98    pub fn body_mut(&mut self) -> &mut Option<Body> {
99        &mut self.body
100    }
101
102    #[inline]
104    pub fn timeout(&self) -> Option<&Duration> {
105        self.timeout.as_ref()
106    }
107
108    #[inline]
110    pub fn timeout_mut(&mut self) -> &mut Option<Duration> {
111        &mut self.timeout
112    }
113
114    #[inline]
116    pub fn version(&self) -> Version {
117        self.version
118    }
119
120    #[inline]
122    pub fn version_mut(&mut self) -> &mut Version {
123        &mut self.version
124    }
125
126    pub fn try_clone(&self) -> Option<Request> {
130        let body = match self.body.as_ref() {
131            Some(body) => Some(body.try_clone()?),
132            None => None,
133        };
134        let mut req = Request::new(self.method().clone(), self.url().clone());
135        *req.timeout_mut() = self.timeout().copied();
136        *req.headers_mut() = self.headers().clone();
137        *req.version_mut() = self.version();
138        req.body = body;
139        Some(req)
140    }
141
142    pub(super) fn pieces(
143        self,
144    ) -> (
145        Method,
146        Url,
147        HeaderMap,
148        Option<Body>,
149        Option<Duration>,
150        Version,
151    ) {
152        (
153            self.method,
154            self.url,
155            self.headers,
156            self.body,
157            self.timeout,
158            self.version,
159        )
160    }
161}
162
163impl RequestBuilder {
164    pub(super) fn new(client: Client, request: crate::Result<Request>) -> RequestBuilder {
165        let mut builder = RequestBuilder { client, request };
166
167        let auth = builder
168            .request
169            .as_mut()
170            .ok()
171            .and_then(|req| extract_authority(&mut req.url));
172
173        if let Some((username, password)) = auth {
174            builder.basic_auth(username, password)
175        } else {
176            builder
177        }
178    }
179
180    pub fn from_parts(client: Client, request: Request) -> RequestBuilder {
182        RequestBuilder {
183            client,
184            request: crate::Result::Ok(request),
185        }
186    }
187
188    pub fn header<K, V>(self, key: K, value: V) -> RequestBuilder
190    where
191        HeaderName: TryFrom<K>,
192        <HeaderName as TryFrom<K>>::Error: Into<http::Error>,
193        HeaderValue: TryFrom<V>,
194        <HeaderValue as TryFrom<V>>::Error: Into<http::Error>,
195    {
196        self.header_sensitive(key, value, false)
197    }
198
199    fn header_sensitive<K, V>(mut self, key: K, value: V, sensitive: bool) -> RequestBuilder
201    where
202        HeaderName: TryFrom<K>,
203        <HeaderName as TryFrom<K>>::Error: Into<http::Error>,
204        HeaderValue: TryFrom<V>,
205        <HeaderValue as TryFrom<V>>::Error: Into<http::Error>,
206    {
207        let mut error = None;
208        if let Ok(ref mut req) = self.request {
209            match <HeaderName as TryFrom<K>>::try_from(key) {
210                Ok(key) => match <HeaderValue as TryFrom<V>>::try_from(value) {
211                    Ok(mut value) => {
212                        if sensitive {
216                            value.set_sensitive(true);
217                        }
218                        req.headers_mut().append(key, value);
219                    }
220                    Err(e) => error = Some(crate::error::builder(e.into())),
221                },
222                Err(e) => error = Some(crate::error::builder(e.into())),
223            };
224        }
225        if let Some(err) = error {
226            self.request = Err(err);
227        }
228        self
229    }
230
231    pub fn headers(mut self, headers: crate::header::HeaderMap) -> RequestBuilder {
235        if let Ok(ref mut req) = self.request {
236            crate::util::replace_headers(req.headers_mut(), headers);
237        }
238        self
239    }
240
241    pub fn basic_auth<U, P>(self, username: U, password: Option<P>) -> RequestBuilder
256    where
257        U: fmt::Display,
258        P: fmt::Display,
259    {
260        let header_value = crate::util::basic_auth(username, password);
261        self.header_sensitive(crate::header::AUTHORIZATION, header_value, true)
262    }
263
264    pub fn bearer_auth<T>(self, token: T) -> RequestBuilder
266    where
267        T: fmt::Display,
268    {
269        let header_value = format!("Bearer {token}");
270        self.header_sensitive(crate::header::AUTHORIZATION, header_value, true)
271    }
272
273    pub fn body<T: Into<Body>>(mut self, body: T) -> RequestBuilder {
275        if let Ok(ref mut req) = self.request {
276            *req.body_mut() = Some(body.into());
277        }
278        self
279    }
280
281    pub fn timeout(mut self, timeout: Duration) -> RequestBuilder {
287        if let Ok(ref mut req) = self.request {
288            *req.timeout_mut() = Some(timeout);
289        }
290        self
291    }
292
293    #[cfg(feature = "multipart")]
313    #[cfg_attr(docsrs, doc(cfg(feature = "multipart")))]
314    pub fn multipart(self, mut multipart: multipart::Form) -> RequestBuilder {
315        let mut builder = self.header(
316            CONTENT_TYPE,
317            format!("multipart/form-data; boundary={}", multipart.boundary()).as_str(),
318        );
319
320        builder = match multipart.compute_length() {
321            Some(length) => builder.header(CONTENT_LENGTH, length),
322            None => builder,
323        };
324
325        if let Ok(ref mut req) = builder.request {
326            *req.body_mut() = Some(multipart.stream())
327        }
328        builder
329    }
330
331    pub fn query<T: Serialize + ?Sized>(mut self, query: &T) -> RequestBuilder {
350        let mut error = None;
351        if let Ok(ref mut req) = self.request {
352            let url = req.url_mut();
353            let mut pairs = url.query_pairs_mut();
354            let serializer = serde_urlencoded::Serializer::new(&mut pairs);
355
356            if let Err(err) = query.serialize(serializer) {
357                error = Some(crate::error::builder(err));
358            }
359        }
360        if let Ok(ref mut req) = self.request {
361            if let Some("") = req.url().query() {
362                req.url_mut().set_query(None);
363            }
364        }
365        if let Some(err) = error {
366            self.request = Err(err);
367        }
368        self
369    }
370
371    pub fn version(mut self, version: Version) -> RequestBuilder {
373        if let Ok(ref mut req) = self.request {
374            req.version = version;
375        }
376        self
377    }
378
379    pub fn form<T: Serialize + ?Sized>(mut self, form: &T) -> RequestBuilder {
407        let mut error = None;
408        if let Ok(ref mut req) = self.request {
409            match serde_urlencoded::to_string(form) {
410                Ok(body) => {
411                    req.headers_mut().insert(
412                        CONTENT_TYPE,
413                        HeaderValue::from_static("application/x-www-form-urlencoded"),
414                    );
415                    *req.body_mut() = Some(body.into());
416                }
417                Err(err) => error = Some(crate::error::builder(err)),
418            }
419        }
420        if let Some(err) = error {
421            self.request = Err(err);
422        }
423        self
424    }
425
426    #[cfg(feature = "json")]
437    #[cfg_attr(docsrs, doc(cfg(feature = "json")))]
438    pub fn json<T: Serialize + ?Sized>(mut self, json: &T) -> RequestBuilder {
439        let mut error = None;
440        if let Ok(ref mut req) = self.request {
441            match serde_json::to_vec(json) {
442                Ok(body) => {
443                    if !req.headers().contains_key(CONTENT_TYPE) {
444                        req.headers_mut()
445                            .insert(CONTENT_TYPE, HeaderValue::from_static("application/json"));
446                    }
447                    *req.body_mut() = Some(body.into());
448                }
449                Err(err) => error = Some(crate::error::builder(err)),
450            }
451        }
452        if let Some(err) = error {
453            self.request = Err(err);
454        }
455        self
456    }
457
458    pub fn fetch_mode_no_cors(self) -> RequestBuilder {
468        self
469    }
470
471    pub fn build(self) -> crate::Result<Request> {
474        self.request
475    }
476
477    pub fn build_split(self) -> (Client, crate::Result<Request>) {
483        (self.client, self.request)
484    }
485
486    pub fn send(self) -> impl Future<Output = Result<Response, crate::Error>> {
508        match self.request {
509            Ok(req) => self.client.execute_request(req),
510            Err(err) => Pending::new_err(err),
511        }
512    }
513
514    pub fn try_clone(&self) -> Option<RequestBuilder> {
534        self.request
535            .as_ref()
536            .ok()
537            .and_then(|req| req.try_clone())
538            .map(|req| RequestBuilder {
539                client: self.client.clone(),
540                request: Ok(req),
541            })
542    }
543}
544
545impl fmt::Debug for Request {
546    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
547        fmt_request_fields(&mut f.debug_struct("Request"), self).finish()
548    }
549}
550
551impl fmt::Debug for RequestBuilder {
552    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
553        let mut builder = f.debug_struct("RequestBuilder");
554        match self.request {
555            Ok(ref req) => fmt_request_fields(&mut builder, req).finish(),
556            Err(ref err) => builder.field("error", err).finish(),
557        }
558    }
559}
560
561fn fmt_request_fields<'a, 'b>(
562    f: &'a mut fmt::DebugStruct<'a, 'b>,
563    req: &Request,
564) -> &'a mut fmt::DebugStruct<'a, 'b> {
565    f.field("method", &req.method)
566        .field("url", &req.url)
567        .field("headers", &req.headers)
568}
569
570pub(crate) fn extract_authority(url: &mut Url) -> Option<(String, Option<String>)> {
573    use percent_encoding::percent_decode;
574
575    if url.has_authority() {
576        let username: String = percent_decode(url.username().as_bytes())
577            .decode_utf8()
578            .ok()?
579            .into();
580        let password = url.password().and_then(|pass| {
581            percent_decode(pass.as_bytes())
582                .decode_utf8()
583                .ok()
584                .map(String::from)
585        });
586        if !username.is_empty() || password.is_some() {
587            url.set_username("")
588                .expect("has_authority means set_username shouldn't fail");
589            url.set_password(None)
590                .expect("has_authority means set_password shouldn't fail");
591            return Some((username, password));
592        }
593    }
594
595    None
596}
597
598impl<T> TryFrom<HttpRequest<T>> for Request
599where
600    T: Into<Body>,
601{
602    type Error = crate::Error;
603
604    fn try_from(req: HttpRequest<T>) -> crate::Result<Self> {
605        let (parts, body) = req.into_parts();
606        let Parts {
607            method,
608            uri,
609            headers,
610            version,
611            ..
612        } = parts;
613        let url = Url::parse(&uri.to_string()).map_err(crate::error::builder)?;
614        Ok(Request {
615            method,
616            url,
617            headers,
618            body: Some(body.into()),
619            timeout: None,
620            version,
621        })
622    }
623}
624
625impl TryFrom<Request> for HttpRequest<Body> {
626    type Error = crate::Error;
627
628    fn try_from(req: Request) -> crate::Result<Self> {
629        let Request {
630            method,
631            url,
632            headers,
633            body,
634            version,
635            ..
636        } = req;
637
638        let mut req = HttpRequest::builder()
639            .version(version)
640            .method(method)
641            .uri(url.as_str())
642            .body(body.unwrap_or_else(Body::empty))
643            .map_err(crate::error::builder)?;
644
645        *req.headers_mut() = headers;
646        Ok(req)
647    }
648}
649
650#[cfg(test)]
651mod tests {
652    use super::{Client, HttpRequest, Request, RequestBuilder, Version};
653    use crate::Method;
654    use serde::Serialize;
655    use std::collections::BTreeMap;
656    use std::convert::TryFrom;
657
658    #[test]
659    fn add_query_append() {
660        let client = Client::new();
661        let some_url = "https://google.com/";
662        let r = client.get(some_url);
663
664        let r = r.query(&[("foo", "bar")]);
665        let r = r.query(&[("qux", 3)]);
666
667        let req = r.build().expect("request is valid");
668        assert_eq!(req.url().query(), Some("foo=bar&qux=3"));
669    }
670
671    #[test]
672    fn add_query_append_same() {
673        let client = Client::new();
674        let some_url = "https://google.com/";
675        let r = client.get(some_url);
676
677        let r = r.query(&[("foo", "a"), ("foo", "b")]);
678
679        let req = r.build().expect("request is valid");
680        assert_eq!(req.url().query(), Some("foo=a&foo=b"));
681    }
682
683    #[test]
684    fn add_query_struct() {
685        #[derive(Serialize)]
686        struct Params {
687            foo: String,
688            qux: i32,
689        }
690
691        let client = Client::new();
692        let some_url = "https://google.com/";
693        let r = client.get(some_url);
694
695        let params = Params {
696            foo: "bar".into(),
697            qux: 3,
698        };
699
700        let r = r.query(¶ms);
701
702        let req = r.build().expect("request is valid");
703        assert_eq!(req.url().query(), Some("foo=bar&qux=3"));
704    }
705
706    #[test]
707    fn add_query_map() {
708        let mut params = BTreeMap::new();
709        params.insert("foo", "bar");
710        params.insert("qux", "three");
711
712        let client = Client::new();
713        let some_url = "https://google.com/";
714        let r = client.get(some_url);
715
716        let r = r.query(¶ms);
717
718        let req = r.build().expect("request is valid");
719        assert_eq!(req.url().query(), Some("foo=bar&qux=three"));
720    }
721
722    #[test]
723    fn test_replace_headers() {
724        use http::HeaderMap;
725
726        let mut headers = HeaderMap::new();
727        headers.insert("foo", "bar".parse().unwrap());
728        headers.append("foo", "baz".parse().unwrap());
729
730        let client = Client::new();
731        let req = client
732            .get("https://hyper.rs")
733            .header("im-a", "keeper")
734            .header("foo", "pop me")
735            .headers(headers)
736            .build()
737            .expect("request build");
738
739        assert_eq!(req.headers()["im-a"], "keeper");
740
741        let foo = req.headers().get_all("foo").iter().collect::<Vec<_>>();
742        assert_eq!(foo.len(), 2);
743        assert_eq!(foo[0], "bar");
744        assert_eq!(foo[1], "baz");
745    }
746
747    #[test]
748    fn normalize_empty_query() {
749        let client = Client::new();
750        let some_url = "https://google.com/";
751        let empty_query: &[(&str, &str)] = &[];
752
753        let req = client
754            .get(some_url)
755            .query(empty_query)
756            .build()
757            .expect("request build");
758
759        assert_eq!(req.url().query(), None);
760        assert_eq!(req.url().as_str(), "https://google.com/");
761    }
762
763    #[test]
764    fn try_clone_reusable() {
765        let client = Client::new();
766        let builder = client
767            .post("http://httpbin.org/post")
768            .header("foo", "bar")
769            .body("from a &str!");
770        let req = builder
771            .try_clone()
772            .expect("clone successful")
773            .build()
774            .expect("request is valid");
775        assert_eq!(req.url().as_str(), "http://httpbin.org/post");
776        assert_eq!(req.method(), Method::POST);
777        assert_eq!(req.headers()["foo"], "bar");
778    }
779
780    #[test]
781    fn try_clone_no_body() {
782        let client = Client::new();
783        let builder = client.get("http://httpbin.org/get");
784        let req = builder
785            .try_clone()
786            .expect("clone successful")
787            .build()
788            .expect("request is valid");
789        assert_eq!(req.url().as_str(), "http://httpbin.org/get");
790        assert_eq!(req.method(), Method::GET);
791        assert!(req.body().is_none());
792    }
793
794    #[test]
795    #[cfg(feature = "stream")]
796    fn try_clone_stream() {
797        let chunks: Vec<Result<_, ::std::io::Error>> = vec![Ok("hello"), Ok(" "), Ok("world")];
798        let stream = futures_util::stream::iter(chunks);
799        let client = Client::new();
800        let builder = client
801            .get("http://httpbin.org/get")
802            .body(super::Body::wrap_stream(stream));
803        let clone = builder.try_clone();
804        assert!(clone.is_none());
805    }
806
807    #[test]
808    fn convert_url_authority_into_basic_auth() {
809        let client = Client::new();
810        let some_url = "https://Aladdin:open sesame@localhost/";
811
812        let req = client.get(some_url).build().expect("request build");
813
814        assert_eq!(req.url().as_str(), "https://localhost/");
815        assert_eq!(
816            req.headers()["authorization"],
817            "Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=="
818        );
819    }
820
821    #[test]
822    fn test_basic_auth_sensitive_header() {
823        let client = Client::new();
824        let some_url = "https://localhost/";
825
826        let req = client
827            .get(some_url)
828            .basic_auth("Aladdin", Some("open sesame"))
829            .build()
830            .expect("request build");
831
832        assert_eq!(req.url().as_str(), "https://localhost/");
833        assert_eq!(
834            req.headers()["authorization"],
835            "Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=="
836        );
837        assert!(req.headers()["authorization"].is_sensitive());
838    }
839
840    #[test]
841    fn test_bearer_auth_sensitive_header() {
842        let client = Client::new();
843        let some_url = "https://localhost/";
844
845        let req = client
846            .get(some_url)
847            .bearer_auth("Hold my bear")
848            .build()
849            .expect("request build");
850
851        assert_eq!(req.url().as_str(), "https://localhost/");
852        assert_eq!(req.headers()["authorization"], "Bearer Hold my bear");
853        assert!(req.headers()["authorization"].is_sensitive());
854    }
855
856    #[test]
857    fn test_explicit_sensitive_header() {
858        let client = Client::new();
859        let some_url = "https://localhost/";
860
861        let mut header = http::HeaderValue::from_static("in plain sight");
862        header.set_sensitive(true);
863
864        let req = client
865            .get(some_url)
866            .header("hiding", header)
867            .build()
868            .expect("request build");
869
870        assert_eq!(req.url().as_str(), "https://localhost/");
871        assert_eq!(req.headers()["hiding"], "in plain sight");
872        assert!(req.headers()["hiding"].is_sensitive());
873    }
874
875    #[test]
876    fn convert_from_http_request() {
877        let http_request = HttpRequest::builder()
878            .method("GET")
879            .uri("http://localhost/")
880            .header("User-Agent", "my-awesome-agent/1.0")
881            .body("test test test")
882            .unwrap();
883        let req: Request = Request::try_from(http_request).unwrap();
884        assert!(req.body().is_some());
885        let test_data = b"test test test";
886        assert_eq!(req.body().unwrap().as_bytes(), Some(&test_data[..]));
887        let headers = req.headers();
888        assert_eq!(headers.get("User-Agent").unwrap(), "my-awesome-agent/1.0");
889        assert_eq!(req.method(), Method::GET);
890        assert_eq!(req.url().as_str(), "http://localhost/");
891    }
892
893    #[test]
894    fn set_http_request_version() {
895        let http_request = HttpRequest::builder()
896            .method("GET")
897            .uri("http://localhost/")
898            .header("User-Agent", "my-awesome-agent/1.0")
899            .version(Version::HTTP_11)
900            .body("test test test")
901            .unwrap();
902        let req: Request = Request::try_from(http_request).unwrap();
903        assert!(req.body().is_some());
904        let test_data = b"test test test";
905        assert_eq!(req.body().unwrap().as_bytes(), Some(&test_data[..]));
906        let headers = req.headers();
907        assert_eq!(headers.get("User-Agent").unwrap(), "my-awesome-agent/1.0");
908        assert_eq!(req.method(), Method::GET);
909        assert_eq!(req.url().as_str(), "http://localhost/");
910        assert_eq!(req.version(), Version::HTTP_11);
911    }
912
913    #[test]
914    fn builder_split_reassemble() {
915        let builder = {
916            let client = Client::new();
917            client.get("http://example.com")
918        };
919        let (client, inner) = builder.build_split();
920        let request = inner.unwrap();
921        let builder = RequestBuilder::from_parts(client, request);
922        builder.build().unwrap();
923    }
924
925    }