2021-10-26 13:36:12 +00:00
|
|
|
use log::{error, info};
|
2021-10-21 08:43:59 +00:00
|
|
|
use std::collections::{HashMap, HashSet};
|
|
|
|
use std::net::SocketAddr;
|
|
|
|
use std::sync::Arc;
|
2023-10-04 22:23:34 +00:00
|
|
|
|
2021-10-21 08:43:59 +00:00
|
|
|
use tokio::task::JoinHandle;
|
|
|
|
|
2021-10-26 13:36:12 +00:00
|
|
|
mod protocol;
|
2023-08-16 07:32:05 +00:00
|
|
|
pub(crate) mod upstream_address;
|
|
|
|
|
2023-10-05 11:32:24 +00:00
|
|
|
use crate::config::ParsedConfigV1;
|
2023-10-04 22:23:34 +00:00
|
|
|
use crate::upstreams::Upstream;
|
2023-06-02 15:35:29 +00:00
|
|
|
use protocol::tcp;
|
2021-10-21 08:43:59 +00:00
|
|
|
|
|
|
|
#[derive(Debug)]
|
2023-08-16 07:32:05 +00:00
|
|
|
pub(crate) struct Server {
|
2021-10-21 08:43:59 +00:00
|
|
|
pub proxies: Vec<Arc<Proxy>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
2023-08-16 07:32:05 +00:00
|
|
|
pub(crate) struct Proxy {
|
2021-10-21 08:43:59 +00:00
|
|
|
pub name: String,
|
|
|
|
pub listen: SocketAddr,
|
2021-10-26 13:36:12 +00:00
|
|
|
pub protocol: String,
|
2021-10-21 08:43:59 +00:00
|
|
|
pub tls: bool,
|
|
|
|
pub sni: Option<HashMap<String, String>>,
|
2023-08-16 07:31:20 +00:00
|
|
|
pub default_action: String,
|
2021-10-31 11:21:32 +00:00
|
|
|
pub upstream: HashMap<String, Upstream>,
|
2021-10-21 08:43:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Server {
|
2023-10-05 11:32:24 +00:00
|
|
|
pub fn new_from_v1_config(config: ParsedConfigV1) -> Self {
|
2021-10-21 08:43:59 +00:00
|
|
|
let mut new_server = Server {
|
|
|
|
proxies: Vec::new(),
|
|
|
|
};
|
|
|
|
|
|
|
|
for (name, proxy) in config.servers.iter() {
|
2021-10-26 13:36:12 +00:00
|
|
|
let protocol = proxy.protocol.clone().unwrap_or_else(|| "tcp".to_string());
|
2021-10-21 08:43:59 +00:00
|
|
|
let tls = proxy.tls.unwrap_or(false);
|
|
|
|
let sni = proxy.sni.clone();
|
|
|
|
let default = proxy.default.clone().unwrap_or_else(|| "ban".to_string());
|
|
|
|
let upstream = config.upstream.clone();
|
|
|
|
let mut upstream_set: HashSet<String> = HashSet::new();
|
2021-10-21 09:01:52 +00:00
|
|
|
for key in upstream.keys() {
|
2021-10-21 08:43:59 +00:00
|
|
|
if key.eq("ban") || key.eq("echo") {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
upstream_set.insert(key.clone());
|
|
|
|
}
|
|
|
|
for listen in proxy.listen.clone() {
|
|
|
|
let listen_addr: SocketAddr = match listen.parse() {
|
|
|
|
Ok(addr) => addr,
|
|
|
|
Err(_) => {
|
|
|
|
error!("Invalid listen address: {}", listen);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
};
|
2021-10-31 11:21:32 +00:00
|
|
|
|
2021-10-21 08:43:59 +00:00
|
|
|
let proxy = Proxy {
|
|
|
|
name: name.clone(),
|
|
|
|
listen: listen_addr,
|
2021-10-26 13:36:12 +00:00
|
|
|
protocol: protocol.clone(),
|
2021-10-21 08:43:59 +00:00
|
|
|
tls,
|
|
|
|
sni: sni.clone(),
|
2023-08-16 07:31:20 +00:00
|
|
|
default_action: default.clone(),
|
2021-10-21 08:43:59 +00:00
|
|
|
upstream: upstream.clone(),
|
|
|
|
};
|
|
|
|
new_server.proxies.push(Arc::new(proxy));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
new_server
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
pub async fn run(&mut self) -> Result<(), Box<dyn std::error::Error>> {
|
|
|
|
let proxies = self.proxies.clone();
|
|
|
|
let mut handles: Vec<JoinHandle<()>> = Vec::new();
|
|
|
|
|
|
|
|
for config in proxies {
|
2021-10-26 13:36:12 +00:00
|
|
|
info!(
|
|
|
|
"Starting {} server {} on {}",
|
|
|
|
config.protocol, config.name, config.listen
|
|
|
|
);
|
2021-10-21 08:43:59 +00:00
|
|
|
let handle = tokio::spawn(async move {
|
2021-10-26 13:36:12 +00:00
|
|
|
match config.protocol.as_ref() {
|
2023-10-05 11:32:59 +00:00
|
|
|
"tcp" | "tcp4" | "tcp6" => {
|
2023-06-02 15:35:29 +00:00
|
|
|
let res = tcp::proxy(config.clone()).await;
|
|
|
|
if res.is_err() {
|
|
|
|
error!("Failed to start {}: {}", config.name, res.err().unwrap());
|
|
|
|
}
|
|
|
|
}
|
2021-10-26 13:36:12 +00:00
|
|
|
_ => {
|
|
|
|
error!("Invalid protocol: {}", config.protocol)
|
|
|
|
}
|
|
|
|
}
|
2021-10-21 08:43:59 +00:00
|
|
|
});
|
|
|
|
handles.push(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
for handle in handles {
|
|
|
|
handle.await?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-25 14:12:44 +00:00
|
|
|
#[cfg(test)]
|
2021-10-31 11:21:32 +00:00
|
|
|
mod tests {
|
2021-10-25 14:12:44 +00:00
|
|
|
use std::thread::{self, sleep};
|
|
|
|
use std::time::Duration;
|
2021-10-26 13:36:12 +00:00
|
|
|
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
2023-06-02 15:35:29 +00:00
|
|
|
use tokio::net::TcpListener;
|
2021-10-25 14:12:44 +00:00
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
2021-10-26 13:36:12 +00:00
|
|
|
#[tokio::main]
|
|
|
|
async fn tcp_mock_server() {
|
|
|
|
let server_addr: SocketAddr = "127.0.0.1:54599".parse().unwrap();
|
|
|
|
let listener = TcpListener::bind(server_addr).await.unwrap();
|
2021-10-27 00:36:24 +00:00
|
|
|
loop {
|
|
|
|
let (mut stream, _) = listener.accept().await.unwrap();
|
|
|
|
let mut buf = [0u8; 2];
|
|
|
|
let mut n = stream.read(&mut buf).await.unwrap();
|
|
|
|
while n > 0 {
|
2023-10-04 20:13:49 +00:00
|
|
|
let _ = stream.write(b"hello").await.unwrap();
|
2021-10-27 00:36:24 +00:00
|
|
|
if buf.eq(b"by") {
|
|
|
|
stream.shutdown().await.unwrap();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
n = stream.read(&mut buf).await.unwrap();
|
|
|
|
}
|
|
|
|
stream.shutdown().await.unwrap();
|
2021-10-26 15:52:07 +00:00
|
|
|
}
|
2021-10-26 13:36:12 +00:00
|
|
|
}
|
|
|
|
|
2021-10-25 14:12:44 +00:00
|
|
|
#[tokio::test]
|
2021-10-27 00:36:24 +00:00
|
|
|
async fn test_proxy() {
|
2023-10-05 11:32:24 +00:00
|
|
|
use crate::config::ConfigV1;
|
|
|
|
let config = ConfigV1::new("tests/config.yaml").unwrap();
|
|
|
|
let mut server = Server::new_from_v1_config(config.base);
|
2021-10-26 13:36:12 +00:00
|
|
|
thread::spawn(move || {
|
|
|
|
tcp_mock_server();
|
|
|
|
});
|
|
|
|
sleep(Duration::from_secs(1)); // wait for server to start
|
|
|
|
thread::spawn(move || {
|
|
|
|
let _ = server.run();
|
|
|
|
});
|
|
|
|
sleep(Duration::from_secs(1)); // wait for server to start
|
|
|
|
|
2021-10-27 00:36:24 +00:00
|
|
|
// test TCP proxy
|
2023-06-02 15:35:29 +00:00
|
|
|
let mut conn = tokio::net::TcpStream::connect("127.0.0.1:54500")
|
|
|
|
.await
|
|
|
|
.unwrap();
|
2021-10-27 00:36:24 +00:00
|
|
|
let mut buf = [0u8; 5];
|
2023-10-04 20:13:49 +00:00
|
|
|
let _ = conn.write(b"hi").await.unwrap();
|
|
|
|
let _ = conn.read(&mut buf).await.unwrap();
|
2021-10-27 00:36:24 +00:00
|
|
|
assert_eq!(&buf, b"hello");
|
|
|
|
conn.shutdown().await.unwrap();
|
2021-10-26 15:52:07 +00:00
|
|
|
|
2021-10-27 00:36:24 +00:00
|
|
|
// test TCP echo
|
2023-06-02 15:35:29 +00:00
|
|
|
let mut conn = tokio::net::TcpStream::connect("127.0.0.1:54956")
|
|
|
|
.await
|
|
|
|
.unwrap();
|
2021-10-25 14:12:44 +00:00
|
|
|
let mut buf = [0u8; 1];
|
2021-10-26 15:52:07 +00:00
|
|
|
for i in 0..=10u8 {
|
2023-10-04 20:13:49 +00:00
|
|
|
let _ = conn.write(&[i]).await.unwrap();
|
|
|
|
let _ = conn.read(&mut buf).await.unwrap();
|
2021-10-25 14:12:44 +00:00
|
|
|
assert_eq!(&buf, &[i]);
|
|
|
|
}
|
|
|
|
conn.shutdown().await.unwrap();
|
2021-10-26 13:36:12 +00:00
|
|
|
}
|
2021-10-25 14:12:44 +00:00
|
|
|
}
|