4

我对 golang 和 docker 还是很陌生,所以在查看了许多线程和互联网资源之后,我最终对应该如何设置感到困惑。

我的问题是由于未找到本地依赖项,我正面临 golang 应用程序的 docker build 错误。

仅供参考(本地机器):go 版本 go1.13.4 linux/amd64

这是我的问题的背景:

我有一个存储库,它将保存将在 Kubernetes 中运行的简单微服务(用 Go 编写)的源代码。所以存储库中的每个文件夹都代表一个服务。我还有一个文件夹“工具”,它重新组合了许多服务共享的辅助功能和资源。我的存储库不在任何 Go 特殊路径中,它位于我的一个驱动器的根目录中。所以这是我的回购的样子:

.
├── service1
│   ├── main.go
│   ├── Dockerfile
│   ├── go.mod
│   └── go.sum
├── service2
├── service3
│   ...
├── serviceX
├── tools
│   ├── helpers.go
│   ├── ressources.go
│   ├── go.mod
└── └── go.sum

因此,每个文件夹/服务都是一个独立处理其依赖关系的 go 模块。文件夹工具也是一个模块。我初始化了每个模块go mod init FOLDER_NAME

在服务service1 中,我引用了模块工具来使用一些功能。这是我实现它的方式:(./service1/main.go 的 src)

package main

import (
    "fmt"

    st "../tools"

    // other modules imports
)

func main() {
    st.ExecHelperFunc()
    // http server inits
}

当我在本地运行service1 的main.go 或者如果我在本地构建它并在之后运行二进制文件时,服务工作正常。

但是当我尝试构建 service1 的dockerfile时,我得到 golang build 错误: build _/go/src/app/tools: cannot find module for path _/go/src/app/tools

这是我的码头文件:

FROM golang:1.13 as builder
ENV GO111MODULE=on
WORKDIR /go/src/app
COPY ./tools ./tools
COPY ./service1 ./service1

WORKDIR /go/src/app/tools
RUN go mod download
WORKDIR /go/src/app/service1
RUN go mod download

WORKDIR /go/src/app

RUN go build -o server /go/src/app/service1/main.go

FROM centos:7
RUN yum -y update && yum clean all
COPY --from=builder /go/src/app/server .
EXPOSE 3000
CMD ["./server"]

所以我不明白为什么docker无法构建服务?显然他无法定位/识别模块工具,但为什么呢?我为 dockerfile 尝试了许多不同的设置,但我仍然无法弄清楚。

预先感谢您的帮助。

更新:

正如答案中所建议的那样,将dockerfile中 GO111MODULE 的值从 on 更改为 auto 并不能修复构建错误,而是让我得到一个新错误:

unexpected directory layout:
    import path: _/go/src/app/tools
    root: /go/src
    dir: /go/src/app/tools
    expand root: /go
    expand dir: /go/src/app/tools
    separator: /
4

2 回答 2

0

对于面临类似问题的任何人 - 我使用该vendor目录解决了类似问题。基本上,您通过运行下载主机上的依赖项,go mod vendor然后它们将自动复制到 docker 容器中。

从 Dockerfile 中,您必须删除RUN go mod download,并且还需要使用-mod=vendor标志修改构建命令。

要构建 docker 映像,您需要运行这些命令

go mod vendor
docker build . -t image-name

我希望这会对某人有所帮助。我在这里找到了这个解决方案:https ://smartystreets.com/blog/2018/09/private-dependencies-in-docker-and-go/

于 2021-04-23T19:27:04.233 回答
0

您应该将环境变量更改GO111MODULE为 auto。

最终的 Dockerfile 是:

FROM golang:1.13 as builder
ENV GO111MODULE=auto
WORKDIR /go/src/app
COPY ./tools ./tools
COPY ./service1 ./service1

WORKDIR /go/src/app/tools
RUN go mod download
WORKDIR /go/src/app/service1
RUN go mod download

WORKDIR /go/src/app

RUN go build -o server /go/src/app/service1/main.go

FROM centos:7
RUN yum -y update && yum clean all
COPY --from=builder /go/src/app/server .
EXPOSE 3000
CMD ["./server"]

更新

在我看来,如果 GO111MODULE 等于 'auto',golang 将禁用模块功能,并在 GOPATH 中查找第三方包。在编译代码之前,您应该go get -u github.com/op/go-logging. 我想这不是你想要的。

根据此处的示例,我在此处更新了我的存储。您可以尝试在 docker 容器中构建代码。它可以构建成功,但您应该更改包名称。

于 2019-12-30T13:02:06.077 回答