当 运行 在 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”嵌入到二进制文件中的……正如我所说,如果您在本地构建和 运行 它工作正常。
我已经尝试了所有我能想到的...下面列出了一些步骤:
- 各种不同的 docker 图像(alpine、ubuntu、golang、golang alpine 等)
- 以不同的方式使用 go:embed,即不同的模式
- 我在服务器代码中添加了一些基本的登录...但似乎return 没有任何错误,所以没有帮助
- 已更改 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
}
})
}
我正在用 Go 构建一个小型的基本 Web 服务器。如果我在本地编译并 运行 它,效果很好——没问题。页面显示,它可以从本地主机访问,样式完好无损 - 一切都很好。
如果我随后在 Docker 容器中执行此操作,则它不起作用。它 return 是“找不到 404 页面”。好像它没有任何静态资产……但这肯定不可能——静态资产是有意使用“//go:embed”嵌入到二进制文件中的……正如我所说,如果您在本地构建和 运行 它工作正常。
我已经尝试了所有我能想到的...下面列出了一些步骤:
- 各种不同的 docker 图像(alpine、ubuntu、golang、golang alpine 等)
- 以不同的方式使用 go:embed,即不同的模式
- 我在服务器代码中添加了一些基本的登录...但似乎return 没有任何错误,所以没有帮助
- 已更改 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
}
})
}