2019-03-30 05:13:39 +01:00
|
|
|
use std::fmt::Write as FmtWrite;
|
2019-03-26 05:58:01 +01:00
|
|
|
use std::io::Write;
|
|
|
|
use std::rc::Rc;
|
2019-09-12 06:40:56 +02:00
|
|
|
use std::time::Duration;
|
2019-04-20 03:03:44 +02:00
|
|
|
use std::{fmt, net};
|
2019-03-26 05:58:01 +01:00
|
|
|
|
|
|
|
use bytes::{BufMut, Bytes, BytesMut};
|
2019-09-12 06:40:56 +02:00
|
|
|
use futures::Stream;
|
2019-08-13 19:48:11 +02:00
|
|
|
use percent_encoding::percent_encode;
|
2019-03-26 05:58:01 +01:00
|
|
|
use serde::Serialize;
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
use actix_http::body::Body;
|
2019-08-13 19:48:11 +02:00
|
|
|
use actix_http::cookie::{Cookie, CookieJar, USERINFO};
|
2019-09-12 06:40:56 +02:00
|
|
|
use actix_http::http::header::{self, Header, IntoHeaderValue};
|
2019-03-26 05:58:01 +01:00
|
|
|
use actix_http::http::{
|
2019-04-02 21:51:16 +02:00
|
|
|
uri, ConnectionType, Error as HttpError, HeaderMap, HeaderName, HeaderValue,
|
|
|
|
HttpTryFrom, Method, Uri, Version,
|
2019-03-26 05:58:01 +01:00
|
|
|
};
|
2019-09-12 06:40:56 +02:00
|
|
|
use actix_http::{Error, RequestHead};
|
2019-03-26 05:58:01 +01:00
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
use crate::error::{FreezeRequestError, InvalidUrl};
|
|
|
|
use crate::frozen::FrozenClientRequest;
|
|
|
|
use crate::sender::{PrepForSendingError, RequestSender, SendClientRequest};
|
2019-03-29 06:33:41 +01:00
|
|
|
use crate::ClientConfig;
|
2019-03-27 04:45:00 +01:00
|
|
|
|
|
|
|
#[cfg(any(feature = "brotli", feature = "flate2-zlib", feature = "flate2-rust"))]
|
|
|
|
const HTTPS_ENCODING: &str = "br, gzip, deflate";
|
|
|
|
#[cfg(all(
|
|
|
|
any(feature = "flate2-zlib", feature = "flate2-rust"),
|
|
|
|
not(feature = "brotli")
|
|
|
|
))]
|
|
|
|
const HTTPS_ENCODING: &str = "gzip, deflate";
|
2019-03-26 05:58:01 +01:00
|
|
|
|
|
|
|
/// An HTTP Client request builder
|
|
|
|
///
|
|
|
|
/// This type can be used to construct an instance of `ClientRequest` through a
|
|
|
|
/// builder-like pattern.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use futures::future::{Future, lazy};
|
|
|
|
/// use actix_rt::System;
|
|
|
|
///
|
|
|
|
/// fn main() {
|
|
|
|
/// System::new("test").block_on(lazy(|| {
|
2019-03-26 19:41:38 +01:00
|
|
|
/// awc::Client::new()
|
|
|
|
/// .get("http://www.rust-lang.org") // <- Create request builder
|
2019-03-26 05:58:01 +01:00
|
|
|
/// .header("User-Agent", "Actix-web")
|
2019-03-26 19:41:38 +01:00
|
|
|
/// .send() // <- Send http request
|
2019-03-26 05:58:01 +01:00
|
|
|
/// .map_err(|_| ())
|
2019-03-26 19:41:38 +01:00
|
|
|
/// .and_then(|response| { // <- server http response
|
2019-03-26 05:58:01 +01:00
|
|
|
/// println!("Response: {:?}", response);
|
|
|
|
/// Ok(())
|
|
|
|
/// })
|
|
|
|
/// }));
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
pub struct ClientRequest {
|
2019-04-02 21:51:16 +02:00
|
|
|
pub(crate) head: RequestHead,
|
2019-03-26 05:58:01 +01:00
|
|
|
err: Option<HttpError>,
|
2019-04-20 03:03:44 +02:00
|
|
|
addr: Option<net::SocketAddr>,
|
2019-03-26 05:58:01 +01:00
|
|
|
cookies: Option<CookieJar>,
|
2019-03-27 04:45:00 +01:00
|
|
|
response_decompress: bool,
|
2019-03-29 22:07:37 +01:00
|
|
|
timeout: Option<Duration>,
|
2019-03-29 06:33:41 +01:00
|
|
|
config: Rc<ClientConfig>,
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ClientRequest {
|
|
|
|
/// Create new client request builder.
|
2019-03-29 06:33:41 +01:00
|
|
|
pub(crate) fn new<U>(method: Method, uri: U, config: Rc<ClientConfig>) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
Uri: HttpTryFrom<U>,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
ClientRequest {
|
2019-03-29 06:33:41 +01:00
|
|
|
config,
|
2019-04-02 21:51:16 +02:00
|
|
|
head: RequestHead::default(),
|
2019-03-30 00:27:18 +01:00
|
|
|
err: None,
|
2019-04-20 03:03:44 +02:00
|
|
|
addr: None,
|
2019-03-26 05:58:01 +01:00
|
|
|
cookies: None,
|
2019-03-29 22:07:37 +01:00
|
|
|
timeout: None,
|
2019-03-27 04:45:00 +01:00
|
|
|
response_decompress: true,
|
2019-04-02 21:51:16 +02:00
|
|
|
}
|
|
|
|
.method(method)
|
|
|
|
.uri(uri)
|
2019-03-30 00:27:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set HTTP URI of request.
|
|
|
|
#[inline]
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn uri<U>(mut self, uri: U) -> Self
|
2019-03-30 00:27:18 +01:00
|
|
|
where
|
|
|
|
Uri: HttpTryFrom<U>,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
match Uri::try_from(uri) {
|
|
|
|
Ok(uri) => self.head.uri = uri,
|
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-30 00:27:18 +01:00
|
|
|
}
|
|
|
|
self
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
|
2019-09-10 06:29:32 +02:00
|
|
|
/// Get HTTP URI of request
|
|
|
|
pub fn get_uri(&self) -> &Uri {
|
|
|
|
&self.head.uri
|
|
|
|
}
|
|
|
|
|
2019-04-20 03:03:44 +02:00
|
|
|
/// Set socket address of the server.
|
|
|
|
///
|
|
|
|
/// This address is used for connection. If address is not
|
|
|
|
/// provided url's host name get resolved.
|
|
|
|
pub fn address(mut self, addr: net::SocketAddr) -> Self {
|
|
|
|
self.addr = Some(addr);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
/// Set HTTP method of this request.
|
|
|
|
#[inline]
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn method(mut self, method: Method) -> Self {
|
|
|
|
self.head.method = method;
|
2019-03-26 05:58:01 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-09-10 06:29:32 +02:00
|
|
|
/// Get HTTP method of this request
|
|
|
|
pub fn get_method(&self) -> &Method {
|
|
|
|
&self.head.method
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
#[doc(hidden)]
|
|
|
|
/// Set HTTP version of this request.
|
|
|
|
///
|
|
|
|
/// By default requests's HTTP version depends on network stream
|
|
|
|
#[inline]
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn version(mut self, version: Version) -> Self {
|
|
|
|
self.head.version = version;
|
2019-03-26 05:58:01 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-04-02 21:51:16 +02:00
|
|
|
#[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
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
/// Set a header.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// fn main() {
|
|
|
|
/// # actix_rt::System::new("test").block_on(futures::future::lazy(|| {
|
|
|
|
/// let req = awc::Client::new()
|
|
|
|
/// .get("http://www.rust-lang.org")
|
|
|
|
/// .set(awc::http::header::Date::now())
|
|
|
|
/// .set(awc::http::header::ContentType(mime::TEXT_HTML));
|
|
|
|
/// # Ok::<_, ()>(())
|
|
|
|
/// # }));
|
|
|
|
/// }
|
|
|
|
/// ```
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn set<H: Header>(mut self, hdr: H) -> Self {
|
|
|
|
match hdr.try_into() {
|
|
|
|
Ok(value) => {
|
|
|
|
self.head.headers.insert(H::name(), value);
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
2019-04-02 21:51:16 +02:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Append a header.
|
|
|
|
///
|
|
|
|
/// Header gets appended to existing header.
|
|
|
|
/// To override header use `set_header()` method.
|
|
|
|
///
|
|
|
|
/// ```rust
|
2019-03-27 06:33:01 +01:00
|
|
|
/// use awc::{http, Client};
|
2019-03-26 05:58:01 +01:00
|
|
|
///
|
|
|
|
/// fn main() {
|
2019-03-26 19:41:38 +01:00
|
|
|
/// # actix_rt::System::new("test").block_on(futures::future::lazy(|| {
|
2019-03-27 06:33:01 +01:00
|
|
|
/// let req = Client::new()
|
2019-03-26 19:41:38 +01:00
|
|
|
/// .get("http://www.rust-lang.org")
|
2019-03-26 05:58:01 +01:00
|
|
|
/// .header("X-TEST", "value")
|
2019-03-26 19:41:38 +01:00
|
|
|
/// .header(http::header::CONTENT_TYPE, "application/json");
|
|
|
|
/// # Ok::<_, ()>(())
|
|
|
|
/// # }));
|
2019-03-26 05:58:01 +01:00
|
|
|
/// }
|
|
|
|
/// ```
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn header<K, V>(mut self, key: K, value: V) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
HeaderName: HttpTryFrom<K>,
|
|
|
|
V: IntoHeaderValue,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
match HeaderName::try_from(key) {
|
|
|
|
Ok(key) => match value.try_into() {
|
2019-07-17 11:08:30 +02:00
|
|
|
Ok(value) => self.head.headers.append(key, value),
|
2019-03-26 05:58:01 +01:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-04-02 21:51:16 +02:00
|
|
|
},
|
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Insert a header, replaces existing header.
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn set_header<K, V>(mut self, key: K, value: V) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
HeaderName: HttpTryFrom<K>,
|
|
|
|
V: IntoHeaderValue,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
match HeaderName::try_from(key) {
|
|
|
|
Ok(key) => match value.try_into() {
|
2019-07-17 11:08:30 +02:00
|
|
|
Ok(value) => self.head.headers.insert(key, value),
|
2019-03-26 05:58:01 +01:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-04-02 21:51:16 +02:00
|
|
|
},
|
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Insert a header only if it is not yet set.
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn set_header_if_none<K, V>(mut self, key: K, value: V) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
HeaderName: HttpTryFrom<K>,
|
|
|
|
V: IntoHeaderValue,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
match HeaderName::try_from(key) {
|
|
|
|
Ok(key) => {
|
|
|
|
if !self.head.headers.contains_key(&key) {
|
|
|
|
match value.try_into() {
|
2019-07-17 11:08:30 +02:00
|
|
|
Ok(value) => self.head.headers.insert(key, value),
|
2019-04-02 21:51:16 +02:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-02 21:51:16 +02:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-04-24 20:27:57 +02:00
|
|
|
/// Send headers in `Camel-Case` form.
|
2019-04-24 19:48:49 +02:00
|
|
|
#[inline]
|
2019-04-24 20:27:57 +02:00
|
|
|
pub fn camel_case(mut self) -> Self {
|
|
|
|
self.head.set_camel_case_headers(true);
|
2019-04-24 19:48:49 +02:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-04-02 23:08:30 +02:00
|
|
|
/// Force close connection instead of returning it back to connections pool.
|
2019-04-02 00:19:34 +02:00
|
|
|
/// This setting affect only http/1 connections.
|
2019-03-26 05:58:01 +01:00
|
|
|
#[inline]
|
2019-04-02 23:08:30 +02:00
|
|
|
pub fn force_close(mut self) -> Self {
|
2019-04-02 21:51:16 +02:00
|
|
|
self.head.set_connection_type(ConnectionType::Close);
|
2019-03-26 05:58:01 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set request's content type
|
|
|
|
#[inline]
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn content_type<V>(mut self, value: V) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
HeaderValue: HttpTryFrom<V>,
|
|
|
|
{
|
2019-04-02 21:51:16 +02:00
|
|
|
match HeaderValue::try_from(value) {
|
2019-07-17 11:08:30 +02:00
|
|
|
Ok(value) => self.head.headers.insert(header::CONTENT_TYPE, value),
|
2019-04-02 21:51:16 +02:00
|
|
|
Err(e) => self.err = Some(e.into()),
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set content length
|
|
|
|
#[inline]
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn content_length(self, len: u64) -> Self {
|
2019-03-26 05:58:01 +01:00
|
|
|
let mut wrt = BytesMut::new().writer();
|
|
|
|
let _ = write!(wrt, "{}", len);
|
|
|
|
self.header(header::CONTENT_LENGTH, wrt.get_mut().take().freeze())
|
|
|
|
}
|
|
|
|
|
2019-03-27 05:31:18 +01:00
|
|
|
/// Set HTTP basic authorization header
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn basic_auth<U>(self, username: U, password: Option<&str>) -> Self
|
2019-03-27 04:57:06 +01:00
|
|
|
where
|
|
|
|
U: fmt::Display,
|
|
|
|
{
|
|
|
|
let auth = match password {
|
|
|
|
Some(password) => format!("{}:{}", username, password),
|
2019-07-01 05:34:42 +02:00
|
|
|
None => format!("{}:", username),
|
2019-03-27 04:57:06 +01:00
|
|
|
};
|
|
|
|
self.header(
|
|
|
|
header::AUTHORIZATION,
|
|
|
|
format!("Basic {}", base64::encode(&auth)),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-03-27 05:31:18 +01:00
|
|
|
/// Set HTTP bearer authentication header
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn bearer_auth<T>(self, token: T) -> Self
|
2019-03-27 04:57:06 +01:00
|
|
|
where
|
|
|
|
T: fmt::Display,
|
|
|
|
{
|
|
|
|
self.header(header::AUTHORIZATION, format!("Bearer {}", token))
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
/// Set a cookie
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// # use actix_rt::System;
|
|
|
|
/// # use futures::future::{lazy, Future};
|
|
|
|
/// fn main() {
|
|
|
|
/// System::new("test").block_on(lazy(|| {
|
|
|
|
/// awc::Client::new().get("https://www.rust-lang.org")
|
|
|
|
/// .cookie(
|
|
|
|
/// awc::http::Cookie::build("name", "value")
|
|
|
|
/// .domain("www.rust-lang.org")
|
|
|
|
/// .path("/")
|
|
|
|
/// .secure(true)
|
|
|
|
/// .http_only(true)
|
|
|
|
/// .finish(),
|
|
|
|
/// )
|
|
|
|
/// .send()
|
|
|
|
/// .map_err(|_| ())
|
|
|
|
/// .and_then(|response| {
|
|
|
|
/// println!("Response: {:?}", response);
|
|
|
|
/// Ok(())
|
|
|
|
/// })
|
|
|
|
/// }));
|
|
|
|
/// }
|
|
|
|
/// ```
|
2019-07-17 11:08:30 +02:00
|
|
|
pub fn cookie(mut self, cookie: Cookie<'_>) -> Self {
|
2019-03-26 05:58:01 +01:00
|
|
|
if self.cookies.is_none() {
|
|
|
|
let mut jar = CookieJar::new();
|
|
|
|
jar.add(cookie.into_owned());
|
|
|
|
self.cookies = Some(jar)
|
|
|
|
} else {
|
|
|
|
self.cookies.as_mut().unwrap().add(cookie.into_owned());
|
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-03-27 04:45:00 +01:00
|
|
|
/// Disable automatic decompress of response's body
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn no_decompress(mut self) -> Self {
|
2019-03-27 04:45:00 +01:00
|
|
|
self.response_decompress = false;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-03-29 22:07:37 +01:00
|
|
|
/// Set request timeout. Overrides client wide timeout setting.
|
|
|
|
///
|
|
|
|
/// Request timeout is the total time before a response must be received.
|
|
|
|
/// Default value is 5 seconds.
|
2019-04-02 21:51:16 +02:00
|
|
|
pub fn timeout(mut self, timeout: Duration) -> Self {
|
2019-03-29 22:07:37 +01:00
|
|
|
self.timeout = Some(timeout);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
/// This method calls provided closure with builder reference if
|
|
|
|
/// value is `true`.
|
2019-04-17 19:28:27 +02:00
|
|
|
pub fn if_true<F>(self, value: bool, f: F) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
2019-04-17 00:43:55 +02:00
|
|
|
F: FnOnce(ClientRequest) -> ClientRequest,
|
2019-03-26 05:58:01 +01:00
|
|
|
{
|
|
|
|
if value {
|
2019-04-17 00:43:55 +02:00
|
|
|
f(self)
|
|
|
|
} else {
|
|
|
|
self
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This method calls provided closure with builder reference if
|
|
|
|
/// value is `Some`.
|
2019-04-17 19:28:27 +02:00
|
|
|
pub fn if_some<T, F>(self, value: Option<T>, f: F) -> Self
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
2019-04-17 00:43:55 +02:00
|
|
|
F: FnOnce(T, ClientRequest) -> ClientRequest,
|
2019-03-26 05:58:01 +01:00
|
|
|
{
|
|
|
|
if let Some(val) = value {
|
2019-04-17 00:43:55 +02:00
|
|
|
f(val, self)
|
|
|
|
} else {
|
|
|
|
self
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
/// Freeze request builder and construct `FrozenClientRequest`,
|
|
|
|
/// which could be used for sending same request multiple times.
|
2019-09-10 06:29:32 +02:00
|
|
|
pub fn freeze(self) -> Result<FrozenClientRequest, FreezeRequestError> {
|
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return Err(e.into()),
|
|
|
|
};
|
|
|
|
|
|
|
|
let request = FrozenClientRequest {
|
|
|
|
head: Rc::new(slf.head),
|
|
|
|
addr: slf.addr,
|
|
|
|
response_decompress: slf.response_decompress,
|
|
|
|
timeout: slf.timeout,
|
|
|
|
config: slf.config,
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(request)
|
|
|
|
}
|
|
|
|
|
2019-03-26 05:58:01 +01:00
|
|
|
/// Complete request construction and send body.
|
2019-09-12 06:40:56 +02:00
|
|
|
pub fn send_body<B>(self, body: B) -> SendClientRequest
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
|
|
|
B: Into<Body>,
|
|
|
|
{
|
2019-09-10 06:29:32 +02:00
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return e.into(),
|
|
|
|
};
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
RequestSender::Owned(slf.head).send_body(
|
|
|
|
slf.addr,
|
|
|
|
slf.response_decompress,
|
|
|
|
slf.timeout,
|
|
|
|
slf.config.as_ref(),
|
|
|
|
body,
|
|
|
|
)
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set a JSON body and generate `ClientRequest`
|
2019-09-12 06:40:56 +02:00
|
|
|
pub fn send_json<T: Serialize>(self, value: &T) -> SendClientRequest {
|
2019-09-10 06:29:32 +02:00
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return e.into(),
|
|
|
|
};
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
RequestSender::Owned(slf.head).send_json(
|
|
|
|
slf.addr,
|
|
|
|
slf.response_decompress,
|
|
|
|
slf.timeout,
|
|
|
|
slf.config.as_ref(),
|
|
|
|
value,
|
|
|
|
)
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set a urlencoded body and generate `ClientRequest`
|
|
|
|
///
|
|
|
|
/// `ClientRequestBuilder` can not be used after this call.
|
2019-09-12 06:40:56 +02:00
|
|
|
pub fn send_form<T: Serialize>(self, value: &T) -> SendClientRequest {
|
2019-09-10 06:29:32 +02:00
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return e.into(),
|
|
|
|
};
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
RequestSender::Owned(slf.head).send_form(
|
|
|
|
slf.addr,
|
|
|
|
slf.response_decompress,
|
|
|
|
slf.timeout,
|
|
|
|
slf.config.as_ref(),
|
|
|
|
value,
|
|
|
|
)
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set an streaming body and generate `ClientRequest`.
|
2019-09-12 06:40:56 +02:00
|
|
|
pub fn send_stream<S, E>(self, stream: S) -> SendClientRequest
|
2019-09-10 06:29:32 +02:00
|
|
|
where
|
|
|
|
S: Stream<Item = Bytes, Error = E> + 'static,
|
|
|
|
E: Into<Error> + 'static,
|
|
|
|
{
|
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return e.into(),
|
|
|
|
};
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
RequestSender::Owned(slf.head).send_stream(
|
|
|
|
slf.addr,
|
|
|
|
slf.response_decompress,
|
|
|
|
slf.timeout,
|
|
|
|
slf.config.as_ref(),
|
|
|
|
stream,
|
|
|
|
)
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set an empty body and generate `ClientRequest`.
|
2019-09-12 06:40:56 +02:00
|
|
|
pub fn send(self) -> SendClientRequest {
|
2019-09-10 06:29:32 +02:00
|
|
|
let slf = match self.prep_for_sending() {
|
|
|
|
Ok(slf) => slf,
|
|
|
|
Err(e) => return e.into(),
|
|
|
|
};
|
|
|
|
|
2019-09-12 06:40:56 +02:00
|
|
|
RequestSender::Owned(slf.head).send(
|
|
|
|
slf.addr,
|
|
|
|
slf.response_decompress,
|
|
|
|
slf.timeout,
|
|
|
|
slf.config.as_ref(),
|
|
|
|
)
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn prep_for_sending(mut self) -> Result<Self, PrepForSendingError> {
|
|
|
|
if let Some(e) = self.err {
|
|
|
|
return Err(e.into());
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
|
2019-03-27 04:45:00 +01:00
|
|
|
// validate uri
|
2019-04-02 21:51:16 +02:00
|
|
|
let uri = &self.head.uri;
|
2019-03-27 04:45:00 +01:00
|
|
|
if uri.host().is_none() {
|
2019-09-10 06:29:32 +02:00
|
|
|
return Err(InvalidUrl::MissingHost.into());
|
2019-03-27 04:45:00 +01:00
|
|
|
} else if uri.scheme_part().is_none() {
|
2019-09-10 06:29:32 +02:00
|
|
|
return Err(InvalidUrl::MissingScheme.into());
|
2019-03-27 04:45:00 +01:00
|
|
|
} else if let Some(scheme) = uri.scheme_part() {
|
|
|
|
match scheme.as_str() {
|
|
|
|
"http" | "ws" | "https" | "wss" => (),
|
2019-09-10 06:29:32 +02:00
|
|
|
_ => return Err(InvalidUrl::UnknownScheme.into()),
|
2019-03-27 04:45:00 +01:00
|
|
|
}
|
|
|
|
} else {
|
2019-09-10 06:29:32 +02:00
|
|
|
return Err(InvalidUrl::UnknownScheme.into());
|
2019-03-27 04:45:00 +01:00
|
|
|
}
|
2019-03-26 05:58:01 +01:00
|
|
|
|
2019-03-30 05:13:39 +01:00
|
|
|
// set cookies
|
2019-04-02 00:19:34 +02:00
|
|
|
if let Some(ref mut jar) = self.cookies {
|
2019-03-30 05:13:39 +01:00
|
|
|
let mut cookie = String::new();
|
|
|
|
for c in jar.delta() {
|
2019-08-13 19:48:11 +02:00
|
|
|
let name = percent_encode(c.name().as_bytes(), USERINFO);
|
|
|
|
let value = percent_encode(c.value().as_bytes(), USERINFO);
|
2019-03-30 05:13:39 +01:00
|
|
|
let _ = write!(&mut cookie, "; {}={}", name, value);
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
2019-04-02 21:51:16 +02:00
|
|
|
self.head.headers.insert(
|
2019-03-30 05:13:39 +01:00
|
|
|
header::COOKIE,
|
|
|
|
HeaderValue::from_str(&cookie.as_str()[2..]).unwrap(),
|
|
|
|
);
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|
|
|
|
|
2019-04-08 20:09:57 +02:00
|
|
|
let mut slf = self;
|
2019-04-02 21:51:16 +02:00
|
|
|
|
|
|
|
// enable br only for https
|
|
|
|
#[cfg(any(
|
|
|
|
feature = "brotli",
|
|
|
|
feature = "flate2-zlib",
|
|
|
|
feature = "flate2-rust"
|
|
|
|
))]
|
2019-04-08 20:09:57 +02:00
|
|
|
{
|
|
|
|
if slf.response_decompress {
|
|
|
|
let https = slf
|
|
|
|
.head
|
|
|
|
.uri
|
|
|
|
.scheme_part()
|
|
|
|
.map(|s| s == &uri::Scheme::HTTPS)
|
|
|
|
.unwrap_or(true);
|
|
|
|
|
|
|
|
if https {
|
|
|
|
slf = slf.set_header_if_none(header::ACCEPT_ENCODING, HTTPS_ENCODING)
|
|
|
|
} else {
|
|
|
|
#[cfg(any(feature = "flate2-zlib", feature = "flate2-rust"))]
|
2019-09-12 06:40:56 +02:00
|
|
|
{
|
|
|
|
slf = slf
|
|
|
|
.set_header_if_none(header::ACCEPT_ENCODING, "gzip, deflate")
|
|
|
|
}
|
2019-04-08 20:09:57 +02:00
|
|
|
};
|
2019-04-02 21:51:16 +02:00
|
|
|
}
|
2019-04-08 20:09:57 +02:00
|
|
|
}
|
2019-04-02 21:51:16 +02:00
|
|
|
|
2019-09-10 06:29:32 +02:00
|
|
|
Ok(slf)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for ClientRequest {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
writeln!(
|
|
|
|
f,
|
|
|
|
"\nClientRequest {:?} {}:{}",
|
|
|
|
self.head.version, self.head.method, self.head.uri
|
|
|
|
)?;
|
|
|
|
writeln!(f, " headers:")?;
|
|
|
|
for (key, val) in self.head.headers.iter() {
|
|
|
|
writeln!(f, " {:?}: {:?}", key, val)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-29 05:48:35 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2019-04-15 05:20:33 +02:00
|
|
|
use std::time::SystemTime;
|
|
|
|
|
2019-03-29 05:48:35 +01:00
|
|
|
use super::*;
|
2019-04-14 16:43:53 +02:00
|
|
|
use crate::Client;
|
2019-03-29 05:48:35 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_debug() {
|
2019-04-14 16:43:53 +02:00
|
|
|
let request = Client::new().get("/").header("x-test", "111");
|
|
|
|
let repr = format!("{:?}", request);
|
|
|
|
assert!(repr.contains("ClientRequest"));
|
|
|
|
assert!(repr.contains("x-test"));
|
2019-03-29 05:48:35 +01:00
|
|
|
}
|
|
|
|
|
2019-04-15 05:20:33 +02:00
|
|
|
#[test]
|
|
|
|
fn test_basics() {
|
|
|
|
let mut req = Client::new()
|
|
|
|
.put("/")
|
|
|
|
.version(Version::HTTP_2)
|
|
|
|
.set(header::Date(SystemTime::now().into()))
|
|
|
|
.content_type("plain/text")
|
2019-04-19 06:28:23 +02:00
|
|
|
.if_true(true, |req| req.header(header::SERVER, "awc"))
|
|
|
|
.if_true(false, |req| req.header(header::EXPECT, "awc"))
|
|
|
|
.if_some(Some("server"), |val, req| {
|
|
|
|
req.header(header::USER_AGENT, val)
|
|
|
|
})
|
|
|
|
.if_some(Option::<&str>::None, |_, req| {
|
|
|
|
req.header(header::ALLOW, "1")
|
|
|
|
})
|
2019-04-15 05:20:33 +02:00
|
|
|
.content_length(100);
|
|
|
|
assert!(req.headers().contains_key(header::CONTENT_TYPE));
|
|
|
|
assert!(req.headers().contains_key(header::DATE));
|
2019-04-19 06:28:23 +02:00
|
|
|
assert!(req.headers().contains_key(header::SERVER));
|
|
|
|
assert!(req.headers().contains_key(header::USER_AGENT));
|
|
|
|
assert!(!req.headers().contains_key(header::ALLOW));
|
|
|
|
assert!(!req.headers().contains_key(header::EXPECT));
|
2019-04-15 05:20:33 +02:00
|
|
|
assert_eq!(req.head.version, Version::HTTP_2);
|
|
|
|
let _ = req.headers_mut();
|
|
|
|
let _ = req.send_body("");
|
|
|
|
}
|
|
|
|
|
2019-03-29 05:48:35 +01:00
|
|
|
#[test]
|
|
|
|
fn test_client_header() {
|
2019-04-14 16:43:53 +02:00
|
|
|
let req = Client::build()
|
|
|
|
.header(header::CONTENT_TYPE, "111")
|
|
|
|
.finish()
|
|
|
|
.get("/");
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
req.head
|
|
|
|
.headers
|
|
|
|
.get(header::CONTENT_TYPE)
|
|
|
|
.unwrap()
|
|
|
|
.to_str()
|
|
|
|
.unwrap(),
|
|
|
|
"111"
|
|
|
|
);
|
2019-03-29 05:48:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_client_header_override() {
|
2019-04-14 16:43:53 +02:00
|
|
|
let req = Client::build()
|
|
|
|
.header(header::CONTENT_TYPE, "111")
|
|
|
|
.finish()
|
|
|
|
.get("/")
|
|
|
|
.set_header(header::CONTENT_TYPE, "222");
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
req.head
|
|
|
|
.headers
|
|
|
|
.get(header::CONTENT_TYPE)
|
|
|
|
.unwrap()
|
|
|
|
.to_str()
|
|
|
|
.unwrap(),
|
|
|
|
"222"
|
|
|
|
);
|
2019-03-29 05:48:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn client_basic_auth() {
|
2019-04-14 16:43:53 +02:00
|
|
|
let req = Client::new()
|
|
|
|
.get("/")
|
|
|
|
.basic_auth("username", Some("password"));
|
|
|
|
assert_eq!(
|
|
|
|
req.head
|
|
|
|
.headers
|
|
|
|
.get(header::AUTHORIZATION)
|
|
|
|
.unwrap()
|
|
|
|
.to_str()
|
|
|
|
.unwrap(),
|
|
|
|
"Basic dXNlcm5hbWU6cGFzc3dvcmQ="
|
|
|
|
);
|
2019-03-29 05:48:35 +01:00
|
|
|
|
2019-04-14 16:43:53 +02:00
|
|
|
let req = Client::new().get("/").basic_auth("username", None);
|
|
|
|
assert_eq!(
|
|
|
|
req.head
|
|
|
|
.headers
|
|
|
|
.get(header::AUTHORIZATION)
|
|
|
|
.unwrap()
|
|
|
|
.to_str()
|
|
|
|
.unwrap(),
|
2019-07-01 05:34:42 +02:00
|
|
|
"Basic dXNlcm5hbWU6"
|
2019-04-14 16:43:53 +02:00
|
|
|
);
|
2019-03-29 05:48:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn client_bearer_auth() {
|
2019-04-14 16:43:53 +02:00
|
|
|
let req = Client::new().get("/").bearer_auth("someS3cr3tAutht0k3n");
|
|
|
|
assert_eq!(
|
|
|
|
req.head
|
|
|
|
.headers
|
|
|
|
.get(header::AUTHORIZATION)
|
|
|
|
.unwrap()
|
|
|
|
.to_str()
|
|
|
|
.unwrap(),
|
|
|
|
"Bearer someS3cr3tAutht0k3n"
|
|
|
|
);
|
2019-03-29 05:48:35 +01:00
|
|
|
}
|
|
|
|
}
|