aboutsummaryrefslogtreecommitdiffstats
path: root/src/main.rs
blob: 342836fe87922b7d9a6e4bf6b1c6601ef9e8177b (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
107
108
109
110
111
112
113
114
115
use hyper::service::{make_service_fn, service_fn};
use hyper::{Body, Client, Request, Response, Server, StatusCode};
use std::env;
use std::net::SocketAddr;
use toml;

#[macro_use]
extern crate log;

use es_public_proxy::{filter_request, ProxyConfig};

async fn upstream_req(
    req: Request<Body>,
    config: ProxyConfig,
) -> Result<Response<Body>, hyper::Error> {
    info!("hit: {}", req.uri());
    let parsed = filter_request(req, &config).await;
    let resp = match parsed {
        Ok(upstream_req) => {
            debug!("sending request...");
            Client::new().request(upstream_req).await?
        }
        Err(other) => Response::builder()
            .status(StatusCode::INTERNAL_SERVER_ERROR)
            .header("Content-Type", "application/json; charset=UTF-8")
            .body(serde_json::to_string(&other.to_json()).unwrap().into())
            .unwrap(),
    };
    debug!("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(),
    };

    info!("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 {
        error!("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;
    let mut allow_all_indices = false;

    // first parse CLI arg
    match args.as_slice() {
        [_] | [] => {}
        [_, "-h"] | [_, "--help"] => {
            println!("{}", usage());
            std::process::exit(0);
        }
        [_, "--example-config"] => {
            println!("{}", include_str!("../example_config.toml"));
            std::process::exit(0);
        },
        [_, "--config", p] => config_path = Some(p.to_string()),
        [_, "--allow-all-indices"] => allow_all_indices = true,
        _ => {
            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
    let mut 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()
    };
    if allow_all_indices {
        config.allow_all_indices = Some(true);
    }
    config
}

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