首页 > 解决方案 > HTTP 静态文件处理程序不断显示目录列表

问题描述

当我实现一个静态服务器处理程序时,如果我访问根路径,它将显示整个目录,如下所示:

访问根文件夹

我的代码是:

package main

import (
    "flag"
    "log"
    "net/http"
    "strings"
)

func main() {
    port := flag.String("p", "3000", "port to serve on")
    directory := flag.String("d", ".", "the directory of static file to host")
    flag.Parse()

    http.Handle("/statics/", http.StripPrefix(strings.TrimRight("/statics/", "/"), http.FileServer(http.Dir(*directory))))

    log.Printf("Serving %s on HTTP port: %s\n", *directory, *port)
    log.Fatal(http.ListenAndServe(":"+*port, nil))
}

转到:http://locahost:3000/statics/

标签: httpgostatic-files

解决方案


幸运与否,默认行为http.FileServer是,如果路径表示没有index.html文件的目录,则将提供目录列表。而且它没有提供一种简单的方法来关闭它。但...

http.FileServer在接口描述的虚拟文件系统上运行。http.FileSystem

这个接口有一个方法告诉如何打开一个文件,并获得它的http.File“视图”:

type FileSystem interface {
        Open(name string) (File, error)
}

如果你想禁用目录列表,你所要做的就是提供你自己的实现http.FileSystem,当一个目录被定位时,你只需报告/返回一个错误。这就是它所需要的。

当然,您不必自己做这一切。您可以创建自己的FileSystem使用 / 利用http.Dir,这是使用本机文件系统的默认实现(仅限于特定的目录树)。

type myfs struct {
    http.Dir
}

func (m myfs) Open(name string) (result http.File, err error) {
    f, err := m.Dir.Open(name)
    if err != nil {
        return
    }

    fi, err := f.Stat()
    if err != nil {
        return
    }
    if fi.IsDir() {
        // Return a response that would have been if directory would not exist:
        return m.Dir.Open("does-not-exist")
    }
    return f, nil
}

使用上面的自定义实现:

handler := http.FileServer(myfs{http.Dir(*directory)})
http.Handle(
    "/statics/",
    http.StripPrefix(strings.TrimRight("/statics/", "/"), handler),
)

就这样。尝试浏览http://locahost:3000/statics/将导致默认响应:

404 page not found

笔记:

上述实现执行第二次Dir.Open()调用以返回错误,这始终是相同的。为了“加快速度”,我们可以存储这个响应并重用它:

var notFoundFile, notFoundErr = http.Dir("dummy").Open("does-not-exist")

当我们在myfs.Open()方法中检测到目录时:

if fi.IsDir() {
    // Return a response that "belogns" to opening a non-existing folder:
    return notFoundFile, notFoundErr
}

推荐阅读