overtls 0.3.8

A simple proxy tunnel, minimalist tool for bypassing the GFW.
Documentation
use overtls::{BoxError, CmdOpt, Config, Result, async_main};

fn main() -> Result<(), BoxError> {
    let opt = CmdOpt::parse_cmd();

    if opt.c_api {
        if opt.is_server() {
            return Err("C API is not supported for server".into());
        }

        let ctrlc_fired = std::sync::Arc::new(std::sync::atomic::AtomicBool::new(false));
        let ctrlc_fired_clone = ctrlc_fired.clone();
        let ctrl_handle = ctrlc2::set_handler(move || {
            log::info!("Ctrl-C received, exiting...");
            ctrlc_fired_clone.store(true, std::sync::atomic::Ordering::SeqCst);
            unsafe { overtls::over_tls_client_stop() };
            true
        })?;

        unsafe extern "C" fn log_cb(_: overtls::ArgVerbosity, msg: *const std::os::raw::c_char, _ctx: *mut std::os::raw::c_void) {
            println!("{:?}", unsafe { std::ffi::CStr::from_ptr(msg).to_str() });
        }
        unsafe { overtls::overtls_set_log_callback(true, Some(log_cb), std::ptr::null_mut()) };

        unsafe extern "C" fn port_cb(port: i32, _ctx: *mut std::os::raw::c_void) {
            log::info!("Listening on {port}");
        }

        if let Some(cfg) = opt.config.as_ref() {
            // Test the C API usage
            let config_path_str = cfg.as_path().to_string_lossy().into_owned();
            let c_string = std::ffi::CString::new(config_path_str)?;
            let config_path: *const std::os::raw::c_char = c_string.as_ptr();

            unsafe { overtls::over_tls_client_run(config_path, opt.verbosity, Some(port_cb), std::ptr::null_mut()) };

            if ctrlc_fired.load(std::sync::atomic::Ordering::SeqCst) {
                ctrl_handle.join().map_err(|e| format!("{e:?}"))?;
            }
        } else if let Some(url) = opt.url_of_node.as_ref() {
            let url_str = std::ffi::CString::new(url.as_str())?;
            let url_ptr = url_str.as_ptr();

            let listen_addr = opt.listen_addr.unwrap_or(std::net::SocketAddr::from(([127, 0, 0, 1], 1080)));
            let listen_addr = std::ffi::CString::new(listen_addr.to_string())?;
            let listen_addr = listen_addr.as_ptr();

            unsafe { overtls::over_tls_client_run_with_ssr_url(url_ptr, listen_addr, opt.verbosity, Some(port_cb), std::ptr::null_mut()) };

            if ctrlc_fired.load(std::sync::atomic::Ordering::SeqCst) {
                ctrl_handle.join().map_err(|e| format!("{e:?}"))?;
            }
        } else {
            return Err("Config file or node URL is required".into());
        }
        return Ok(());
    }

    dotenvy::dotenv().ok();

    let level = format!("overtls={:?},connection_pool={:?}", opt.verbosity, opt.verbosity);
    env_logger::Builder::from_env(env_logger::Env::default().default_filter_or(level)).init();

    let is_server = opt.is_server();

    let mut config = if let Some(file) = opt.config {
        Config::from_config_file(file)?
    } else if let Some(ref url_of_node) = opt.url_of_node {
        let mut cfg = Config::from_ssr_url(url_of_node)?;
        cfg.set_listen_addr(opt.listen_addr.unwrap_or(std::net::SocketAddr::from(([127, 0, 0, 1], 1080))));
        cfg
    } else {
        return Err("Config file or node URL is required".into());
    };
    config.set_cache_dns(opt.cache_dns);

    if let Some(size) = opt.pool_max_size
        && let Some(client) = config.client.as_mut()
    {
        client.pool_max_size = Some(size);
    }

    if opt.generate_url {
        let url = config.generate_ssr_url()?;
        println!("{url}");
        return Ok(());
    }

    config.check_correctness(is_server)?;

    #[cfg(unix)]
    if opt.daemonize {
        let stdout = std::fs::File::create("/tmp/overtls.out")?;
        let stderr = std::fs::File::create("/tmp/overtls.err")?;
        let daemonize = daemonize::Daemonize::new()
            .working_directory("/tmp")
            .umask(0o777)
            .stdout(stdout)
            .stderr(stderr)
            .privileged_action(|| "Executed before drop privileges");
        let _ = daemonize.start()?;
    }

    #[cfg(target_os = "windows")]
    if opt.daemonize {
        overtls::win_svc::start_service()?;
        return Ok(());
    }

    let shutdown_token = overtls::CancellationToken::new();
    let rt = tokio::runtime::Builder::new_multi_thread().enable_all().build()?;
    rt.block_on(async_main(config, true, shutdown_token))?;
    Ok(())
}