上一篇文章介紹 cobra 的時候提到了 viper,今天咱們就來介紹一下這個庫。
viper 是一個配置解決方案,擁有豐富的特性:mysql
io.Reader
中讀取配置;安裝:git
$ go get github.com/spf13/viper
使用:github
package main import ( "fmt" "log" "github.com/spf13/viper" ) func main() { viper.SetConfigName("config") viper.SetConfigType("toml") viper.AddConfigPath(".") viper.SetDefault("redis.port", 6381) err := viper.ReadInConfig() if err != nil { log.Fatal("read config failed: %v", err) } fmt.Println(viper.Get("app_name")) fmt.Println(viper.Get("log_level")) fmt.Println("mysql ip: ", viper.Get("mysql.ip")) fmt.Println("mysql port: ", viper.Get("mysql.port")) fmt.Println("mysql user: ", viper.Get("mysql.user")) fmt.Println("mysql password: ", viper.Get("mysql.password")) fmt.Println("mysql database: ", viper.Get("mysql.database")) fmt.Println("redis ip: ", viper.Get("redis.ip")) fmt.Println("redis port: ", viper.Get("redis.port")) }
咱們使用以前Go 每日一庫之 go-ini一文中使用的配置,不過改成 toml 格式。
toml 的語法很簡單,快速入門請看learn X in Y minutes。golang
app_name = "awesome web" # possible values: DEBUG, INFO, WARNING, ERROR, FATAL log_level = "DEBUG" [mysql] ip = "127.0.0.1" port = 3306 user = "dj" password = 123456 database = "awesome" [redis] ip = "127.0.0.1" port = 7381
viper 的使用很是簡單,它須要不多的設置。設置文件名(SetConfigName
)、配置類型(SetConfigType
)和搜索路徑(AddConfigPath
),而後調用ReadInConfig
。
viper會自動根據類型來讀取配置。使用時調用viper.Get
方法獲取鍵值。web
編譯、運行程序:redis
awesome web DEBUG mysql ip: 127.0.0.1 mysql port: 3306 mysql user: dj mysql password: 123456 mysql database: awesome redis ip: 127.0.0.1 redis port: 7381
有幾點須要注意:sql
section.key
的形式,即傳入嵌套的鍵名;viper.SetDefault
設置。viper 提供了多種形式的讀取方法。在上面的例子中,咱們看到了Get
方法的用法。Get
方法返回一個interface{}
的值,使用有所不便。服務器
GetType
系列方法能夠返回指定類型的值。
其中,Type 能夠爲Bool/Float64/Int/String/Time/Duration/IntSlice/StringSlice
。
可是請注意,若是指定的鍵不存在或類型不正確,GetType
方法返回對應類型的零值。微信
若是要判斷某個鍵是否存在,使用IsSet
方法。
另外,GetStringMap
和GetStringMapString
直接以 map 返回某個鍵下面全部的鍵值對,前者返回map[string]interface{}
,後者返回map[string]string
。
AllSettings
以map[string]interface{}
返回全部設置。網絡
// 省略包名和 import 部分 func main() { viper.SetConfigName("config") viper.SetConfigType("toml") viper.AddConfigPath(".") err := viper.ReadInConfig() if err != nil { log.Fatal("read config failed: %v", err) } fmt.Println("protocols: ", viper.GetStringSlice("server.protocols")) fmt.Println("ports: ", viper.GetIntSlice("server.ports")) fmt.Println("timeout: ", viper.GetDuration("server.timeout")) fmt.Println("mysql ip: ", viper.GetString("mysql.ip")) fmt.Println("mysql port: ", viper.GetInt("mysql.port")) if viper.IsSet("redis.port") { fmt.Println("redis.port is set") } else { fmt.Println("redis.port is not set") } fmt.Println("mysql settings: ", viper.GetStringMap("mysql")) fmt.Println("redis settings: ", viper.GetStringMap("redis")) fmt.Println("all settings: ", viper.AllSettings()) }
咱們在配置文件 config.toml 中添加protocols
和ports
配置:
[server] protocols = ["http", "https", "port"] ports = [10000, 10001, 10002] timeout = 3s
編譯、運行程序,輸出:
protocols: [http https port] ports: [10000 10001 10002] timeout: 3s mysql ip: 127.0.0.1 mysql port: 3306 redis.port is set mysql settings: map[database:awesome ip:127.0.0.1 password:123456 port:3306 user:dj] redis settings: map[ip:127.0.0.1 port:7381] all settings: map[app_name:awesome web log_level:DEBUG mysql:map[database:awesome ip:127.0.0.1 password:123456 port:3306 user:dj] redis:map[ip:127.0.0.1 port:7381] server:map[ports:[10000 10001 10002] protocols:[http https port]]]
若是將配置中的redis.port
註釋掉,將輸出redis.port is not set
。
上面的示例中還演示瞭如何使用time.Duration
類型,只要是time.ParseDuration
接受的格式均可以,例如3s
、2min
、1min30s
等。
viper 支持在多個地方設置,使用下面的順序依次讀取:
Set
顯示設置的;viper.Set
若是某個鍵經過viper.Set
設置了值,那麼這個值的優先級最高。
viper.Set("redis.port", 5381)
若是將上面這行代碼放到程序中,運行程序,輸出的redis.port
將是 5381。
若是一個鍵沒有經過viper.Set
顯示設置值,那麼獲取時將嘗試從命令行選項中讀取。
若是有,優先使用。viper 使用 pflag 庫來解析選項。
咱們首先在init
方法中定義選項,而且調用viper.BindPFlags
綁定選項到配置中:
func init() { pflag.Int("redis.port", 8381, "Redis port to connect") // 綁定命令行 viper.BindPFlags(pflag.CommandLine) }
而後,在main
方法開頭處調用pflag.Parse
解析選項。
編譯、運行程序:
$ ./main.exe --redis.port 9381 awesome web DEBUG mysql ip: 127.0.0.1 mysql port: 3306 mysql user: dj mysql password: 123456 mysql database: awesome redis ip: 127.0.0.1 redis port: 9381
如何不傳入選項:
$ ./main.exe awesome web DEBUG mysql ip: 127.0.0.1 mysql port: 3306 mysql user: dj mysql password: 123456 mysql database: awesome redis ip: 127.0.0.1 redis port: 7381
注意,這裏並不會使用選項redis.port
的默認值。
可是,若是經過下面的方法都沒法得到鍵值,那麼返回選項默認值(若是有)。試試註釋掉配置文件中redis.port
看看效果。
若是前面都沒有獲取到鍵值,將嘗試從環境變量中讀取。咱們既能夠一個個綁定,也能夠自動所有綁定。
在init
方法中調用AutomaticEnv
方法綁定所有環境變量:
func init() { // 綁定環境變量 viper.AutomaticEnv() }
爲了驗證是否綁定成功,咱們在main
方法中將環境變量 GOPATH 打印出來:
func main() { // 省略部分代碼 fmt.Println("GOPATH: ", viper.Get("GOPATH")) }
經過 系統 -> 高級設置 -> 新建 建立一個名爲redis.port
的環境變量,值爲 10381。
運行程序,輸出的redis.port
值爲 10381,而且輸出中有 GOPATH 信息。
也能夠單獨綁定環境變量:
func init() { // 綁定環境變量 viper.BindEnv("redis.port") viper.BindEnv("go.path", "GOPATH") } func main() { // 省略部分代碼 fmt.Println("go path: ", viper.Get("go.path")) }
調用BindEnv
方法,若是隻傳入一個參數,則這個參數既表示鍵名,又表示環境變量名。
若是傳入兩個參數,則第一個參數表示鍵名,第二個參數表示環境變量名。
還能夠經過viper.SetEnvPrefix
方法設置環境變量前綴,這樣一來,經過AutomaticEnv
和一個參數的BindEnv
綁定的環境變量,
在使用Get
的時候,viper 會自動加上這個前綴再從環境變量中查找。
若是對應的環境變量不存在,viper 會自動將鍵名所有轉爲大寫再查找一次。因此,使用鍵名gopath
也能讀取環境變量GOPATH
的值。
若是通過前面的途徑都沒能找到該鍵,viper 接下來會嘗試從配置文件中查找。
爲了不環境變量的影響,須要刪除redis.port
這個環境變量。
看快速使用中的示例。
在上面的快速使用一節,咱們已經看到了如何設置默認值,這裏就不贅述了。
io.Reader
中讀取viper 支持從io.Reader
中讀取配置。這種形式很靈活,來源能夠是文件,也能夠是程序中生成的字符串,甚至能夠從網絡鏈接中讀取的字節流。
package main import ( "bytes" "fmt" "log" "github.com/spf13/viper" ) func main() { viper.SetConfigType("toml") tomlConfig := []byte(` app_name = "awesome web" # possible values: DEBUG, INFO, WARNING, ERROR, FATAL log_level = "DEBUG" [mysql] ip = "127.0.0.1" port = 3306 user = "dj" password = 123456 database = "awesome" [redis] ip = "127.0.0.1" port = 7381 `) err := viper.ReadConfig(bytes.NewBuffer(tomlConfig)) if err != nil { log.Fatal("read config failed: %v", err) } fmt.Println("redis port: ", viper.GetInt("redis.port")) }
Unmarshal
viper 支持將配置Unmarshal
到一個結構體中,爲結構體中的對應字段賦值。
package main import ( "fmt" "log" "github.com/spf13/viper" ) type Config struct { AppName string LogLevel string MySQL MySQLConfig Redis RedisConfig } type MySQLConfig struct { IP string Port int User string Password string Database string } type RedisConfig struct { IP string Port int } func main() { viper.SetConfigName("config") viper.SetConfigType("toml") viper.AddConfigPath(".") err := viper.ReadInConfig() if err != nil { log.Fatal("read config failed: %v", err) } var c Config viper.Unmarshal(&c) fmt.Println(c.MySQL) }
編譯,運行程序,輸出:
{127.0.0.1 3306 dj 123456 awesome}
有時候,咱們想要將程序中生成的配置,或者所作的修改保存下來。viper 提供了接口!
WriteConfig
:將當前的 viper 配置寫到預約義路徑,若是沒有預約義路徑,返回錯誤。將會覆蓋當前配置;SafeWriteConfig
:與上面功能同樣,可是若是配置文件存在,則不覆蓋;WriteConfigAs
:保存配置到指定路徑,若是文件存在,則覆蓋;SafeWriteConfig
:與上面功能同樣,可是入股配置文件存在,則不覆蓋。下面咱們經過程序生成一個config.toml
配置:
package main import ( "log" "github.com/spf13/viper" ) func main() { viper.SetConfigName("config") viper.SetConfigType("toml") viper.AddConfigPath(".") viper.Set("app_name", "awesome web") viper.Set("log_level", "DEBUG") viper.Set("mysql.ip", "127.0.0.1") viper.Set("mysql.port", 3306) viper.Set("mysql.user", "root") viper.Set("mysql.password", "123456") viper.Set("mysql.database", "awesome") viper.Set("redis.ip", "127.0.0.1") viper.Set("redis.port", 6381) err := viper.SafeWriteConfig() if err != nil { log.Fatal("write config failed: ", err) } }
編譯、運行程序,生成的文件以下:
app_name = "awesome web" log_level = "DEBUG" [mysql] database = "awesome" ip = "127.0.0.1" password = "123456" port = 3306 user = "root" [redis] ip = "127.0.0.1" port = 6381
viper 能夠監聽文件修改,熱加載配置。所以不須要重啓服務器,就能讓配置生效。
package main import ( "fmt" "log" "time" "github.com/spf13/viper" ) func main() { viper.SetConfigName("config") viper.SetConfigType("toml") viper.AddConfigPath(".") err := viper.ReadInConfig() if err != nil { log.Fatal("read config failed: %v", err) } viper.WatchConfig() fmt.Println("redis port before sleep: ", viper.Get("redis.port")) time.Sleep(time.Second * 10) fmt.Println("redis port after sleep: ", viper.Get("redis.port")) }
只須要調用viper.WatchConfig
,viper 會自動監聽配置修改。若是有修改,從新加載的配置。
上面程序中,咱們先打印redis.port
的值,而後Sleep
10s。在這期間修改配置中redis.port
的值,Sleep
結束後再次打印。
發現打印出修改後的值:
redis port before sleep: 7381 redis port after sleep: 73810
另外,還能夠爲配置修改增長一個回調:
viper.OnConfigChange(func(e fsnotify.Event) { fmt.Printf("Config file:%s Op:%s\n", e.Name, e.Op) })
這樣文件修改時會執行這個回調。
viper 使用fsnotify這個庫來實現監聽文件修改的功能。
完整示例代碼見 GitHub。
歡迎關注個人微信公衆號【GoUpUp】,共同窗習,一塊兒進步~
本文由博客一文多發平臺 OpenWrite 發佈!