cleanup
This commit is contained in:
parent
ac76f52f6d
commit
b9bf0f005d
6 changed files with 231 additions and 152 deletions
121
src/main.rs
121
src/main.rs
|
@ -8,6 +8,7 @@ mod tls;
|
||||||
mod tunnel;
|
mod tunnel;
|
||||||
mod udp;
|
mod udp;
|
||||||
|
|
||||||
|
use anyhow::anyhow;
|
||||||
use base64::Engine;
|
use base64::Engine;
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use futures_util::{stream, TryStreamExt};
|
use futures_util::{stream, TryStreamExt};
|
||||||
|
@ -26,6 +27,8 @@ use std::str::FromStr;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use std::{fmt, io};
|
use std::{fmt, io};
|
||||||
|
use tokio::io::{AsyncRead, AsyncWrite};
|
||||||
|
use tokio::net::TcpStream;
|
||||||
|
|
||||||
use tokio_rustls::rustls::server::DnsName;
|
use tokio_rustls::rustls::server::DnsName;
|
||||||
use tokio_rustls::rustls::{Certificate, PrivateKey, ServerName};
|
use tokio_rustls::rustls::{Certificate, PrivateKey, ServerName};
|
||||||
|
@ -33,7 +36,8 @@ use tokio_rustls::rustls::{Certificate, PrivateKey, ServerName};
|
||||||
use tracing::{error, info};
|
use tracing::{error, info};
|
||||||
|
|
||||||
use crate::dns::DnsResolver;
|
use crate::dns::DnsResolver;
|
||||||
use crate::tunnel::to_host_port;
|
use crate::tunnel::{to_host_port, RemoteAddr};
|
||||||
|
use crate::udp::MyUdpSocket;
|
||||||
use tracing_subscriber::filter::Directive;
|
use tracing_subscriber::filter::Directive;
|
||||||
use tracing_subscriber::EnvFilter;
|
use tracing_subscriber::EnvFilter;
|
||||||
use url::{Host, Url};
|
use url::{Host, Url};
|
||||||
|
@ -656,11 +660,10 @@ async fn main() {
|
||||||
let client_config = Arc::new(client_config);
|
let client_config = Arc::new(client_config);
|
||||||
|
|
||||||
// Start tunnels
|
// Start tunnels
|
||||||
for mut tunnel in args.remote_to_local.into_iter() {
|
for tunnel in args.remote_to_local.into_iter() {
|
||||||
let client_config = client_config.clone();
|
let client_config = client_config.clone();
|
||||||
match &tunnel.local_protocol {
|
match &tunnel.local_protocol {
|
||||||
LocalProtocol::Tcp => {
|
LocalProtocol::Tcp => {
|
||||||
tunnel.local_protocol = LocalProtocol::ReverseTcp;
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let remote = tunnel.remote.clone();
|
let remote = tunnel.remote.clone();
|
||||||
let cfg = client_config.clone();
|
let cfg = client_config.clone();
|
||||||
|
@ -675,43 +678,82 @@ async fn main() {
|
||||||
.await
|
.await
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let (host, port) = to_host_port(tunnel.local);
|
||||||
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::ReverseTcp,
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
if let Err(err) =
|
if let Err(err) =
|
||||||
tunnel::client::run_reverse_tunnel(client_config, tunnel, connect_to_dest).await
|
tunnel::client::run_reverse_tunnel(client_config, remote, connect_to_dest).await
|
||||||
{
|
{
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
LocalProtocol::Udp { timeout } => {
|
LocalProtocol::Udp { timeout } => {
|
||||||
tunnel.local_protocol = LocalProtocol::ReverseUdp { timeout: *timeout };
|
let timeout = *timeout;
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let cfg = client_config.clone();
|
let cfg = client_config.clone();
|
||||||
let remote = tunnel.remote.clone();
|
let (host, port) = to_host_port(tunnel.local);
|
||||||
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::ReverseUdp { timeout },
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
let connect_to_dest = |_| async {
|
let connect_to_dest = |_| async {
|
||||||
udp::connect(&remote.0, remote.1, cfg.timeout_connect, &cfg.dns_resolver).await
|
udp::connect(&tunnel.remote.0, tunnel.remote.1, cfg.timeout_connect, &cfg.dns_resolver)
|
||||||
|
.await
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Err(err) =
|
if let Err(err) =
|
||||||
tunnel::client::run_reverse_tunnel(client_config, tunnel, connect_to_dest).await
|
tunnel::client::run_reverse_tunnel(client_config, remote, connect_to_dest).await
|
||||||
{
|
{
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
LocalProtocol::Socks5 { .. } => {
|
LocalProtocol::Socks5 { .. } => {
|
||||||
tunnel.local_protocol = LocalProtocol::ReverseSocks5;
|
trait T: AsyncWrite + AsyncRead + Unpin + Send {}
|
||||||
|
impl T for TcpStream {}
|
||||||
|
impl T for MyUdpSocket {}
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let cfg = client_config.clone();
|
let cfg = client_config.clone();
|
||||||
let connect_to_dest = |remote: (Host, u16)| {
|
let (host, port) = to_host_port(tunnel.local);
|
||||||
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::ReverseSocks5,
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
let connect_to_dest = |remote: Option<RemoteAddr>| {
|
||||||
let so_mark = cfg.socket_so_mark;
|
let so_mark = cfg.socket_so_mark;
|
||||||
let timeout = cfg.timeout_connect;
|
let timeout = cfg.timeout_connect;
|
||||||
let dns_resolver = &cfg.dns_resolver;
|
let dns_resolver = &cfg.dns_resolver;
|
||||||
async move { tcp::connect(&remote.0, remote.1, so_mark, timeout, dns_resolver).await }
|
async move {
|
||||||
|
let Some(remote) = remote else {
|
||||||
|
return Err(anyhow!("Missing remote destination for reverse socks5"));
|
||||||
|
};
|
||||||
|
|
||||||
|
match remote.protocol {
|
||||||
|
LocalProtocol::Tcp => {
|
||||||
|
tcp::connect(&remote.host, remote.port, so_mark, timeout, dns_resolver)
|
||||||
|
.await
|
||||||
|
.map(|s| Box::new(s) as Box<dyn T>)
|
||||||
|
}
|
||||||
|
LocalProtocol::Udp { .. } => {
|
||||||
|
udp::connect(&remote.host, remote.port, timeout, dns_resolver)
|
||||||
|
.await
|
||||||
|
.map(|s| Box::new(s) as Box<dyn T>)
|
||||||
|
}
|
||||||
|
_ => Err(anyhow!("Invalid protocol for reverse socks5 {:?}", remote.protocol)),
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Err(err) =
|
if let Err(err) =
|
||||||
tunnel::client::run_reverse_tunnel(client_config, tunnel, connect_to_dest).await
|
tunnel::client::run_reverse_tunnel(client_config, remote, connect_to_dest).await
|
||||||
{
|
{
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
|
@ -732,12 +774,16 @@ async fn main() {
|
||||||
.unwrap_or_else(|err| panic!("Cannot start TCP server on {}: {}", tunnel.local, err))
|
.unwrap_or_else(|err| panic!("Cannot start TCP server on {}: {}", tunnel.local, err))
|
||||||
.map_err(anyhow::Error::new)
|
.map_err(anyhow::Error::new)
|
||||||
.map_ok(move |stream| {
|
.map_ok(move |stream| {
|
||||||
let remote = remote.clone();
|
let remote = RemoteAddr {
|
||||||
(stream.into_split(), (LocalProtocol::Tcp, remote.0, remote.1))
|
protocol: LocalProtocol::Tcp,
|
||||||
|
host: remote.0.clone(),
|
||||||
|
port: remote.1,
|
||||||
|
};
|
||||||
|
(stream.into_split(), remote)
|
||||||
});
|
});
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(client_config, tunnel, server).await {
|
if let Err(err) = tunnel::client::run_tunnel(client_config, server).await {
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -751,11 +797,16 @@ async fn main() {
|
||||||
.map_ok(move |stream| {
|
.map_ok(move |stream| {
|
||||||
// In TProxy mode local destination is the final ip:port destination
|
// In TProxy mode local destination is the final ip:port destination
|
||||||
let (host, port) = to_host_port(stream.local_addr().unwrap());
|
let (host, port) = to_host_port(stream.local_addr().unwrap());
|
||||||
(stream.into_split(), (LocalProtocol::Tcp, host, port))
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::Tcp,
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
(stream.into_split(), remote)
|
||||||
});
|
});
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(client_config, tunnel, server).await {
|
if let Err(err) = tunnel::client::run_tunnel(client_config, server).await {
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -773,11 +824,16 @@ async fn main() {
|
||||||
.map_ok(move |stream| {
|
.map_ok(move |stream| {
|
||||||
// In TProxy mode local destination is the final ip:port destination
|
// In TProxy mode local destination is the final ip:port destination
|
||||||
let (host, port) = to_host_port(stream.local_addr().unwrap());
|
let (host, port) = to_host_port(stream.local_addr().unwrap());
|
||||||
(tokio::io::split(stream), (LocalProtocol::Udp { timeout }, host, port))
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::Udp { timeout },
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
(tokio::io::split(stream), remote)
|
||||||
});
|
});
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(client_config, tunnel, server).await {
|
if let Err(err) = tunnel::client::run_tunnel(client_config, server).await {
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -794,11 +850,16 @@ async fn main() {
|
||||||
.unwrap_or_else(|err| panic!("Cannot start UDP server on {}: {}", tunnel.local, err))
|
.unwrap_or_else(|err| panic!("Cannot start UDP server on {}: {}", tunnel.local, err))
|
||||||
.map_err(anyhow::Error::new)
|
.map_err(anyhow::Error::new)
|
||||||
.map_ok(move |stream| {
|
.map_ok(move |stream| {
|
||||||
(tokio::io::split(stream), (LocalProtocol::Udp { timeout }, host.clone(), port))
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::Udp { timeout },
|
||||||
|
host: host.clone(),
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
(tokio::io::split(stream), remote)
|
||||||
});
|
});
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(client_config, tunnel, server).await {
|
if let Err(err) = tunnel::client::run_tunnel(client_config, server).await {
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -808,12 +869,16 @@ async fn main() {
|
||||||
.await
|
.await
|
||||||
.unwrap_or_else(|err| panic!("Cannot start Socks5 server on {}: {}", tunnel.local, err))
|
.unwrap_or_else(|err| panic!("Cannot start Socks5 server on {}: {}", tunnel.local, err))
|
||||||
.map_ok(|(stream, (host, port))| {
|
.map_ok(|(stream, (host, port))| {
|
||||||
let proto = stream.local_protocol();
|
let remote = RemoteAddr {
|
||||||
(tokio::io::split(stream), (proto, host, port))
|
protocol: stream.local_protocol(),
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
(tokio::io::split(stream), remote)
|
||||||
});
|
});
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(client_config, tunnel, server).await {
|
if let Err(err) = tunnel::client::run_tunnel(client_config, server).await {
|
||||||
error!("{:?}", err);
|
error!("{:?}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -826,9 +891,13 @@ async fn main() {
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = tunnel::client::run_tunnel(
|
if let Err(err) = tunnel::client::run_tunnel(
|
||||||
client_config,
|
client_config,
|
||||||
tunnel.clone(),
|
|
||||||
stream::once(async move {
|
stream::once(async move {
|
||||||
Ok((server, (LocalProtocol::Tcp, tunnel.remote.0, tunnel.remote.1)))
|
let remote = RemoteAddr {
|
||||||
|
protocol: LocalProtocol::Tcp,
|
||||||
|
host: tunnel.remote.0,
|
||||||
|
port: tunnel.remote.1,
|
||||||
|
};
|
||||||
|
Ok((server, remote))
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
|
|
|
@ -58,7 +58,7 @@ pub async fn run_server(bind: SocketAddr, timeout: Option<Duration>) -> Result<S
|
||||||
cfg.set_execute_command(false);
|
cfg.set_execute_command(false);
|
||||||
cfg.set_udp_support(true);
|
cfg.set_udp_support(true);
|
||||||
|
|
||||||
let udp_server = socks5_udp::run_server(bind, timeout, |_| Ok(()), |s| Ok(s.clone())).await?;
|
let udp_server = socks5_udp::run_server(bind, timeout).await?;
|
||||||
let server = server.with_config(cfg);
|
let server = server.with_config(cfg);
|
||||||
let stream = stream::unfold((server, Box::pin(udp_server)), move |(server, mut udp_server)| async move {
|
let stream = stream::unfold((server, Box::pin(udp_server)), move |(server, mut udp_server)| async move {
|
||||||
let mut acceptor = server.incoming();
|
let mut acceptor = server.incoming();
|
||||||
|
|
|
@ -195,8 +195,6 @@ impl AsyncWrite for Socks5UdpStream {
|
||||||
pub async fn run_server(
|
pub async fn run_server(
|
||||||
bind: SocketAddr,
|
bind: SocketAddr,
|
||||||
timeout: Option<Duration>,
|
timeout: Option<Duration>,
|
||||||
configure_listener: impl Fn(&UdpSocket) -> anyhow::Result<()>,
|
|
||||||
mk_send_socket: impl Fn(&Arc<UdpSocket>) -> anyhow::Result<Arc<UdpSocket>>,
|
|
||||||
) -> Result<impl Stream<Item = io::Result<Socks5UdpStream>>, anyhow::Error> {
|
) -> Result<impl Stream<Item = io::Result<Socks5UdpStream>>, anyhow::Error> {
|
||||||
info!(
|
info!(
|
||||||
"Starting SOCKS5 UDP server listening cnx on {} with cnx timeout of {}s",
|
"Starting SOCKS5 UDP server listening cnx on {} with cnx timeout of {}s",
|
||||||
|
@ -207,14 +205,11 @@ pub async fn run_server(
|
||||||
let listener = UdpSocket::bind(bind)
|
let listener = UdpSocket::bind(bind)
|
||||||
.await
|
.await
|
||||||
.with_context(|| format!("Cannot create UDP server {:?}", bind))?;
|
.with_context(|| format!("Cannot create UDP server {:?}", bind))?;
|
||||||
configure_listener(&listener)?;
|
|
||||||
|
|
||||||
let udp_server = Socks5UdpServer::new(listener, timeout);
|
let udp_server = Socks5UdpServer::new(listener, timeout);
|
||||||
static MAX_PACKET_LENGTH: usize = 64 * 1024;
|
static MAX_PACKET_LENGTH: usize = 64 * 1024;
|
||||||
let buffer = BytesMut::with_capacity(MAX_PACKET_LENGTH * 10);
|
let buffer = BytesMut::with_capacity(MAX_PACKET_LENGTH * 10);
|
||||||
let stream = stream::unfold(
|
let stream = stream::unfold((udp_server, buffer), |(mut server, mut buf)| async move {
|
||||||
(udp_server, mk_send_socket, buffer),
|
|
||||||
|(mut server, mk_send_socket, mut buf)| async move {
|
|
||||||
loop {
|
loop {
|
||||||
server.clean_dead_keys();
|
server.clean_dead_keys();
|
||||||
if buf.remaining_mut() < MAX_PACKET_LENGTH {
|
if buf.remaining_mut() < MAX_PACKET_LENGTH {
|
||||||
|
@ -248,7 +243,7 @@ pub async fn run_server(
|
||||||
None => {
|
None => {
|
||||||
info!("New UDP connection for {}", destination_addr);
|
info!("New UDP connection for {}", destination_addr);
|
||||||
let (udp_client, io) = Socks5UdpStream::new(
|
let (udp_client, io) = Socks5UdpStream::new(
|
||||||
mk_send_socket(&server.listener).ok()?,
|
server.listener.clone(),
|
||||||
peer_addr,
|
peer_addr,
|
||||||
destination_addr.clone(),
|
destination_addr.clone(),
|
||||||
server.cnx_timeout,
|
server.cnx_timeout,
|
||||||
|
@ -256,12 +251,11 @@ pub async fn run_server(
|
||||||
);
|
);
|
||||||
let _ = io.sender.send(data).await;
|
let _ = io.sender.send(data).await;
|
||||||
server.peers.insert(destination_addr, io);
|
server.peers.insert(destination_addr, io);
|
||||||
return Some((Ok(udp_client), (server, mk_send_socket, buf)));
|
return Some((Ok(udp_client), (server, buf)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
});
|
||||||
);
|
|
||||||
|
|
||||||
Ok(stream)
|
Ok(stream)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
use super::{to_host_port, JwtTunnelConfig, JWT_HEADER_PREFIX, JWT_KEY};
|
use super::{tunnel_to_jwt_token, JwtTunnelConfig, RemoteAddr, JWT_DECODE, JWT_HEADER_PREFIX};
|
||||||
use crate::{LocalProtocol, LocalToRemote, WsClientConfig};
|
use crate::WsClientConfig;
|
||||||
use anyhow::{anyhow, Context};
|
use anyhow::{anyhow, Context};
|
||||||
|
|
||||||
use base64::Engine;
|
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use fastwebsockets::WebSocket;
|
use fastwebsockets::WebSocket;
|
||||||
use futures_util::pin_mut;
|
use futures_util::pin_mut;
|
||||||
|
@ -13,6 +12,7 @@ use hyper::header::{CONNECTION, HOST, SEC_WEBSOCKET_KEY};
|
||||||
use hyper::upgrade::Upgraded;
|
use hyper::upgrade::Upgraded;
|
||||||
use hyper::{Request, Response};
|
use hyper::{Request, Response};
|
||||||
use hyper_util::rt::{TokioExecutor, TokioIo};
|
use hyper_util::rt::{TokioExecutor, TokioIo};
|
||||||
|
use jsonwebtoken::TokenData;
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
use std::ops::{Deref, DerefMut};
|
use std::ops::{Deref, DerefMut};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
@ -21,24 +21,18 @@ use tokio::sync::oneshot;
|
||||||
use tokio_stream::{Stream, StreamExt};
|
use tokio_stream::{Stream, StreamExt};
|
||||||
use tracing::log::debug;
|
use tracing::log::debug;
|
||||||
use tracing::{error, span, Instrument, Level, Span};
|
use tracing::{error, span, Instrument, Level, Span};
|
||||||
use url::{Host, Url};
|
use url::Host;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
fn tunnel_to_jwt_token(request_id: Uuid, tunnel: &LocalToRemote) -> String {
|
|
||||||
let cfg = JwtTunnelConfig::new(request_id, tunnel);
|
|
||||||
let (alg, secret) = JWT_KEY.deref();
|
|
||||||
jsonwebtoken::encode(alg, &cfg, secret).unwrap_or_default()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn connect(
|
pub async fn connect(
|
||||||
request_id: Uuid,
|
request_id: Uuid,
|
||||||
client_cfg: &WsClientConfig,
|
client_cfg: &WsClientConfig,
|
||||||
tunnel_cfg: &LocalToRemote,
|
dest_addr: &RemoteAddr,
|
||||||
) -> anyhow::Result<(WebSocket<TokioIo<Upgraded>>, Response<Incoming>)> {
|
) -> anyhow::Result<(WebSocket<TokioIo<Upgraded>>, Response<Incoming>)> {
|
||||||
let mut pooled_cnx = match client_cfg.cnx_pool().get().await {
|
let mut pooled_cnx = match client_cfg.cnx_pool().get().await {
|
||||||
Ok(tcp_stream) => tcp_stream,
|
Ok(cnx) => Ok(cnx),
|
||||||
Err(err) => Err(anyhow!("failed to get a connection to the server from the pool: {err:?}"))?,
|
Err(err) => Err(anyhow!("failed to get a connection to the server from the pool: {err:?}")),
|
||||||
};
|
}?;
|
||||||
|
|
||||||
let mut req = Request::builder()
|
let mut req = Request::builder()
|
||||||
.method("GET")
|
.method("GET")
|
||||||
|
@ -50,7 +44,7 @@ pub async fn connect(
|
||||||
.header(SEC_WEBSOCKET_VERSION, "13")
|
.header(SEC_WEBSOCKET_VERSION, "13")
|
||||||
.header(
|
.header(
|
||||||
SEC_WEBSOCKET_PROTOCOL,
|
SEC_WEBSOCKET_PROTOCOL,
|
||||||
format!("v1, {}{}", JWT_HEADER_PREFIX, tunnel_to_jwt_token(request_id, tunnel_cfg)),
|
format!("v1, {}{}", JWT_HEADER_PREFIX, tunnel_to_jwt_token(request_id, dest_addr)),
|
||||||
)
|
)
|
||||||
.version(hyper::Version::HTTP_11);
|
.version(hyper::Version::HTTP_11);
|
||||||
|
|
||||||
|
@ -79,7 +73,7 @@ pub async fn connect(
|
||||||
async fn connect_to_server<R, W>(
|
async fn connect_to_server<R, W>(
|
||||||
request_id: Uuid,
|
request_id: Uuid,
|
||||||
client_cfg: &WsClientConfig,
|
client_cfg: &WsClientConfig,
|
||||||
remote_cfg: &LocalToRemote,
|
remote_cfg: &RemoteAddr,
|
||||||
duplex_stream: (R, W),
|
duplex_stream: (R, W),
|
||||||
) -> anyhow::Result<()>
|
) -> anyhow::Result<()>
|
||||||
where
|
where
|
||||||
|
@ -105,32 +99,25 @@ where
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn run_tunnel<T, R, W>(
|
pub async fn run_tunnel<T, R, W>(client_config: Arc<WsClientConfig>, incoming_cnx: T) -> anyhow::Result<()>
|
||||||
client_config: Arc<WsClientConfig>,
|
|
||||||
tunnel_cfg: LocalToRemote,
|
|
||||||
incoming_cnx: T,
|
|
||||||
) -> anyhow::Result<()>
|
|
||||||
where
|
where
|
||||||
T: Stream<Item = anyhow::Result<((R, W), (LocalProtocol, Host, u16))>>,
|
T: Stream<Item = anyhow::Result<((R, W), RemoteAddr)>>,
|
||||||
R: AsyncRead + Send + 'static,
|
R: AsyncRead + Send + 'static,
|
||||||
W: AsyncWrite + Send + 'static,
|
W: AsyncWrite + Send + 'static,
|
||||||
{
|
{
|
||||||
pin_mut!(incoming_cnx);
|
pin_mut!(incoming_cnx);
|
||||||
while let Some(Ok((cnx_stream, remote_dest))) = incoming_cnx.next().await {
|
while let Some(Ok((cnx_stream, remote_addr))) = incoming_cnx.next().await {
|
||||||
let request_id = Uuid::now_v7();
|
let request_id = Uuid::now_v7();
|
||||||
let span = span!(
|
let span = span!(
|
||||||
Level::INFO,
|
Level::INFO,
|
||||||
"tunnel",
|
"tunnel",
|
||||||
id = request_id.to_string(),
|
id = request_id.to_string(),
|
||||||
remote = format!("{}:{}", remote_dest.1, remote_dest.2)
|
remote = format!("{}:{}", remote_addr.host, remote_addr.port)
|
||||||
);
|
);
|
||||||
let mut tunnel_cfg = tunnel_cfg.clone();
|
|
||||||
tunnel_cfg.local_protocol = remote_dest.0;
|
|
||||||
tunnel_cfg.remote = (remote_dest.1, remote_dest.2);
|
|
||||||
let client_config = client_config.clone();
|
let client_config = client_config.clone();
|
||||||
|
|
||||||
let tunnel = async move {
|
let tunnel = async move {
|
||||||
let _ = connect_to_server(request_id, &client_config, &tunnel_cfg, cnx_stream)
|
let _ = connect_to_server(request_id, &client_config, &remote_addr, cnx_stream)
|
||||||
.await
|
.await
|
||||||
.map_err(|err| error!("{:?}", err));
|
.map_err(|err| error!("{:?}", err));
|
||||||
}
|
}
|
||||||
|
@ -144,18 +131,14 @@ where
|
||||||
|
|
||||||
pub async fn run_reverse_tunnel<F, Fut, T>(
|
pub async fn run_reverse_tunnel<F, Fut, T>(
|
||||||
client_config: Arc<WsClientConfig>,
|
client_config: Arc<WsClientConfig>,
|
||||||
mut tunnel_cfg: LocalToRemote,
|
remote_addr: RemoteAddr,
|
||||||
connect_to_dest: F,
|
connect_to_dest: F,
|
||||||
) -> anyhow::Result<()>
|
) -> anyhow::Result<()>
|
||||||
where
|
where
|
||||||
F: Fn((Host, u16)) -> Fut,
|
F: Fn(Option<RemoteAddr>) -> Fut,
|
||||||
Fut: Future<Output = anyhow::Result<T>>,
|
Fut: Future<Output = anyhow::Result<T>>,
|
||||||
T: AsyncRead + AsyncWrite + Send + 'static,
|
T: AsyncRead + AsyncWrite + Send + 'static,
|
||||||
{
|
{
|
||||||
// Invert local with remote
|
|
||||||
let remote_ori = tunnel_cfg.remote;
|
|
||||||
tunnel_cfg.remote = to_host_port(tunnel_cfg.local);
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let client_config = client_config.clone();
|
let client_config = client_config.clone();
|
||||||
let request_id = Uuid::now_v7();
|
let request_id = Uuid::now_v7();
|
||||||
|
@ -163,12 +146,12 @@ where
|
||||||
Level::INFO,
|
Level::INFO,
|
||||||
"tunnel",
|
"tunnel",
|
||||||
id = request_id.to_string(),
|
id = request_id.to_string(),
|
||||||
remote = format!("{}:{}", tunnel_cfg.remote.0, tunnel_cfg.remote.1)
|
remote = format!("{}:{}", remote_addr.host, remote_addr.port)
|
||||||
);
|
);
|
||||||
let _span = span.enter();
|
let _span = span.enter();
|
||||||
|
|
||||||
// Correctly configure tunnel cfg
|
// Correctly configure tunnel cfg
|
||||||
let (mut ws, response) = connect(request_id, &client_config, &tunnel_cfg)
|
let (mut ws, response) = connect(request_id, &client_config, &remote_addr)
|
||||||
.instrument(span.clone())
|
.instrument(span.clone())
|
||||||
.await?;
|
.await?;
|
||||||
ws.set_auto_apply_mask(client_config.websocket_mask_frame);
|
ws.set_auto_apply_mask(client_config.websocket_mask_frame);
|
||||||
|
@ -178,18 +161,21 @@ where
|
||||||
.headers()
|
.headers()
|
||||||
.get(COOKIE)
|
.get(COOKIE)
|
||||||
.and_then(|h| h.to_str().ok())
|
.and_then(|h| h.to_str().ok())
|
||||||
.and_then(|h| base64::engine::general_purpose::STANDARD.decode(h).ok())
|
.and_then(|h| {
|
||||||
.and_then(|h| Url::parse(&String::from_utf8_lossy(&h)).ok())
|
let (validation, decode_key) = JWT_DECODE.deref();
|
||||||
.and_then(|url| match (url.host(), url.port_or_known_default()) {
|
let jwt: Option<TokenData<JwtTunnelConfig>> = jsonwebtoken::decode(h, decode_key, validation).ok();
|
||||||
(Some(h), Some(p)) => Some((h.to_owned(), p)),
|
jwt
|
||||||
_ => None,
|
|
||||||
})
|
})
|
||||||
.unwrap_or(remote_ori.clone());
|
.map(|jwt| RemoteAddr {
|
||||||
|
protocol: jwt.claims.p,
|
||||||
|
host: Host::parse(&jwt.claims.r).unwrap_or_else(|_| Host::Domain(String::new())),
|
||||||
|
port: jwt.claims.rp,
|
||||||
|
});
|
||||||
|
|
||||||
let stream = match connect_to_dest(remote.clone()).instrument(span.clone()).await {
|
let stream = match connect_to_dest(remote).instrument(span.clone()).await {
|
||||||
Ok(s) => s,
|
Ok(s) => s,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!("Cannot connect to {remote:?}: {err:?}");
|
error!("Cannot connect to xxxx: {err:?}");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -3,7 +3,7 @@ mod io;
|
||||||
pub mod server;
|
pub mod server;
|
||||||
mod tls_reloader;
|
mod tls_reloader;
|
||||||
|
|
||||||
use crate::{tcp, tls, LocalProtocol, LocalToRemote, WsClientConfig};
|
use crate::{tcp, tls, LocalProtocol, WsClientConfig};
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use bb8::ManageConnection;
|
use bb8::ManageConnection;
|
||||||
use jsonwebtoken::{Algorithm, DecodingKey, EncodingKey, Header, Validation};
|
use jsonwebtoken::{Algorithm, DecodingKey, EncodingKey, Header, Validation};
|
||||||
|
@ -12,6 +12,7 @@ use serde::{Deserialize, Serialize};
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
use std::io::{Error, IoSlice};
|
use std::io::{Error, IoSlice};
|
||||||
use std::net::{IpAddr, SocketAddr};
|
use std::net::{IpAddr, SocketAddr};
|
||||||
|
use std::ops::Deref;
|
||||||
use std::pin::Pin;
|
use std::pin::Pin;
|
||||||
use std::task::{Context, Poll};
|
use std::task::{Context, Poll};
|
||||||
use tokio::io::{AsyncRead, AsyncWrite, ReadBuf};
|
use tokio::io::{AsyncRead, AsyncWrite, ReadBuf};
|
||||||
|
@ -29,26 +30,32 @@ struct JwtTunnelConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl JwtTunnelConfig {
|
impl JwtTunnelConfig {
|
||||||
fn new(request_id: Uuid, tunnel: &LocalToRemote) -> Self {
|
fn new(request_id: Uuid, dest: &RemoteAddr) -> Self {
|
||||||
Self {
|
Self {
|
||||||
id: request_id.to_string(),
|
id: request_id.to_string(),
|
||||||
p: match tunnel.local_protocol {
|
p: match dest.protocol {
|
||||||
LocalProtocol::Tcp => LocalProtocol::Tcp,
|
LocalProtocol::Tcp => LocalProtocol::Tcp,
|
||||||
LocalProtocol::Udp { .. } => tunnel.local_protocol,
|
LocalProtocol::Udp { .. } => dest.protocol,
|
||||||
LocalProtocol::Stdio => LocalProtocol::Tcp,
|
LocalProtocol::Stdio => LocalProtocol::Tcp,
|
||||||
LocalProtocol::Socks5 { .. } => LocalProtocol::Tcp,
|
LocalProtocol::Socks5 { .. } => LocalProtocol::Tcp,
|
||||||
LocalProtocol::ReverseTcp => LocalProtocol::ReverseTcp,
|
LocalProtocol::ReverseTcp => LocalProtocol::ReverseTcp,
|
||||||
LocalProtocol::ReverseUdp { .. } => tunnel.local_protocol,
|
LocalProtocol::ReverseUdp { .. } => dest.protocol,
|
||||||
LocalProtocol::ReverseSocks5 => LocalProtocol::ReverseSocks5,
|
LocalProtocol::ReverseSocks5 => LocalProtocol::ReverseSocks5,
|
||||||
LocalProtocol::TProxyTcp => LocalProtocol::Tcp,
|
LocalProtocol::TProxyTcp => LocalProtocol::Tcp,
|
||||||
LocalProtocol::TProxyUdp { timeout } => LocalProtocol::Udp { timeout },
|
LocalProtocol::TProxyUdp { timeout } => LocalProtocol::Udp { timeout },
|
||||||
},
|
},
|
||||||
r: tunnel.remote.0.to_string(),
|
r: dest.host.to_string(),
|
||||||
rp: tunnel.remote.1,
|
rp: dest.port,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn tunnel_to_jwt_token(request_id: Uuid, tunnel: &RemoteAddr) -> String {
|
||||||
|
let cfg = JwtTunnelConfig::new(request_id, tunnel);
|
||||||
|
let (alg, secret) = JWT_KEY.deref();
|
||||||
|
jsonwebtoken::encode(alg, &cfg, secret).unwrap_or_default()
|
||||||
|
}
|
||||||
|
|
||||||
static JWT_HEADER_PREFIX: &str = "authorization.bearer.";
|
static JWT_HEADER_PREFIX: &str = "authorization.bearer.";
|
||||||
static JWT_SECRET: &[u8; 15] = b"champignonfrais";
|
static JWT_SECRET: &[u8; 15] = b"champignonfrais";
|
||||||
static JWT_KEY: Lazy<(Header, EncodingKey)> =
|
static JWT_KEY: Lazy<(Header, EncodingKey)> =
|
||||||
|
@ -60,6 +67,13 @@ static JWT_DECODE: Lazy<(Validation, DecodingKey)> = Lazy::new(|| {
|
||||||
(validation, DecodingKey::from_secret(JWT_SECRET))
|
(validation, DecodingKey::from_secret(JWT_SECRET))
|
||||||
});
|
});
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct RemoteAddr {
|
||||||
|
pub protocol: LocalProtocol,
|
||||||
|
pub host: Host,
|
||||||
|
pub port: u16,
|
||||||
|
}
|
||||||
|
|
||||||
pub enum TransportStream {
|
pub enum TransportStream {
|
||||||
Plain(TcpStream),
|
Plain(TcpStream),
|
||||||
Tls(TlsStream<TcpStream>),
|
Tls(TlsStream<TcpStream>),
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
use ahash::{HashMap, HashMapExt};
|
use ahash::{HashMap, HashMapExt};
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
use base64::Engine;
|
|
||||||
use futures_util::{pin_mut, FutureExt, Stream, StreamExt};
|
use futures_util::{pin_mut, FutureExt, Stream, StreamExt};
|
||||||
use std::cmp::min;
|
use std::cmp::min;
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
|
@ -10,7 +9,7 @@ use std::pin::Pin;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use super::{JwtTunnelConfig, JWT_DECODE, JWT_HEADER_PREFIX};
|
use super::{tunnel_to_jwt_token, JwtTunnelConfig, RemoteAddr, JWT_DECODE, JWT_HEADER_PREFIX};
|
||||||
use crate::{socks5, tcp, tls, udp, LocalProtocol, TlsServerConfig, WsServerConfig};
|
use crate::{socks5, tcp, tls, udp, LocalProtocol, TlsServerConfig, WsServerConfig};
|
||||||
use hyper::body::Incoming;
|
use hyper::body::Incoming;
|
||||||
use hyper::header::{COOKIE, SEC_WEBSOCKET_PROTOCOL};
|
use hyper::header::{COOKIE, SEC_WEBSOCKET_PROTOCOL};
|
||||||
|
@ -32,17 +31,12 @@ use tokio::sync::{mpsc, oneshot};
|
||||||
use tokio_rustls::TlsAcceptor;
|
use tokio_rustls::TlsAcceptor;
|
||||||
use tracing::{error, info, span, warn, Instrument, Level, Span};
|
use tracing::{error, info, span, warn, Instrument, Level, Span};
|
||||||
use url::Host;
|
use url::Host;
|
||||||
|
use uuid::Uuid;
|
||||||
|
|
||||||
async fn run_tunnel(
|
async fn run_tunnel(
|
||||||
server_config: &WsServerConfig,
|
server_config: &WsServerConfig,
|
||||||
jwt: TokenData<JwtTunnelConfig>,
|
jwt: TokenData<JwtTunnelConfig>,
|
||||||
) -> anyhow::Result<(
|
) -> anyhow::Result<(RemoteAddr, Pin<Box<dyn AsyncRead + Send>>, Pin<Box<dyn AsyncWrite + Send>>)> {
|
||||||
LocalProtocol,
|
|
||||||
Host,
|
|
||||||
u16,
|
|
||||||
Pin<Box<dyn AsyncRead + Send>>,
|
|
||||||
Pin<Box<dyn AsyncWrite + Send>>,
|
|
||||||
)> {
|
|
||||||
match jwt.claims.p {
|
match jwt.claims.p {
|
||||||
LocalProtocol::Udp { timeout, .. } => {
|
LocalProtocol::Udp { timeout, .. } => {
|
||||||
let host = Host::parse(&jwt.claims.r)?;
|
let host = Host::parse(&jwt.claims.r)?;
|
||||||
|
@ -53,13 +47,13 @@ async fn run_tunnel(
|
||||||
&server_config.dns_resolver,
|
&server_config.dns_resolver,
|
||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
Ok((
|
|
||||||
LocalProtocol::Udp { timeout: None },
|
let remote = RemoteAddr {
|
||||||
|
protocol: jwt.claims.p,
|
||||||
host,
|
host,
|
||||||
jwt.claims.rp,
|
port: jwt.claims.rp,
|
||||||
Box::pin(cnx.clone()),
|
};
|
||||||
Box::pin(cnx),
|
Ok((remote, Box::pin(cnx.clone()), Box::pin(cnx)))
|
||||||
))
|
|
||||||
}
|
}
|
||||||
LocalProtocol::Tcp => {
|
LocalProtocol::Tcp => {
|
||||||
let host = Host::parse(&jwt.claims.r)?;
|
let host = Host::parse(&jwt.claims.r)?;
|
||||||
|
@ -74,7 +68,12 @@ async fn run_tunnel(
|
||||||
.await?
|
.await?
|
||||||
.into_split();
|
.into_split();
|
||||||
|
|
||||||
Ok((jwt.claims.p, host, port, Box::pin(rx), Box::pin(tx)))
|
let remote = RemoteAddr {
|
||||||
|
protocol: jwt.claims.p,
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
};
|
||||||
|
Ok((remote, Box::pin(rx), Box::pin(tx)))
|
||||||
}
|
}
|
||||||
LocalProtocol::ReverseTcp => {
|
LocalProtocol::ReverseTcp => {
|
||||||
#[allow(clippy::type_complexity)]
|
#[allow(clippy::type_complexity)]
|
||||||
|
@ -87,7 +86,12 @@ async fn run_tunnel(
|
||||||
let tcp = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
let tcp = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
||||||
let (local_rx, local_tx) = tcp.into_split();
|
let (local_rx, local_tx) = tcp.into_split();
|
||||||
|
|
||||||
Ok((jwt.claims.p, local_srv.0, local_srv.1, Box::pin(local_rx), Box::pin(local_tx)))
|
let remote = RemoteAddr {
|
||||||
|
protocol: jwt.claims.p,
|
||||||
|
host: local_srv.0,
|
||||||
|
port: local_srv.1,
|
||||||
|
};
|
||||||
|
Ok((remote, Box::pin(local_rx), Box::pin(local_tx)))
|
||||||
}
|
}
|
||||||
LocalProtocol::ReverseUdp { timeout } => {
|
LocalProtocol::ReverseUdp { timeout } => {
|
||||||
#[allow(clippy::type_complexity)]
|
#[allow(clippy::type_complexity)]
|
||||||
|
@ -101,7 +105,12 @@ async fn run_tunnel(
|
||||||
let udp = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
let udp = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
||||||
let (local_rx, local_tx) = tokio::io::split(udp);
|
let (local_rx, local_tx) = tokio::io::split(udp);
|
||||||
|
|
||||||
Ok((jwt.claims.p, local_srv.0, local_srv.1, Box::pin(local_rx), Box::pin(local_tx)))
|
let remote = RemoteAddr {
|
||||||
|
protocol: jwt.claims.p,
|
||||||
|
host: local_srv.0,
|
||||||
|
port: local_srv.1,
|
||||||
|
};
|
||||||
|
Ok((remote, Box::pin(local_rx), Box::pin(local_tx)))
|
||||||
}
|
}
|
||||||
LocalProtocol::ReverseSocks5 => {
|
LocalProtocol::ReverseSocks5 => {
|
||||||
#[allow(clippy::type_complexity)]
|
#[allow(clippy::type_complexity)]
|
||||||
|
@ -112,10 +121,15 @@ async fn run_tunnel(
|
||||||
let bind = format!("{}:{}", local_srv.0, local_srv.1);
|
let bind = format!("{}:{}", local_srv.0, local_srv.1);
|
||||||
let listening_server = socks5::run_server(bind.parse()?, None);
|
let listening_server = socks5::run_server(bind.parse()?, None);
|
||||||
let (stream, local_srv) = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
let (stream, local_srv) = run_listening_server(&local_srv, SERVERS.deref(), listening_server).await?;
|
||||||
let proto = stream.local_protocol();
|
let protocol = stream.local_protocol();
|
||||||
let (local_rx, local_tx) = tokio::io::split(stream);
|
let (local_rx, local_tx) = tokio::io::split(stream);
|
||||||
|
|
||||||
Ok((proto, local_srv.0, local_srv.1, Box::pin(local_rx), Box::pin(local_tx)))
|
let remote = RemoteAddr {
|
||||||
|
protocol,
|
||||||
|
host: local_srv.0,
|
||||||
|
port: local_srv.1,
|
||||||
|
};
|
||||||
|
Ok((remote, Box::pin(local_rx), Box::pin(local_tx)))
|
||||||
}
|
}
|
||||||
_ => Err(anyhow::anyhow!("Invalid upgrade request")),
|
_ => Err(anyhow::anyhow!("Invalid upgrade request")),
|
||||||
}
|
}
|
||||||
|
@ -308,6 +322,7 @@ async fn server_upgrade(server_config: Arc<WsServerConfig>, mut req: Request<Inc
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let req_protocol = jwt.claims.p;
|
||||||
let tunnel = match run_tunnel(&server_config, jwt).await {
|
let tunnel = match run_tunnel(&server_config, jwt).await {
|
||||||
Ok(ret) => ret,
|
Ok(ret) => ret,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
|
@ -319,8 +334,11 @@ async fn server_upgrade(server_config: Arc<WsServerConfig>, mut req: Request<Inc
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let (protocol, dest, port, local_rx, local_tx) = tunnel;
|
let (remote_addr, local_rx, local_tx) = tunnel;
|
||||||
info!("connected to {:?} {:?} {:?}", protocol, dest, port);
|
info!(
|
||||||
|
"connected to {:?} {:?} {:?}",
|
||||||
|
remote_addr.protocol, remote_addr.host, remote_addr.port
|
||||||
|
);
|
||||||
let (mut response, fut) = match fastwebsockets::upgrade::upgrade(&mut req) {
|
let (mut response, fut) = match fastwebsockets::upgrade::upgrade(&mut req) {
|
||||||
Ok(ret) => ret,
|
Ok(ret) => ret,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
|
@ -351,11 +369,9 @@ async fn server_upgrade(server_config: Arc<WsServerConfig>, mut req: Request<Inc
|
||||||
.instrument(Span::current()),
|
.instrument(Span::current()),
|
||||||
);
|
);
|
||||||
|
|
||||||
if protocol == LocalProtocol::ReverseSocks5 {
|
if req_protocol == LocalProtocol::ReverseSocks5 {
|
||||||
let Ok(header_val) = HeaderValue::from_str(
|
let Ok(header_val) = HeaderValue::from_str(&tunnel_to_jwt_token(Uuid::from_u128(0), &remote_addr)) else {
|
||||||
&base64::engine::general_purpose::STANDARD.encode(format!("https://{}:{}", dest, port)),
|
error!("Bad headervalue for reverse socks5: {} {}", remote_addr.host, remote_addr.port);
|
||||||
) else {
|
|
||||||
error!("Bad headervalue for reverse socks5: {} {}", dest, port);
|
|
||||||
return http::Response::builder()
|
return http::Response::builder()
|
||||||
.status(StatusCode::BAD_REQUEST)
|
.status(StatusCode::BAD_REQUEST)
|
||||||
.body("Invalid upgrade request".to_string())
|
.body("Invalid upgrade request".to_string())
|
||||||
|
|
Loading…
Reference in a new issue