use s3s_fs::FileSystem;
use s3s_fs::Result;
use s3s::auth::SimpleAuth;
use s3s::host::MultiDomain;
use s3s::service::S3ServiceBuilder;
use std::io::IsTerminal;
use std::ops::Not;
use std::path::PathBuf;
use tokio::net::TcpListener;
use clap::{CommandFactory, Parser};
use tracing::info;
use hyper_util::rt::{TokioExecutor, TokioIo};
use hyper_util::server::conn::auto::Builder as ConnBuilder;
#[derive(Debug, Parser)]
#[command(version)]
struct Opt {
#[arg(long, default_value = "localhost")]
host: String,
#[arg(long, default_value = "8014")] port: u16,
#[arg(long)]
access_key: Option<String>,
#[arg(long)]
secret_key: Option<String>,
#[arg(long)]
domain: Vec<String>,
root: PathBuf,
}
fn setup_tracing() {
use tracing_subscriber::EnvFilter;
let env_filter = EnvFilter::from_default_env();
let enable_color = std::io::stdout().is_terminal();
tracing_subscriber::fmt()
.pretty()
.with_env_filter(env_filter)
.with_ansi(enable_color)
.init();
}
fn check_cli_args(opt: &Opt) {
use clap::error::ErrorKind;
let mut cmd = Opt::command();
if let (Some(_), None) | (None, Some(_)) = (&opt.access_key, &opt.secret_key) {
let msg = "access key and secret key must be specified together";
cmd.error(ErrorKind::MissingRequiredArgument, msg).exit();
}
for s in &opt.domain {
if s.contains('/') {
let msg = format!("expected domain name, found URL-like string: {s:?}");
cmd.error(ErrorKind::InvalidValue, msg).exit();
}
}
}
fn main() -> Result {
let opt = Opt::parse();
check_cli_args(&opt);
setup_tracing();
run(opt)
}
#[tokio::main]
async fn run(opt: Opt) -> Result {
let fs = FileSystem::new(opt.root)?;
let service = {
let mut b = S3ServiceBuilder::new(fs);
if let (Some(ak), Some(sk)) = (opt.access_key, opt.secret_key) {
b.set_auth(SimpleAuth::from_single(ak, sk));
info!("authentication is enabled");
}
if opt.domain.is_empty().not() {
b.set_host(MultiDomain::new(&opt.domain)?);
info!("virtual-hosted-style requests are enabled");
}
b.build()
};
let listener = TcpListener::bind((opt.host.as_str(), opt.port)).await?;
let local_addr = listener.local_addr()?;
let http_server = ConnBuilder::new(TokioExecutor::new());
let graceful = hyper_util::server::graceful::GracefulShutdown::new();
let mut ctrl_c = std::pin::pin!(tokio::signal::ctrl_c());
info!("server is running at http://{local_addr}");
loop {
let (socket, _) = tokio::select! {
res = listener.accept() => {
match res {
Ok(conn) => conn,
Err(err) => {
tracing::error!("error accepting connection: {err}");
continue;
}
}
}
_ = ctrl_c.as_mut() => {
break;
}
};
let conn = http_server.serve_connection(TokioIo::new(socket), service.clone());
let conn = graceful.watch(conn.into_owned());
tokio::spawn(async move {
let _ = conn.await;
});
}
tokio::select! {
() = graceful.shutdown() => {
tracing::debug!("Gracefully shutdown!");
},
() = tokio::time::sleep(std::time::Duration::from_secs(10)) => {
tracing::debug!("Waited 10 seconds for graceful shutdown, aborting...");
}
}
info!("server is stopped");
Ok(())
}