1
0
mirror of https://github.com/fafhrd91/actix-web synced 2024-11-24 00:21:08 +01:00
actix-web/awc/src/connect.rs

231 lines
6.5 KiB
Rust
Raw Normal View History

use std::rc::Rc;
2019-09-12 17:52:46 +02:00
use std::{fmt, io, net};
2019-03-28 02:53:19 +01:00
use actix_codec::{AsyncRead, AsyncWrite, Framed};
2019-03-26 05:58:01 +01:00
use actix_http::body::Body;
use actix_http::client::{
Connect as ClientConnect, ConnectError, Connection, SendRequestError,
};
2019-03-28 02:53:19 +01:00
use actix_http::h1::ClientCodec;
use actix_http::http::HeaderMap;
2019-09-12 17:52:46 +02:00
use actix_http::{RequestHead, RequestHeadType, ResponseHead};
2019-03-26 05:58:01 +01:00
use actix_service::Service;
2019-03-28 02:53:19 +01:00
use futures::{Future, Poll};
2019-03-26 05:58:01 +01:00
use crate::response::ClientResponse;
2019-03-26 05:58:01 +01:00
pub(crate) struct ConnectorWrapper<T>(pub T);
pub(crate) trait Connect {
fn send_request(
&mut self,
head: RequestHead,
body: Body,
addr: Option<net::SocketAddr>,
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>>;
2019-03-28 02:53:19 +01:00
fn send_request_extra(
&mut self,
head: Rc<RequestHead>,
extra_headers: Option<HeaderMap>,
body: Body,
addr: Option<net::SocketAddr>,
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>>;
2019-03-28 02:53:19 +01:00
/// Send request, returns Response and Framed
fn open_tunnel(
&mut self,
head: RequestHead,
addr: Option<net::SocketAddr>,
2019-03-28 02:53:19 +01:00
) -> Box<
2019-07-17 11:48:37 +02:00
dyn Future<
2019-03-28 02:53:19 +01:00
Item = (ResponseHead, Framed<BoxedSocket, ClientCodec>),
Error = SendRequestError,
>,
>;
/// Send request and extra headers, returns Response and Framed
fn open_tunnel_extra(
&mut self,
head: Rc<RequestHead>,
extra_headers: Option<HeaderMap>,
addr: Option<net::SocketAddr>,
) -> Box<
dyn Future<
Item = (ResponseHead, Framed<BoxedSocket, ClientCodec>),
Error = SendRequestError,
>,
>;
2019-03-26 05:58:01 +01:00
}
impl<T> Connect for ConnectorWrapper<T>
where
T: Service<Request = ClientConnect, Error = ConnectError>,
2019-03-26 05:58:01 +01:00
T::Response: Connection,
2019-03-28 02:53:19 +01:00
<T::Response as Connection>::Io: 'static,
2019-03-26 05:58:01 +01:00
<T::Response as Connection>::Future: 'static,
2019-03-28 02:53:19 +01:00
<T::Response as Connection>::TunnelFuture: 'static,
2019-03-26 05:58:01 +01:00
T::Future: 'static,
{
fn send_request(
&mut self,
head: RequestHead,
body: Body,
addr: Option<net::SocketAddr>,
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>> {
2019-03-26 05:58:01 +01:00
Box::new(
self.0
// connect to the host
.call(ClientConnect {
uri: head.uri.clone(),
addr,
})
2019-03-26 05:58:01 +01:00
.from_err()
// send request
2019-09-12 17:52:46 +02:00
.and_then(move |connection| {
connection.send_request(RequestHeadType::from(head), body)
})
.map(|(head, payload)| ClientResponse::new(head, payload)),
)
}
fn send_request_extra(
&mut self,
head: Rc<RequestHead>,
extra_headers: Option<HeaderMap>,
body: Body,
addr: Option<net::SocketAddr>,
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>> {
Box::new(
self.0
// connect to the host
.call(ClientConnect {
uri: head.uri.clone(),
addr,
})
.from_err()
// send request
2019-09-12 17:52:46 +02:00
.and_then(move |connection| {
connection
.send_request(RequestHeadType::Rc(head, extra_headers), body)
})
.map(|(head, payload)| ClientResponse::new(head, payload)),
2019-03-26 05:58:01 +01:00
)
}
2019-03-28 02:53:19 +01:00
fn open_tunnel(
&mut self,
head: RequestHead,
addr: Option<net::SocketAddr>,
2019-03-28 02:53:19 +01:00
) -> Box<
2019-07-17 11:48:37 +02:00
dyn Future<
2019-03-28 02:53:19 +01:00
Item = (ResponseHead, Framed<BoxedSocket, ClientCodec>),
Error = SendRequestError,
>,
> {
Box::new(
self.0
// connect to the host
.call(ClientConnect {
uri: head.uri.clone(),
addr,
})
2019-03-28 02:53:19 +01:00
.from_err()
// send request
2019-09-12 17:52:46 +02:00
.and_then(move |connection| {
connection.open_tunnel(RequestHeadType::from(head))
})
.map(|(head, framed)| {
let framed = framed.map_io(|io| BoxedSocket(Box::new(Socket(io))));
(head, framed)
}),
)
}
fn open_tunnel_extra(
&mut self,
head: Rc<RequestHead>,
extra_headers: Option<HeaderMap>,
addr: Option<net::SocketAddr>,
) -> Box<
dyn Future<
Item = (ResponseHead, Framed<BoxedSocket, ClientCodec>),
Error = SendRequestError,
>,
> {
Box::new(
self.0
// connect to the host
.call(ClientConnect {
uri: head.uri.clone(),
addr,
})
.from_err()
// send request
2019-09-12 17:52:46 +02:00
.and_then(move |connection| {
connection.open_tunnel(RequestHeadType::Rc(head, extra_headers))
})
2019-03-28 02:53:19 +01:00
.map(|(head, framed)| {
let framed = framed.map_io(|io| BoxedSocket(Box::new(Socket(io))));
(head, framed)
}),
)
}
}
trait AsyncSocket {
2019-07-17 11:48:37 +02:00
fn as_read(&self) -> &dyn AsyncRead;
fn as_read_mut(&mut self) -> &mut dyn AsyncRead;
fn as_write(&mut self) -> &mut dyn AsyncWrite;
2019-03-28 02:53:19 +01:00
}
struct Socket<T: AsyncRead + AsyncWrite>(T);
impl<T: AsyncRead + AsyncWrite> AsyncSocket for Socket<T> {
2019-07-17 11:48:37 +02:00
fn as_read(&self) -> &dyn AsyncRead {
2019-03-28 02:53:19 +01:00
&self.0
}
2019-07-17 11:48:37 +02:00
fn as_read_mut(&mut self) -> &mut dyn AsyncRead {
2019-03-28 02:53:19 +01:00
&mut self.0
}
2019-07-17 11:48:37 +02:00
fn as_write(&mut self) -> &mut dyn AsyncWrite {
2019-03-28 02:53:19 +01:00
&mut self.0
}
}
pub struct BoxedSocket(Box<dyn AsyncSocket>);
2019-04-12 01:01:54 +02:00
impl fmt::Debug for BoxedSocket {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "BoxedSocket")
}
}
2019-03-28 02:53:19 +01:00
impl io::Read for BoxedSocket {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
self.0.as_read_mut().read(buf)
}
}
impl AsyncRead for BoxedSocket {
unsafe fn prepare_uninitialized_buffer(&self, buf: &mut [u8]) -> bool {
self.0.as_read().prepare_uninitialized_buffer(buf)
}
}
impl io::Write for BoxedSocket {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.0.as_write().write(buf)
}
fn flush(&mut self) -> io::Result<()> {
self.0.as_write().flush()
}
}
impl AsyncWrite for BoxedSocket {
fn shutdown(&mut self) -> Poll<(), io::Error> {
self.0.as_write().shutdown()
}
2019-03-26 05:58:01 +01:00
}