使用go path問題
- 程式碼開發必須在go path src目錄下,不然,就有問題。
- 依賴手動管理
- 依賴包沒有版本可言
從這個看, go path不算包管理工具
govendor
- 解決了包依賴,一個配置檔案就管理
- 依賴包全都下載到專案vendor下,每個專案都把有一份。拉取專案時,開始懷疑人生。
go mod介紹
go modules 是 golang 1.11 新加的特性。現在1.12 已經發布了,是時候用起來了。Modules官方定義為:
模組是相關Go包的集合。modules是原始碼交換和版本控制的單元。 go命令直接支援使用modules,包括記錄和解析對其他模組的依賴性。modules替換舊的基於GOPATH的方法來指定在給定構建中使用哪些原始檔。
如何使用go mod
首先,必須升級go到1.11,目前版本是1.15
下面我以我自己升級演示:
### 解除安裝舊版本,刪除對應檔案
brew uninstall -f go
### 更新一下brew
brew update
### 安裝go
brew install go
上面升級完了,使用 go version
看下版本
go version go1.14.1 darwin/amd64
下面設定go mod和go proxy
go env -w GOBIN=/Users/youdi/go/bin
go env -w GO111MODULE=on
go env -w GOPROXY=https://goproxy.cn,direct // 使用七牛雲的
注意: go env -w會將配置寫到 GOENV="/Users/youdi/Library/Application Support/go/env"
下面看下我的配置
GO111MODULE="on"
GOARCH="amd64"
GOBIN="/Users/youdi/go/bin"
GOCACHE="/Users/youdi/Library/Caches/go-build"
GOENV="/Users/youdi/Library/Application Support/go/env"
GOEXE=""
GOFLAGS=""
GOHOSTARCH="amd64"
GOHOSTOS="darwin"
GOINSECURE=""
GONOPROXY=""
GONOSUMDB=""
GOOS="darwin"
GOPATH="/Users/youdi/go"
GOPRIVATE=""
GOPROXY="https://goproxy.cn,direct"
GOROOT="/usr/local/go"
GOSUMDB="off"
GOTMPDIR=""
GOTOOLDIR="/usr/local/go/pkg/tool/darwin_amd64"
GCCGO="gccgo"
AR="ar"
CC="clang"
CXX="clang++"
CGO_ENABLED="1"
GOMOD="/dev/null"
CGO_CFLAGS="-g -O2"
CGO_CPPFLAGS=""
CGO_CXXFLAGS="-g -O2"
CGO_FFLAGS="-g -O2"
CGO_LDFLAGS="-g -O2"
PKG_CONFIG="pkg-config"
GOGCCFLAGS="-fPIC -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/8m/v_1j4dgs7rzgqq4p_4_8k_nr0000gn/T/go-build221113671=/tmp/go-build -gno-record-gcc-switches -fno-common"
我們看一下,我修改的內容
cat /Users/youdi/Library/Application Support/go/env
GO111MODULE=on
GOBIN=/Users/youdi/go/bin
GOPROXY=https://goproxy.cn,direct
GOSUMDB=off
GO111MODULE
GO111MODULE 有三個值:off, on和auto(預設值)。
GO111MODULE=off,go命令列將不會支援module功能,尋找依賴包的方式將會沿用舊版本那種通過vendor目錄或者GOPATH模式來查詢。
GO111MODULE=on,go命令列會使用modules,而一點也不會去GOPATH目錄下查詢。
GO111MODULE=auto,預設值,go命令列將會根據當前目錄來決定是否啟用module功能。這種情況下可以分為兩種情形:
當前目錄在GOPATH/src之外且該目錄包含go.mod檔案
當前檔案在包含go.mod檔案的目錄下面。
當modules功能啟用時,依賴包的存放位置變更為$GOPATH/pkg,允許同一個package多個版本並存,且多個專案可以共享快取的 module
我們看下目錄:
cd /Users/youdi/go/pkg
├── darwin_amd64
│ ├── github.com
│ ├── go.etcd.io
│ ├── golang
│ ├── golang.org
│ ├── gopkg.in
│ ├── quickstart
│ └── uc.a
├── mod
│ ├── cache
│ ├── github.com
│ ├── golang.org
│ ├── google.golang.org
│ └── gopkg.in
└── sumdb
└── sum.golang.org
go mod命令
golang 提供了 go mod
命令來管理包。
go help mod
Go mod provides access to operations on modules.
Note that support for modules is built into all the go commands,
not just 'go mod'. For example, day-to-day adding, removing, upgrading,
and downgrading of dependencies should be done using 'go get'.
See 'go help modules' for an overview of module functionality.
Usage:
go mod <command> [arguments]
The commands are:
download download modules to local cache
edit edit go.mod from tools or scripts
graph print module requirement graph
init initialize new module in current directory
tidy add missing and remove unused modules
vendor make vendored copy of dependencies
verify verify dependencies have expected content
why explain why packages or modules are needed
Use "go help mod <command>" for more information about a command.
go mod 有以下命令:
命令 | 說明 |
---|---|
download | download modules to local cache(下載依賴包) |
edit | edit go.mod from tools or scripts(編輯go.mod) |
graph | print module requirement graph (列印模組依賴圖) |
verify | initialize new module in current directory(在當前目錄初始化mod) |
tidy | add missing and remove unused modules(拉取缺少的模組,移除不用的模組) |
vendor | make vendored copy of dependencies(將依賴複製到vendor下) |
verify | verify dependencies have expected content (驗證依賴是否正確) |
why | explain why packages or modules are needed(解釋為什麼需要依賴) |
比較常用的是 init
,tidy
, edit
使用go mod管理一個新專案
1. 初始化專案
可以隨便找一個目錄建立專案,我使用習慣用IDEA進行建立
mkdir Gone
cd Gone
go mod init Gone
檢視一下 go.mod檔案
module Gone
go 1.14
go.mod檔案一旦建立後,它的內容將會被go toolchain全面掌控。go toolchain會在各類命令執行時,比如go get、go build、go mod等修改和維護go.mod檔案。
go.mod 提供了module, require、replace和exclude 四個命令
module
語句指定包的名字(路徑)require
語句指定的依賴項模組replace
語句可以替換依賴項模組exclude
語句可以忽略依賴項模組
2. 新增依賴
建立 main.go檔案
package main
import (
"github.com/gin-gonic/gin"
)
func main() {
r := gin.Default()
r.GET("/ping", func(c *gin.Context) {
c.JSON(200, gin.H{
"message": "pong",
})
})
r.Run() // listen and serve on 0.0.0.0:8080 (for windows "localhost:8080")
}
執行 go run main.go 執行程式碼會發現 go mod 會自動查詢依賴自動下載
再檢視 go.mod
module Gone
go 1.14
require github.com/gin-gonic/gin v1.6.3
go module 安裝 package 的原則是先拉最新的 release tag,若無tag則拉最新的commit
go 會自動生成一個 go.sum 檔案來記錄 dependency tree
再次執行指令碼 go run main.go發現跳過了檢查並安裝依賴的步驟。
可以使用命令 go list -m -u all 來檢查可以升級的package,使用go get -u need-upgrade-package 升級後會將新的依賴版本更新到go.mod * 也可以使用 go get -u 升級所有依賴
去mod包快取下看看
/Users/youdi/go/pkg/mod/github.com/gin-gonic/gin@v1.6.3
go get升級
- 執行 go get -u 將會升級到最新的次要版本或者修訂版本(x.y.z, z是修訂版本號, y是次要版本號)
- 執行 go get -u=patch 將會升級到最新的修訂版本
- 執行 go get package@version 將會升級到指定的版本號version
- 執行go get如果有版本的更改,那麼go.mod檔案也會更改
使用replace替換無法直接獲取的package
由於某些已知的原因,並不是所有的package都能成功下載,比如:golang.org下的包。
modules 可以通過在 go.mod 檔案中使用 replace 指令替換成github上對應的庫,比如:
replace (
golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a => github.com/golang/crypto v0.0.0-20190313024323-a1f597ede03a
)
go mod釋出和使用
參考Roberto Selbach寫的go mod入門文章,文末,我給出連結
Creating a Module
如果你設定好go mod了,那你就可以在任何目錄下隨便建立
$mkdir gomodone
$cd gomodone
在這個目錄下建立一個檔案say.go
package gomodone
import "fmt"
// say Hi to someone
func SayHi(name string) string {
return fmt.Sprintf("Hi, %s", name)
}
初始化一個 go.mod
檔案
$ go mod init github.com/jacksonyoudi/gomodone
go: creating new go.mod: module github.com/jacksonyoudi/gomodone
檢視 go.mod內容如下:
github.com/jacksonyoudi/gomodone
go 1.14
下面我們要將這個module釋出到github上,然後在另外一個程式使用
$git init
$vim .gitiiignore
$git commit -am "init"
// github建立對應的repo
$git remote add origin git@github.com:jacksonyoudi/gomodone.git
$git push -u origin master
執行完,上面我們就相當於釋出完了。
如果有人需要使用,就可以使用
go get github.com/jacksonyoudi/gomodone
這個時候沒有加tag,所以,沒有版本的控制。預設是v0.0.0後面接上時間和commitid。如下:
gomodone@v0.0.0-20200517004046-ee882713fd1e
官方不建議這樣做,沒有進行版本控制管理。
module versioning
使用tag,進行版本控制
making a release
git tag v1.0.0
git push --tags
操作完,我們的module就釋出了一個v1.0.0的版本了。
推薦在這個狀態下,再切出一個分支,用於後續v1.0.0的修復推送,不要直接在master分支修復
$git checkout -b v1
$git push -u origin v1
use our module
上面已經發布了一個v1.0.0的版本,我們可以在另一個專案中使用,建立一個go的專案
$mkdir Gone
$cd Gone
$vim main.go
package main
import (
"fmt"
"github.com/jacksonyoudi/gomodone"
)
func main() {
fmt.Println(gomodone.SayHi("Roberto"))
}
程式碼寫好了,我們生成 go mod檔案
go mod init Gone
上面命令執行完,會生成 go mod檔案
看下mod檔案:
module Gone
go 1.14
require (
github.com/jacksonyoudi/gomodone v1.0.0
)
$go mod tidy
go: finding module for package github.com/jacksonyoudi/gomodone
go: found github.com/jacksonyoudi/gomodone in github.com/jacksonyoudi/gomodone v1.0.0
同時還生成了go.sum, 其中包含軟體包的雜湊值,以確保我們具有正確的版本和檔案。
github.com/jacksonyoudi/gomodone v1.0.1 h1:jFd+qZlAB0R3zqrC9kwO8IgPrAdayMUS0rSHMDc/uG8=
github.com/jacksonyoudi/gomodone v1.0.1/go.mod h1:XWi+BLbuiuC2YM8Qz4yQzTSPtHt3T3hrlNN2pNlyA94=
github.com/jacksonyoudi/gomodone/v2 v2.0.0 h1:GpzGeXCx/Xv2ueiZJ8hEhFwLu7xjxLBjkOYSmg8Ya/w=
github.com/jacksonyoudi/gomodone/v2 v2.0.0/go.mod h1:L8uFPSZNHoAhpaePWUfKmGinjufYdw9c2i70xtBorSw=
這個內容是下面的,需要操作執行的結果
go run main.go就可以執行了
Making a bugfix release
假如fix一個bug,我們在v1版本上進行修復
修改程式碼如下:
// say Hi to someone
func SayHi(name string) string {
- return fmt.Sprintf("Hi, %s", name)
+ return fmt.Sprintf("Hi, %s!", name)
}
修復好,我們開始push
$ git commit -m "Emphasize our friendliness" say.go
$ git tag v1.0.1
$ git push --tags origin v1
Updating modules
剛才fix bug,所以要在我們使用專案中更新
這個需要我們手動執行更新module操作
我們通過使用我們的好朋友來做到這一點go get:
- 執行
go get -u
以使用最新的 minor 版本或修補程式版本(即它將從1.0.0更新到例如1.0.1,或者,如果可用,則更新為1.1.0) - 執行 go get -u=patch 以使用最新的 修補程式 版本(即,將更新為1.0.1但不更新 為1.1.0)
- 執行go get package@version 以更新到特定版本(例如github.com/jacksonyoudi/gomodone@v1.0.1)
目前module最新的也是v1.0.1
// 更新最新
$go get -u
$go get -u=patch
//指定包,指定版本
$go get github.com/jacksonyoudi/gomodone@v1.0.1
操作完,go.mod檔案會修改如下:
module Gone
go 1.14
require (
github.com/jacksonyoudi/gomodone v1.0.1
)
Major versions
根據語義版本語義,主要版本與次要版本 不同。主要版本可能會破壞向後相容性。從Go模組的角度來看,主要版本是 完全不同的軟體包。乍一看這聽起來很奇怪,但這是有道理的:兩個不相容的庫版本是兩個不同的庫。
比如下面修改,完全破壞了相容性。
package gomodone
import (
"errors"
"fmt"
)
// Hi returns a friendly greeting
// Hi returns a friendly greeting in language lang
func SayHi(name, lang string) (string, error) {
switch lang {
case "en":
return fmt.Sprintf("Hi, %s!", name), nil
case "pt":
return fmt.Sprintf("Oi, %s!", name), nil
case "es":
return fmt.Sprintf("¡Hola, %s!", name), nil
case "fr":
return fmt.Sprintf("Bonjour, %s!", name), nil
default:
return "", errors.New("unknown language")
}
}
如上,我們需要不同的大版本,這種情況下
修改 go.mod如下
module github.com/jacksonyoudi/gomodone/v2
go 1.14
然後,重新tag,push
$ git commit say.go -m "Change Hi to allow multilang"
$ git checkout -b v2 # 用於v2版本,後續修復v2
$ git commit go.mod -m "Bump version to v2"
$ git tag v2.0.0
$ git push --tags origin v2
Updating to a major version
即使釋出了庫的新不相容版本,現有軟體 也不會中斷,因為它將繼續使用現有版本1.0.1。go get -u 將不會獲得版本2.0.0。
如果想使用v2.0.0,程式碼改成如下:
package main
import (
"fmt"
"github.com/jacksonyoudi/gomodone/v2"
)
func main() {
g, err := gomodone.SayHi("Roberto", "pt")
if err != nil {
panic(err)
}
fmt.Println(g)
}
執行 go mod tidy
go: finding module for package github.com/jacksonyoudi/gomodone/v2
go: downloading github.com/jacksonyoudi/gomodone/v2 v2.0.0
go: found github.com/jacksonyoudi/gomodone/v2 in github.com/jacksonyoudi/gomodone/v2 v2.0.0
當然,兩個版本都可以同時使用, 使用別名
如下:
package main
import (
"fmt"
"github.com/jacksonyoudi/gomodone"
mv2 "github.com/jacksonyoudi/gomodone/v2"
)
func main() {
g, err := mv2.SayHi("Roberto", "pt")
if err != nil {
panic(err)
}
fmt.Println(g)
fmt.Println(gomodone.SayHi("Roberto"))
}
執行一下 go mod tidy
Vendoring
預設是忽略vendor的,如果想在專案目錄下有vendor可以執行下面命令
$go vendor
當然,如果構建程式的時候,希望使用vendor中的依賴,
$ go build -mod vendor
IDEA下開發GO
建立go專案
建立完專案,會自動生成go mod檔案
如果需要修改,可以手動修改,加入git等操作寫業務邏輯程式碼
解決依賴,更新go.mod
go build
本作品採用《CC 協議》,轉載必須註明作者和本文連結