2021-02-07 19:56:39 +01:00
|
|
|
use std::{
|
2021-02-28 19:17:08 +01:00
|
|
|
fmt,
|
|
|
|
future::Future,
|
|
|
|
io, net,
|
2021-02-07 19:56:39 +01:00
|
|
|
pin::Pin,
|
|
|
|
task::{Context, Poll},
|
|
|
|
};
|
2019-03-28 02:53:19 +01:00
|
|
|
|
2021-01-04 00:47:04 +01:00
|
|
|
use actix_codec::{AsyncRead, AsyncWrite, Framed, ReadBuf};
|
2021-02-07 19:56:39 +01:00
|
|
|
use actix_http::{
|
|
|
|
body::Body,
|
|
|
|
client::{Connect as ClientConnect, ConnectError, Connection, SendRequestError},
|
|
|
|
h1::ClientCodec,
|
2021-02-28 19:17:08 +01:00
|
|
|
Payload, RequestHead, RequestHeadType, ResponseHead,
|
2019-04-20 03:03:44 +02:00
|
|
|
};
|
2019-03-26 05:58:01 +01:00
|
|
|
use actix_service::Service;
|
2021-02-28 19:17:08 +01:00
|
|
|
use futures_core::{future::LocalBoxFuture, ready};
|
2019-03-26 05:58:01 +01:00
|
|
|
|
2019-03-26 19:41:38 +01:00
|
|
|
use crate::response::ClientResponse;
|
|
|
|
|
2021-02-28 19:17:08 +01:00
|
|
|
pub type ConnectorService = Box<
|
2021-02-17 18:10:46 +01:00
|
|
|
dyn Service<
|
|
|
|
ConnectRequest,
|
|
|
|
Response = ConnectResponse,
|
|
|
|
Error = SendRequestError,
|
|
|
|
Future = LocalBoxFuture<'static, Result<ConnectResponse, SendRequestError>>,
|
|
|
|
>,
|
|
|
|
>;
|
|
|
|
|
|
|
|
pub enum ConnectRequest {
|
|
|
|
Client(RequestHeadType, Body, Option<net::SocketAddr>),
|
|
|
|
Tunnel(RequestHead, Option<net::SocketAddr>),
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum ConnectResponse {
|
|
|
|
Client(ClientResponse),
|
|
|
|
Tunnel(ResponseHead, Framed<BoxedSocket, ClientCodec>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ConnectResponse {
|
|
|
|
pub fn into_client_response(self) -> ClientResponse {
|
|
|
|
match self {
|
|
|
|
ConnectResponse::Client(res) => res,
|
|
|
|
_ => panic!(
|
|
|
|
"ClientResponse only reachable with ConnectResponse::ClientResponse variant"
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn into_tunnel_response(self) -> (ResponseHead, Framed<BoxedSocket, ClientCodec>) {
|
|
|
|
match self {
|
|
|
|
ConnectResponse::Tunnel(head, framed) => (head, framed),
|
|
|
|
_ => panic!(
|
|
|
|
"TunnelResponse only reachable with ConnectResponse::TunnelResponse variant"
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-28 19:17:08 +01:00
|
|
|
pub(crate) struct DefaultConnector<S> {
|
|
|
|
connector: S,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<S> DefaultConnector<S> {
|
|
|
|
pub(crate) fn new(connector: S) -> Self {
|
|
|
|
Self { connector }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<S> Service<ConnectRequest> for DefaultConnector<S>
|
2019-03-26 05:58:01 +01:00
|
|
|
where
|
2021-02-28 19:17:08 +01:00
|
|
|
S: Service<ClientConnect, Error = ConnectError>,
|
|
|
|
S::Response: Connection,
|
|
|
|
<S::Response as Connection>::Io: 'static,
|
2019-03-26 05:58:01 +01:00
|
|
|
{
|
2021-02-17 18:10:46 +01:00
|
|
|
type Response = ConnectResponse;
|
|
|
|
type Error = SendRequestError;
|
2021-02-28 19:17:08 +01:00
|
|
|
type Future = ConnectRequestFuture<S::Future, <S::Response as Connection>::Io>;
|
2019-11-19 04:55:17 +01:00
|
|
|
|
2021-02-17 18:10:46 +01:00
|
|
|
actix_service::forward_ready!(connector);
|
2019-11-19 04:55:17 +01:00
|
|
|
|
2021-02-17 18:10:46 +01:00
|
|
|
fn call(&self, req: ConnectRequest) -> Self::Future {
|
2019-11-19 04:55:17 +01:00
|
|
|
// connect to the host
|
2021-02-17 18:10:46 +01:00
|
|
|
let fut = match req {
|
|
|
|
ConnectRequest::Client(ref head, .., addr) => self.connector.call(ClientConnect {
|
|
|
|
uri: head.as_ref().uri.clone(),
|
|
|
|
addr,
|
|
|
|
}),
|
|
|
|
ConnectRequest::Tunnel(ref head, addr) => self.connector.call(ClientConnect {
|
|
|
|
uri: head.uri.clone(),
|
|
|
|
addr,
|
|
|
|
}),
|
|
|
|
};
|
2019-11-19 04:55:17 +01:00
|
|
|
|
2021-02-28 19:17:08 +01:00
|
|
|
ConnectRequestFuture::Connection {
|
|
|
|
fut,
|
|
|
|
req: Some(req),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-17 18:10:46 +01:00
|
|
|
|
2021-02-28 19:17:08 +01:00
|
|
|
pin_project_lite::pin_project! {
|
|
|
|
#[project = ConnectRequestProj]
|
|
|
|
pub(crate) enum ConnectRequestFuture<Fut, Io> {
|
|
|
|
Connection {
|
|
|
|
#[pin]
|
|
|
|
fut: Fut,
|
|
|
|
req: Option<ConnectRequest>
|
|
|
|
},
|
|
|
|
Client {
|
|
|
|
fut: LocalBoxFuture<'static, Result<(ResponseHead, Payload), SendRequestError>>
|
|
|
|
},
|
|
|
|
Tunnel {
|
|
|
|
fut: LocalBoxFuture<
|
|
|
|
'static,
|
|
|
|
Result<(ResponseHead, Framed<Io, ClientCodec>), SendRequestError>,
|
|
|
|
>,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-17 18:10:46 +01:00
|
|
|
|
2021-02-28 19:17:08 +01:00
|
|
|
impl<Fut, C, Io> Future for ConnectRequestFuture<Fut, Io>
|
|
|
|
where
|
|
|
|
Fut: Future<Output = Result<C, ConnectError>>,
|
|
|
|
C: Connection<Io = Io>,
|
|
|
|
Io: AsyncRead + AsyncWrite + Unpin + 'static,
|
|
|
|
{
|
|
|
|
type Output = Result<ConnectResponse, SendRequestError>;
|
|
|
|
|
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
match self.as_mut().project() {
|
|
|
|
ConnectRequestProj::Connection { fut, req } => {
|
|
|
|
let connection = ready!(fut.poll(cx))?;
|
|
|
|
let req = req.take().unwrap();
|
|
|
|
match req {
|
|
|
|
ConnectRequest::Client(head, body, ..) => {
|
|
|
|
// send request
|
|
|
|
let fut = ConnectRequestFuture::Client {
|
|
|
|
fut: connection.send_request(head, body),
|
|
|
|
};
|
|
|
|
self.as_mut().set(fut);
|
|
|
|
}
|
|
|
|
ConnectRequest::Tunnel(head, ..) => {
|
|
|
|
// send request
|
|
|
|
let fut = ConnectRequestFuture::Tunnel {
|
|
|
|
fut: connection.open_tunnel(RequestHeadType::from(head)),
|
|
|
|
};
|
|
|
|
self.as_mut().set(fut);
|
|
|
|
}
|
2021-02-17 18:10:46 +01:00
|
|
|
}
|
2021-02-28 19:17:08 +01:00
|
|
|
self.poll(cx)
|
2021-02-17 18:10:46 +01:00
|
|
|
}
|
2021-02-28 19:17:08 +01:00
|
|
|
ConnectRequestProj::Client { fut } => {
|
|
|
|
let (head, payload) = ready!(fut.as_mut().poll(cx))?;
|
|
|
|
Poll::Ready(Ok(ConnectResponse::Client(ClientResponse::new(
|
|
|
|
head, payload,
|
|
|
|
))))
|
|
|
|
}
|
|
|
|
ConnectRequestProj::Tunnel { fut } => {
|
|
|
|
let (head, framed) = ready!(fut.as_mut().poll(cx))?;
|
|
|
|
let framed = framed.into_map_io(|io| BoxedSocket(Box::new(Socket(io))));
|
|
|
|
Poll::Ready(Ok(ConnectResponse::Tunnel(head, framed)))
|
|
|
|
}
|
|
|
|
}
|
2019-09-10 06:29:32 +02:00
|
|
|
}
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
trait AsyncSocket {
|
2019-11-19 04:55:17 +01:00
|
|
|
fn as_read(&self) -> &(dyn AsyncRead + Unpin);
|
|
|
|
fn as_read_mut(&mut self) -> &mut (dyn AsyncRead + Unpin);
|
|
|
|
fn as_write(&mut self) -> &mut (dyn AsyncWrite + Unpin);
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
|
|
|
|
2019-11-19 04:55:17 +01:00
|
|
|
struct Socket<T: AsyncRead + AsyncWrite + Unpin>(T);
|
2019-03-28 02:53:19 +01:00
|
|
|
|
2019-11-19 04:55:17 +01:00
|
|
|
impl<T: AsyncRead + AsyncWrite + Unpin> AsyncSocket for Socket<T> {
|
|
|
|
fn as_read(&self) -> &(dyn AsyncRead + Unpin) {
|
2019-03-28 02:53:19 +01:00
|
|
|
&self.0
|
|
|
|
}
|
2019-11-19 04:55:17 +01:00
|
|
|
fn as_read_mut(&mut self) -> &mut (dyn AsyncRead + Unpin) {
|
2019-03-28 02:53:19 +01:00
|
|
|
&mut self.0
|
|
|
|
}
|
2019-11-19 04:55:17 +01:00
|
|
|
fn as_write(&mut self) -> &mut (dyn AsyncWrite + Unpin) {
|
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 {
|
2019-12-08 07:31:16 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2019-04-12 01:01:54 +02:00
|
|
|
write!(f, "BoxedSocket")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 02:53:19 +01:00
|
|
|
impl AsyncRead for BoxedSocket {
|
2019-11-19 04:55:17 +01:00
|
|
|
fn poll_read(
|
|
|
|
self: Pin<&mut Self>,
|
|
|
|
cx: &mut Context<'_>,
|
2021-01-04 00:47:04 +01:00
|
|
|
buf: &mut ReadBuf<'_>,
|
|
|
|
) -> Poll<io::Result<()>> {
|
2019-11-19 04:55:17 +01:00
|
|
|
Pin::new(self.get_mut().0.as_read_mut()).poll_read(cx, buf)
|
|
|
|
}
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
|
|
|
|
2019-11-19 04:55:17 +01:00
|
|
|
impl AsyncWrite for BoxedSocket {
|
|
|
|
fn poll_write(
|
|
|
|
self: Pin<&mut Self>,
|
|
|
|
cx: &mut Context<'_>,
|
|
|
|
buf: &[u8],
|
|
|
|
) -> Poll<io::Result<usize>> {
|
|
|
|
Pin::new(self.get_mut().0.as_write()).poll_write(cx, buf)
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
|
|
|
|
2019-11-19 04:55:17 +01:00
|
|
|
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
|
|
|
Pin::new(self.get_mut().0.as_write()).poll_flush(cx)
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
|
|
|
|
2021-02-12 00:03:17 +01:00
|
|
|
fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
|
2019-11-19 04:55:17 +01:00
|
|
|
Pin::new(self.get_mut().0.as_write()).poll_shutdown(cx)
|
2019-03-28 02:53:19 +01:00
|
|
|
}
|
2019-03-26 05:58:01 +01:00
|
|
|
}
|