summaryrefslogtreecommitdiffstats
path: root/src/main.rs
blob: 017b8c86853575c2e8a5d0e8d01dff00f54c6342 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106

use hyper::service::{make_service_fn, service_fn};
use hyper::{Body, Client, Request, Response, Server, Uri, StatusCode};
use std::net::SocketAddr;
use std::env;
use toml;

use es_public_proxy::{ProxyConfig, ParsedRequest, parse_request};

async fn upstream_req(req: Request<Body>, config: ProxyConfig) -> Result<Response<Body>, hyper::Error> {
    println!("hit: {}", req.uri());
    let parsed = parse_request(req, &config);
    let resp = match parsed {
        ParsedRequest::Allowed(upstream_req) => {
            println!("sending request...");
            Client::new().request(upstream_req).await?
        }
        other => {
            Response::builder()
                .status(StatusCode::NOT_FOUND)
                .body(format!("oh noooo! {:?}", other).into())
                .unwrap()
        },
    };
    println!("resp!");
    Ok(resp)
}

async fn shutdown_signal() {
    // Wait for the CTRL+C signal
    tokio::signal::ctrl_c()
        .await
        .expect("failed to install CTRL+C signal handler");
}

async fn run_server(config: ProxyConfig) {

    let addr = match &config.bind_addr {
        None => SocketAddr::from(([127, 0, 0, 1], 9292)),
        Some(addr) => addr.parse().unwrap(),
    };

    println!("Listening on http://{}", addr);

    // TODO: possible to avoid cloning config on every connection?
    let make_svc = make_service_fn(move |_| {
        let inner = config.clone();
        async move {
            Ok::<_, hyper::Error>(service_fn(move |req| {
                upstream_req(req, inner.clone())
            }))
        }
    });
    let serve_future = Server::bind(&addr).serve(make_svc);
    let graceful = serve_future.with_graceful_shutdown(shutdown_signal());

    if let Err(e) = graceful.await {
        eprintln!("server error: {}", e);
    }
}

fn usage() -> String {
    "es-public-proxy [--config CONFIG_FILE] [--help]".to_string()
}

fn load_config() -> ProxyConfig {

    let args: Vec<String> = env::args().collect();
    let args: Vec<&str> = args.iter().map(|x| x.as_str()).collect();
    let mut config_path: Option<String> = None;

    // first parse CLI arg
    match args.as_slice() {
        [_] | [] => {},
        [_, "-h"] | [_, "--help"] => {
            println!("{}", usage());
            std::process::exit(0);
        },
        [_, "--config", p] => { config_path = Some(p.to_string()) },
        _ => {
            eprintln!("{}", usage());
            eprintln!("couldn't parse arguments");
            std::process::exit(1);
        }
    }

    // then try environment variables
    if let None = config_path {
        config_path = std::env::var("ES_PUBLIC_PROXY_CONFIG_PATH").ok();
    }

    // then either load config file (TOML), or use default config
    if let Some(config_path) = config_path {
        let config_toml = std::fs::read_to_string(config_path).unwrap();
        let config: ProxyConfig = toml::from_str(&config_toml).unwrap();
        config
    } else {
        ProxyConfig::default()
    }
}

#[tokio::main]
async fn main() {
    let config = load_config();
    run_server(config).await;
}