odproxy/main.rs

79 lines
1.9 KiB
Rust
Executable file

mod conf;
mod data;
mod services;
use std::str::FromStr;
use data::HOST_MAP;
use hyperlocal::UnixClientExt;
use services::check_service;
use tower::make::Shared;
use hyper::{service::service_fn, Body, Client, Request, Response, Server};
use crate::{conf::CONFIG, services::prepare_services};
async fn run(req: Request<Body>) -> Result<Response<Body>, hyper::Error> {
let host = req.headers().get("host");
let host_index = data::get_proxy_index(host);
let proxy = data::get_proxy(host_index);
match proxy {
Some(p) => {
check_service(host_index.unwrap().clone(), p).await;
// Create new Request
let mut request_builder = Request::builder().method(req.method());
let path = req.uri().path_and_query().unwrap().as_str();
let is_socket = p.socket.unwrap_or(false);
if is_socket {
request_builder = request_builder.uri(hyperlocal::Uri::new(&p.target, path));
} else {
let url = p.target.clone() + path;
request_builder = request_builder.uri(hyper::Uri::from_str(url.as_str()).expect("[!] Wrong url address!"));
}
// Copy all the headers
for (name, value) in req.headers().iter() {
request_builder = request_builder.header(name, value);
}
// Copy body
let body = req.into_body();
let nreq = request_builder.body(body).unwrap();
if is_socket {
Client::unix().request(nreq).await
} else {
Client::new().request(nreq).await
}
},
None => {
println!("Unknown host accessed: {:?}", host.unwrap());
return Ok(Response::new(Body::empty()));
}
}
}
#[tokio::main]
async fn main() {
prepare_services().await;
let make_service = Shared::new(service_fn(run));
let server = Server::bind(&CONFIG.listen).serve(make_service);
let host_count = HOST_MAP.len();
let service_count = CONFIG.proxy.len();
println!("odproxy is listening on {} with {} hosts and {} services", CONFIG.listen, host_count, service_count);
if let Err(e) = server.await {
println!("error: {}", e);
}
}