当 运行 在 Docker 容器中时,为什么我的 Go Web 服务器 return “找不到 404 页面”?

Why does my Go web server return "404 page not found" when run inside a Docker container?

我正在用 Go 构建一个小型的基本 Web 服务器。如果我在本地编译并 运行 它,效果很好——没问题。页面显示,它可以从本地主机访问,样式完好无损 - 一切都很好。

如果我随后在 Docker 容器中执行此操作,则它不起作用。它 return 是“找不到 404 页面”。好像它没有任何静态资产……但这肯定不可能——静态资产是有意使用“//go:embed”嵌入到二进制文件中的……正如我所说,如果您在本地构建和 运行 它工作正常。

我已经尝试了所有我能想到的...下面列出了一些步骤:

  1. 各种不同的 docker 图像(alpine、ubuntu、golang、golang alpine 等)
  2. 以不同的方式使用 go:embed,即不同的模式
  3. 我在服务器代码中添加了一些基本的登录...但似乎return 没有任何错误,所以没有帮助
  4. 已更改 ports/address

这只是我尝试过的一些方法,没有成功。

我已经排除了下面的 css,它并不真正相关,因为索引页甚至没有显示任何样式。

代码:server code

DOCKERFILE:

FROM golang:1.16.0-alpine3.13 AS build

WORKDIR /app

COPY . .

RUN go build -o server .

FROM golang:1.16.0-alpine3.13

WORKDIR /app

COPY --from=build /app/server .

EXPOSE 8080

CMD ["./server"]

HTML:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>GOOO</title>
    <link rel="stylesheet" href="/assets/static/css/style.css">
</head>
<body>
    <h1>GO!!!</h1>
</body>
</html>

目录结构:

├── Dockerfile
├── go.mod
├── go.sum
├── server.go
├── server_test.go
├── static
│   ├── css
│   │   └── style.css
│   └── index.html

根据评论,您遇到的主要问题是您从文件系统(不是嵌入式文件系统)提供 index.html 服务,而该文件不存在。

第二个问题是嵌入式文件系统将包含一个目录 static 所以你需要使用像 s, err := fs.Sub(static, "static") 这样的东西,这样 s.Open("index.html") 才能工作(否则你需要 static.Open("static/index.html") - 这适用于您的 http.FileServer 以及服务 index.html).

注意:您可能不需要以下内容,因为您可以 运行 http.FileServer 作为 / 路径(因此它服务于 index.html 和子目录中的文件).如果 url.

中没有提供文件名,http.FileServer 将自动提供 index.html

要从嵌入式文件系统提供 index.html,您可以将函数重写为(未经测试!):

// default/root handler which serves the index page and associated styling
func indexHandler(w http.ResponseWriter, r *http.Request) {
    f, err := s.Open("index.html") // Using `s` from above and assumes its global; better to use handlerfunc and pass filesystem in

    if err != nil {
        // Send whatever error you want (as file is embedded open should never fail)
        return
    }
    defer f.Close()

    w.Header().Set("Content-Type", "text/html")
    if _, err := io.Copy(w, f); err != nil { // Write out the file
        // Handle error
    }
}

上面依赖于一个全局变量(我不喜欢)所以我将其转换成类似的东西:

func IndexHandlerFunc(fs fs.FS, h http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        f, err := fs.Open("index.html")

        if err != nil {
            // Send whatever error you want (as file is embedded open should never fail)
            return
        }
        defer f.Close()

        w.Header().Set("Content-Type", "text/html")
        if _, err := io.Copy(w, f); err != nil { // Write out the file
            // Handle error
        }
    })
}