From d208b5cb6ba4331d7f1c58275a579316bb325b80 Mon Sep 17 00:00:00 2001 From: sigoden Date: Thu, 2 Jun 2022 08:32:31 +0800 Subject: [PATCH] feat: support path prefix --- src/args.rs | 9 +++++++++ src/server.rs | 17 ++++++++++++++++- 2 files changed, 25 insertions(+), 1 deletion(-) diff --git a/src/args.rs b/src/args.rs index c419329..eb1b3e0 100644 --- a/src/args.rs +++ b/src/args.rs @@ -33,6 +33,12 @@ fn app() -> clap::Command<'static> { .allow_invalid_utf8(true) .help("Path to a root directory for serving files"), ) + .arg( + Arg::new("path-prefix") + .long("path-prefix") + .value_name("path") + .help("Specify an url path prefix"), + ) .arg( Arg::new("allow-all") .short('A') @@ -92,6 +98,7 @@ pub struct Args { pub address: String, pub port: u16, pub path: PathBuf, + pub path_prefix: Option, pub auth: Option, pub no_auth_read: bool, pub allow_upload: bool, @@ -111,6 +118,7 @@ impl Args { let address = matches.value_of("address").unwrap_or_default().to_owned(); let port = matches.value_of_t::("port")?; let path = Args::parse_path(matches.value_of_os("path").unwrap_or_default())?; + let path_prefix = matches.value_of("path-prefix").map(|v| v.to_owned()); let cors = matches.is_present("cors"); let auth = matches.value_of("auth").map(|v| v.to_owned()); let no_auth_read = matches.is_present("no-auth-read"); @@ -124,6 +132,7 @@ impl Args { address, port, path, + path_prefix, auth, no_auth_read, cors, diff --git a/src/server.rs b/src/server.rs index 4e37c11..17f20c3 100644 --- a/src/server.rs +++ b/src/server.rs @@ -506,7 +506,22 @@ impl InnerService { } else { decoded_path.into_owned() }; - Some(self.args.path.join(&slashes_switched)) + let stripped_path = match self.strip_path_prefix(&slashes_switched) { + Some(path) => path, + None => return None, + }; + Some(self.args.path.join(&stripped_path)) + } + + fn strip_path_prefix<'a, P: AsRef>(&self, path: &'a P) -> Option<&'a Path> { + let path = path.as_ref(); + match self.args.path_prefix.as_deref() { + Some(prefix) => { + let prefix = prefix.trim_start_matches('/'); + path.strip_prefix(prefix).ok() + } + None => Some(path), + } } async fn to_pathitem>(