mirror of https://github.com/breard-r/acmed.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
155 lines
4.9 KiB
155 lines
4.9 KiB
use acme_common::error::Error;
|
|
use clap::{App, Arg, ArgMatches};
|
|
use log::{debug, error, info};
|
|
use std::fs::File;
|
|
use std::io::{self, Read};
|
|
|
|
mod certificate;
|
|
mod server;
|
|
|
|
const APP_NAME: &str = env!("CARGO_PKG_NAME");
|
|
const APP_VERSION: &str = env!("CARGO_PKG_VERSION");
|
|
const APP_ORG: &str = "ACMEd";
|
|
const DEFAULT_PID_FILE: &str = "/var/run/admed.pid";
|
|
const DEFAULT_LISTEN_ADDR: &str = "127.0.0.1:5001";
|
|
const ALPN_ACME_PROTO_NAME: &[u8] = b"\x0aacme-tls/1";
|
|
|
|
fn read_line(path: Option<&str>) -> Result<String, Error> {
|
|
let mut input = String::new();
|
|
match path {
|
|
Some(p) => File::open(p)?.read_to_string(&mut input)?,
|
|
None => io::stdin().read_line(&mut input)?,
|
|
};
|
|
let line = input.trim().to_string();
|
|
Ok(line)
|
|
}
|
|
|
|
fn get_acme_value(cnf: &ArgMatches, opt: &str, opt_file: &str) -> Result<String, Error> {
|
|
match cnf.value_of(opt) {
|
|
Some(v) => Ok(v.to_string()),
|
|
None => {
|
|
debug!(
|
|
"Reading {} from {}",
|
|
opt,
|
|
cnf.value_of(opt_file).unwrap_or("stdin")
|
|
);
|
|
read_line(cnf.value_of(opt_file))
|
|
}
|
|
}
|
|
}
|
|
|
|
fn init(cnf: &ArgMatches) -> Result<(), Error> {
|
|
acme_common::init_server(
|
|
cnf.is_present("foregroung"),
|
|
cnf.value_of("pid-file").unwrap_or(DEFAULT_PID_FILE),
|
|
);
|
|
let domain = get_acme_value(cnf, "domain", "domain-file")?;
|
|
let ext = get_acme_value(cnf, "acme-ext", "acme-ext-file")?;
|
|
let listen_addr = cnf.value_of("listen").unwrap_or(DEFAULT_LISTEN_ADDR);
|
|
let (pk, cert) = certificate::gen_certificate(&domain, &ext)?;
|
|
info!("Starting {} on {} for {}", APP_NAME, listen_addr, domain);
|
|
server::start(listen_addr, &cert, &pk)?;
|
|
Ok(())
|
|
}
|
|
|
|
fn main() {
|
|
let matches = App::new(APP_NAME)
|
|
.version(APP_VERSION)
|
|
.arg(
|
|
Arg::with_name("listen")
|
|
.long("listen")
|
|
.short("l")
|
|
.help("Specifies the host and port to listen on")
|
|
.takes_value(true)
|
|
.value_name("host:port|unix:path"),
|
|
)
|
|
.arg(
|
|
Arg::with_name("domain")
|
|
.long("domain")
|
|
.short("d")
|
|
.help("The domain that is being validated")
|
|
.takes_value(true)
|
|
.value_name("STRING")
|
|
.conflicts_with("domain-file")
|
|
)
|
|
.arg(
|
|
Arg::with_name("domain-file")
|
|
.long("domain-file")
|
|
.help("File from which is read the domain that is being validated")
|
|
.takes_value(true)
|
|
.value_name("FILE")
|
|
.conflicts_with("domain")
|
|
)
|
|
.arg(
|
|
Arg::with_name("acme-ext")
|
|
.long("acme-ext")
|
|
.short("e")
|
|
.help("The acmeIdentifier extension to set in the self-signed certificate")
|
|
.takes_value(true)
|
|
.value_name("STRING")
|
|
.conflicts_with("acme-ext-file")
|
|
)
|
|
.arg(
|
|
Arg::with_name("acme-ext-file")
|
|
.long("acme-ext-file")
|
|
.help("File from which is read the acmeIdentifier extension to set in the self-signed certificate")
|
|
.takes_value(true)
|
|
.value_name("FILE")
|
|
.conflicts_with("acme-ext-file")
|
|
)
|
|
.arg(
|
|
Arg::with_name("log-level")
|
|
.long("log-level")
|
|
.help("Specify the log level")
|
|
.takes_value(true)
|
|
.value_name("LEVEL")
|
|
.possible_values(&["error", "warn", "info", "debug", "trace"]),
|
|
)
|
|
.arg(
|
|
Arg::with_name("to-syslog")
|
|
.long("log-syslog")
|
|
.help("Sends log messages via syslog")
|
|
.conflicts_with("to-stderr"),
|
|
)
|
|
.arg(
|
|
Arg::with_name("to-stderr")
|
|
.long("log-stderr")
|
|
.help("Prints log messages to the standard error output")
|
|
.conflicts_with("log-syslog"),
|
|
)
|
|
.arg(
|
|
Arg::with_name("foregroung")
|
|
.long("foregroung")
|
|
.short("f")
|
|
.help("Runs in the foregroung"),
|
|
)
|
|
.arg(
|
|
Arg::with_name("pid-file")
|
|
.long("pid-file")
|
|
.help("Specifies the location of the PID file")
|
|
.takes_value(true)
|
|
.value_name("FILE")
|
|
.conflicts_with("foregroung"),
|
|
)
|
|
.get_matches();
|
|
|
|
match acme_common::logs::set_log_system(
|
|
matches.value_of("log-level"),
|
|
matches.is_present("log-syslog"),
|
|
matches.is_present("to-stderr"),
|
|
) {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
eprintln!("Error: {}", e);
|
|
std::process::exit(2);
|
|
}
|
|
};
|
|
|
|
match init(&matches) {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
error!("{}", e);
|
|
std::process::exit(1);
|
|
}
|
|
};
|
|
}
|