我正在尝试为我的 go 服务器创建一个 dockerfile,但它一直失败,因为它无法识别一些本地依赖项(它们是代码本身的模块,而不是外部依赖项)。
例子:
import (
"<private-repo-url>/src/cmd/http-api/bootstrap" // this a local module that's part of the server
"go.uber.org/fx"
)
func main() {
fx.New(bootstrap.Module).Run()
}
这是错误:
=> ERROR [7/7] RUN go build -a -o ./server 0.3s
------
> [7/7] RUN go build -a -o ./server:
#10 0.295 server.go:4:2: no required module provides package <private-repo-url>/src/cmd/http-api/bootstrap; to add it:
#10 0.295 go get <private-repo-url>/src/cmd/http-api/bootstrap
------
executor failed running [/bin/sh -c go build -a -o ./server]: exit code: 1
请注意,此private-repo-url对应于此应用程序的存储库(它不是外部依赖项)。
这是 Dockerfile
FROM golang:1.17
WORKDIR /balrog
# Copy dependency definitions and download them
ADD go.mod .
ADD go.sum .
RUN go mod download
# Build the binary
ADD ./src .
ENV CGO_ENABLED=0
ENV GOOS=linux
ENV GOARCH=amd64
RUN go build -a -o ./server
#Run the server
CMD ["/server"]
还有 mod.go 文件:
module <private-repo-url>
go 1.16
require (
github.com/gin-gonic/gin v1.7.7
github.com/google/uuid v1.3.0
github.com/kelseyhightower/envconfig v1.4.0
github.com/sirupsen/logrus v1.8.1
go.uber.org/fx v1.15.0
)
我读过GO111MODULE说它应该打开,并且我还读到它从 1.17 开始默认启用(这里)。
同样根据官方 docker 映像(在dockerhub中),正确的方法是在复制所有文件后使用go get和go install 。这种方法使我遇到了一个稍微不同的问题,即 docker 无法访问存储库(因为它是私有的)并且我想避免向 docker 添加凭据。
我尝试使用环境变量GOVCS设置它的值,例如:
ENV GOVCS=github.com:git,gitlab.com:off
但它仍然失败并出现同样的错误。
最后我尝试了替换,我认为如果我从本地依赖项中删除它会起作用,所以我执行了(在 Dockerfile 中)这个:
RUN go mod edit -replace <private-repo-url>=./
它再次失败了:
=> ERROR [builder 10/10] RUN go build -a -o ./server 0.3s
------
> [builder 10/10] RUN go build -a -o ./server:
#17 0.299 server.go:4:2: module <private-repo-url>/src provides package <private-repo-url>/src/cmd/http-api/bootstrap and is replaced but not required; to add it:
#17 0.299 go get <private-repo-url>/src
#17 0.299 server.go:5:2: no required module provides package go.uber.org/fx; to add it:
#17 0.299 go get go.uber.org/fx
有什么方法可以防止 go builder/package installer 在外部查找这些文件?由于go mod和go get + go install 都尝试访问此私有存储库,但由于他们无权访问而失败。但是他们不应该首先尝试访问它,因为它是应用程序的存储库......或者我做错了什么(显然或者我不会在这里),遗漏了什么?