mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-26 10:21:52 +00:00
* squashfs-wip * sdk fixes * misc fixes * bump sdk * Include StartTunnel installation command Added installation instructions for StartTunnel. * CA instead of leaf for StartTunnel (#3046) * updated docs for CA instead of cert * generate ca instead of self-signed in start-tunnel * Fix formatting in START-TUNNEL.md installation instructions * Fix formatting in START-TUNNEL.md * fix infinite loop * add success message to install * hide loopback and bridge gateways --------- Co-authored-by: Aiden McClelland <me@drbonez.dev> Co-authored-by: Aiden McClelland <3732071+dr-bonez@users.noreply.github.com> * prevent gateways from getting stuck empty * fix set-password * misc networking fixes * build and efi fixes * efi fixes * alpha.13 * remove cross * fix tests * provide path to upgrade * fix networkmanager issues * remove squashfs before creating --------- Co-authored-by: Matt Hill <MattDHill@users.noreply.github.com>
201 lines
6.8 KiB
Rust
201 lines
6.8 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 helpers::NonDetachingJoinHandle;
|
|
use rpc_toolkit::CliApp;
|
|
use tokio::signal::unix::signal;
|
|
use tracing::instrument;
|
|
use visit_rs::Visit;
|
|
|
|
use crate::context::CliContext;
|
|
use crate::context::config::ClientConfig;
|
|
use crate::net::gateway::{Bind, BindTcp};
|
|
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::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(
|
|
BindTcp.bind(addr)?,
|
|
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!("error adding ssl listener: {e}");
|
|
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!("error updating webserver bind: {e}");
|
|
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("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(),
|
|
)
|
|
.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);
|
|
}
|
|
}
|