2022-05-26 11:17:55 +03:00
|
|
|
use clap::crate_description;
|
|
|
|
use clap::{Arg, ArgMatches};
|
2022-06-02 06:06:41 +03:00
|
|
|
use rustls::{Certificate, PrivateKey};
|
2022-05-26 11:17:55 +03:00
|
|
|
use std::net::SocketAddr;
|
|
|
|
use std::path::{Path, PathBuf};
|
2022-06-02 06:06:41 +03:00
|
|
|
use std::{env, fs, io};
|
2022-05-26 11:17:55 +03:00
|
|
|
|
|
|
|
use crate::BoxResult;
|
|
|
|
|
|
|
|
const ABOUT: &str = concat!("\n", crate_description!()); // Add extra newline.
|
|
|
|
|
|
|
|
fn app() -> clap::Command<'static> {
|
|
|
|
clap::command!()
|
|
|
|
.about(ABOUT)
|
2022-05-29 12:01:30 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("address")
|
|
|
|
.short('b')
|
|
|
|
.long("bind")
|
2022-06-03 06:18:46 +03:00
|
|
|
.default_value("0.0.0.0")
|
2022-05-29 12:01:30 +03:00
|
|
|
.help("Specify bind address")
|
|
|
|
.value_name("address"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("port")
|
|
|
|
.short('p')
|
|
|
|
.long("port")
|
|
|
|
.default_value("5000")
|
|
|
|
.help("Specify port to listen on")
|
|
|
|
.value_name("port"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("path")
|
|
|
|
.default_value(".")
|
|
|
|
.allow_invalid_utf8(true)
|
2022-05-31 15:53:14 +03:00
|
|
|
.help("Path to a root directory for serving files"),
|
2022-05-29 12:01:30 +03:00
|
|
|
)
|
2022-06-02 03:32:31 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("path-prefix")
|
|
|
|
.long("path-prefix")
|
|
|
|
.value_name("path")
|
|
|
|
.help("Specify an url path prefix"),
|
|
|
|
)
|
2022-05-29 12:01:30 +03:00
|
|
|
.arg(
|
2022-05-31 03:38:30 +03:00
|
|
|
Arg::new("allow-all")
|
|
|
|
.short('A')
|
|
|
|
.long("allow-all")
|
|
|
|
.help("Allow all operations"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("allow-upload")
|
|
|
|
.long("allow-upload")
|
2022-06-02 14:32:19 +03:00
|
|
|
.help("Allow upload files/folders"),
|
2022-05-31 03:38:30 +03:00
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("allow-delete")
|
2022-05-31 17:38:00 +03:00
|
|
|
.long("allow-delete")
|
2022-06-02 14:32:19 +03:00
|
|
|
.help("Allow delete files/folders"),
|
2022-05-29 12:01:30 +03:00
|
|
|
)
|
2022-05-31 15:53:14 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("allow-symlink")
|
2022-05-31 17:38:00 +03:00
|
|
|
.long("allow-symlink")
|
2022-06-02 14:32:19 +03:00
|
|
|
.help("Allow symlink to files/folders outside root directory"),
|
2022-05-31 15:53:14 +03:00
|
|
|
)
|
2022-06-01 17:49:55 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("render-index")
|
|
|
|
.long("render-index")
|
2022-06-02 12:06:22 +03:00
|
|
|
.help("Render index.html when requesting a directory"),
|
2022-06-01 17:49:55 +03:00
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("render-spa")
|
|
|
|
.long("render-spa")
|
2022-06-02 12:06:22 +03:00
|
|
|
.help("Render for single-page application"),
|
2022-06-01 17:49:55 +03:00
|
|
|
)
|
2022-05-29 12:01:30 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("auth")
|
|
|
|
.short('a')
|
2022-06-02 14:32:19 +03:00
|
|
|
.display_order(1)
|
2022-05-29 12:01:30 +03:00
|
|
|
.long("auth")
|
2022-06-03 01:51:03 +03:00
|
|
|
.help("Use HTTP authentication")
|
2022-05-29 12:01:30 +03:00
|
|
|
.value_name("user:pass"),
|
|
|
|
)
|
2022-05-30 07:40:57 +03:00
|
|
|
.arg(
|
2022-06-03 01:51:03 +03:00
|
|
|
Arg::new("no-auth-access")
|
2022-06-02 14:32:19 +03:00
|
|
|
.display_order(1)
|
2022-06-03 01:51:03 +03:00
|
|
|
.long("no-auth-access")
|
|
|
|
.help("Not required auth when access static files"),
|
2022-05-30 07:40:57 +03:00
|
|
|
)
|
2022-05-29 12:33:21 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("cors")
|
|
|
|
.long("cors")
|
|
|
|
.help("Enable CORS, sets `Access-Control-Allow-Origin: *`"),
|
|
|
|
)
|
2022-06-02 06:06:41 +03:00
|
|
|
.arg(
|
|
|
|
Arg::new("tls-cert")
|
|
|
|
.long("tls-cert")
|
|
|
|
.value_name("path")
|
|
|
|
.help("Path to an SSL/TLS certificate to serve with HTTPS"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::new("tls-key")
|
|
|
|
.long("tls-key")
|
|
|
|
.value_name("path")
|
|
|
|
.help("Path to the SSL/TLS certificate's private key"),
|
|
|
|
)
|
2022-05-26 11:17:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn matches() -> ArgMatches {
|
|
|
|
app().get_matches()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Eq, PartialEq)]
|
|
|
|
pub struct Args {
|
|
|
|
pub address: String,
|
|
|
|
pub port: u16,
|
|
|
|
pub path: PathBuf,
|
2022-06-04 14:08:18 +03:00
|
|
|
pub path_prefix: String,
|
|
|
|
pub uri_prefix: String,
|
2022-05-26 13:06:52 +03:00
|
|
|
pub auth: Option<String>,
|
2022-06-03 01:51:03 +03:00
|
|
|
pub no_auth_access: bool,
|
2022-05-31 03:38:30 +03:00
|
|
|
pub allow_upload: bool,
|
|
|
|
pub allow_delete: bool,
|
2022-05-31 15:53:14 +03:00
|
|
|
pub allow_symlink: bool,
|
2022-06-01 17:49:55 +03:00
|
|
|
pub render_index: bool,
|
|
|
|
pub render_spa: bool,
|
2022-05-29 12:33:21 +03:00
|
|
|
pub cors: bool,
|
2022-06-02 06:06:41 +03:00
|
|
|
pub tls: Option<(Vec<Certificate>, PrivateKey)>,
|
2022-05-26 11:17:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Args {
|
|
|
|
/// Parse command-line arguments.
|
|
|
|
///
|
|
|
|
/// If a parsing error ocurred, exit the process and print out informative
|
|
|
|
/// error message to user.
|
|
|
|
pub fn parse(matches: ArgMatches) -> BoxResult<Args> {
|
|
|
|
let address = matches.value_of("address").unwrap_or_default().to_owned();
|
|
|
|
let port = matches.value_of_t::<u16>("port")?;
|
2022-05-31 03:38:30 +03:00
|
|
|
let path = Args::parse_path(matches.value_of_os("path").unwrap_or_default())?;
|
2022-06-04 07:51:56 +03:00
|
|
|
let path_prefix = matches
|
|
|
|
.value_of("path-prefix")
|
2022-06-04 14:08:18 +03:00
|
|
|
.map(|v| v.trim_matches('/').to_owned())
|
|
|
|
.unwrap_or_default();
|
|
|
|
let uri_prefix = if path_prefix.is_empty() {
|
|
|
|
"/".to_owned()
|
|
|
|
} else {
|
|
|
|
format!("/{}/", &path_prefix)
|
|
|
|
};
|
2022-05-29 12:33:21 +03:00
|
|
|
let cors = matches.is_present("cors");
|
2022-05-26 13:06:52 +03:00
|
|
|
let auth = matches.value_of("auth").map(|v| v.to_owned());
|
2022-06-03 01:51:03 +03:00
|
|
|
let no_auth_access = matches.is_present("no-auth-access");
|
2022-05-31 03:38:30 +03:00
|
|
|
let allow_upload = matches.is_present("allow-all") || matches.is_present("allow-upload");
|
|
|
|
let allow_delete = matches.is_present("allow-all") || matches.is_present("allow-delete");
|
2022-05-31 15:53:14 +03:00
|
|
|
let allow_symlink = matches.is_present("allow-all") || matches.is_present("allow-symlink");
|
2022-06-01 17:49:55 +03:00
|
|
|
let render_index = matches.is_present("render-index");
|
|
|
|
let render_spa = matches.is_present("render-spa");
|
2022-06-02 06:06:41 +03:00
|
|
|
let tls = match (matches.value_of("tls-cert"), matches.value_of("tls-key")) {
|
|
|
|
(Some(certs_file), Some(key_file)) => {
|
|
|
|
let certs = load_certs(certs_file)?;
|
|
|
|
let key = load_private_key(key_file)?;
|
|
|
|
Some((certs, key))
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
};
|
2022-05-26 11:17:55 +03:00
|
|
|
|
|
|
|
Ok(Args {
|
|
|
|
address,
|
|
|
|
port,
|
|
|
|
path,
|
2022-06-02 03:32:31 +03:00
|
|
|
path_prefix,
|
2022-06-04 14:08:18 +03:00
|
|
|
uri_prefix,
|
2022-05-26 13:06:52 +03:00
|
|
|
auth,
|
2022-06-03 01:51:03 +03:00
|
|
|
no_auth_access,
|
2022-05-29 12:33:21 +03:00
|
|
|
cors,
|
2022-05-31 03:38:30 +03:00
|
|
|
allow_delete,
|
|
|
|
allow_upload,
|
2022-05-31 15:53:14 +03:00
|
|
|
allow_symlink,
|
2022-06-01 17:49:55 +03:00
|
|
|
render_index,
|
|
|
|
render_spa,
|
2022-06-02 06:06:41 +03:00
|
|
|
tls,
|
2022-05-26 11:17:55 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse path.
|
|
|
|
fn parse_path<P: AsRef<Path>>(path: P) -> BoxResult<PathBuf> {
|
|
|
|
let path = path.as_ref();
|
|
|
|
if !path.exists() {
|
2022-06-03 06:18:46 +03:00
|
|
|
return Err(format!("Path `{}` doesn't exist", path.display()).into());
|
2022-05-26 11:17:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
env::current_dir()
|
|
|
|
.and_then(|mut p| {
|
|
|
|
p.push(path); // If path is absolute, it replaces the current path.
|
2022-05-31 17:38:00 +03:00
|
|
|
std::fs::canonicalize(p)
|
2022-05-26 11:17:55 +03:00
|
|
|
})
|
2022-06-03 06:18:46 +03:00
|
|
|
.map_err(|err| format!("Failed to access path `{}`: {}", path.display(), err,).into())
|
2022-05-26 11:17:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Construct socket address from arguments.
|
|
|
|
pub fn address(&self) -> BoxResult<SocketAddr> {
|
|
|
|
format!("{}:{}", self.address, self.port)
|
|
|
|
.parse()
|
2022-06-03 06:18:46 +03:00
|
|
|
.map_err(|_| format!("Invalid bind address `{}:{}`", self.address, self.port).into())
|
2022-05-26 11:17:55 +03:00
|
|
|
}
|
|
|
|
}
|
2022-06-02 06:06:41 +03:00
|
|
|
|
|
|
|
// Load public certificate from file.
|
|
|
|
pub fn load_certs(filename: &str) -> BoxResult<Vec<Certificate>> {
|
|
|
|
// Open certificate file.
|
|
|
|
let certfile =
|
|
|
|
fs::File::open(&filename).map_err(|e| format!("Failed to open {}: {}", &filename, e))?;
|
|
|
|
let mut reader = io::BufReader::new(certfile);
|
|
|
|
|
|
|
|
// Load and return certificate.
|
|
|
|
let certs = rustls_pemfile::certs(&mut reader).map_err(|_| "Failed to load certificate")?;
|
|
|
|
if certs.is_empty() {
|
|
|
|
return Err("Expected at least one certificate".into());
|
|
|
|
}
|
|
|
|
Ok(certs.into_iter().map(Certificate).collect())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load private key from file.
|
|
|
|
pub fn load_private_key(filename: &str) -> BoxResult<PrivateKey> {
|
|
|
|
// Open keyfile.
|
|
|
|
let keyfile =
|
|
|
|
fs::File::open(&filename).map_err(|e| format!("Failed to open {}: {}", &filename, e))?;
|
|
|
|
let mut reader = io::BufReader::new(keyfile);
|
|
|
|
|
|
|
|
// Load and return a single private key.
|
|
|
|
let keys = rustls_pemfile::rsa_private_keys(&mut reader)
|
|
|
|
.map_err(|e| format!("There was a problem with reading private key: {:?}", e))?;
|
|
|
|
|
|
|
|
if keys.len() != 1 {
|
|
|
|
return Err("Expected a single private key".into());
|
|
|
|
}
|
|
|
|
Ok(PrivateKey(keys[0].to_owned()))
|
|
|
|
}
|