2022-04-27 20:19:04 +02:00
|
|
|
use clap::{Arg, Command};
|
2022-03-15 20:00:43 +01:00
|
|
|
use diflouroborane::{config, Diflouroborane};
|
|
|
|
|
2022-03-11 22:13:54 +01:00
|
|
|
use std::str::FromStr;
|
|
|
|
use std::{env, io, io::Write, path::PathBuf};
|
2022-03-15 20:00:43 +01:00
|
|
|
|
2022-03-11 22:13:54 +01:00
|
|
|
use nix::NixPath;
|
2021-10-19 11:16:24 +02:00
|
|
|
|
2022-03-11 22:13:54 +01:00
|
|
|
fn main() -> anyhow::Result<()> {
|
2021-10-19 11:16:24 +02:00
|
|
|
// Argument parsing
|
|
|
|
// values for the name, description and version are pulled from `Cargo.toml`.
|
2022-03-11 22:13:54 +01:00
|
|
|
let matches = Command::new(clap::crate_name!())
|
|
|
|
.version(clap::crate_version!())
|
2022-05-18 17:01:03 +02:00
|
|
|
.long_version(&*format!("{version}\n\
|
|
|
|
FabAccess {apiver}\n\
|
|
|
|
\t[{build_kind} build built on {build_time}]\n\
|
|
|
|
\t {rustc_version}\n\t {cargo_version}",
|
|
|
|
version=diflouroborane::env::PKG_VERSION,
|
|
|
|
apiver="0.3",
|
|
|
|
rustc_version=diflouroborane::env::RUST_VERSION,
|
|
|
|
cargo_version=diflouroborane::env::CARGO_VERSION,
|
|
|
|
build_time=diflouroborane::env::BUILD_TIME_3339,
|
|
|
|
build_kind=diflouroborane::env::BUILD_RUST_CHANNEL))
|
2022-03-11 22:13:54 +01:00
|
|
|
.about(clap::crate_description!())
|
2022-05-18 17:01:03 +02:00
|
|
|
.arg(Arg::new("config")
|
2022-03-11 22:13:54 +01:00
|
|
|
.help("Path to the config file to use")
|
|
|
|
.long("config")
|
|
|
|
.short('c')
|
2022-05-18 17:01:03 +02:00
|
|
|
.takes_value(true))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(Arg::new("verbosity")
|
|
|
|
.help("Increase logging verbosity")
|
|
|
|
.long("verbose")
|
|
|
|
.short('v')
|
|
|
|
.multiple_occurrences(true)
|
|
|
|
.max_occurrences(3)
|
2022-05-18 17:01:03 +02:00
|
|
|
.conflicts_with("quiet"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(Arg::new("quiet")
|
|
|
|
.help("Decrease logging verbosity")
|
|
|
|
.long("quiet")
|
2022-05-18 17:01:03 +02:00
|
|
|
.conflicts_with("verbosity"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(Arg::new("log format")
|
|
|
|
.help("Use an alternative log formatter. Available: Full, Compact, Pretty")
|
|
|
|
.long("log-format")
|
|
|
|
.takes_value(true)
|
|
|
|
.ignore_case(true)
|
|
|
|
.possible_values(["Full", "Compact", "Pretty"]))
|
2022-04-27 17:30:04 +02:00
|
|
|
.arg(Arg::new("log level")
|
|
|
|
.help("Set the desired log levels.")
|
|
|
|
.long("log-level")
|
|
|
|
.takes_value(true))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(
|
|
|
|
Arg::new("print default")
|
|
|
|
.help("Print a default config to stdout instead of running")
|
2022-05-18 17:01:03 +02:00
|
|
|
.long("print-default"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(
|
|
|
|
Arg::new("check config")
|
|
|
|
.help("Check config for validity")
|
2022-05-18 17:01:03 +02:00
|
|
|
.long("check"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(
|
|
|
|
Arg::new("dump")
|
|
|
|
.help("Dump all internal databases")
|
|
|
|
.long("dump")
|
2022-05-18 17:01:03 +02:00
|
|
|
.conflicts_with("load"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(
|
|
|
|
Arg::new("load")
|
|
|
|
.help("Load values into the internal databases")
|
|
|
|
.long("load")
|
2022-03-13 22:50:37 +01:00
|
|
|
.takes_value(true)
|
2022-05-18 17:01:03 +02:00
|
|
|
.conflicts_with("dump"))
|
2022-03-11 22:13:54 +01:00
|
|
|
.arg(Arg::new("keylog")
|
|
|
|
.help("log TLS keys into PATH. If no path is specified the value of the envvar SSLKEYLOGFILE is used.")
|
|
|
|
.long("tls-key-log")
|
|
|
|
.value_name("PATH")
|
|
|
|
.takes_value(true)
|
|
|
|
.max_values(1)
|
|
|
|
.min_values(0)
|
2022-05-18 17:01:03 +02:00
|
|
|
.default_missing_value(""))
|
2022-03-11 22:13:54 +01:00
|
|
|
.get_matches();
|
|
|
|
|
|
|
|
let configpath = matches
|
|
|
|
.value_of("config")
|
|
|
|
.unwrap_or("/etc/diflouroborane.dhall");
|
2021-10-19 11:16:24 +02:00
|
|
|
|
|
|
|
// Check for the --print-default option first because we don't need to do anything else in that
|
|
|
|
// case.
|
|
|
|
if matches.is_present("print default") {
|
|
|
|
let config = config::Config::default();
|
|
|
|
let encoded = serde_dhall::serialize(&config).to_string().unwrap();
|
|
|
|
|
|
|
|
// Direct writing to fd 1 is faster but also prevents any print-formatting that could
|
|
|
|
// invalidate the generated TOML
|
|
|
|
let stdout = io::stdout();
|
|
|
|
let mut handle = stdout.lock();
|
2022-03-20 21:22:15 +01:00
|
|
|
handle.write_all(encoded.as_bytes()).unwrap();
|
2021-10-19 11:16:24 +02:00
|
|
|
|
|
|
|
// Early return to exit.
|
|
|
|
return Ok(());
|
|
|
|
} else if matches.is_present("check config") {
|
|
|
|
match config::read(&PathBuf::from_str(configpath).unwrap()) {
|
2022-03-15 19:13:55 +01:00
|
|
|
Ok(c) => {
|
2022-03-20 21:22:15 +01:00
|
|
|
let formatted = format!("{:#?}", c);
|
|
|
|
|
|
|
|
// Direct writing to fd 1 is faster but also prevents any print-formatting that could
|
|
|
|
// invalidate the generated TOML
|
|
|
|
let stdout = io::stdout();
|
|
|
|
let mut handle = stdout.lock();
|
|
|
|
handle.write_all(formatted.as_bytes()).unwrap();
|
|
|
|
|
|
|
|
// Early return to exit.
|
|
|
|
return Ok(());
|
2021-10-19 11:16:24 +02:00
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
eprintln!("{}", e);
|
|
|
|
std::process::exit(-1);
|
|
|
|
}
|
|
|
|
}
|
2022-03-13 22:50:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut config = config::read(&PathBuf::from_str(configpath).unwrap()).unwrap();
|
|
|
|
|
|
|
|
if matches.is_present("dump") {
|
2022-03-16 18:10:59 +01:00
|
|
|
unimplemented!()
|
2022-03-11 22:13:54 +01:00
|
|
|
} else if matches.is_present("load") {
|
2022-03-15 19:13:55 +01:00
|
|
|
let bffh = Diflouroborane::new(config)?;
|
2022-05-05 15:50:44 +02:00
|
|
|
if bffh
|
|
|
|
.users
|
|
|
|
.load_file(matches.value_of("load").unwrap())
|
|
|
|
.is_ok()
|
|
|
|
{
|
2022-03-19 05:54:18 +01:00
|
|
|
tracing::info!("loaded users from {}", matches.value_of("load").unwrap());
|
|
|
|
} else {
|
2022-05-05 15:50:44 +02:00
|
|
|
tracing::error!(
|
|
|
|
"failed to load users from {}",
|
|
|
|
matches.value_of("load").unwrap()
|
|
|
|
);
|
2022-03-19 05:54:18 +01:00
|
|
|
}
|
2022-05-05 15:50:44 +02:00
|
|
|
return Ok(());
|
2022-03-11 22:13:54 +01:00
|
|
|
} else {
|
|
|
|
let keylog = matches.value_of("keylog");
|
|
|
|
// When passed an empty string (i.e no value) take the value from the env
|
|
|
|
let keylog = if let Some("") = keylog {
|
|
|
|
let v = env::var_os("SSLKEYLOGFILE").map(PathBuf::from);
|
|
|
|
if v.is_none() || v.as_ref().unwrap().is_empty() {
|
|
|
|
eprintln!("--tls-key-log set but no path configured!");
|
|
|
|
return Ok(());
|
2021-10-20 12:58:05 +02:00
|
|
|
}
|
2022-03-11 22:13:54 +01:00
|
|
|
v
|
2021-10-20 18:37:50 +02:00
|
|
|
} else {
|
2022-03-11 22:13:54 +01:00
|
|
|
keylog.map(PathBuf::from)
|
|
|
|
};
|
2021-10-19 11:16:24 +02:00
|
|
|
|
2022-03-11 22:13:54 +01:00
|
|
|
config.tlskeylog = keylog;
|
|
|
|
config.verbosity = matches.occurrences_of("verbosity") as isize;
|
|
|
|
if config.verbosity == 0 && matches.is_present("quiet") {
|
|
|
|
config.verbosity = -1;
|
2021-10-20 18:37:50 +02:00
|
|
|
}
|
2022-03-20 21:22:15 +01:00
|
|
|
config.logging.format = matches.value_of("log format").unwrap_or("full").to_string();
|
2022-03-11 22:13:54 +01:00
|
|
|
|
2022-03-15 19:13:55 +01:00
|
|
|
let mut bffh = Diflouroborane::new(config)?;
|
|
|
|
bffh.run()?;
|
2022-03-11 22:13:54 +01:00
|
|
|
}
|
2021-10-20 12:58:05 +02:00
|
|
|
|
|
|
|
Ok(())
|
2022-03-11 22:13:54 +01:00
|
|
|
}
|