使用“go get”下载二进制文件而不将它们添加到 go.mod

Ahm*_*gle 13 go go-modules

我在我的项目和我的构建系统(例如 Travis CI)中使用 Go 模块我正在下载一个命令行实用程序(用 Go 编写)go get来协助我的构建过程,例如:

go get github.com/mitchellh/gox
Run Code Online (Sandbox Code Playgroud)

但是,这go get会导致文件被添加到我的go.mod文件中。这会污染构建环境,导致它“脏”(因为 git 中跟踪的某些文件发生了更改,在本例中为 go.mod 和 go.sum),我用它git describe --always --dirty --tag来描述我的构建,它显示为“肮脏的”。

有没有办法“去获取”一个二进制文件只是为了下载它,而不将它添加到 go.mod/go.sum 中?

我已经尝试将 GOPATH 设置为其他地方,即使如此,也会go get更新 go.mod/go.sum 以将其添加为// indirect依赖项。

go get github.com/mitchellh/gox
Run Code Online (Sandbox Code Playgroud)

tru*_*ted 12

从 1.16 开始

Go 1.16(2021 年 2 月发布)包含一项更改,可以在不影响go.mod.

问题 40276跟踪该提案:

cmd/go: 'go install' 应该在模块外以模块模式安装可执行文件

这是在CL 254365中实现的。作为此更改的一部分,您可以运行例如:

go install golang.org/x/tools/cmd/goimports@latest
Run Code Online (Sandbox Code Playgroud)

安装二进制文件而不影响go.mod.

要安装特定版本,请替换@latest为 eg @v0.1.5

  • 请注意,需要“@latest”或类似的内容才能正常工作 (3认同)

typ*_*182 5

希望在 Go 1.14 中会有一个新的标志来go get满足你的要求。这在问题#30515 “cmd/go:提供一致的全局安装命令”中进行了跟踪。

在此之前,您有几个不同的选择。

Go 1.12 和 1.13:更改目录

如果您使用的是 Go 1.12 或更高版本,最简单的解决方案可能是在不执行go.mod之前将当前模块移到目录之外go get,例如:

$ cd /tmp                              
$ go get github.com/foo/bar@v1.2.3
$ cd -                                # return to prior directory
Run Code Online (Sandbox Code Playgroud)

Go 1.11、1.12、1.13+:gobin

gobin是一个模块感知命​​令,用于安装或运行二进制文件,提供额外的灵活性,包括在不改变当前模块的go.mod. 有关更多详细信息,请参阅gobin 自述文件常见问题解答

Go 1.11:临时模块

如果您使用带有模块的 Go 1.11,第一步可能是升级到 Go 1.12 或 1.13,因为模块有很多改进。如果您需要使用 Go 1.11 并希望使用@version语法而不更新当前模块的go.mod,那么一种方法是创建一个临时模块:

cd $(mktemp -d) && go mod init tempmod && go get github.com/foo/bar@v1.2.3
Run Code Online (Sandbox Code Playgroud)

这是因为在 Go 1.11 中,@version除非您在模块中,否则无法使用语法,而在 Go 1.12 中则放宽了这一点。这种方法已经被自动化的简单的shell脚本@rogpeppe

额外细节

一般情况go下,module-module 中的命令总是确定它“在”哪个模块,这是基于调用go命令时的当前工作目录。(您可以类比在make没有任何 args 的情况下如何在当前工作目录中查找 makefile,或者在历史上go build没有任何 args 将如何构建当前工作目录等)。

使用模块,在当前工作目录或其任何父目录中go get查找go.mod文件,go get并将使用 any 中列出的约束go.mod作为解决版本的一部分,并go.mod根据执行go get. 这就是为什么go.mod如果您go get从现有模块中运行您的文件会被更新。

在另一方面,从围棋1.12,如果你在一个目录中没有任何模块(也就是目录不具有的一部分go.mod,也不做任何的父母),那么就没有go.mod进行更新,但该go命令仍然能够在模块模式下运行并使用@version语法。

Go 1.12 发行说明

当 GO111MODULE 设置为 on 时,go 命令现在支持模块目录之外的模块感知操作,前提是这些操作不需要解析相对于当前目录的导入路径或明确编辑 go.mod 文件。go get、go list 和 go mod download 等命令的行为就像在具有初始为空要求的模块中一样。在这种模式下, go env GOMOD 报告系统的空设备(/dev/null 或 NUL)。