1use std::convert::TryFrom;
2use std::fmt;
3use std::time::Duration;
4
5use http::{request::Parts, Request as HttpRequest, Version};
6#[cfg(any(feature = "query", feature = "form", feature = "json"))]
7use serde::Serialize;
8#[cfg(feature = "json")]
9use serde_json;
10
11use super::body::{self, Body};
12#[cfg(feature = "multipart")]
13use super::multipart;
14use super::Client;
15#[cfg(any(feature = "multipart", feature = "form", feature = "json"))]
16use crate::header::CONTENT_TYPE;
17use crate::header::{HeaderMap, HeaderName, HeaderValue};
18use crate::{async_impl, Method, Url};
19
20pub struct Request {
22 body: Option<Body>,
23 inner: async_impl::Request,
24}
25
26#[derive(Debug)]
30#[must_use = "RequestBuilder does nothing until you 'send' it"]
31pub struct RequestBuilder {
32 client: Client,
33 request: crate::Result<Request>,
34}
35
36impl Request {
37 #[inline]
39 pub fn new(method: Method, url: Url) -> Self {
40 Request {
41 body: None,
42 inner: async_impl::Request::new(method, url),
43 }
44 }
45
46 #[inline]
48 pub fn method(&self) -> &Method {
49 self.inner.method()
50 }
51
52 #[inline]
54 pub fn method_mut(&mut self) -> &mut Method {
55 self.inner.method_mut()
56 }
57
58 #[inline]
60 pub fn url(&self) -> &Url {
61 self.inner.url()
62 }
63
64 #[inline]
66 pub fn url_mut(&mut self) -> &mut Url {
67 self.inner.url_mut()
68 }
69
70 #[inline]
72 pub fn headers(&self) -> &HeaderMap {
73 self.inner.headers()
74 }
75
76 #[inline]
78 pub fn headers_mut(&mut self) -> &mut HeaderMap {
79 self.inner.headers_mut()
80 }
81
82 #[inline]
84 pub fn version(&self) -> Version {
85 self.inner.version()
86 }
87
88 #[inline]
90 pub fn version_mut(&mut self) -> &mut Version {
91 self.inner.version_mut()
92 }
93
94 #[inline]
96 pub fn body(&self) -> Option<&Body> {
97 self.body.as_ref()
98 }
99
100 #[inline]
102 pub fn body_mut(&mut self) -> &mut Option<Body> {
103 &mut self.body
104 }
105
106 #[inline]
108 pub fn timeout(&self) -> Option<&Duration> {
109 self.inner.timeout()
110 }
111
112 #[inline]
114 pub fn timeout_mut(&mut self) -> &mut Option<Duration> {
115 self.inner.timeout_mut()
116 }
117
118 pub fn try_clone(&self) -> Option<Request> {
123 let body = if let Some(ref body) = self.body.as_ref() {
124 if let Some(body) = body.try_clone() {
125 Some(body)
126 } else {
127 return None;
128 }
129 } else {
130 None
131 };
132 let mut req = Request::new(self.method().clone(), self.url().clone());
133 *req.timeout_mut() = self.timeout().copied();
134 *req.headers_mut() = self.headers().clone();
135 *req.version_mut() = self.version().clone();
136 req.body = body;
137 Some(req)
138 }
139
140 pub(crate) fn into_async(self) -> (async_impl::Request, Option<body::Sender>) {
141 use crate::header::CONTENT_LENGTH;
142
143 let mut req_async = self.inner;
144 let body = self.body.and_then(|body| {
145 let (tx, body, len) = body.into_async();
146 if let Some(len) = len {
147 req_async.headers_mut().insert(CONTENT_LENGTH, len.into());
148 }
149 *req_async.body_mut() = Some(body);
150 tx
151 });
152 (req_async, body)
153 }
154}
155
156impl RequestBuilder {
157 pub(crate) fn new(client: Client, request: crate::Result<Request>) -> RequestBuilder {
158 let mut builder = RequestBuilder { client, request };
159
160 let auth = builder
161 .request
162 .as_mut()
163 .ok()
164 .and_then(|req| async_impl::request::extract_authority(req.url_mut()));
165
166 if let Some((username, password)) = auth {
167 builder.basic_auth(username, password)
168 } else {
169 builder
170 }
171 }
172
173 pub fn from_parts(client: Client, request: Request) -> RequestBuilder {
175 RequestBuilder {
176 client,
177 request: crate::Result::Ok(request),
178 }
179 }
180
181 pub fn header<K, V>(self, key: K, value: V) -> RequestBuilder
195 where
196 HeaderName: TryFrom<K>,
197 HeaderValue: TryFrom<V>,
198 <HeaderName as TryFrom<K>>::Error: Into<http::Error>,
199 <HeaderValue as TryFrom<V>>::Error: Into<http::Error>,
200 {
201 self.header_sensitive(key, value, false)
202 }
203
204 fn header_sensitive<K, V>(mut self, key: K, value: V, sensitive: bool) -> RequestBuilder
206 where
207 HeaderName: TryFrom<K>,
208 HeaderValue: TryFrom<V>,
209 <HeaderName as TryFrom<K>>::Error: Into<http::Error>,
210 <HeaderValue as TryFrom<V>>::Error: Into<http::Error>,
211 {
212 let mut error = None;
213 if let Ok(ref mut req) = self.request {
214 match <HeaderName as TryFrom<K>>::try_from(key) {
215 Ok(key) => match <HeaderValue as TryFrom<V>>::try_from(value) {
216 Ok(mut value) => {
217 if sensitive {
221 value.set_sensitive(true);
222 }
223 req.headers_mut().append(key, value);
224 }
225 Err(e) => error = Some(crate::error::builder(e.into())),
226 },
227 Err(e) => error = Some(crate::error::builder(e.into())),
228 };
229 }
230 if let Some(err) = error {
231 self.request = Err(err);
232 }
233 self
234 }
235
236 pub fn headers(mut self, headers: crate::header::HeaderMap) -> RequestBuilder {
262 if let Ok(ref mut req) = self.request {
263 crate::util::replace_headers(req.headers_mut(), headers);
264 }
265 self
266 }
267
268 pub fn basic_auth<U, P>(self, username: U, password: Option<P>) -> RequestBuilder
280 where
281 U: fmt::Display,
282 P: fmt::Display,
283 {
284 let header_value = crate::util::basic_auth(username, password);
285 self.header_sensitive(crate::header::AUTHORIZATION, header_value, true)
286 }
287
288 pub fn bearer_auth<T>(self, token: T) -> RequestBuilder
300 where
301 T: fmt::Display,
302 {
303 let header_value = format!("Bearer {token}");
304 self.header_sensitive(crate::header::AUTHORIZATION, &*header_value, true)
305 }
306
307 pub fn body<T: Into<Body>>(mut self, body: T) -> RequestBuilder {
351 if let Ok(ref mut req) = self.request {
352 *req.body_mut() = Some(body.into());
353 }
354 self
355 }
356
357 pub fn timeout(mut self, timeout: Duration) -> RequestBuilder {
363 if let Ok(ref mut req) = self.request {
364 *req.timeout_mut() = Some(timeout);
365 }
366 self
367 }
368
369 #[cfg(feature = "query")]
404 #[cfg_attr(docsrs, doc(cfg(feature = "query")))]
405 pub fn query<T: Serialize + ?Sized>(mut self, query: &T) -> RequestBuilder {
406 let mut error = None;
407 if let Ok(ref mut req) = self.request {
408 let url = req.url_mut();
409 let mut pairs = url.query_pairs_mut();
410 let serializer = serde_urlencoded::Serializer::new(&mut pairs);
411
412 if let Err(err) = query.serialize(serializer) {
413 error = Some(crate::error::builder(err));
414 }
415 }
416 if let Ok(ref mut req) = self.request {
417 if let Some("") = req.url().query() {
418 req.url_mut().set_query(None);
419 }
420 }
421 if let Some(err) = error {
422 self.request = Err(err);
423 }
424 self
425 }
426
427 pub fn version(mut self, version: Version) -> RequestBuilder {
429 if let Ok(ref mut req) = self.request {
430 *req.version_mut() = version;
431 }
432 self
433 }
434
435 #[cfg(feature = "form")]
466 #[cfg_attr(docsrs, doc(cfg(feature = "form")))]
467 pub fn form<T: Serialize + ?Sized>(mut self, form: &T) -> RequestBuilder {
468 let mut error = None;
469 if let Ok(ref mut req) = self.request {
470 match serde_urlencoded::to_string(form) {
471 Ok(body) => {
472 req.headers_mut()
473 .entry(CONTENT_TYPE)
474 .or_insert(HeaderValue::from_static(
475 "application/x-www-form-urlencoded",
476 ));
477 *req.body_mut() = Some(body.into());
478 }
479 Err(err) => error = Some(crate::error::builder(err)),
480 }
481 }
482 if let Some(err) = error {
483 self.request = Err(err);
484 }
485 self
486 }
487
488 #[cfg(feature = "json")]
520 #[cfg_attr(docsrs, doc(cfg(feature = "json")))]
521 pub fn json<T: Serialize + ?Sized>(mut self, json: &T) -> RequestBuilder {
522 let mut error = None;
523 if let Ok(ref mut req) = self.request {
524 match serde_json::to_vec(json) {
525 Ok(body) => {
526 req.headers_mut()
527 .entry(CONTENT_TYPE)
528 .or_insert_with(|| HeaderValue::from_static("application/json"));
529 *req.body_mut() = Some(body.into());
530 }
531 Err(err) => error = Some(crate::error::builder(err)),
532 }
533 }
534 if let Some(err) = error {
535 self.request = Err(err);
536 }
537 self
538 }
539
540 #[cfg(feature = "multipart")]
560 #[cfg_attr(docsrs, doc(cfg(feature = "multipart")))]
561 pub fn multipart(self, mut multipart: multipart::Form) -> RequestBuilder {
562 let mut builder = self.header(
563 CONTENT_TYPE,
564 format!("multipart/form-data; boundary={}", multipart.boundary()).as_str(),
565 );
566 if let Ok(ref mut req) = builder.request {
567 *req.body_mut() = Some(match multipart.compute_length() {
568 Some(length) => Body::sized(multipart.reader(), length),
569 None => Body::new(multipart.reader()),
570 })
571 }
572 builder
573 }
574
575 pub fn build(self) -> crate::Result<Request> {
578 self.request
579 }
580
581 pub fn build_split(self) -> (Client, crate::Result<Request>) {
587 (self.client, self.request)
588 }
589
590 pub fn send(self) -> crate::Result<super::Response> {
597 self.client.execute(self.request?)
598 }
599
600 pub fn try_clone(&self) -> Option<RequestBuilder> {
645 self.request
646 .as_ref()
647 .ok()
648 .and_then(|req| req.try_clone())
649 .map(|req| RequestBuilder {
650 client: self.client.clone(),
651 request: Ok(req),
652 })
653 }
654}
655
656impl<T> TryFrom<HttpRequest<T>> for Request
657where
658 T: Into<Body>,
659{
660 type Error = crate::Error;
661
662 fn try_from(req: HttpRequest<T>) -> crate::Result<Self> {
663 let (parts, body) = req.into_parts();
664 let Parts {
665 method,
666 uri,
667 headers,
668 ..
669 } = parts;
670 let url = Url::parse(&uri.to_string()).map_err(crate::error::builder)?;
671 let mut inner = async_impl::Request::new(method, url);
672 crate::util::replace_headers(inner.headers_mut(), headers);
673 Ok(Request {
674 body: Some(body.into()),
675 inner,
676 })
677 }
678}
679
680impl fmt::Debug for Request {
681 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
682 fmt_request_fields(&mut f.debug_struct("Request"), self).finish()
683 }
684}
685
686fn fmt_request_fields<'a, 'b>(
687 f: &'a mut fmt::DebugStruct<'a, 'b>,
688 req: &Request,
689) -> &'a mut fmt::DebugStruct<'a, 'b> {
690 f.field("method", req.method())
691 .field("url", req.url())
692 .field("headers", req.headers())
693}
694
695#[cfg(test)]
696mod tests {
697 use super::*;
698 use crate::header::{ACCEPT, HOST};
699 #[cfg(feature = "query")]
700 use std::collections::BTreeMap;
701 #[cfg(feature = "form")]
702 use std::collections::HashMap;
703
704 #[test]
705 fn basic_get_request() {
706 let client = Client::new();
707 let some_url = "https://google.com/";
708 let r = client.get(some_url).build().unwrap();
709
710 assert_eq!(r.method(), &Method::GET);
711 assert_eq!(r.url().as_str(), some_url);
712 }
713
714 #[test]
715 fn basic_head_request() {
716 let client = Client::new();
717 let some_url = "https://google.com/";
718 let r = client.head(some_url).build().unwrap();
719
720 assert_eq!(r.method(), &Method::HEAD);
721 assert_eq!(r.url().as_str(), some_url);
722 }
723
724 #[test]
725 fn basic_post_request() {
726 let client = Client::new();
727 let some_url = "https://google.com/";
728 let r = client.post(some_url).build().unwrap();
729
730 assert_eq!(r.method(), &Method::POST);
731 assert_eq!(r.url().as_str(), some_url);
732 }
733
734 #[test]
735 fn basic_put_request() {
736 let client = Client::new();
737 let some_url = "https://google.com/";
738 let r = client.put(some_url).build().unwrap();
739
740 assert_eq!(r.method(), &Method::PUT);
741 assert_eq!(r.url().as_str(), some_url);
742 }
743
744 #[test]
745 fn basic_patch_request() {
746 let client = Client::new();
747 let some_url = "https://google.com/";
748 let r = client.patch(some_url).build().unwrap();
749
750 assert_eq!(r.method(), &Method::PATCH);
751 assert_eq!(r.url().as_str(), some_url);
752 }
753
754 #[test]
755 fn basic_delete_request() {
756 let client = Client::new();
757 let some_url = "https://google.com/";
758 let r = client.delete(some_url).build().unwrap();
759
760 assert_eq!(r.method(), &Method::DELETE);
761 assert_eq!(r.url().as_str(), some_url);
762 }
763
764 #[test]
765 fn add_header() {
766 let client = Client::new();
767 let some_url = "https://google.com/";
768 let r = client.post(some_url);
769
770 let header = HeaderValue::from_static("google.com");
771
772 let r = r.header(HOST, header.clone()).build().unwrap();
774
775 assert_eq!(r.headers().get(HOST), Some(&header));
777 }
778
779 #[test]
780 fn add_headers() {
781 let client = Client::new();
782 let some_url = "https://google.com/";
783 let r = client.post(some_url);
784
785 let header = HeaderValue::from_static("google.com");
786
787 let mut headers = HeaderMap::new();
788 headers.insert(HOST, header);
789
790 let r = r.headers(headers.clone()).build().unwrap();
792
793 assert_eq!(r.headers(), &headers);
795 }
796
797 #[test]
798 fn add_headers_multi() {
799 let client = Client::new();
800 let some_url = "https://google.com/";
801 let r = client.post(some_url);
802
803 let header_json = HeaderValue::from_static("application/json");
804 let header_xml = HeaderValue::from_static("application/xml");
805
806 let mut headers = HeaderMap::new();
807 headers.append(ACCEPT, header_json);
808 headers.append(ACCEPT, header_xml);
809
810 let r = r.headers(headers.clone()).build().unwrap();
812
813 assert_eq!(r.headers(), &headers);
815 let mut all_values = r.headers().get_all(ACCEPT).iter();
816 assert_eq!(all_values.next().unwrap(), &"application/json");
817 assert_eq!(all_values.next().unwrap(), &"application/xml");
818 assert_eq!(all_values.next(), None);
819 }
820
821 #[test]
822 fn add_body() {
823 let client = Client::new();
824 let some_url = "https://google.com/";
825 let r = client.post(some_url);
826
827 let body = "Some interesting content";
828
829 let mut r = r.body(body).build().unwrap();
830
831 let buf = body::read_to_string(r.body_mut().take().unwrap()).unwrap();
832
833 assert_eq!(buf, body);
834 }
835
836 #[test]
837 #[cfg(feature = "query")]
838 fn add_query_append() {
839 let client = Client::new();
840 let some_url = "https://google.com/";
841 let mut r = client.get(some_url);
842
843 r = r.query(&[("foo", "bar")]);
844 r = r.query(&[("qux", 3)]);
845
846 let req = r.build().expect("request is valid");
847 assert_eq!(req.url().query(), Some("foo=bar&qux=3"));
848 }
849
850 #[test]
851 #[cfg(feature = "query")]
852 fn add_query_append_same() {
853 let client = Client::new();
854 let some_url = "https://google.com/";
855 let mut r = client.get(some_url);
856
857 r = r.query(&[("foo", "a"), ("foo", "b")]);
858
859 let req = r.build().expect("request is valid");
860 assert_eq!(req.url().query(), Some("foo=a&foo=b"));
861 }
862
863 #[test]
864 #[cfg(feature = "query")]
865 fn add_query_struct() {
866 #[derive(Serialize)]
867 struct Params {
868 foo: String,
869 qux: i32,
870 }
871
872 let client = Client::new();
873 let some_url = "https://google.com/";
874 let mut r = client.get(some_url);
875
876 let params = Params {
877 foo: "bar".into(),
878 qux: 3,
879 };
880
881 r = r.query(¶ms);
882
883 let req = r.build().expect("request is valid");
884 assert_eq!(req.url().query(), Some("foo=bar&qux=3"));
885 }
886
887 #[test]
888 #[cfg(feature = "query")]
889 fn add_query_map() {
890 let mut params = BTreeMap::new();
891 params.insert("foo", "bar");
892 params.insert("qux", "three");
893
894 let client = Client::new();
895 let some_url = "https://google.com/";
896 let mut r = client.get(some_url);
897
898 r = r.query(¶ms);
899
900 let req = r.build().expect("request is valid");
901 assert_eq!(req.url().query(), Some("foo=bar&qux=three"));
902 }
903
904 #[test]
905 #[cfg(feature = "form")]
906 fn add_form() {
907 let client = Client::new();
908 let some_url = "https://google.com/";
909 let r = client.post(some_url);
910
911 let mut form_data = HashMap::new();
912 form_data.insert("foo", "bar");
913
914 let mut r = r.form(&form_data).build().unwrap();
915
916 assert_eq!(
918 r.headers().get(CONTENT_TYPE).unwrap(),
919 &"application/x-www-form-urlencoded"
920 );
921
922 let buf = body::read_to_string(r.body_mut().take().unwrap()).unwrap();
923
924 let body_should_be = serde_urlencoded::to_string(&form_data).unwrap();
925 assert_eq!(buf, body_should_be);
926 }
927
928 #[test]
929 #[cfg(feature = "json")]
930 fn add_json() {
931 let client = Client::new();
932 let some_url = "https://google.com/";
933 let r = client.post(some_url);
934
935 let mut json_data = HashMap::new();
936 json_data.insert("foo", "bar");
937
938 let mut r = r.json(&json_data).build().unwrap();
939
940 assert_eq!(r.headers().get(CONTENT_TYPE).unwrap(), &"application/json");
942
943 let buf = body::read_to_string(r.body_mut().take().unwrap()).unwrap();
944
945 let body_should_be = serde_json::to_string(&json_data).unwrap();
946 assert_eq!(buf, body_should_be);
947 }
948
949 #[test]
950 #[cfg(feature = "json")]
951 fn add_json_fail() {
952 use serde::ser::Error as _;
953 use serde::{Serialize, Serializer};
954 use std::error::Error as _;
955 struct MyStruct;
956 impl Serialize for MyStruct {
957 fn serialize<S>(&self, _serializer: S) -> Result<S::Ok, S::Error>
958 where
959 S: Serializer,
960 {
961 Err(S::Error::custom("nope"))
962 }
963 }
964
965 let client = Client::new();
966 let some_url = "https://google.com/";
967 let r = client.post(some_url);
968 let json_data = MyStruct;
969 let err = r.json(&json_data).build().unwrap_err();
970 assert!(err.is_builder()); assert!(err.source().unwrap().is::<serde_json::Error>());
972 }
973
974 #[test]
975 fn test_replace_headers() {
976 use http::HeaderMap;
977
978 let mut headers = HeaderMap::new();
979 headers.insert("foo", "bar".parse().unwrap());
980 headers.append("foo", "baz".parse().unwrap());
981
982 let client = Client::new();
983 let req = client
984 .get("https://hyper.rs")
985 .header("im-a", "keeper")
986 .header("foo", "pop me")
987 .headers(headers)
988 .build()
989 .expect("request build");
990
991 assert_eq!(req.headers()["im-a"], "keeper");
992
993 let foo = req.headers().get_all("foo").iter().collect::<Vec<_>>();
994 assert_eq!(foo.len(), 2);
995 assert_eq!(foo[0], "bar");
996 assert_eq!(foo[1], "baz");
997 }
998
999 #[test]
1000 #[cfg(feature = "query")]
1001 fn normalize_empty_query() {
1002 let client = Client::new();
1003 let some_url = "https://google.com/";
1004 let empty_query: &[(&str, &str)] = &[];
1005
1006 let req = client
1007 .get(some_url)
1008 .query(empty_query)
1009 .build()
1010 .expect("request build");
1011
1012 assert_eq!(req.url().query(), None);
1013 assert_eq!(req.url().as_str(), "https://google.com/");
1014 }
1015
1016 #[test]
1017 fn convert_url_authority_into_basic_auth() {
1018 let client = Client::new();
1019 let some_url = "https://Aladdin:open sesame@localhost/";
1020
1021 let req = client.get(some_url).build().expect("request build");
1022
1023 assert_eq!(req.url().as_str(), "https://localhost/");
1024 assert_eq!(
1025 req.headers()["authorization"],
1026 "Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=="
1027 );
1028 }
1029
1030 #[test]
1031 fn convert_from_http_request() {
1032 let http_request = HttpRequest::builder()
1033 .method("GET")
1034 .uri("http://localhost/")
1035 .header("User-Agent", "my-awesome-agent/1.0")
1036 .body("test test test")
1037 .unwrap();
1038 let req: Request = Request::try_from(http_request).unwrap();
1039 assert_eq!(req.body().is_none(), false);
1040 let test_data = b"test test test";
1041 assert_eq!(req.body().unwrap().as_bytes(), Some(&test_data[..]));
1042 let headers = req.headers();
1043 assert_eq!(headers.get("User-Agent").unwrap(), "my-awesome-agent/1.0");
1044 assert_eq!(req.method(), Method::GET);
1045 assert_eq!(req.url().as_str(), "http://localhost/");
1046 }
1047
1048 #[test]
1049 fn set_http_request_version() {
1050 let http_request = HttpRequest::builder()
1051 .method("GET")
1052 .uri("http://localhost/")
1053 .header("User-Agent", "my-awesome-agent/1.0")
1054 .version(Version::HTTP_11)
1055 .body("test test test")
1056 .unwrap();
1057 let req: Request = Request::try_from(http_request).unwrap();
1058 assert_eq!(req.body().is_none(), false);
1059 let test_data = b"test test test";
1060 assert_eq!(req.body().unwrap().as_bytes(), Some(&test_data[..]));
1061 let headers = req.headers();
1062 assert_eq!(headers.get("User-Agent").unwrap(), "my-awesome-agent/1.0");
1063 assert_eq!(req.method(), Method::GET);
1064 assert_eq!(req.url().as_str(), "http://localhost/");
1065 assert_eq!(req.version(), Version::HTTP_11);
1066 }
1067
1068 #[test]
1069 fn test_basic_auth_sensitive_header() {
1070 let client = Client::new();
1071 let some_url = "https://localhost/";
1072
1073 let req = client
1074 .get(some_url)
1075 .basic_auth("Aladdin", Some("open sesame"))
1076 .build()
1077 .expect("request build");
1078
1079 assert_eq!(req.url().as_str(), "https://localhost/");
1080 assert_eq!(
1081 req.headers()["authorization"],
1082 "Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=="
1083 );
1084 assert_eq!(req.headers()["authorization"].is_sensitive(), true);
1085 }
1086
1087 #[test]
1088 fn test_bearer_auth_sensitive_header() {
1089 let client = Client::new();
1090 let some_url = "https://localhost/";
1091
1092 let req = client
1093 .get(some_url)
1094 .bearer_auth("Hold my bear")
1095 .build()
1096 .expect("request build");
1097
1098 assert_eq!(req.url().as_str(), "https://localhost/");
1099 assert_eq!(req.headers()["authorization"], "Bearer Hold my bear");
1100 assert_eq!(req.headers()["authorization"].is_sensitive(), true);
1101 }
1102
1103 #[test]
1104 fn test_request_cloning() {
1105 let mut request = Request::new(Method::GET, "https://example.com".try_into().unwrap());
1106 *request.timeout_mut() = Some(Duration::from_secs(42));
1107 *request.version_mut() = Version::HTTP_11;
1108
1109 let clone = request.try_clone().unwrap();
1110 assert_eq!(request.version(), clone.version());
1111 assert_eq!(request.headers(), clone.headers());
1112 assert_eq!(request.timeout(), clone.timeout());
1113 }
1114}