oauth2 3.0.0

An extensible, strongly-typed implementation of OAuth2
Documentation
use failure::Fail;

///
/// Error type returned by failed reqwest HTTP requests.
///
#[derive(Debug, Fail)]
pub enum Error<T>
where
    T: std::error::Error + Send + Sync + 'static,
{
    /// Error returned by reqwest crate.
    #[fail(display = "request failed")]
    Reqwest(#[cause] T),
    /// Non-reqwest HTTP error.
    #[fail(display = "HTTP error")]
    Http(#[cause] http::Error),
    /// I/O error.
    #[fail(display = "I/O error")]
    Io(#[cause] std::io::Error),
    /// Other error.
    #[fail(display = "Other error: {}", _0)]
    Other(String),
}

#[cfg(any(feature = "reqwest-09", feature = "reqwest-010"))]
pub use blocking::http_client;
///
/// Error type returned by failed reqwest blocking HTTP requests.
///
#[cfg(any(feature = "reqwest-09", feature = "reqwest-010"))]
pub type HttpClientError = Error<blocking::reqwest::Error>;

#[cfg(all(feature = "futures-01", feature = "reqwest-09"))]
pub use future_client::future_http_client;
///
/// Error type returned by failed reqwest futures HTTP requests.
///
#[cfg(all(feature = "futures-01", feature = "reqwest-09"))]
pub type FutureHttpClientError = Error<reqwest_0_9::Error>;

#[cfg(all(feature = "futures-03", feature = "reqwest-010"))]
pub use async_client::async_http_client;
///
/// Error type returned by failed reqwest async HTTP requests.
///
#[cfg(all(feature = "futures-03", feature = "reqwest-010"))]
pub type AsyncHttpClientError = Error<reqwest_0_10::Error>;

#[cfg(any(feature = "reqwest-09", feature = "reqwest-010"))]
mod blocking {
    use super::super::{HttpRequest, HttpResponse};
    use super::Error;

    #[cfg(all(feature = "reqwest-09", not(feature = "reqwest-010")))]
    use reqwest_0_9 as blocking;
    #[cfg(all(feature = "reqwest-09", not(feature = "reqwest-010")))]
    pub use reqwest_0_9 as reqwest;
    #[cfg(all(feature = "reqwest-09", not(feature = "reqwest-010")))]
    use reqwest_0_9::RedirectPolicy;

    #[cfg(feature = "reqwest-010")]
    pub use reqwest_0_10 as reqwest;
    #[cfg(feature = "reqwest-010")]
    use reqwest_0_10::blocking;
    #[cfg(feature = "reqwest-010")]
    use reqwest_0_10::redirect::Policy as RedirectPolicy;

    use std::io::Read;

    ///
    /// Synchronous HTTP client.
    ///
    pub fn http_client(request: HttpRequest) -> Result<HttpResponse, Error<reqwest::Error>> {
        let client = blocking::Client::builder()
            // Following redirects opens the client up to SSRF vulnerabilities.
            .redirect(RedirectPolicy::none())
            .build()
            .map_err(Error::Reqwest)?;

        #[cfg(all(feature = "reqwest-09", not(feature = "reqwest-010")))]
        let mut request_builder = client
            .request(request.method, request.url.as_str())
            .body(request.body);
        #[cfg(feature = "reqwest-010")]
        let mut request_builder = client
            .request(
                http_0_2::Method::from_bytes(request.method.as_str().as_ref())
                    .expect("failed to convert Method from http 0.2 to 0.1"),
                request.url.as_str(),
            )
            .body(request.body);

        for (name, value) in &request.headers {
            request_builder = request_builder.header(name.as_str(), value.as_bytes());
        }
        let mut response = client
            .execute(request_builder.build().map_err(Error::Reqwest)?)
            .map_err(Error::Reqwest)?;

        let mut body = Vec::new();
        response.read_to_end(&mut body).map_err(Error::Io)?;

        #[cfg(all(feature = "reqwest-09", not(feature = "reqwest-010")))]
        {
            Ok(HttpResponse {
                status_code: response.status(),
                headers: response.headers().clone(),
                body,
            })
        }
        #[cfg(feature = "reqwest-010")]
        {
            let headers = response
                .headers()
                .iter()
                .map(|(name, value)| {
                    (
                        http::header::HeaderName::from_bytes(name.as_str().as_ref())
                            .expect("failed to convert HeaderName from http 0.2 to 0.1"),
                        http::HeaderValue::from_bytes(value.as_bytes())
                            .expect("failed to convert HeaderValue from http 0.2 to 0.1"),
                    )
                })
                .collect::<http::HeaderMap>();
            Ok(HttpResponse {
                status_code: http::StatusCode::from_u16(response.status().as_u16())
                    .expect("failed to convert StatusCode from http 0.2 to 0.1"),
                headers,
                body,
            })
        }
    }
}

#[cfg(all(feature = "futures-01", feature = "reqwest-09"))]
mod future_client {
    use super::super::{HttpRequest, HttpResponse};
    use super::Error;

    use futures_0_1::{Future, IntoFuture, Stream};
    use reqwest_0_9 as reqwest;
    use reqwest_0_9::r#async::Client as AsyncClient;

    ///
    /// Asynchronous HTTP client.
    ///
    pub fn future_http_client(
        request: HttpRequest,
    ) -> impl Future<Item = HttpResponse, Error = Error<reqwest::Error>> {
        AsyncClient::builder()
            // Following redirects opens the client up to SSRF vulnerabilities.
            .redirect(reqwest::RedirectPolicy::none())
            .build()
            .map_err(Error::Reqwest)
            .into_future()
            .and_then(|client| {
                let mut request_builder = client
                    .request(request.method, request.url.as_str())
                    .body(request.body);
                for (name, value) in &request.headers {
                    request_builder = request_builder.header(name, value);
                }
                request_builder
                    .build()
                    .map_err(Error::Reqwest)
                    .into_future()
                    .and_then(move |request| {
                        client
                            .execute(request)
                            .and_then(|response| {
                                let status_code = response.status();
                                let headers = response.headers().clone();
                                response
                                    .into_body()
                                    .map(|chunk| chunk.as_ref().to_vec())
                                    .collect()
                                    .map(move |body| HttpResponse {
                                        status_code,
                                        headers,
                                        body: body.into_iter().flatten().collect::<_>(),
                                    })
                            })
                            .map_err(Error::Reqwest)
                    })
            })
    }
}

#[cfg(all(feature = "reqwest-010", feature = "futures-03"))]
mod async_client;