mod embedded_certificate; mod stdio; mod tcp; mod tls; mod transport; mod udp; use base64::Engine; use clap::Parser; use futures_util::{pin_mut, stream, Stream, StreamExt, TryStreamExt}; use hyper::http::HeaderValue; use serde::{Deserialize, Serialize}; use std::borrow::Cow; use std::collections::{BTreeMap, HashMap}; use std::io; use std::io::ErrorKind; use std::net::{IpAddr, Ipv4Addr, Ipv6Addr, SocketAddr}; use std::path::PathBuf; use std::str::FromStr; use std::sync::Arc; use std::time::Duration; use tokio::io::{AsyncRead, AsyncWrite}; use tokio::net::TcpStream; use tokio_rustls::rustls::server::DnsName; use tokio_rustls::rustls::{Certificate, PrivateKey, ServerName}; use tracing::{debug, error, field, instrument, Instrument, Span}; use tracing_subscriber::EnvFilter; use url::{Host, Url}; use uuid::Uuid; /// Use the websockets protocol to tunnel {TCP,UDP} traffic /// wsTunnelClient <---> wsTunnelServer <---> RemoteHost /// Use secure connection (wss://) to bypass proxies #[derive(clap::Parser, Debug)] #[command(author, version, about, verbatim_doc_comment, long_about = None)] struct Wstunnel { #[command(subcommand)] commands: Commands, } #[derive(clap::Subcommand, Debug)] enum Commands { Client(Client), Server(Server), } #[derive(clap::Args, Debug)] struct Client { /// Listen on local and forwards traffic from remote /// Can be specified multiple times #[arg(short='L', long, value_name = "{tcp,udp}://[BIND:]PORT:HOST:PORT", value_parser = parse_env_var)] local_to_remote: Vec, /// (linux only) Mark network packet with SO_MARK sockoption with the specified value. /// You need to use {root, sudo, capabilities} to run wstunnel when using this option #[arg(long, value_name = "INT", verbatim_doc_comment)] socket_so_mark: Option, /// Domain name that will be use as SNI during TLS handshake /// Warning: If you are behind a CDN (i.e: Cloudflare) you must set this domain also in the http HOST header. /// or it will be flag as fishy as your request rejected #[arg(long, value_name = "DOMAIN_NAME", value_parser = parse_sni_override, verbatim_doc_comment)] tls_sni_override: Option, /// Enable TLS certificate verification. /// Disabled by default. The client will happily connect to any server with self signed certificate. #[arg(long, verbatim_doc_comment)] tls_verify_certificate: bool, /// Use a specific prefix that will show up in the http path during the upgrade request. /// Useful if you need to route requests server side but don't have vhosts #[arg(long, default_value = "morille", verbatim_doc_comment)] http_upgrade_path_prefix: String, /// Pass authorization header with basic auth credentials during the upgrade request. /// If you need more customization, you can use the http_headers option. #[arg(long, value_name = "USER[:PASS]", value_parser = parse_http_credentials, verbatim_doc_comment)] http_upgrade_credentials: Option, /// Frequency at which the client will send websocket ping to the server. #[arg(long, value_name = "seconds", default_value = "30", value_parser = parse_duration_sec, verbatim_doc_comment)] websocket_ping_frequency_sec: Option, /// Enable the masking of websocket frames. Default is false /// Enable this option only if you use unsecure (non TLS) websocket server and you see some issues. Otherwise, it is just overhead. #[arg(long, default_value = "false", verbatim_doc_comment)] websocket_mask_frame: bool, /// Send custom headers in the upgrade request /// Can be specified multiple time #[arg(short='H', long, value_name = "HEADER_NAME: HEADER_VALUE", value_parser = parse_http_headers, verbatim_doc_comment)] http_headers: Vec<(String, HeaderValue)>, /// Address of the wstunnel server /// Example: With TLS wss://wstunnel.example.com or without ws://wstunnel.example.com #[arg(value_name = "ws[s]://wstunnel.server.com[:port]", value_parser = parse_server_url, verbatim_doc_comment)] remote_addr: Url, } #[derive(clap::Args, Debug)] struct Server { /// Address of the wstunnel server to bind to /// Example: With TLS wss://0.0.0.0:8080 or without ws://[::]:8080 #[arg(value_name = "ws[s]://0.0.0.0[:port]", value_parser = parse_server_url, verbatim_doc_comment)] remote_addr: Url, /// (linux only) Mark network packet with SO_MARK sockoption with the specified value. /// You need to use {root, sudo, capabilities} to run wstunnel when using this option #[arg(long, value_name = "INT", verbatim_doc_comment)] socket_so_mark: Option, /// Frequency at which the server will send websocket ping to client. #[arg(long, value_name = "seconds", value_parser = parse_duration_sec, verbatim_doc_comment)] websocket_ping_frequency_sec: Option, /// Enable the masking of websocket frames. Default is false /// Enable this option only if you use unsecure (non TLS) websocket server and you see some issues. Otherwise, it is just overhead. #[arg(long, default_value = "false", verbatim_doc_comment)] websocket_mask_frame: bool, /// Server will only accept connection from the specified tunnel information. /// Can be specified multiple time /// Example: --restrict-to "google.com:443" --restrict-to "localhost:22" #[arg(long, value_name = "DEST:PORT", verbatim_doc_comment)] restrict_to: Option>, /// [Optional] Use custom certificate (.crt) instead of the default embedded self signed certificate. #[arg(long, value_name = "FILE_PATH", verbatim_doc_comment)] tls_certificate: Option, /// [Optional] Use a custom tls key (.key) that the server will use instead of the default embedded one #[arg(long, value_name = "FILE_PATH", verbatim_doc_comment)] tls_private_key: Option, } #[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize, Deserialize)] enum L4Protocol { Tcp, Udp { timeout: Option }, Stdio, } impl L4Protocol { fn new_udp() -> L4Protocol { L4Protocol::Udp { timeout: Some(Duration::from_secs(30)), } } } #[derive(Clone, Debug)] pub struct LocalToRemote { socket_so_mark: Option, protocol: L4Protocol, local: SocketAddr, remote: (Host, u16), } fn parse_duration_sec(arg: &str) -> Result { use std::io::Error; let Ok(secs) = arg.parse::() else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot duration of seconds from {}", arg), )); }; Ok(Duration::from_secs(secs)) } fn parse_env_var(arg: &str) -> Result { use std::io::Error; let (mut protocol, arg) = match &arg[..6] { "tcp://" => (L4Protocol::Tcp, &arg[6..]), "udp://" => (L4Protocol::new_udp(), &arg[6..]), _ => match &arg[..8] { "stdio://" => (L4Protocol::Stdio, &arg[8..]), _ => (L4Protocol::Tcp, arg), }, }; let (bind, remaining) = if arg.starts_with('[') { // ipv6 bind let Some((ipv6_str, remaining)) = arg.split_once(']') else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse IPv6 bind from {}", arg), )); }; let Ok(ipv6_addr) = Ipv6Addr::from_str(&ipv6_str[1..]) else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse IPv6 bind from {}", ipv6_str), )); }; (IpAddr::V6(ipv6_addr), remaining) } else { // Maybe ipv4 addr let Some((ipv4_str, remaining)) = arg.split_once(':') else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse IPv4 bind from {}", arg), )); }; match Ipv4Addr::from_str(ipv4_str) { Ok(ip4_addr) => (IpAddr::V4(ip4_addr), remaining), // Must be the port, so we default to ipv6 bind Err(_) => (IpAddr::V4(Ipv4Addr::from_str("127.0.0.1").unwrap()), arg), } }; let Some((port_str, remaining)) = remaining.trim_start_matches(':').split_once(':') else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse bind port from {}", remaining), )); }; let Ok(bind_port): Result = port_str.parse() else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse bind port from {}", port_str), )); }; let Ok(remote) = Url::parse(&format!("fake://{}", remaining)) else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse remote from {}", remaining), )); }; let Some(remote_host) = remote.host() else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse remote host from {}", remaining), )); }; let Some(remote_port) = remote.port() else { return Err(Error::new( ErrorKind::InvalidInput, format!("cannot parse remote port from {}", remaining), )); }; let options: BTreeMap, Cow<'_, str>> = remote.query_pairs().collect(); match &mut protocol { L4Protocol::Stdio => {} L4Protocol::Tcp => {} L4Protocol::Udp { ref mut timeout, .. } => { if let Some(duration) = options .get("timeout_sec") .and_then(|x| x.parse::().ok()) .map(|d| { if d == 0 { None } else { Some(Duration::from_secs(d)) } }) { *timeout = duration; } } }; Ok(LocalToRemote { socket_so_mark: options .get("socket_so_mark") .and_then(|x| x.parse::().ok()), protocol, local: SocketAddr::new(bind, bind_port), remote: (remote_host.to_owned(), remote_port), }) } fn parse_sni_override(arg: &str) -> Result { match DnsName::try_from(arg.to_string()) { Ok(val) => Ok(val), Err(err) => Err(io::Error::new( ErrorKind::InvalidInput, format!("Invalid sni override: {}", err), )), } } fn parse_http_headers(arg: &str) -> Result<(String, HeaderValue), io::Error> { let Some((key, value)) = arg.split_once(':') else { return Err(io::Error::new( ErrorKind::InvalidInput, format!("cannot parse http header from {}", arg), )); }; let value = match HeaderValue::from_str(value.trim()) { Ok(value) => value, Err(err) => { return Err(io::Error::new( ErrorKind::InvalidInput, format!( "cannot parse http header value from {} due to {:?}", value, err ), )) } }; Ok((key.to_owned(), value)) } fn parse_http_credentials(arg: &str) -> Result { let encoded = base64::engine::general_purpose::STANDARD.encode(arg.trim().as_bytes()); let Ok(header) = HeaderValue::from_str(&format!("Basic {}", encoded)) else { return Err(io::Error::new( ErrorKind::InvalidInput, format!("cannot parse http credentials {}", arg), )); }; Ok(header) } fn parse_server_url(arg: &str) -> Result { let Ok(url) = Url::parse(arg) else { return Err(io::Error::new( ErrorKind::InvalidInput, format!("cannot parse server url {}", arg), )); }; if url.scheme() != "ws" && url.scheme() != "wss" { return Err(io::Error::new( ErrorKind::InvalidInput, format!("invalid scheme {}", url.scheme()), )); } if url.host().is_none() { return Err(io::Error::new( ErrorKind::InvalidInput, format!("invalid server host {}", arg), )); } Ok(url) } #[derive(Clone, Debug)] pub struct TlsClientConfig { pub tls_sni_override: Option, pub tls_verify_certificate: bool, } #[derive(Clone, Debug)] pub struct TlsServerConfig { pub tls_certificate: Vec, pub tls_key: PrivateKey, } #[derive(Clone, Debug)] pub struct WsServerConfig { pub socket_so_mark: Option, pub bind: SocketAddr, pub restrict_to: Option>, pub websocket_ping_frequency: Option, pub timeout_connect: Duration, pub websocket_mask_frame: bool, pub tls: Option, } #[derive(Clone, Debug)] pub struct WsClientConfig { pub remote_addr: (Host, u16), pub tls: Option, pub http_upgrade_path_prefix: String, pub http_upgrade_credentials: Option, pub http_headers: HashMap, pub timeout_connect: Duration, pub websocket_ping_frequency: Duration, pub websocket_mask_frame: bool, } impl WsClientConfig { pub fn websocket_scheme(&self) -> &'static str { match self.tls { None => "ws", Some(_) => "wss", } } pub fn websocket_host_url(&self) -> String { format!("{}:{}", self.remote_addr.0, self.remote_addr.1) } pub fn tls_server_name(&self) -> ServerName { match self .tls .as_ref() .and_then(|tls| tls.tls_sni_override.as_ref()) { None => match &self.remote_addr.0 { Host::Domain(domain) => { ServerName::DnsName(DnsName::try_from(domain.clone()).unwrap()) } Host::Ipv4(ip) => ServerName::IpAddress(IpAddr::V4(*ip)), Host::Ipv6(ip) => ServerName::IpAddress(IpAddr::V6(*ip)), }, Some(sni_override) => ServerName::DnsName(sni_override.clone()), } } } #[tokio::main] async fn main() { let args = Wstunnel::parse(); // Setup logging match &args.commands { // Disable logging if there is a stdio tunnel Commands::Client(args) if args .local_to_remote .iter() .filter(|x| x.protocol == L4Protocol::Stdio) .count() > 0 => {} _ => { tracing_subscriber::fmt() .with_ansi(true) .with_env_filter(EnvFilter::from_default_env()) .init(); } } match args.commands { Commands::Client(args) => { let tls = match args.remote_addr.scheme() { "ws" => None, "wss" => Some(TlsClientConfig { tls_sni_override: args.tls_sni_override, tls_verify_certificate: args.tls_verify_certificate, }), _ => panic!("invalid scheme in server url {}", args.remote_addr.scheme()), }; let server_config = Arc::new(WsClientConfig { remote_addr: ( args.remote_addr.host().unwrap().to_owned(), args.remote_addr.port_or_known_default().unwrap(), ), tls, http_upgrade_path_prefix: args.http_upgrade_path_prefix, http_upgrade_credentials: args.http_upgrade_credentials, http_headers: args.http_headers.into_iter().collect(), timeout_connect: Duration::from_secs(10), websocket_ping_frequency: args .websocket_ping_frequency_sec .unwrap_or(Duration::from_secs(30)), websocket_mask_frame: args.websocket_mask_frame, }); // Start tunnels for tunnel in args.local_to_remote.into_iter() { let server_config = server_config.clone(); match &tunnel.protocol { L4Protocol::Tcp => { let server = tcp::run_server(tunnel.local) .await .unwrap_or_else(|err| { panic!("Cannot start TCP server on {}: {}", tunnel.local, err) }) .map_ok(TcpStream::into_split); tokio::spawn(async move { if let Err(err) = run_tunnel(server_config, tunnel, server).await { error!("{:?}", err); } }); } L4Protocol::Udp { timeout } => { let server = udp::run_server(tunnel.local, *timeout) .await .unwrap_or_else(|err| { panic!("Cannot start UDP server on {}: {}", tunnel.local, err) }) .map_ok(tokio::io::split); tokio::spawn(async move { if let Err(err) = run_tunnel(server_config, tunnel, server).await { error!("{:?}", err); } }); } L4Protocol::Stdio => { let server = stdio::run_server().await.unwrap_or_else(|err| { panic!("Cannot start STDIO server: {}", err); }); tokio::spawn(async move { if let Err(err) = run_tunnel( server_config, tunnel, stream::once(async move { Ok(server) }), ) .await { error!("{:?}", err); } }); } } } } Commands::Server(args) => { let tls_config = if args.remote_addr.scheme() == "wss" { let tls_certificate = if let Some(cert_path) = args.tls_certificate { tls::load_certificates_from_pem(&cert_path) .expect("Cannot load tls certificate") } else { embedded_certificate::TLS_CERTIFICATE.clone() }; let tls_key = if let Some(key_path) = args.tls_private_key { tls::load_private_key_from_file(&key_path).expect("Cannot load tls private key") } else { embedded_certificate::TLS_PRIVATE_KEY.clone() }; Some(TlsServerConfig { tls_certificate, tls_key, }) } else { None }; let server_config = WsServerConfig { socket_so_mark: args.socket_so_mark, bind: args.remote_addr.socket_addrs(|| Some(8080)).unwrap()[0], restrict_to: args.restrict_to, websocket_ping_frequency: args.websocket_ping_frequency_sec, timeout_connect: Duration::from_secs(10), websocket_mask_frame: args.websocket_mask_frame, tls: tls_config, }; debug!("{:?}", server_config); transport::run_server(Arc::new(server_config)) .await .unwrap_or_else(|err| { panic!("Cannot start wstunnel server: {:?}", err); }); } } tokio::signal::ctrl_c().await.unwrap(); } #[instrument(name="tunnel", level="info", skip_all, fields(id=field::Empty, remote=field::Empty))] async fn run_tunnel( server_config: Arc, tunnel: LocalToRemote, incoming_cnx: T, ) -> anyhow::Result<()> where T: Stream>, R: AsyncRead + Send + 'static, W: AsyncWrite + Send + 'static, { let span = Span::current(); let request_id = Uuid::now_v7(); span.record("id", request_id.to_string()); span.record( "remote", &format!("{}:{}", tunnel.remote.0, tunnel.remote.1), ); let tunnel = Arc::new(tunnel); pin_mut!(incoming_cnx); while let Some(Ok(cnx_stream)) = incoming_cnx.next().await { let server_config = server_config.clone(); let tunnel = tunnel.clone(); tokio::spawn( async move { let ret = transport::connect_to_server(request_id, &server_config, &tunnel, cnx_stream) .await; if let Err(ret) = ret { error!("{:?}", ret); } anyhow::Ok(()) } .instrument(span.clone()), ); } Ok(()) }