1. 程式人生 > 程式設計 >golang常用庫之配置檔案解析庫-viper使用詳解

golang常用庫之配置檔案解析庫-viper使用詳解

golang常用庫:gorilla/mux-http路由庫使用
golang常用庫:配置檔案解析庫-viper使用
golang常用庫:操作資料庫的orm框架-gorm基本使用
golang常用庫:欄位引數驗證庫-validator使用

一、viper簡介

viper 配置管理解析庫,是由大神 Steve Francia 開發,他在google領導著 golang 的產品開發,他也是 gohugo.io 的創始人之一,命令列解析庫 cobra 開發者。總之,他在golang領域是專家,很牛的一個人。

他的github地址:https://github.com/spf13

viper是一個配置管理的解決方案,它能夠從 json,toml,ini,yaml,hcl,env 等多種格式檔案中,讀取配置內容,它還能從一些遠端配置中心讀取配置檔案,如consul,etcd等;它還能夠監聽檔案的內容變化。

viper的 logo:

golang常用庫之配置檔案解析庫-viper使用詳解

二、viper功能介紹

  • 讀取 json,toml,ini,yaml,hcl,env 等格式的檔案內容
  • 讀取遠端配置檔案,如 consul,etcd 等和監控配置檔案變化
  • 讀取命令列 flag 的值
  • 從 buffer 中讀取值

配置檔案又可以分為不同的環境,比如dev,test,prod等。

viper 可以幫助你專注配置檔案管理。

viper 讀取配置檔案的優先順序,從高到低,如下:

  • 顯式設定的Set函式
  • 命令列引數
  • 環境變數
  • 配置檔案
  • 遠端k-v 儲存系統,如consul,etcd等
  • 預設值

Viper 配置key是不區分大小寫的。

其實,上面的每一種檔案格式,都有一些比較有名的解析庫,如:

  • toml :https://github.com/BurntSushi/toml
  • json :json的解析庫比較多,下面列出幾個常用的

https://github.com/json-iterator/go
https://github.com/mailru/easyjson
https://github.com/bitly/go-simplejson
https://github.com/tidwall/gjson
ini : https://github.com/go-ini/ini
等等單獨檔案格式解析庫。

但是為啥子要用viper,因為它是一個綜合檔案解析庫,包含了上面所有的檔案格式解析,是一個集合體,少了配置多個庫的煩惱。

三、viper使用

安裝viper命令:

go get github.com/spf13/viper

文件: https://github.com/spf13/viper/blob/master/README.md#putting-values-into-viper

通過viper.Set設定值

如果某個鍵通過viper.Set設定了值,那麼這個值讀取的優先順序最高

viper.Set("mysql.info","this is mysql info")

設定預設值

https://github.com/spf13/viper/blob/master/README.md#establishing-defaults

viper 支援預設值的設定。如果配置檔案、環境變數、遠端配置中沒有設定鍵值,就可以通過viper設定一些預設值。

Examples:

viper.SetDefault("ContentDir","content")
viper.SetDefault("LayoutDir","layouts")
viper.SetDefault("Taxonomies",map[string]string{"tag": "tags","category": "categories"})

讀取配置檔案

https://github.com/spf13/viper/blob/master/README.md#reading-config-files

讀取配置檔案說明

讀取配置檔案要求:最少要知道從哪個位置查詢配置檔案。使用者一定要設定這個路徑。

viper可以從多個路徑搜尋配置檔案,單個viper例項只支援單個配置檔案。
viper本身沒有設定預設的搜尋路徑,需要使用者自己設定預設路徑。

viper搜尋和讀取配置檔案例子片段:

viper.SetConfigName("config") // 配置檔案的檔名,沒有副檔名,如 .yaml,.toml 這樣的副檔名
viper.SetConfigType("yaml") // 設定副檔名。在這裡設定檔案的副檔名。另外,如果配置檔案的名稱沒有副檔名,則需要配置這個選項
viper.AddConfigPath("/etc/appname/") // 查詢配置檔案所在路徑
viper.AddConfigPath("$HOME/.appname") // 多次呼叫AddConfigPath,可以新增多個搜尋路徑
viper.AddConfigPath(".") // 還可以在工作目錄中搜索配置檔案
err := viper.ReadInConfig() // 搜尋並讀取配置檔案
if err != nil { // 處理錯誤
 panic(fmt.Errorf("Fatal error config file: %s \n",err))
}

說明:
這裡執行viper.ReadInConfig()之後,viper才能確定到底用哪個檔案,viper按照上面的AddConfigPath() 進行搜尋,找到第一個名為 config.ext (這裡的ext代表副檔名: 如 json,toml,yaml,yml,ini,prop 等副檔名) 的檔案後即停止搜尋。

如果有多個名稱為config的配置檔案,viper怎麼搜尋呢?它會按照如下順序搜尋

  • config.json
  • config.toml
  • config.yaml
  • config.yml
  • config.properties (這種一般是java中的配置檔名)
  • config.props (這種一般是java中的配置檔名)

你還可以處理一些特殊情況:

if err := viper.ReadInConfig(); err != nil {
 if _,ok := err.(viper.ConfigFileNotFoundError); ok { 
 // 配置檔案沒有找到; 如果需要可以忽略
 } else { 
 // 查詢到了配置檔案但是產生了其它的錯誤
 }
}

// 查詢到配置檔案並解析成功

注意[自1.6起]: 你也可以有不帶副檔名的檔案,並以程式設計方式指定其格式。對於位於使用者$HOME目錄中的配置檔案沒有任何副檔名,如.bashrc。

例子1. 讀取配置檔案

config.toml 配置檔案:

# this is a toml 

title = "toml exaples"
redis = "127.0.0.1:3300" # redis

[mysql]
host = "192.168.1.1"
ports = 3306
username = "root"
password = "root123456"

viper_toml.go:

package main

import(
 "fmt"
 "github.com/spf13/viper"
)

// 讀取配置檔案config
type Config struct {
 Redis string
 MySQL MySQLConfig
}

type MySQLConfig struct {
 Port int
 Host string
 Username string
 Password string
}

func main() {
 // 把配置檔案讀取到結構體上
 var config Config
 
 viper.SetConfigName("config")
 viper.AddConfigPath(".")
 err := viper.ReadInConfig()
 if err != nil {
 fmt.Println(err)
 return
 }
 
 viper.Unmarshal(&config) //將配置檔案繫結到config上
 fmt.Println("config: ",config,"redis: ",config.Redis)
}

例子2. 讀取多個配置檔案

在例子1基礎上多增加一個json的配置檔案,config3.json 配置檔案:

{
 "redis": "127.0.0.1:33000","mysql": {
 "port": 3306,"host": "127.0.0.1","username": "root","password": "123456"
 }
}

viper_multi.go

package main

import (
	"fmt"

	"github.com/spf13/viper"
)

type Config struct {
	Redis string
	MySQL MySQLConfig
}

type MySQLConfig struct {
	Port int
	Host string
	Username string
	Password string
}

func main() {
	// 讀取 toml 配置檔案
	var config1 Config

	vtoml := viper.New()
	vtoml.SetConfigName("config")
	vtoml.SetConfigType("toml")
	vtoml.AddConfigPath(".")

	if err := vtoml.ReadInConfig(); err != nil {
		fmt.Println(err)
		return
	}

	vtoml.Unmarshal(&config1)
	fmt.Println("read config.toml")
	fmt.Println("config: ",config1,config1.Redis)

	// 讀取 json 配置檔案
	var config2 Config
	vjson := viper.New()
	vjson.SetConfigName("config3")
	vjson.SetConfigType("json")
	vjson.AddConfigPath(".")

	if err := vjson.ReadInConfig(); err != nil {
		fmt.Println(err)
		return
	}

	vjson.Unmarshal(&config2)
	fmt.Println("read config3.json")
	fmt.Println("config: ",config1.Redis)
}

執行:

$ go run viper_multi.go

read config.toml
config: {127.0.0.1:33000 {0 192.168.1.1 root 123456}} redis: 127.0.0.1:33000
read config3.json
config: {127.0.0.1:33000 {0 192.168.1.1 root 123456}} redis: 127.0.0.1:33000

例子3. 讀取配置項的值

新建資料夾 item, 在裡面建立檔案 config.json,內容如下:

{
 "redis": "127.0.0.1:33000","password": "123456","ports": [
 5799,6029
 ],"metric": {
 "host": "127.0.0.1","port": 2112
 }
 }
}

item/viper_get_item.go 讀取配置項的值

package main

import (
	"fmt"

	"github.com/spf13/viper"
)

func main() {
	viper.SetConfigName("config")
	viper.SetConfigType("json")
	viper.AddConfigPath(".")
	err := viper.ReadInConfig() //根據上面配置載入檔案
	if err != nil {
		fmt.Println(err)
		return
	}

	host := viper.Get("mysql.host")
	username := viper.GetString("mysql.username")
	port := viper.GetInt("mysql.port")
	portsSlice := viper.GetIntSlice("mysql.ports")

	metricPort := viper.GetInt("mysql.metric.port")
	redis := viper.Get("redis")

	mysqlMap := viper.GetStringMapString("mysql")

	if viper.IsSet("mysql.host") {
		fmt.Println("[IsSet()]mysql.host is set")
	} else {
		fmt.Println("[IsSet()]mysql.host is not set")
	}
	fmt.Println("mysql - host: ",host,",username: ",username,port: ",port)
	fmt.Println("mysql ports :",portsSlice)
	fmt.Println("metric port: ",metricPort)
	fmt.Println("redis - ",redis)

	fmt.Println("mysqlmap - ",mysqlMap,mysqlMap["username"])
}

執行:

$ go run viper_get_item.go

[IsSet()]mysql.host is set
mysql - host: 127.0.0.1,username: root,port: 3306
mysql ports : [5799 6029]
metric port: 2112
redis - 127.0.0.1:33000
mysqlmap - map[host:127.0.0.1 metric: password:123456 port:3306 ports: username:root],username: root

如果把上面的檔案config.json寫成toml格式,怎麼解析? 改成config1.toml:

# toml
toml = "toml example"

redis = "127.0.0.1:33000"

[mysql]
port = 3306
host = "127.0.0.1"
username = "root"
password = "123456"
ports = [5799,6029]
[mysql.metric]
host = "127.0.0.1"
port = 2112

其實解析程式碼差不多,只需修改2處,

viper.SetConfigName("config") 裡的 config 改成 config1 ,
viper.SetConfigType("json")裡的 json 改成 toml,其餘程式碼都一樣。解析的效果也一樣。

viper獲取值的方法:

  • Get(key string) : interface{}
  • GetBool(key string) : bool
  • GetFloat64(key string) : float64
  • GetInt(key string) : int
  • GetIntSlice(key string) : []int
  • GetString(key string) : string
  • GetStringMap(key string) : map[string]interface{}
  • GetStringMapString(key string) : map[string]string
  • GetStringSlice(key string) : []string
  • GetTime(key string) : time.Time
  • GetDuration(key string) : time.Duration
  • IsSet(key string) : bool
  • AllSettings() : map[string]interface{}

例子4. 讀取命令列的值

新建資料夾 cmd,然後cmd資料夾裡新建config.json檔案:

{
 "redis":{
 "port": 3301,"host": "127.0.0.1"
 },"password": "123456"
 }
}

go解析檔案,cmd/viper_pflag.go:

package main

import (
	"fmt"

	"github.com/spf13/pflag"
	"github.com/spf13/viper"
)

func main() {
	pflag.Int("redis.port",3302,"redis port")

	viper.BindPFlags(pflag.CommandLine)
	pflag.Parse()

	viper.SetConfigName("config")
	viper.SetConfigType("json")
	viper.AddConfigPath(".")
	err := viper.ReadInConfig() //根據上面配置載入檔案
	if err != nil {
		fmt.Println(err)
		return
	}

	host := viper.Get("mysql.host")
	username := viper.GetString("mysql.username")
	port := viper.GetInt("mysql.port")
	redisHost := viper.GetString("redis.host")
	redisPort := viper.GetInt("redis.port")

	fmt.Println("mysql - host: ",port)
	fmt.Println("redis - host: ",redisHost,redisPort)
}

1.不加命令列引數執行:

$ go run viper_pflag.go

mysql - host: 127.0.0.1,port: 3306
redis - host: 127.0.0.1,port: 3301

說明:redis.port 的值是 3301,是 config.json 配置檔案裡的值。

2.加命令列引數執行

$ go run viper_pflag.go --redis.port 6666

mysql - host: 127.0.0.1,port: 6666

說明:加了命令列引數 --redis.port 6666,這時候redis.port輸出的值為 6666,讀取的是cmd命令列的值

例子5:io.Reader中讀取值

https://github.com/spf13/viper#reading-config-from-ioreader

viper_ioreader.go

package main

import (
	"bytes"
	"fmt"

	"github.com/spf13/viper"
)

func main() {
	viper.SetConfigType("yaml")

	var yaml = []byte(`
Hacker: true
name: steve
hobbies:
- skateboarding
- snowboarding
- go
clothing:
 jacket: leather
 trousers: denim
age: 35
eyes : brown
beard: true
 `)

	err := viper.ReadConfig(bytes.NewBuffer(yaml))
	if err != nil {
		fmt.Println(err)
		return
	}
	hacker := viper.GetBool("Hacker")
	hobbies := viper.GetStringSlice("hobbies")
	jacket := viper.Get("clothing.jacket")
	age := viper.GetInt("age")
	fmt.Println("Hacker: ",hacker,hobbies: ",hobbies,jacket: ",jacket,age: ",age)

}

例子6:寫配置檔案

https://github.com/spf13/viper#writing-config-files

新建檔案 writer/viper_write_config.go:

package main

import (
	"fmt"

	"github.com/spf13/viper"
)

func main() {
	viper.SetConfigName("config")
	viper.SetConfigType("yaml")
	viper.AddConfigPath(".")

	viper.Set("yaml","this is a example of yaml")

	viper.Set("redis.port",4405)
	viper.Set("redis.host","127.0.0.1")

	viper.Set("mysql.port",3306)
	viper.Set("mysql.host","192.168.1.0")
	viper.Set("mysql.username","root123")
	viper.Set("mysql.password","root123")

	if err := viper.WriteConfig(); err != nil {
		fmt.Println(err)
	}
}

執行:

$ go run viper_write_config.go

沒有任何輸出表示生成配置檔案成功

mysql:
 host: 192.168.1.0
 password: root123
 port: 3306
 username: root123
redis:
 host: 127.0.0.1
 port: 4405
yaml: this is a example of yaml

WriteConfig() 和 SafeWriteConfig() 區別:

如果待生成的檔案已經存在,那麼SafeWriteConfig()就會報錯,Config File "config.yaml" Already Exists, 而WriteConfig()則會直接覆蓋同名檔案。

四、參考

  • viper 文件
  • golang json庫gjson的使用

到此這篇關於golang常用庫之配置檔案解析庫-viper使用詳解的文章就介紹到這了,更多相關golang配置檔案解析庫viper內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!