1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2025-01-24 16:08:06 +00:00

Allow to override files listing renderer for #203

This commit is contained in:
Nikolay Kim 2018-05-02 15:53:07 -07:00
parent 35a4078434
commit 32a2866449
3 changed files with 69 additions and 55 deletions

View file

@ -12,6 +12,8 @@
* Allow to access Error's backtrace object * Allow to access Error's backtrace object
* Allow to override files listing renderer for `StaticFiles` #203
* Various extractor usability improvements #207 * Various extractor usability improvements #207

View file

@ -365,11 +365,14 @@ impl Stream for ChunkedReadFile {
} }
} }
type DirectoryRenderer<S> =
Fn(&Directory, &HttpRequest<S>) -> Result<HttpResponse, io::Error>;
/// A directory; responds with the generated directory listing. /// A directory; responds with the generated directory listing.
#[derive(Debug)] #[derive(Debug)]
pub struct Directory { pub struct Directory {
base: PathBuf, pub base: PathBuf,
path: PathBuf, pub path: PathBuf,
} }
impl Directory { impl Directory {
@ -377,7 +380,7 @@ impl Directory {
Directory { base, path } Directory { base, path }
} }
fn can_list(&self, entry: &io::Result<DirEntry>) -> bool { pub fn is_visible(&self, entry: &io::Result<DirEntry>) -> bool {
if let Ok(ref entry) = *entry { if let Ok(ref entry) = *entry {
if let Some(name) = entry.file_name().to_str() { if let Some(name) = entry.file_name().to_str() {
if name.starts_with('.') { if name.starts_with('.') {
@ -393,19 +396,17 @@ impl Directory {
} }
} }
impl Responder for Directory { fn directory_listing<S>(
type Item = HttpResponse; dir: &Directory, req: &HttpRequest<S>,
type Error = io::Error; ) -> Result<HttpResponse, io::Error> {
fn respond_to(self, req: HttpRequest) -> Result<HttpResponse, io::Error> {
let index_of = format!("Index of {}", req.path()); let index_of = format!("Index of {}", req.path());
let mut body = String::new(); let mut body = String::new();
let base = Path::new(req.path()); let base = Path::new(req.path());
for entry in self.path.read_dir()? { for entry in dir.path.read_dir()? {
if self.can_list(&entry) { if dir.is_visible(&entry) {
let entry = entry.unwrap(); let entry = entry.unwrap();
let p = match entry.path().strip_prefix(&self.path) { let p = match entry.path().strip_prefix(&dir.path) {
Ok(p) => base.join(p), Ok(p) => base.join(p),
Err(_) => continue, Err(_) => continue,
}; };
@ -447,7 +448,6 @@ impl Responder for Directory {
Ok(HttpResponse::Ok() Ok(HttpResponse::Ok()
.content_type("text/html; charset=utf-8") .content_type("text/html; charset=utf-8")
.body(html)) .body(html))
}
} }
/// Static files handling /// Static files handling
@ -472,6 +472,7 @@ pub struct StaticFiles<S> {
show_index: bool, show_index: bool,
cpu_pool: CpuPool, cpu_pool: CpuPool,
default: Box<RouteHandler<S>>, default: Box<RouteHandler<S>>,
renderer: Box<DirectoryRenderer<S>>,
_chunk_size: usize, _chunk_size: usize,
_follow_symlinks: bool, _follow_symlinks: bool,
} }
@ -535,6 +536,7 @@ impl<S: 'static> StaticFiles<S> {
default: Box::new(WrapHandler::new(|_| { default: Box::new(WrapHandler::new(|_| {
HttpResponse::new(StatusCode::NOT_FOUND) HttpResponse::new(StatusCode::NOT_FOUND)
})), })),
renderer: Box::new(directory_listing),
_chunk_size: 0, _chunk_size: 0,
_follow_symlinks: false, _follow_symlinks: false,
} }
@ -548,6 +550,17 @@ impl<S: 'static> StaticFiles<S> {
self self
} }
/// Set custom directory renderer
pub fn files_listing_renderer<F>(mut self, f: F) -> Self
where
for<'r, 's> F: Fn(&'r Directory, &'s HttpRequest<S>)
-> Result<HttpResponse, io::Error>
+ 'static,
{
self.renderer = Box::new(f);
self
}
/// Set index file /// Set index file
/// ///
/// Redirects to specific index file for directory "/" instead of /// Redirects to specific index file for directory "/" instead of
@ -601,9 +614,8 @@ impl<S: 'static> Handler<S> for StaticFiles<S> {
.finish() .finish()
.respond_to(req.drop_state()) .respond_to(req.drop_state())
} else if self.show_index { } else if self.show_index {
Directory::new(self.directory.clone(), path) let dir = Directory::new(self.directory.clone(), path);
.respond_to(req.drop_state())? Ok((*self.renderer)(&dir, &req)?.into())
.respond_to(req.drop_state())
} else { } else {
Ok(self.default.handle(req)) Ok(self.default.handle(req))
} }

View file

@ -21,7 +21,7 @@ pub use self::logger::Logger;
/// Middleware start result /// Middleware start result
pub enum Started { pub enum Started {
/// Execution completed /// Middleware is completed, continue to next middleware
Done, Done,
/// New http response got generated. If middleware generates response /// New http response got generated. If middleware generates response
/// handler execution halts. /// handler execution halts.