mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-04-02 05:23:14 +00:00
- Replace DynInterfaceFilter with ForwardRequirements for per-IP forward precision with source-subnet iptables filtering for private forwards - Add WildcardListener (binds [::]:port) to replace the per-gateway NetworkInterfaceListener/SelfContainedNetworkInterfaceListener/ UpgradableListener infrastructure - Update forward-port script with src_subnet and excluded_src env vars - Remove unused filter types and listener infrastructure from gateway.rs - Add availablePorts migration (IdPool -> BTreeMap<u16, bool>) to alpha.20 - Complete version bump to 0.4.0-alpha.20 in SDK and web
208 lines
7.3 KiB
Rust
208 lines
7.3 KiB
Rust
use std::ffi::OsString;
|
|
use std::net::SocketAddr;
|
|
use std::sync::Arc;
|
|
use std::time::Duration;
|
|
|
|
use clap::Parser;
|
|
use futures::FutureExt;
|
|
use rpc_toolkit::CliApp;
|
|
use rust_i18n::t;
|
|
use tokio::signal::unix::signal;
|
|
use tracing::instrument;
|
|
use visit_rs::Visit;
|
|
|
|
use crate::context::CliContext;
|
|
use crate::context::config::ClientConfig;
|
|
use tokio::net::TcpListener;
|
|
use crate::net::tls::TlsListener;
|
|
use crate::net::web_server::{Accept, Acceptor, MetadataVisitor, WebServer};
|
|
use crate::prelude::*;
|
|
use crate::tunnel::context::{TunnelConfig, TunnelContext};
|
|
use crate::tunnel::tunnel_router;
|
|
use crate::tunnel::web::TunnelCertHandler;
|
|
use crate::util::future::NonDetachingJoinHandle;
|
|
use crate::util::logger::LOGGER;
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
|
|
enum WebserverListener {
|
|
Http,
|
|
Https(SocketAddr),
|
|
}
|
|
impl<V: MetadataVisitor> Visit<V> for WebserverListener {
|
|
fn visit(&self, visitor: &mut V) -> <V as visit_rs::Visitor>::Result {
|
|
visitor.visit(self)
|
|
}
|
|
}
|
|
|
|
#[instrument(skip_all)]
|
|
async fn inner_main(config: &TunnelConfig) -> Result<(), Error> {
|
|
let server = async {
|
|
let ctx = TunnelContext::init(config).await?;
|
|
let listen = ctx.listen;
|
|
let server = WebServer::new(
|
|
Acceptor::bind_map_dyn([(WebserverListener::Http, listen)]).await?,
|
|
tunnel_router(ctx.clone()),
|
|
);
|
|
let acceptor_setter = server.acceptor_setter();
|
|
let https_db = ctx.db.clone();
|
|
let https_thread: NonDetachingJoinHandle<()> = tokio::spawn(async move {
|
|
let mut sub = https_db.subscribe("/webserver".parse().unwrap()).await;
|
|
while {
|
|
while let Err(e) = async {
|
|
let webserver = https_db.peek().await.into_webserver();
|
|
if webserver.as_enabled().de()? {
|
|
let addr = webserver.as_listen().de()?.or_not_found("listen address")?;
|
|
acceptor_setter.send_if_modified(|a| {
|
|
let key = WebserverListener::Https(addr);
|
|
if !a.contains_key(&key) {
|
|
match (|| {
|
|
Ok::<_, Error>(TlsListener::new(
|
|
TcpListener::from_std(
|
|
mio::net::TcpListener::bind(addr)
|
|
.with_kind(ErrorKind::Network)?
|
|
.into(),
|
|
)
|
|
.with_kind(ErrorKind::Network)?,
|
|
TunnelCertHandler {
|
|
db: https_db.clone(),
|
|
crypto_provider: Arc::new(tokio_rustls::rustls::crypto::ring::default_provider()),
|
|
},
|
|
))
|
|
})() {
|
|
Ok(l) => {
|
|
a.retain(|k, _| *k == WebserverListener::Http);
|
|
a.insert(key, l.into_dyn());
|
|
|
|
true
|
|
}
|
|
Err(e) => {
|
|
tracing::error!("{}", t!("bins.tunnel.error-adding-ssl-listener", error = e.to_string()));
|
|
tracing::debug!("{e:?}");
|
|
|
|
false
|
|
}
|
|
}
|
|
} else {
|
|
false
|
|
}
|
|
});
|
|
} else {
|
|
acceptor_setter.send_if_modified(|a| {
|
|
let before = a.len();
|
|
a.retain(|k, _| *k == WebserverListener::Http);
|
|
a.len() != before
|
|
});
|
|
}
|
|
|
|
Ok::<_, Error>(())
|
|
}
|
|
.await
|
|
{
|
|
tracing::error!("{}", t!("bins.tunnel.error-updating-webserver-bind", error = e.to_string()));
|
|
tracing::debug!("{e:?}");
|
|
tokio::time::sleep(Duration::from_secs(5)).await;
|
|
}
|
|
sub.recv().await.is_some()
|
|
} {}
|
|
})
|
|
.into();
|
|
|
|
let mut shutdown_recv = ctx.shutdown.subscribe();
|
|
|
|
let sig_handler_ctx = ctx;
|
|
let sig_handler: NonDetachingJoinHandle<()> = tokio::spawn(async move {
|
|
use tokio::signal::unix::SignalKind;
|
|
futures::future::select_all(
|
|
[
|
|
SignalKind::interrupt(),
|
|
SignalKind::quit(),
|
|
SignalKind::terminate(),
|
|
]
|
|
.iter()
|
|
.map(|s| {
|
|
async move {
|
|
signal(*s)
|
|
.unwrap_or_else(|_| panic!("register {:?} handler", s))
|
|
.recv()
|
|
.await
|
|
}
|
|
.boxed()
|
|
}),
|
|
)
|
|
.await;
|
|
sig_handler_ctx
|
|
.shutdown
|
|
.send(())
|
|
.map_err(|_| ())
|
|
.expect("send shutdown signal");
|
|
})
|
|
.into();
|
|
|
|
shutdown_recv
|
|
.recv()
|
|
.await
|
|
.with_kind(crate::ErrorKind::Unknown)?;
|
|
|
|
sig_handler.wait_for_abort().await.with_kind(ErrorKind::Unknown)?;
|
|
https_thread.wait_for_abort().await.with_kind(ErrorKind::Unknown)?;
|
|
|
|
Ok::<_, Error>(server)
|
|
}
|
|
.await?;
|
|
server.shutdown().await;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn main(args: impl IntoIterator<Item = OsString>) {
|
|
LOGGER.enable();
|
|
|
|
let config = TunnelConfig::parse_from(args).load().unwrap();
|
|
|
|
let res = {
|
|
let rt = tokio::runtime::Builder::new_multi_thread()
|
|
.enable_all()
|
|
.build()
|
|
.expect(&t!("bins.tunnel.failed-to-initialize-runtime"));
|
|
rt.block_on(inner_main(&config))
|
|
};
|
|
|
|
match res {
|
|
Ok(()) => (),
|
|
Err(e) => {
|
|
eprintln!("{}", e.source);
|
|
tracing::debug!("{:?}", e.source);
|
|
drop(e.source);
|
|
std::process::exit(e.kind as i32)
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn cli(args: impl IntoIterator<Item = OsString>) {
|
|
LOGGER.enable();
|
|
|
|
if let Err(e) = CliApp::new(
|
|
|cfg: ClientConfig| Ok(CliContext::init(cfg.load()?)?),
|
|
crate::tunnel::api::tunnel_api(),
|
|
)
|
|
.mutate_command(super::translate_cli)
|
|
.run(args)
|
|
{
|
|
match e.data {
|
|
Some(serde_json::Value::String(s)) => eprintln!("{}: {}", e.message, s),
|
|
Some(serde_json::Value::Object(o)) => {
|
|
if let Some(serde_json::Value::String(s)) = o.get("details") {
|
|
eprintln!("{}: {}", e.message, s);
|
|
if let Some(serde_json::Value::String(s)) = o.get("debug") {
|
|
tracing::debug!("{}", s)
|
|
}
|
|
}
|
|
}
|
|
Some(a) => eprintln!("{}: {}", e.message, a),
|
|
None => eprintln!("{}", e.message),
|
|
}
|
|
|
|
std::process::exit(e.code);
|
|
}
|
|
}
|