Browse Source

added support for authentication with username and password

pull/8/head
KaranGauswami 3 years ago
parent
commit
a06e3bc81e
No known key found for this signature in database GPG Key ID: 521E5CB2B400CA51
  1. 49
      src/main.rs

49
src/main.rs

@ -5,7 +5,7 @@ use hyper::client::HttpConnector;
use hyper::service::{make_service_fn, service_fn};
use hyper::upgrade::Upgraded;
use hyper::{Body, Request, Response, Server};
use hyper_socks2::SocksConnector;
use hyper_socks2::{Auth, SocksConnector};
use log::debug;
use std::convert::Infallible;
use std::net::SocketAddr;
@ -22,6 +22,14 @@ struct Cli {
/// Socks5 proxy address
#[clap(short, long, default_value = "127.0.0.1:1080")]
socks_address: SocketAddr,
/// Socks5 username
#[clap(short = 'u', long)]
username: Option<String>,
/// Socks5 username
#[clap(short = 'P', long)]
password: Option<String>,
}
#[tokio::main]
@ -31,9 +39,18 @@ async fn main() -> Result<()> {
let args = Cli::parse();
let socks_address = args.socks_address;
let port = args.port;
let username = args.username;
let password = args.password;
let auth = if let (Some(username), Some(password)) = (username, password) {
Some(Auth::new(username, password))
} else {
None
};
let auth = &*Box::leak(Box::new(auth));
let addr = SocketAddr::from(([0, 0, 0, 0], port));
let make_service = make_service_fn(move |_| async move {
Ok::<_, Infallible>(service_fn(move |req| proxy(req, socks_address)))
Ok::<_, Infallible>(service_fn(move |req| proxy(req, socks_address, auth)))
});
let server = Server::bind(&addr)
.http1_preserve_header_case(true)
@ -48,7 +65,11 @@ async fn main() -> Result<()> {
fn host_addr(uri: &http::Uri) -> Option<String> {
uri.authority().map(|auth| auth.to_string())
}
async fn proxy(req: Request<Body>, socks_address: SocketAddr) -> Result<Response<Body>> {
async fn proxy(
req: Request<Body>,
socks_address: SocketAddr,
auth: &'static Option<Auth>,
) -> Result<Response<Body>> {
let mut connector = HttpConnector::new();
connector.enforce_http(false);
let proxy_addr = Box::leak(Box::new(format!("socks://{}", socks_address)));
@ -58,7 +79,7 @@ async fn proxy(req: Request<Body>, socks_address: SocketAddr) -> Result<Response
tokio::task::spawn(async move {
match hyper::upgrade::on(req).await {
Ok(upgraded) => {
if let Err(e) = tunnel(upgraded, plain, socks_address).await {
if let Err(e) = tunnel(upgraded, plain, socks_address, auth).await {
debug!("server io error: {}", e);
};
}
@ -68,7 +89,8 @@ async fn proxy(req: Request<Body>, socks_address: SocketAddr) -> Result<Response
Ok(Response::new(Body::empty()))
} else {
let connector = SocksConnector {
auth: None,
// TODO: Can we remove this clone ?
auth: auth.clone(),
proxy_addr,
connector,
};
@ -77,18 +99,29 @@ async fn proxy(req: Request<Body>, socks_address: SocketAddr) -> Result<Response
Ok(response.expect("Cannot make HTTP request"))
}
} else {
let mut resp = Response::new(Body::from("CONNECT must be to a socket address"));
let mut resp = Response::new("CONNECT must be to a socket address".into());
*resp.status_mut() = http::StatusCode::BAD_REQUEST;
Ok(resp)
}
}
async fn tunnel<'t, P, T>(mut upgraded: Upgraded, plain: T, socks_address: P) -> Result<()>
async fn tunnel<'t, P, T>(
mut upgraded: Upgraded,
plain: T,
socks_address: P,
auth: &Option<Auth>,
) -> Result<()>
where
P: ToProxyAddrs,
T: IntoTargetAddr<'t>,
{
let mut stream = Socks5Stream::connect(socks_address, plain).await?;
let mut stream = if let Some(auth) = auth {
let username = &auth.username;
let password = &auth.password;
Socks5Stream::connect_with_password(socks_address, plain, username, password).await?
} else {
Socks5Stream::connect(socks_address, plain).await?
};
// Proxying data
let (from_client, from_server) =

Loading…
Cancel
Save