use crate::{Args, BoxResult}; use async_walkdir::WalkDir; use async_zip::write::{EntryOptions, ZipFileWriter}; use async_zip::Compression; use futures::stream::StreamExt; use futures::TryStreamExt; use headers::{ AccessControlAllowHeaders, AccessControlAllowOrigin, ContentType, ETag, HeaderMapExt, IfModifiedSince, IfNoneMatch, LastModified, }; use hyper::header::{HeaderValue, ACCEPT, CONTENT_TYPE, ORIGIN, RANGE, WWW_AUTHENTICATE}; use hyper::service::{make_service_fn, service_fn}; use hyper::{Body, Method, StatusCode}; use percent_encoding::percent_decode; use serde::Serialize; use std::convert::Infallible; use std::path::{Path, PathBuf}; use std::sync::Arc; use std::time::SystemTime; use tokio::fs::File; use tokio::io::AsyncWrite; use tokio::{fs, io}; use tokio_util::codec::{BytesCodec, FramedRead}; use tokio_util::io::{ReaderStream, StreamReader}; type Request = hyper::Request; type Response = hyper::Response; macro_rules! status_code { ($status:expr) => { hyper::Response::builder() .status($status) .body($status.canonical_reason().unwrap_or_default().into()) .unwrap() }; } const INDEX_HTML: &str = include_str!("index.html"); const INDEX_CSS: &str = include_str!("index.css"); const BUF_SIZE: usize = 1024 * 16; pub async fn serve(args: Args) -> BoxResult<()> { let address = args.address()?; let inner = Arc::new(InnerService::new(args)); let make_svc = make_service_fn(move |_| { let inner = inner.clone(); async { Ok::<_, Infallible>(service_fn(move |req| { let inner = inner.clone(); inner.call(req) })) } }); let server = hyper::Server::try_bind(&address)?.serve(make_svc); let address = server.local_addr(); eprintln!("Files served on http://{}", address); server.await?; Ok(()) } struct InnerService { args: Args, } impl InnerService { pub fn new(args: Args) -> Self { Self { args } } pub async fn call(self: Arc, req: Request) -> Result { let method = req.method().clone(); let uri = req.uri().clone(); let cors = self.args.cors; let mut res = self .handle(req) .await .unwrap_or_else(|_| status_code!(StatusCode::INTERNAL_SERVER_ERROR)); info!(r#""{} {}" - {}"#, method, uri, res.status()); if cors { add_cors(&mut res); } Ok(res) } pub async fn handle(self: Arc, req: Request) -> BoxResult { if !self.auth_guard(&req).unwrap_or_default() { let mut res = status_code!(StatusCode::UNAUTHORIZED); res.headers_mut() .insert(WWW_AUTHENTICATE, HeaderValue::from_static("Basic")); return Ok(res); } match *req.method() { Method::GET => self.handle_static(req).await, Method::PUT => { if self.args.readonly { return Ok(status_code!(StatusCode::FORBIDDEN)); } self.handle_upload(req).await } Method::OPTIONS => Ok(status_code!(StatusCode::NO_CONTENT)), Method::DELETE => self.handle_delete(req).await, _ => Ok(status_code!(StatusCode::NOT_FOUND)), } } async fn handle_static(&self, req: Request) -> BoxResult { let req_path = req.uri().path(); let path = match self.get_file_path(req_path)? { Some(path) => path, None => return Ok(status_code!(StatusCode::FORBIDDEN)), }; match fs::metadata(&path).await { Ok(meta) => { if meta.is_dir() { let req_query = req.uri().query().unwrap_or_default(); if req_query == "zip" { return self.handle_send_dir_zip(path.as_path()).await; } if let Some(q) = req_query.strip_prefix("q=") { return self.handle_query_dir(path.as_path(), q).await; } self.handle_ls_dir(path.as_path(), true).await } else { self.handle_send_file(&req, path.as_path()).await } } Err(_) => { if req_path.ends_with('/') { self.handle_ls_dir(path.as_path(), false).await } else { Ok(status_code!(StatusCode::NOT_FOUND)) } } } } async fn handle_upload(&self, mut req: Request) -> BoxResult { let forbidden = status_code!(StatusCode::FORBIDDEN); let path = match self.get_file_path(req.uri().path())? { Some(path) => path, None => return Ok(forbidden), }; match path.parent() { Some(parent) => match fs::metadata(parent).await { Ok(meta) => { if !meta.is_dir() { return Ok(forbidden); } } Err(_) => fs::create_dir_all(parent).await?, }, None => return Ok(forbidden), } let mut file = fs::File::create(path).await?; let body_with_io_error = req .body_mut() .map_err(|err| io::Error::new(io::ErrorKind::Other, err)); let body_reader = StreamReader::new(body_with_io_error); futures::pin_mut!(body_reader); io::copy(&mut body_reader, &mut file).await?; return Ok(status_code!(StatusCode::OK)); } async fn handle_delete(&self, req: Request) -> BoxResult { let path = match self.get_file_path(req.uri().path())? { Some(path) => path, None => return Ok(status_code!(StatusCode::FORBIDDEN)), }; let meta = fs::metadata(&path).await?; if meta.is_file() { fs::remove_file(path).await?; } else { fs::remove_dir_all(path).await?; } Ok(status_code!(StatusCode::OK)) } async fn handle_ls_dir(&self, path: &Path, exist: bool) -> BoxResult { let mut paths: Vec = vec![]; if exist { let mut rd = fs::read_dir(path).await?; while let Some(entry) = rd.next_entry().await? { let entry_path = entry.path(); if let Ok(item) = get_path_item(entry_path, path.to_path_buf()).await { paths.push(item); } } } self.send_index(path, paths) } async fn handle_query_dir(&self, path: &Path, q: &str) -> BoxResult { let mut paths: Vec = vec![]; let mut walkdir = WalkDir::new(path); while let Some(entry) = walkdir.next().await { if let Ok(entry) = entry { if !entry .file_name() .to_string_lossy() .to_lowercase() .contains(&q.to_lowercase()) { continue; } if fs::symlink_metadata(entry.path()).await.is_err() { continue; } if let Ok(item) = get_path_item(entry.path(), path.to_path_buf()).await { paths.push(item); } } } self.send_index(path, paths) } async fn handle_send_dir_zip(&self, path: &Path) -> BoxResult { let (mut writer, reader) = tokio::io::duplex(BUF_SIZE); let path = path.to_owned(); tokio::spawn(async move { if let Err(e) = dir_zip(&mut writer, &path).await { error!("Fail to zip {}, {}", path.display(), e.to_string()); } }); let stream = ReaderStream::new(reader); let body = Body::wrap_stream(stream); Ok(Response::new(body)) } async fn handle_send_file(&self, req: &Request, path: &Path) -> BoxResult { let (file, meta) = tokio::join!(fs::File::open(path), fs::metadata(path),); let (file, meta) = (file?, meta?); let mut res = Response::default(); if let Ok(mtime) = meta.modified() { let mtime_value = get_timestamp(&mtime); let size = meta.len(); let etag = format!(r#""{}-{}""#, mtime_value, size) .parse::() .unwrap(); let last_modified = LastModified::from(mtime); let fresh = { // `If-None-Match` takes presedence over `If-Modified-Since`. if let Some(if_none_match) = req.headers().typed_get::() { !if_none_match.precondition_passes(&etag) } else if let Some(if_modified_since) = req.headers().typed_get::() { !if_modified_since.is_modified(mtime) } else { false } }; res.headers_mut().typed_insert(last_modified); res.headers_mut().typed_insert(etag); if fresh { *res.status_mut() = StatusCode::NOT_MODIFIED; return Ok(res); } } if let Some(mime) = mime_guess::from_path(&path).first() { res.headers_mut().typed_insert(ContentType::from(mime)); } let stream = FramedRead::new(file, BytesCodec::new()); let body = Body::wrap_stream(stream); *res.body_mut() = body; Ok(res) } fn send_index(&self, path: &Path, mut paths: Vec) -> BoxResult { paths.sort_unstable(); let breadcrumb = self.get_breadcrumb(path); let data = IndexData { breadcrumb, paths, readonly: self.args.readonly, }; let data = serde_json::to_string(&data).unwrap(); let mut output = INDEX_HTML.replace("__STYLE__", &format!("", INDEX_CSS)); output = output.replace("__DATA__", &data); Ok(Response::new(output.into())) } fn auth_guard(&self, req: &Request) -> BoxResult { if let Some(auth) = &self.args.auth { if let Some(value) = req.headers().get("Authorization") { let value = value.to_str()?; let value = if value.contains("Basic ") { &value[6..] } else { return Ok(false); }; let value = base64::decode(value)?; let value = std::str::from_utf8(&value)?; return Ok(value == auth); } else { return Ok(false); } } Ok(true) } fn get_breadcrumb(&self, path: &Path) -> String { let path = match self.args.path.parent() { Some(p) => path.strip_prefix(p).unwrap(), None => path, }; normalize_path(path) } fn get_file_path(&self, path: &str) -> BoxResult> { let decoded_path = percent_decode(path[1..].as_bytes()).decode_utf8()?; let slashes_switched = if cfg!(windows) { decoded_path.replace('/', "\\") } else { decoded_path.into_owned() }; let path = self.args.path.join(&slashes_switched); if path.starts_with(&self.args.path) { Ok(Some(path)) } else { Ok(None) } } } #[derive(Debug, Serialize, Eq, PartialEq, Ord, PartialOrd)] struct IndexData { breadcrumb: String, paths: Vec, readonly: bool, } #[derive(Debug, Serialize, Eq, PartialEq, Ord, PartialOrd)] struct PathItem { path_type: PathType, name: String, mtime: u64, size: Option, } #[derive(Debug, Serialize, Eq, PartialEq, Ord, PartialOrd)] enum PathType { Dir, SymlinkDir, File, SymlinkFile, } async fn get_path_item>(path: P, base_path: P) -> BoxResult { let path = path.as_ref(); let rel_path = path.strip_prefix(base_path).unwrap(); let meta = fs::metadata(&path).await?; let s_meta = fs::symlink_metadata(&path).await?; let is_dir = meta.is_dir(); let is_symlink = s_meta.file_type().is_symlink(); let path_type = match (is_symlink, is_dir) { (true, true) => PathType::SymlinkDir, (false, true) => PathType::Dir, (true, false) => PathType::SymlinkFile, (false, false) => PathType::File, }; let mtime = get_timestamp(&meta.modified()?); let size = match path_type { PathType::Dir | PathType::SymlinkDir => None, PathType::File | PathType::SymlinkFile => Some(meta.len()), }; let name = normalize_path(rel_path); Ok(PathItem { path_type, name, mtime, size, }) } fn get_timestamp(time: &SystemTime) -> u64 { time.duration_since(SystemTime::UNIX_EPOCH) .unwrap() .as_millis() as u64 } fn normalize_path>(path: P) -> String { let path = path.as_ref().to_str().unwrap_or_default(); if cfg!(windows) { path.replace('\\', "/") } else { path.to_string() } } fn add_cors(res: &mut Response) { res.headers_mut() .typed_insert(AccessControlAllowOrigin::ANY); res.headers_mut().typed_insert( vec![RANGE, CONTENT_TYPE, ACCEPT, ORIGIN, WWW_AUTHENTICATE] .into_iter() .collect::(), ); } async fn dir_zip(writer: &mut W, dir: &Path) -> BoxResult<()> { let mut writer = ZipFileWriter::new(writer); let mut walkdir = WalkDir::new(dir); while let Some(entry) = walkdir.next().await { if let Ok(entry) = entry { let meta = match fs::symlink_metadata(entry.path()).await { Ok(meta) => meta, Err(_) => continue, }; if meta.is_file() { let filepath = entry.path(); let filename = match filepath.strip_prefix(dir).ok().and_then(|v| v.to_str()) { Some(v) => v, None => continue, }; let entry_options = EntryOptions::new(filename.to_owned(), Compression::Deflate); let mut file = File::open(&filepath).await?; let mut file_writer = writer.write_entry_stream(entry_options).await?; io::copy(&mut file, &mut file_writer).await?; file_writer.close().await?; } } } writer.close().await?; Ok(()) }