2022-06-04 19:34:53 +00:00
|
|
|
use clap::Parser;
|
2022-06-17 11:43:23 +00:00
|
|
|
use servers::{
|
|
|
|
plugins::loader,
|
2022-06-25 14:22:33 +00:00
|
|
|
tcp::{handle_connection, handle_websocket, Client},
|
2022-06-17 11:43:23 +00:00
|
|
|
};
|
2022-07-01 19:32:20 +00:00
|
|
|
use tokio::net::TcpListener;
|
2022-07-28 20:31:10 +00:00
|
|
|
use tracing::{info, error};
|
2022-06-04 19:34:53 +00:00
|
|
|
|
|
|
|
#[derive(Parser)]
|
|
|
|
#[clap(
|
|
|
|
name = "servers",
|
2022-06-17 11:43:23 +00:00
|
|
|
about = "A simple TCP server for client which can be extended with plugins."
|
2022-06-04 19:34:53 +00:00
|
|
|
)]
|
|
|
|
struct Cli {
|
|
|
|
#[clap(
|
|
|
|
short = 'h',
|
|
|
|
long = "host",
|
|
|
|
default_value = "0.0.0.0",
|
|
|
|
help = "Tcp server host",
|
|
|
|
display_order = 1
|
|
|
|
)]
|
|
|
|
host: String,
|
|
|
|
#[clap(
|
|
|
|
short = 'p',
|
|
|
|
long = "port",
|
|
|
|
default_value = "9999",
|
|
|
|
help = "Tcp server port [set 0 to random]",
|
|
|
|
display_order = 2
|
|
|
|
)]
|
|
|
|
port: String,
|
2022-06-25 14:22:33 +00:00
|
|
|
|
|
|
|
#[clap(
|
|
|
|
short = 'w',
|
|
|
|
long = "ws-port",
|
|
|
|
default_value = "9998",
|
2022-06-26 10:35:58 +00:00
|
|
|
help = "WebSocket server port [set 0 to random]",
|
|
|
|
display_order = 3
|
2022-06-25 14:22:33 +00:00
|
|
|
)]
|
|
|
|
ws_port: String,
|
2022-06-26 10:35:58 +00:00
|
|
|
|
|
|
|
#[clap(
|
|
|
|
long = "disable-websocket",
|
|
|
|
help = "Disable WebSocket proxy to Tcp",
|
|
|
|
display_order = 4
|
|
|
|
)]
|
|
|
|
ws_disable: bool,
|
2022-06-04 11:58:21 +00:00
|
|
|
}
|
|
|
|
|
2022-06-25 14:22:33 +00:00
|
|
|
#[tokio::main]
|
|
|
|
async fn main() -> anyhow::Result<()> {
|
2022-06-26 10:59:48 +00:00
|
|
|
// init better panic
|
|
|
|
better_panic::install();
|
2022-06-05 20:10:23 +00:00
|
|
|
// init logger
|
2022-07-28 20:31:10 +00:00
|
|
|
tracing_subscriber::fmt().init();
|
2022-06-05 14:08:13 +00:00
|
|
|
|
|
|
|
// parse cli args
|
2022-06-04 19:34:53 +00:00
|
|
|
let cli = Cli::parse();
|
2022-06-04 11:58:21 +00:00
|
|
|
|
2022-06-26 10:35:58 +00:00
|
|
|
// if enabled start WebSocket server
|
|
|
|
if !cli.ws_disable {
|
|
|
|
tokio::spawn(start_ws_server(
|
|
|
|
cli.host.clone(),
|
|
|
|
cli.ws_port,
|
|
|
|
cli.port.clone(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2022-06-04 19:34:53 +00:00
|
|
|
// start tcp server
|
2022-06-25 14:22:33 +00:00
|
|
|
start_tcp_server(cli.host, cli.port).await?;
|
2022-06-04 11:58:21 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-06-17 11:43:23 +00:00
|
|
|
|
|
|
|
/// Start tcp server
|
2022-06-25 14:22:33 +00:00
|
|
|
async fn start_tcp_server(host: String, port: String) -> anyhow::Result<()> {
|
2022-06-17 11:43:23 +00:00
|
|
|
// listen Tcp server
|
2022-07-01 19:32:20 +00:00
|
|
|
let listener = TcpListener::bind(format!("{host}:{port}")).await?;
|
2022-06-17 11:43:23 +00:00
|
|
|
|
2022-06-26 10:35:58 +00:00
|
|
|
info!("Tcp server started at: {}", listener.local_addr()?);
|
2022-06-17 11:43:23 +00:00
|
|
|
|
|
|
|
// load plugins, commands and events
|
2022-06-20 21:22:45 +00:00
|
|
|
let plugin_manager = loader()?;
|
2022-06-17 11:43:23 +00:00
|
|
|
|
|
|
|
// Accepts a new incoming connection from this listener.
|
2022-07-01 19:32:20 +00:00
|
|
|
while let Ok((stream, _address)) = listener.accept().await {
|
2022-06-17 11:43:23 +00:00
|
|
|
let client = Client::new(stream);
|
2022-06-26 10:59:48 +00:00
|
|
|
let plugin_manager = plugin_manager.clone();
|
2022-06-17 11:43:23 +00:00
|
|
|
|
|
|
|
// handle client connection in new thread
|
2022-06-26 10:59:48 +00:00
|
|
|
tokio::spawn(async move {
|
|
|
|
let ip = client.stream.peer_addr().unwrap();
|
|
|
|
|
|
|
|
match handle_connection(client, plugin_manager).await {
|
|
|
|
Ok(_) => (),
|
|
|
|
Err(err) => error!("Client {}, {}", ip, err),
|
|
|
|
}
|
|
|
|
});
|
2022-06-17 11:43:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// server for a unexpectedly reason be terminated
|
|
|
|
panic!("Server unexpectedly terminated!")
|
|
|
|
}
|
2022-06-25 14:22:33 +00:00
|
|
|
|
|
|
|
/// Start WebSocket server
|
|
|
|
async fn start_ws_server(host: String, port: String, tcp_port: String) -> anyhow::Result<()> {
|
|
|
|
// listen Tcp server
|
|
|
|
let listener = tokio::net::TcpListener::bind(format!("{host}:{port}")).await?;
|
|
|
|
|
2022-06-26 10:35:58 +00:00
|
|
|
info!("WebSocket server started at: {}", listener.local_addr()?);
|
2022-06-25 14:22:33 +00:00
|
|
|
|
|
|
|
// Accepts a new incoming connection from this listener.
|
|
|
|
while let Ok((stream, _address)) = listener.accept().await {
|
|
|
|
let tcp_port = tcp_port.clone();
|
2022-06-26 10:59:48 +00:00
|
|
|
tokio::spawn(async {
|
|
|
|
let ip = stream.peer_addr().unwrap();
|
|
|
|
|
|
|
|
match handle_websocket(stream, tcp_port).await {
|
|
|
|
Ok(_) => (),
|
|
|
|
Err(err) => error!("Client {}, {}", ip, err),
|
|
|
|
}
|
|
|
|
});
|
2022-06-25 14:22:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// server for a unexpectedly reason be terminated
|
|
|
|
panic!("Server unexpectedly terminated!")
|
|
|
|
}
|