mirror of
https://github.com/fafhrd91/actix-web
synced 2025-01-18 05:41:50 +01:00
return back consuming builder
This commit is contained in:
parent
49a499ce74
commit
e282ef7925
@ -5,8 +5,6 @@
|
|||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
* Added `Deref<Target = RequestHead>` for `ClientRequest`.
|
|
||||||
|
|
||||||
* Export `MessageBody` type
|
* Export `MessageBody` type
|
||||||
|
|
||||||
* `ClientResponse::json()` - Loads and parse `application/json` encoded body
|
* `ClientResponse::json()` - Loads and parse `application/json` encoded body
|
||||||
@ -14,8 +12,6 @@
|
|||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
* Use non-consuming builder pattern for `ClientRequest`.
|
|
||||||
|
|
||||||
* `ClientRequest::json()` accepts reference instead of object.
|
* `ClientRequest::json()` accepts reference instead of object.
|
||||||
|
|
||||||
* `ClientResponse::body()` does not consume response object.
|
* `ClientResponse::body()` does not consume response object.
|
||||||
|
@ -105,7 +105,7 @@ impl Client {
|
|||||||
let mut req = ClientRequest::new(method, url, self.0.clone());
|
let mut req = ClientRequest::new(method, url, self.0.clone());
|
||||||
|
|
||||||
for (key, value) in &self.0.headers {
|
for (key, value) in &self.0.headers {
|
||||||
req.set_header_if_none(key.clone(), value.clone());
|
req = req.set_header_if_none(key.clone(), value.clone());
|
||||||
}
|
}
|
||||||
req
|
req
|
||||||
}
|
}
|
||||||
@ -120,7 +120,7 @@ impl Client {
|
|||||||
{
|
{
|
||||||
let mut req = self.request(head.method.clone(), url);
|
let mut req = self.request(head.method.clone(), url);
|
||||||
for (key, value) in &head.headers {
|
for (key, value) in &head.headers {
|
||||||
req.set_header_if_none(key.clone(), value.clone());
|
req = req.set_header_if_none(key.clone(), value.clone());
|
||||||
}
|
}
|
||||||
req
|
req
|
||||||
}
|
}
|
||||||
|
@ -17,8 +17,8 @@ use actix_http::cookie::{Cookie, CookieJar};
|
|||||||
use actix_http::encoding::Decoder;
|
use actix_http::encoding::Decoder;
|
||||||
use actix_http::http::header::{self, ContentEncoding, Header, IntoHeaderValue};
|
use actix_http::http::header::{self, ContentEncoding, Header, IntoHeaderValue};
|
||||||
use actix_http::http::{
|
use actix_http::http::{
|
||||||
uri, ConnectionType, Error as HttpError, HeaderName, HeaderValue, HttpTryFrom,
|
uri, ConnectionType, Error as HttpError, HeaderMap, HeaderName, HeaderValue,
|
||||||
Method, Uri, Version,
|
HttpTryFrom, Method, Uri, Version,
|
||||||
};
|
};
|
||||||
use actix_http::{Error, Payload, RequestHead};
|
use actix_http::{Error, Payload, RequestHead};
|
||||||
|
|
||||||
@ -58,7 +58,7 @@ const HTTPS_ENCODING: &str = "gzip, deflate";
|
|||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub struct ClientRequest {
|
pub struct ClientRequest {
|
||||||
pub(crate) head: Option<RequestHead>,
|
pub(crate) head: RequestHead,
|
||||||
err: Option<HttpError>,
|
err: Option<HttpError>,
|
||||||
cookies: Option<CookieJar>,
|
cookies: Option<CookieJar>,
|
||||||
default_headers: bool,
|
default_headers: bool,
|
||||||
@ -73,40 +73,36 @@ impl ClientRequest {
|
|||||||
where
|
where
|
||||||
Uri: HttpTryFrom<U>,
|
Uri: HttpTryFrom<U>,
|
||||||
{
|
{
|
||||||
let mut req = ClientRequest {
|
ClientRequest {
|
||||||
config,
|
config,
|
||||||
head: Some(RequestHead::default()),
|
head: RequestHead::default(),
|
||||||
err: None,
|
err: None,
|
||||||
cookies: None,
|
cookies: None,
|
||||||
timeout: None,
|
timeout: None,
|
||||||
default_headers: true,
|
default_headers: true,
|
||||||
response_decompress: true,
|
response_decompress: true,
|
||||||
};
|
}
|
||||||
req.method(method).uri(uri);
|
.method(method)
|
||||||
req
|
.uri(uri)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set HTTP URI of request.
|
/// Set HTTP URI of request.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn uri<U>(&mut self, uri: U) -> &mut Self
|
pub fn uri<U>(mut self, uri: U) -> Self
|
||||||
where
|
where
|
||||||
Uri: HttpTryFrom<U>,
|
Uri: HttpTryFrom<U>,
|
||||||
{
|
{
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match Uri::try_from(uri) {
|
match Uri::try_from(uri) {
|
||||||
Ok(uri) => head.uri = uri,
|
Ok(uri) => self.head.uri = uri,
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set HTTP method of this request.
|
/// Set HTTP method of this request.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn method(&mut self, method: Method) -> &mut Self {
|
pub fn method(mut self, method: Method) -> Self {
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
self.head.method = method;
|
||||||
head.method = method;
|
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,13 +111,23 @@ impl ClientRequest {
|
|||||||
///
|
///
|
||||||
/// By default requests's HTTP version depends on network stream
|
/// By default requests's HTTP version depends on network stream
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn version(&mut self, version: Version) -> &mut Self {
|
pub fn version(mut self, version: Version) -> Self {
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
self.head.version = version;
|
||||||
head.version = version;
|
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
/// Returns request's headers.
|
||||||
|
pub fn headers(&self) -> &HeaderMap {
|
||||||
|
&self.head.headers
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
/// Returns request's mutable headers.
|
||||||
|
pub fn headers_mut(&mut self) -> &mut HeaderMap {
|
||||||
|
&mut self.head.headers
|
||||||
|
}
|
||||||
|
|
||||||
/// Set a header.
|
/// Set a header.
|
||||||
///
|
///
|
||||||
/// ```rust
|
/// ```rust
|
||||||
@ -135,15 +141,13 @@ impl ClientRequest {
|
|||||||
/// # }));
|
/// # }));
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub fn set<H: Header>(&mut self, hdr: H) -> &mut Self {
|
pub fn set<H: Header>(mut self, hdr: H) -> Self {
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match hdr.try_into() {
|
match hdr.try_into() {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
head.headers.insert(H::name(), value);
|
self.head.headers.insert(H::name(), value);
|
||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -165,58 +169,53 @@ impl ClientRequest {
|
|||||||
/// # }));
|
/// # }));
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub fn header<K, V>(&mut self, key: K, value: V) -> &mut Self
|
pub fn header<K, V>(mut self, key: K, value: V) -> Self
|
||||||
where
|
where
|
||||||
HeaderName: HttpTryFrom<K>,
|
HeaderName: HttpTryFrom<K>,
|
||||||
V: IntoHeaderValue,
|
V: IntoHeaderValue,
|
||||||
{
|
{
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match HeaderName::try_from(key) {
|
match HeaderName::try_from(key) {
|
||||||
Ok(key) => match value.try_into() {
|
Ok(key) => match value.try_into() {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
head.headers.append(key, value);
|
let _ = self.head.headers.append(key, value);
|
||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
},
|
},
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Insert a header, replaces existing header.
|
/// Insert a header, replaces existing header.
|
||||||
pub fn set_header<K, V>(&mut self, key: K, value: V) -> &mut Self
|
pub fn set_header<K, V>(mut self, key: K, value: V) -> Self
|
||||||
where
|
where
|
||||||
HeaderName: HttpTryFrom<K>,
|
HeaderName: HttpTryFrom<K>,
|
||||||
V: IntoHeaderValue,
|
V: IntoHeaderValue,
|
||||||
{
|
{
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match HeaderName::try_from(key) {
|
match HeaderName::try_from(key) {
|
||||||
Ok(key) => match value.try_into() {
|
Ok(key) => match value.try_into() {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
head.headers.insert(key, value);
|
let _ = self.head.headers.insert(key, value);
|
||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
},
|
},
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Insert a header only if it is not yet set.
|
/// Insert a header only if it is not yet set.
|
||||||
pub fn set_header_if_none<K, V>(&mut self, key: K, value: V) -> &mut Self
|
pub fn set_header_if_none<K, V>(mut self, key: K, value: V) -> Self
|
||||||
where
|
where
|
||||||
HeaderName: HttpTryFrom<K>,
|
HeaderName: HttpTryFrom<K>,
|
||||||
V: IntoHeaderValue,
|
V: IntoHeaderValue,
|
||||||
{
|
{
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match HeaderName::try_from(key) {
|
match HeaderName::try_from(key) {
|
||||||
Ok(key) => {
|
Ok(key) => {
|
||||||
if !head.headers.contains_key(&key) {
|
if !self.head.headers.contains_key(&key) {
|
||||||
match value.try_into() {
|
match value.try_into() {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
head.headers.insert(key, value);
|
let _ = self.head.headers.insert(key, value);
|
||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
@ -224,47 +223,42 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Close connection instead of returning it back to connections pool.
|
/// Close connection instead of returning it back to connections pool.
|
||||||
/// This setting affect only http/1 connections.
|
/// This setting affect only http/1 connections.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn close_connection(&mut self) -> &mut Self {
|
pub fn close_connection(mut self) -> Self {
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
self.head.set_connection_type(ConnectionType::Close);
|
||||||
head.set_connection_type(ConnectionType::Close);
|
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set request's content type
|
/// Set request's content type
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn content_type<V>(&mut self, value: V) -> &mut Self
|
pub fn content_type<V>(mut self, value: V) -> Self
|
||||||
where
|
where
|
||||||
HeaderValue: HttpTryFrom<V>,
|
HeaderValue: HttpTryFrom<V>,
|
||||||
{
|
{
|
||||||
if let Some(head) = parts(&mut self.head, &self.err) {
|
|
||||||
match HeaderValue::try_from(value) {
|
match HeaderValue::try_from(value) {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
let _ = head.headers.insert(header::CONTENT_TYPE, value);
|
let _ = self.head.headers.insert(header::CONTENT_TYPE, value);
|
||||||
}
|
}
|
||||||
Err(e) => self.err = Some(e.into()),
|
Err(e) => self.err = Some(e.into()),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set content length
|
/// Set content length
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn content_length(&mut self, len: u64) -> &mut Self {
|
pub fn content_length(self, len: u64) -> Self {
|
||||||
let mut wrt = BytesMut::new().writer();
|
let mut wrt = BytesMut::new().writer();
|
||||||
let _ = write!(wrt, "{}", len);
|
let _ = write!(wrt, "{}", len);
|
||||||
self.header(header::CONTENT_LENGTH, wrt.get_mut().take().freeze())
|
self.header(header::CONTENT_LENGTH, wrt.get_mut().take().freeze())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set HTTP basic authorization header
|
/// Set HTTP basic authorization header
|
||||||
pub fn basic_auth<U>(&mut self, username: U, password: Option<&str>) -> &mut Self
|
pub fn basic_auth<U>(self, username: U, password: Option<&str>) -> Self
|
||||||
where
|
where
|
||||||
U: fmt::Display,
|
U: fmt::Display,
|
||||||
{
|
{
|
||||||
@ -279,7 +273,7 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Set HTTP bearer authentication header
|
/// Set HTTP bearer authentication header
|
||||||
pub fn bearer_auth<T>(&mut self, token: T) -> &mut Self
|
pub fn bearer_auth<T>(self, token: T) -> Self
|
||||||
where
|
where
|
||||||
T: fmt::Display,
|
T: fmt::Display,
|
||||||
{
|
{
|
||||||
@ -311,7 +305,7 @@ impl ClientRequest {
|
|||||||
/// }));
|
/// }));
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub fn cookie<'c>(&mut self, cookie: Cookie<'c>) -> &mut Self {
|
pub fn cookie<'c>(mut self, cookie: Cookie<'c>) -> Self {
|
||||||
if self.cookies.is_none() {
|
if self.cookies.is_none() {
|
||||||
let mut jar = CookieJar::new();
|
let mut jar = CookieJar::new();
|
||||||
jar.add(cookie.into_owned());
|
jar.add(cookie.into_owned());
|
||||||
@ -324,13 +318,13 @@ impl ClientRequest {
|
|||||||
|
|
||||||
/// Do not add default request headers.
|
/// Do not add default request headers.
|
||||||
/// By default `Date` and `User-Agent` headers are set.
|
/// By default `Date` and `User-Agent` headers are set.
|
||||||
pub fn no_default_headers(&mut self) -> &mut Self {
|
pub fn no_default_headers(mut self) -> Self {
|
||||||
self.default_headers = false;
|
self.default_headers = false;
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Disable automatic decompress of response's body
|
/// Disable automatic decompress of response's body
|
||||||
pub fn no_decompress(&mut self) -> &mut Self {
|
pub fn no_decompress(mut self) -> Self {
|
||||||
self.response_decompress = false;
|
self.response_decompress = false;
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
@ -339,38 +333,38 @@ impl ClientRequest {
|
|||||||
///
|
///
|
||||||
/// Request timeout is the total time before a response must be received.
|
/// Request timeout is the total time before a response must be received.
|
||||||
/// Default value is 5 seconds.
|
/// Default value is 5 seconds.
|
||||||
pub fn timeout(&mut self, timeout: Duration) -> &mut Self {
|
pub fn timeout(mut self, timeout: Duration) -> Self {
|
||||||
self.timeout = Some(timeout);
|
self.timeout = Some(timeout);
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This method calls provided closure with builder reference if
|
/// This method calls provided closure with builder reference if
|
||||||
/// value is `true`.
|
/// value is `true`.
|
||||||
pub fn if_true<F>(&mut self, value: bool, f: F) -> &mut Self
|
pub fn if_true<F>(mut self, value: bool, f: F) -> Self
|
||||||
where
|
where
|
||||||
F: FnOnce(&mut ClientRequest),
|
F: FnOnce(&mut ClientRequest),
|
||||||
{
|
{
|
||||||
if value {
|
if value {
|
||||||
f(self);
|
f(&mut self);
|
||||||
}
|
}
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This method calls provided closure with builder reference if
|
/// This method calls provided closure with builder reference if
|
||||||
/// value is `Some`.
|
/// value is `Some`.
|
||||||
pub fn if_some<T, F>(&mut self, value: Option<T>, f: F) -> &mut Self
|
pub fn if_some<T, F>(mut self, value: Option<T>, f: F) -> Self
|
||||||
where
|
where
|
||||||
F: FnOnce(T, &mut ClientRequest),
|
F: FnOnce(T, &mut ClientRequest),
|
||||||
{
|
{
|
||||||
if let Some(val) = value {
|
if let Some(val) = value {
|
||||||
f(val, self);
|
f(val, &mut self);
|
||||||
}
|
}
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Complete request construction and send body.
|
/// Complete request construction and send body.
|
||||||
pub fn send_body<B>(
|
pub fn send_body<B>(
|
||||||
&mut self,
|
mut self,
|
||||||
body: B,
|
body: B,
|
||||||
) -> impl Future<
|
) -> impl Future<
|
||||||
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
||||||
@ -383,10 +377,8 @@ impl ClientRequest {
|
|||||||
return Either::A(err(e.into()));
|
return Either::A(err(e.into()));
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut head = self.head.take().expect("cannot reuse response builder");
|
|
||||||
|
|
||||||
// validate uri
|
// validate uri
|
||||||
let uri = &head.uri;
|
let uri = &self.head.uri;
|
||||||
if uri.host().is_none() {
|
if uri.host().is_none() {
|
||||||
return Either::A(err(InvalidUrl::MissingHost.into()));
|
return Either::A(err(InvalidUrl::MissingHost.into()));
|
||||||
} else if uri.scheme_part().is_none() {
|
} else if uri.scheme_part().is_none() {
|
||||||
@ -403,18 +395,18 @@ impl ClientRequest {
|
|||||||
// set default headers
|
// set default headers
|
||||||
if self.default_headers {
|
if self.default_headers {
|
||||||
// set request host header
|
// set request host header
|
||||||
if let Some(host) = head.uri.host() {
|
if let Some(host) = self.head.uri.host() {
|
||||||
if !head.headers.contains_key(header::HOST) {
|
if !self.head.headers.contains_key(header::HOST) {
|
||||||
let mut wrt = BytesMut::with_capacity(host.len() + 5).writer();
|
let mut wrt = BytesMut::with_capacity(host.len() + 5).writer();
|
||||||
|
|
||||||
let _ = match head.uri.port_u16() {
|
let _ = match self.head.uri.port_u16() {
|
||||||
None | Some(80) | Some(443) => write!(wrt, "{}", host),
|
None | Some(80) | Some(443) => write!(wrt, "{}", host),
|
||||||
Some(port) => write!(wrt, "{}:{}", host, port),
|
Some(port) => write!(wrt, "{}:{}", host, port),
|
||||||
};
|
};
|
||||||
|
|
||||||
match wrt.get_mut().take().freeze().try_into() {
|
match wrt.get_mut().take().freeze().try_into() {
|
||||||
Ok(value) => {
|
Ok(value) => {
|
||||||
head.headers.insert(header::HOST, value);
|
self.head.headers.insert(header::HOST, value);
|
||||||
}
|
}
|
||||||
Err(e) => return Either::A(err(HttpError::from(e).into())),
|
Err(e) => return Either::A(err(HttpError::from(e).into())),
|
||||||
}
|
}
|
||||||
@ -422,33 +414,12 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// user agent
|
// user agent
|
||||||
self.set_header_if_none(
|
if !self.head.headers.contains_key(&header::USER_AGENT) {
|
||||||
|
self.head.headers.insert(
|
||||||
header::USER_AGENT,
|
header::USER_AGENT,
|
||||||
concat!("awc/", env!("CARGO_PKG_VERSION")),
|
HeaderValue::from_static(concat!("awc/", env!("CARGO_PKG_VERSION"))),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// enable br only for https
|
|
||||||
let https = head
|
|
||||||
.uri
|
|
||||||
.scheme_part()
|
|
||||||
.map(|s| s == &uri::Scheme::HTTPS)
|
|
||||||
.unwrap_or(true);
|
|
||||||
|
|
||||||
#[cfg(any(
|
|
||||||
feature = "brotli",
|
|
||||||
feature = "flate2-zlib",
|
|
||||||
feature = "flate2-rust"
|
|
||||||
))]
|
|
||||||
{
|
|
||||||
if https {
|
|
||||||
self.set_header_if_none(header::ACCEPT_ENCODING, HTTPS_ENCODING);
|
|
||||||
} else {
|
|
||||||
#[cfg(any(feature = "flate2-zlib", feature = "flate2-rust"))]
|
|
||||||
{
|
|
||||||
self.set_header_if_none(header::ACCEPT_ENCODING, "gzip, deflate");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// set cookies
|
// set cookies
|
||||||
@ -459,14 +430,43 @@ impl ClientRequest {
|
|||||||
let value = percent_encode(c.value().as_bytes(), USERINFO_ENCODE_SET);
|
let value = percent_encode(c.value().as_bytes(), USERINFO_ENCODE_SET);
|
||||||
let _ = write!(&mut cookie, "; {}={}", name, value);
|
let _ = write!(&mut cookie, "; {}={}", name, value);
|
||||||
}
|
}
|
||||||
head.headers.insert(
|
self.head.headers.insert(
|
||||||
header::COOKIE,
|
header::COOKIE,
|
||||||
HeaderValue::from_str(&cookie.as_str()[2..]).unwrap(),
|
HeaderValue::from_str(&cookie.as_str()[2..]).unwrap(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
let config = self.config.as_ref();
|
let slf = self;
|
||||||
let response_decompress = self.response_decompress;
|
|
||||||
|
// enable br only for https
|
||||||
|
#[cfg(any(
|
||||||
|
feature = "brotli",
|
||||||
|
feature = "flate2-zlib",
|
||||||
|
feature = "flate2-rust"
|
||||||
|
))]
|
||||||
|
let slf = {
|
||||||
|
let https = slf
|
||||||
|
.head
|
||||||
|
.uri
|
||||||
|
.scheme_part()
|
||||||
|
.map(|s| s == &uri::Scheme::HTTPS)
|
||||||
|
.unwrap_or(true);
|
||||||
|
|
||||||
|
if https {
|
||||||
|
slf.set_header_if_none(header::ACCEPT_ENCODING, HTTPS_ENCODING)
|
||||||
|
} else {
|
||||||
|
#[cfg(any(feature = "flate2-zlib", feature = "flate2-rust"))]
|
||||||
|
{
|
||||||
|
slf.set_header_if_none(header::ACCEPT_ENCODING, "gzip, deflate")
|
||||||
|
}
|
||||||
|
#[cfg(not(any(feature = "flate2-zlib", feature = "flate2-rust")))]
|
||||||
|
slf
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let head = slf.head;
|
||||||
|
let config = slf.config.as_ref();
|
||||||
|
let response_decompress = slf.response_decompress;
|
||||||
|
|
||||||
let fut = config
|
let fut = config
|
||||||
.connector
|
.connector
|
||||||
@ -483,7 +483,7 @@ impl ClientRequest {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// set request timeout
|
// set request timeout
|
||||||
if let Some(timeout) = self.timeout.or_else(|| config.timeout.clone()) {
|
if let Some(timeout) = slf.timeout.or_else(|| config.timeout.clone()) {
|
||||||
Either::B(Either::A(Timeout::new(fut, timeout).map_err(|e| {
|
Either::B(Either::A(Timeout::new(fut, timeout).map_err(|e| {
|
||||||
if let Some(e) = e.into_inner() {
|
if let Some(e) = e.into_inner() {
|
||||||
e
|
e
|
||||||
@ -498,7 +498,7 @@ impl ClientRequest {
|
|||||||
|
|
||||||
/// Set a JSON body and generate `ClientRequest`
|
/// Set a JSON body and generate `ClientRequest`
|
||||||
pub fn send_json<T: Serialize>(
|
pub fn send_json<T: Serialize>(
|
||||||
&mut self,
|
self,
|
||||||
value: &T,
|
value: &T,
|
||||||
) -> impl Future<
|
) -> impl Future<
|
||||||
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
||||||
@ -510,16 +510,16 @@ impl ClientRequest {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// set content-type
|
// set content-type
|
||||||
self.set_header_if_none(header::CONTENT_TYPE, "application/json");
|
let slf = self.set_header_if_none(header::CONTENT_TYPE, "application/json");
|
||||||
|
|
||||||
Either::B(self.send_body(Body::Bytes(Bytes::from(body))))
|
Either::B(slf.send_body(Body::Bytes(Bytes::from(body))))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set a urlencoded body and generate `ClientRequest`
|
/// Set a urlencoded body and generate `ClientRequest`
|
||||||
///
|
///
|
||||||
/// `ClientRequestBuilder` can not be used after this call.
|
/// `ClientRequestBuilder` can not be used after this call.
|
||||||
pub fn send_form<T: Serialize>(
|
pub fn send_form<T: Serialize>(
|
||||||
&mut self,
|
self,
|
||||||
value: &T,
|
value: &T,
|
||||||
) -> impl Future<
|
) -> impl Future<
|
||||||
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
||||||
@ -531,17 +531,17 @@ impl ClientRequest {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// set content-type
|
// set content-type
|
||||||
self.set_header_if_none(
|
let slf = self.set_header_if_none(
|
||||||
header::CONTENT_TYPE,
|
header::CONTENT_TYPE,
|
||||||
"application/x-www-form-urlencoded",
|
"application/x-www-form-urlencoded",
|
||||||
);
|
);
|
||||||
|
|
||||||
Either::B(self.send_body(Body::Bytes(Bytes::from(body))))
|
Either::B(slf.send_body(Body::Bytes(Bytes::from(body))))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set an streaming body and generate `ClientRequest`.
|
/// Set an streaming body and generate `ClientRequest`.
|
||||||
pub fn send_stream<S, E>(
|
pub fn send_stream<S, E>(
|
||||||
&mut self,
|
self,
|
||||||
stream: S,
|
stream: S,
|
||||||
) -> impl Future<
|
) -> impl Future<
|
||||||
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
||||||
@ -556,7 +556,7 @@ impl ClientRequest {
|
|||||||
|
|
||||||
/// Set an empty body and generate `ClientRequest`.
|
/// Set an empty body and generate `ClientRequest`.
|
||||||
pub fn send(
|
pub fn send(
|
||||||
&mut self,
|
self,
|
||||||
) -> impl Future<
|
) -> impl Future<
|
||||||
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
Item = ClientResponse<impl Stream<Item = Bytes, Error = PayloadError>>,
|
||||||
Error = SendRequestError,
|
Error = SendRequestError,
|
||||||
@ -565,48 +565,21 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl std::ops::Deref for ClientRequest {
|
|
||||||
type Target = RequestHead;
|
|
||||||
|
|
||||||
fn deref(&self) -> &RequestHead {
|
|
||||||
self.head.as_ref().expect("cannot reuse response builder")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl std::ops::DerefMut for ClientRequest {
|
|
||||||
fn deref_mut(&mut self) -> &mut RequestHead {
|
|
||||||
self.head.as_mut().expect("cannot reuse response builder")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Debug for ClientRequest {
|
impl fmt::Debug for ClientRequest {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
let head = self.head.as_ref().expect("cannot reuse response builder");
|
|
||||||
|
|
||||||
writeln!(
|
writeln!(
|
||||||
f,
|
f,
|
||||||
"\nClientRequest {:?} {}:{}",
|
"\nClientRequest {:?} {}:{}",
|
||||||
head.version, head.method, head.uri
|
self.head.version, self.head.method, self.head.uri
|
||||||
)?;
|
)?;
|
||||||
writeln!(f, " headers:")?;
|
writeln!(f, " headers:")?;
|
||||||
for (key, val) in head.headers.iter() {
|
for (key, val) in self.head.headers.iter() {
|
||||||
writeln!(f, " {:?}: {:?}", key, val)?;
|
writeln!(f, " {:?}: {:?}", key, val)?;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn parts<'a>(
|
|
||||||
parts: &'a mut Option<RequestHead>,
|
|
||||||
err: &Option<HttpError>,
|
|
||||||
) -> Option<&'a mut RequestHead> {
|
|
||||||
if err.is_some() {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
parts.as_mut()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
@ -615,8 +588,7 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn test_debug() {
|
fn test_debug() {
|
||||||
test::run_on(|| {
|
test::run_on(|| {
|
||||||
let mut request = Client::new().get("/");
|
let request = Client::new().get("/").header("x-test", "111");
|
||||||
request.header("x-test", "111");
|
|
||||||
let repr = format!("{:?}", request);
|
let repr = format!("{:?}", request);
|
||||||
assert!(repr.contains("ClientRequest"));
|
assert!(repr.contains("ClientRequest"));
|
||||||
assert!(repr.contains("x-test"));
|
assert!(repr.contains("x-test"));
|
||||||
@ -633,8 +605,6 @@ mod tests {
|
|||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
req.head
|
req.head
|
||||||
.as_ref()
|
|
||||||
.unwrap()
|
|
||||||
.headers
|
.headers
|
||||||
.get(header::CONTENT_TYPE)
|
.get(header::CONTENT_TYPE)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -648,16 +618,14 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn test_client_header_override() {
|
fn test_client_header_override() {
|
||||||
test::run_on(|| {
|
test::run_on(|| {
|
||||||
let mut req = Client::build()
|
let req = Client::build()
|
||||||
.header(header::CONTENT_TYPE, "111")
|
.header(header::CONTENT_TYPE, "111")
|
||||||
.finish()
|
.finish()
|
||||||
.get("/");
|
.get("/")
|
||||||
req.set_header(header::CONTENT_TYPE, "222");
|
.set_header(header::CONTENT_TYPE, "222");
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
req.head
|
req.head
|
||||||
.as_ref()
|
|
||||||
.unwrap()
|
|
||||||
.headers
|
.headers
|
||||||
.get(header::CONTENT_TYPE)
|
.get(header::CONTENT_TYPE)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -671,12 +639,11 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn client_basic_auth() {
|
fn client_basic_auth() {
|
||||||
test::run_on(|| {
|
test::run_on(|| {
|
||||||
let mut req = Client::new().get("/");
|
let req = Client::new()
|
||||||
req.basic_auth("username", Some("password"));
|
.get("/")
|
||||||
|
.basic_auth("username", Some("password"));
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
req.head
|
req.head
|
||||||
.as_ref()
|
|
||||||
.unwrap()
|
|
||||||
.headers
|
.headers
|
||||||
.get(header::AUTHORIZATION)
|
.get(header::AUTHORIZATION)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -685,12 +652,9 @@ mod tests {
|
|||||||
"Basic dXNlcm5hbWU6cGFzc3dvcmQ="
|
"Basic dXNlcm5hbWU6cGFzc3dvcmQ="
|
||||||
);
|
);
|
||||||
|
|
||||||
let mut req = Client::new().get("/");
|
let req = Client::new().get("/").basic_auth("username", None);
|
||||||
req.basic_auth("username", None);
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
req.head
|
req.head
|
||||||
.as_ref()
|
|
||||||
.unwrap()
|
|
||||||
.headers
|
.headers
|
||||||
.get(header::AUTHORIZATION)
|
.get(header::AUTHORIZATION)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -704,12 +668,9 @@ mod tests {
|
|||||||
#[test]
|
#[test]
|
||||||
fn client_bearer_auth() {
|
fn client_bearer_auth() {
|
||||||
test::run_on(|| {
|
test::run_on(|| {
|
||||||
let mut req = Client::new().get("/");
|
let req = Client::new().get("/").bearer_auth("someS3cr3tAutht0k3n");
|
||||||
req.bearer_auth("someS3cr3tAutht0k3n");
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
req.head
|
req.head
|
||||||
.as_ref()
|
|
||||||
.unwrap()
|
|
||||||
.headers
|
.headers
|
||||||
.get(header::AUTHORIZATION)
|
.get(header::AUTHORIZATION)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user