1 use native_tls::TlsConnector;
2 use std::io::{Read, Write};
3 use std::net::{TcpListener, TcpStream};
5 use std::thread::{spawn, JoinHandle};
7 use crate::configuration::ProxyConfiguration;
9 pub fn create_proxy(configuration: Arc<ProxyConfiguration>) -> JoinHandle<()> {
10 let cloned_configuration = Arc::clone(&configuration);
12 run_proxy(cloned_configuration);
16 fn run_proxy(configuration: Arc<ProxyConfiguration>) {
17 let listener = TcpListener::bind(format!("0.0.0.0:{}", configuration.local_port)).unwrap();
19 println!("Proxy listening on port {}", configuration.local_port);
21 for stream in listener.incoming() {
24 let cloned_configuration = Arc::clone(&configuration);
26 handle_client(stream, cloned_configuration);
30 eprintln!("Failed to accept connection: {}", e);
36 fn handle_client(mut client_stream: TcpStream, configuration: Arc<ProxyConfiguration>) {
37 let connector = TlsConnector::new().unwrap();
38 let remote_stream = TcpStream::connect(format!(
40 configuration.remote_domain, configuration.remote_port
43 let mut remote_stream = connector
44 .connect(&configuration.remote_domain, remote_stream)
47 let mut client_stream_clone = client_stream.try_clone().unwrap();
48 let mut remote_stream_clone = remote_stream.get_ref().try_clone().unwrap();
50 let cloned_configuration = Arc::clone(&configuration);
52 forward_stream(&mut client_stream, &mut remote_stream, cloned_configuration);
55 &mut remote_stream_clone,
56 &mut client_stream_clone,
61 fn forward_stream<R: Read, W: Write>(
64 configuration: Arc<ProxyConfiguration>,
66 let mut buffer = [0; 4096];
68 match from.read(&mut buffer) {
69 Ok(0) => break, // EOF
71 if let Err(e) = to.write_all(&buffer[..n]) {
72 eprintln!("{} proxy write error: {}", configuration.protocol, e);
75 if let Err(e) = to.flush() {
76 eprintln!("{} proxy flush error: {}", configuration.protocol, e);
81 eprintln!("{} proxy read error: {}", configuration.protocol, e);