在 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)
    }