2019-09-10 06:29:32 +02:00
|
|
|
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;
|
2019-04-20 03:03:44 +02:00
|
|
|
use actix_http::client::{
|
|
|
|
Connect as ClientConnect, ConnectError, Connection, SendRequestError,
|
|
|
|
};
|
2019-03-28 02:53:19 +01:00
|
|
|
use actix_http::h1::ClientCodec;
|
2019-09-10 06:29:32 +02:00
|
|
|
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
|
|
|
|
2019-03-26 19:41:38 +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,
|
2019-04-20 03:03:44 +02:00
|
|
|
addr: Option<net::SocketAddr>,
|
2019-07-17 07:44:39 +02:00
|
|
|
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>>;
|
2019-03-28 02:53:19 +01:00
|
|
|
|
2019-09-10 06:29:32 +02: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,
|
2019-04-20 03:03:44 +02:00
|
|
|
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,
|
|
|
|
>,
|
|
|
|
>;
|
2019-09-10 06:29:32 +02:00
|
|
|
|
|
|
|
/// 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
|
2019-04-20 03:03:44 +02:00
|
|
|
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,
|
2019-04-20 03:03:44 +02:00
|
|
|
addr: Option<net::SocketAddr>,
|
2019-07-17 07:44:39 +02:00
|
|
|
) -> Box<dyn Future<Item = ClientResponse, Error = SendRequestError>> {
|
2019-03-26 05:58:01 +01:00
|
|
|
Box::new(
|
|
|
|
self.0
|
|
|
|
// connect to the host
|
2019-04-20 03:03:44 +02:00
|
|
|
.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)
|
|
|
|
})
|
2019-09-10 06:29:32 +02:00
|
|
|
.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)
|
|
|
|
})
|
2019-03-26 19:41:38 +01:00
|
|
|
.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,
|
2019-04-20 03:03:44 +02:00
|
|
|
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
|
2019-04-20 03:03:44 +02:00
|
|
|
.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))
|
|
|
|
})
|
2019-09-10 06:29:32 +02:00
|
|
|
.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
|
|
|
}
|