Go Module 引入本地自定义包
文章转载至 小一辈无产阶级码农
最近由于项目要求,需要对 IPFS
源码进行修改,由于自己在此之前没有接触过 Go 语言,在使用 go mod
导入本地自己开发的工具包的时候折腾了好久才搞定。 记录一下,以备后期查阅。 Go 语言的 Module 新特性是在 go1.11 的发布之后才支持的,这是 Go 语言新的一套依赖管理系统。
文章导读
1. 启用 Go Module
首先在默认情况下,$GOPATH
默认情况下是不支持 go mudules 的,当你执行 go mod init
的时候会遇到如下错误:
go: modules disabled inside GOPATH/src by GO111MODULE=auto; see ‘go help modules’
我们需要在执行 go mod
命令之前,导出 GO111MODULE
环境变量,你可以直接临时一次性导出, 为了后面方便,建议直接在 ~/.bashrc
文件中导出, 在文件末尾加入:
1 | export GO111MODULE=on |
Bash
从这也表明了 go 将来是要利用 modules 机制去消灭 $GOPATH
的。
2. 创建 Go Module
我们现在 $GOPATH
下面先创建一个项目,并初始化 module
1 | mkdir $GOPATH/src/gitee.com/rockyang/testmod -p |
Bash
这时,我们新建的项目已经成为了一个 module 了,我们可以在项目中随便写几个函数导出测试。
Note: 我这里使用的是码云做项目托管,没有使用 github,国内码云确实比 github 快得多。
接下来你可以选择把项目推送到远程仓库,如果你的仓库是公开的话,别人就是可以直接使用 go get
命令去下载你的项目了。
如果是私有项目只想给内部使用,则你可以参考我的这篇博客的做法。Go Module 使用私有仓库作为项目依赖包
3. Go Module 版本规则
go modules 是一个版本化依赖管理系统,版本需要遵循一些规则,打开一个 go.mod
文件,你会发现类似下面的依赖规则:
1 | require ( |
Bash
依赖规则由两个部分组成,前面一部分是包路径,后面一部分表示的是版本号。 你会发现有两种版本号,一种是我们很熟悉的 git 标签,比如 v0.0.2
,另一种就比较复杂一些,它是:版本号 + 时间戳 +hash 比如:v0.0.0-20190212224330-8d79a5489543
,它其实是精准的对应着一个 git log
记录,后面的哈希是去提交哈希的前 12 位。
比如我当前的提交记录是这样的:
1 | git log |
Bash
则我的最新版本号应该为 v0.0.0-20190610103414-4c55783279db
4. 引入本地依赖包
前面铺垫了这么多,接下来回到我们的主题,我该怎样使用我们自己开发的工具包呢? 假设我们有一个新的项目 testmod-demo
,现在想要在新的项目中使用 testmod 中的工具包,那么首先我们需要使用 go mod
初始化该项目:
1 | cd testmod-demo |
Bash
初始化之后会在当前项目根目录生成一个 go.mod
,接下来我们有两种方式去引入 testmod 包,一种是直接修改 go.mod
文件,在 require 配置中添加上
1 | gitee.com/rockyang/testmod v0.0.0-20190610103414-4c55783279db |
Bash
或者使用 go mod edit
命令修改依赖
1 | go mod edit -require="gitee.com/rockyang/testmod@v0.0.0-20190610103414-4c55783279db" |
Bash
5. 使用 replace 将远程包替换为本地包服务
这时如果你执行 go build
的时候会报错,提示找不到 gitee.com/rockyang/testmod
,是因为你没有把仓库推送到远程,所以无法下载。 go module 提供了另外一个方案, 使用 replace, 编辑 go.mod 文件,在最后面添加:replace gitee.com/rockyang/testmod => /gopath/src/gitee.com/rockyang/testmod
1 | module gitee.com/rockyang/testmod-demo |
Bash
这里的 /gopath/src/gitee.com/rockyang/testmod 是本地的包路径
然后再执行 go build
你会看到你想要的结果。
以上为原作者原文,转载这篇文章的原因是因为Golang1.13发布了,看到了其中一个新特新GOPRIVATE
可以搭配 GOPROXY
对私有模块更细粒度的控制。
Go Module 引入本地自定义包