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

时间:2019-07-01 20:57:33

标签: go go-modules

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

go get github.com/mitchellh/gox

但是,此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依赖项。

dir="$(mktemp -d)"; \
  env GOPATH="$dir" go get github.com/mitchellh/gox && \
  mv "$dir/bin/gox" "$(go env GOPATH)"/bin/gox

3 个答案:

答案 0 :(得分:9)

从 1.16 开始

Go 1.16(2021 年 2 月发布)包含一项更改,可以在不影响 go.mod 的情况下安装二进制文件。

问题 40276 跟踪提案:

<块引用>

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

这是在 CL 254365 中实现的。作为此更改的一部分,如果我理解正确,您可以运行:

go install github.com/foo/bar@latest 

在不影响 go.mod 的情况下安装命令。

答案 1 :(得分:2)

希望在Go 1.14中,go get将会有一个新标记,它完全符合您的要求。在问题#30515“ cmd / 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

转到1.11、1.12、1.13+:gobin

gobin是安装或运行二进制文件的模块感知命​​令,它提供了更多的灵活性,包括无需更改当前模块的go.mod即可进行安装的能力。有关更多详细信息,请参见gobin READMEFAQ

转到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

这是因为在Go 1.11中,除非您处于模块中,否则无法使用@version语法,而该模块在Go 1.12中得到了放松。 simple shell script@rogpeppe通过Go 1.12 release notes使这种方法自动化。

其他详细信息

通常,模块-模块中的go命令始终确定其所在的模块,该模块基于您调用go命令时的当前工作目录。 (您可以比喻没有任何参数的make将如何在当前工作目录中查找makefile,或者从历史上来看,go build没有任何参数将如何构建当前工作目录,等等)。

对于模块,go get在当前工作目录或其任何父目录中查找go.mod文件,而go get将使用任何go.mod中列出的约束作为解决版本的一部分,并根据需要进行go.mod更新go get。因此,如果您在现有模块中运行go.mod,则会更新go get文件。

另一方面,从Go 1.12开始,如果您所在的目录不属于任何模块(也就是说,该目录没有go.mod,也没有其任何父目录),则没有go.mod可以更新,但是go命令仍然可以在模块模式下运行并使用@version语法。

来自{{3}}:

  

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

答案 2 :(得分:0)

go help build

  

-mod build标志提供了对go.mod的更新和使用的附加控制。

     

如果使用-mod = readonly调用,则go命令将禁止隐式自动更新go.mod