在 Go 中使用 http.FileServer 禁用目录列表的好方法
Good way to disable directory listing with http.FileServer in Go
如果你在 Go 中使用 http.FileServer 像:
func main() {
port := flag.String("p", "8100", "port to serve on")
directory := flag.String("d", ".", "the directory of static file to host")
flag.Parse()
http.Handle("/", http.FileServer(http.Dir(*directory)))
log.Printf("Serving %s on HTTP port: %s\n", *directory, *port)
log.Fatal(http.ListenAndServe(":"+*port, nil))
}
然后访问一个目录会给你一个文件列表。通常这对于 Web 服务是禁用的,而是以 404 响应,我也想要这种行为。
http.FileServer 没有这个 AFAIK 的选项,我在这里看到了解决这个问题的建议方法 https://groups.google.com/forum/#!topic/golang-nuts/bStLPdIVM6w 他们所做的是包装 http.FileSystem 类型并实现自己的 Open 方法.然而,当路径是一个目录时,这不会给出 404,它只会给出一个空白页面,并且不清楚如何修改它以适应这个。他们是这样做的:
type justFilesFilesystem struct {
fs http.FileSystem
}
func (fs justFilesFilesystem) Open(name string) (http.File, error) {
f, err := fs.fs.Open(name)
if err != nil {
return nil, err
}
return neuteredReaddirFile{f}, nil
}
type neuteredReaddirFile struct {
http.File
}
func (f neuteredReaddirFile) Readdir(count int) ([]os.FileInfo, error) {
return nil, nil
}
func main() {
fs := justFilesFilesystem{http.Dir("/tmp/")}
http.ListenAndServe(":8080", http.FileServer(fs))
}
注意:如果您创建 Readdir return nil, os.ErrNotExist
,那么您会收到 500 响应 "Error reading directory" - 而不是 404。
关于如何巧妙地呈现 404 并仍然保留自动查找 index.html(如果存在)的功能的任何想法?
如果您替换的不是 Readdir
方法,而是 Stat
.
,则可以更改此行为
请查看下面的工作代码。它支持服务 index.html
文件,如果它们在请求的目录和 returns 404
中,如果没有 index.html
并且它是一个目录。
package main
import (
"io"
"net/http"
"os"
)
type justFilesFilesystem struct {
fs http.FileSystem
// readDirBatchSize - configuration parameter for `Readdir` func
readDirBatchSize int
}
func (fs justFilesFilesystem) Open(name string) (http.File, error) {
f, err := fs.fs.Open(name)
if err != nil {
return nil, err
}
return neuteredStatFile{File: f, readDirBatchSize: fs.readDirBatchSize}, nil
}
type neuteredStatFile struct {
http.File
readDirBatchSize int
}
func (e neuteredStatFile) Stat() (os.FileInfo, error) {
s, err := e.File.Stat()
if err != nil {
return nil, err
}
if s.IsDir() {
LOOP:
for {
fl, err := e.File.Readdir(e.readDirBatchSize)
switch err {
case io.EOF:
break LOOP
case nil:
for _, f := range fl {
if f.Name() == "index.html" {
return s, err
}
}
default:
return nil, err
}
}
return nil, os.ErrNotExist
}
return s, err
}
func main() {
fs := justFilesFilesystem{fs: http.Dir("/tmp/"), readDirBatchSize: 2}
fss := http.FileServer(fs)
http.ListenAndServe(":8080", fss)
}
如果你在 Go 中使用 http.FileServer 像:
func main() {
port := flag.String("p", "8100", "port to serve on")
directory := flag.String("d", ".", "the directory of static file to host")
flag.Parse()
http.Handle("/", http.FileServer(http.Dir(*directory)))
log.Printf("Serving %s on HTTP port: %s\n", *directory, *port)
log.Fatal(http.ListenAndServe(":"+*port, nil))
}
然后访问一个目录会给你一个文件列表。通常这对于 Web 服务是禁用的,而是以 404 响应,我也想要这种行为。
http.FileServer 没有这个 AFAIK 的选项,我在这里看到了解决这个问题的建议方法 https://groups.google.com/forum/#!topic/golang-nuts/bStLPdIVM6w 他们所做的是包装 http.FileSystem 类型并实现自己的 Open 方法.然而,当路径是一个目录时,这不会给出 404,它只会给出一个空白页面,并且不清楚如何修改它以适应这个。他们是这样做的:
type justFilesFilesystem struct {
fs http.FileSystem
}
func (fs justFilesFilesystem) Open(name string) (http.File, error) {
f, err := fs.fs.Open(name)
if err != nil {
return nil, err
}
return neuteredReaddirFile{f}, nil
}
type neuteredReaddirFile struct {
http.File
}
func (f neuteredReaddirFile) Readdir(count int) ([]os.FileInfo, error) {
return nil, nil
}
func main() {
fs := justFilesFilesystem{http.Dir("/tmp/")}
http.ListenAndServe(":8080", http.FileServer(fs))
}
注意:如果您创建 Readdir return nil, os.ErrNotExist
,那么您会收到 500 响应 "Error reading directory" - 而不是 404。
关于如何巧妙地呈现 404 并仍然保留自动查找 index.html(如果存在)的功能的任何想法?
如果您替换的不是 Readdir
方法,而是 Stat
.
,则可以更改此行为
请查看下面的工作代码。它支持服务 index.html
文件,如果它们在请求的目录和 returns 404
中,如果没有 index.html
并且它是一个目录。
package main
import (
"io"
"net/http"
"os"
)
type justFilesFilesystem struct {
fs http.FileSystem
// readDirBatchSize - configuration parameter for `Readdir` func
readDirBatchSize int
}
func (fs justFilesFilesystem) Open(name string) (http.File, error) {
f, err := fs.fs.Open(name)
if err != nil {
return nil, err
}
return neuteredStatFile{File: f, readDirBatchSize: fs.readDirBatchSize}, nil
}
type neuteredStatFile struct {
http.File
readDirBatchSize int
}
func (e neuteredStatFile) Stat() (os.FileInfo, error) {
s, err := e.File.Stat()
if err != nil {
return nil, err
}
if s.IsDir() {
LOOP:
for {
fl, err := e.File.Readdir(e.readDirBatchSize)
switch err {
case io.EOF:
break LOOP
case nil:
for _, f := range fl {
if f.Name() == "index.html" {
return s, err
}
}
default:
return nil, err
}
}
return nil, os.ErrNotExist
}
return s, err
}
func main() {
fs := justFilesFilesystem{fs: http.Dir("/tmp/"), readDirBatchSize: 2}
fss := http.FileServer(fs)
http.ListenAndServe(":8080", fss)
}