Go 每日一庫之 viper

簡介

上一篇文章介紹 cobra 的時候提到了 viper,今天咱們就來介紹一下這個庫。
viper 是一個配置解決方案,擁有豐富的特性:mysql

  • 支持 JSON/TOML/YAML/HCL/envfile/Java properties 等多種格式的配置文件;
  • 能夠設置監聽配置文件的修改,修改時自動加載新的配置;
  • 從環境變量、命令行選項和io.Reader中讀取配置;
  • 從遠程配置系統中讀取和監聽修改,如 etcd/Consul;
  • 代碼邏輯中顯示設置鍵值。

快速使用

安裝: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 minutesgolang

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

  • 設置文件名時不要帶後綴;
  • 搜索路徑能夠設置多個,viper 會根據設置順序依次查找;
  • viper 獲取值時使用section.key的形式,即傳入嵌套的鍵名;
  • 默認值能夠調用viper.SetDefault設置。

讀取鍵

viper 提供了多種形式的讀取方法。在上面的例子中,咱們看到了Get方法的用法。Get方法返回一個interface{}的值,使用有所不便。服務器

GetType系列方法能夠返回指定類型的值。
其中,Type 能夠爲Bool/Float64/Int/String/Time/Duration/IntSlice/StringSlice
可是請注意,若是指定的鍵不存在或類型不正確,GetType方法返回對應類型的零值微信

若是要判斷某個鍵是否存在,使用IsSet方法。
另外,GetStringMapGetStringMapString直接以 map 返回某個鍵下面全部的鍵值對,前者返回map[string]interface{},後者返回map[string]string
AllSettingsmap[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 中添加protocolsports配置:

[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接受的格式均可以,例如3s2min1min30s等。

設置鍵值

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

參考

  1. viper GitHub 倉庫

個人博客

歡迎關注個人微信公衆號【GoUpUp】,共同窗習,一塊兒進步~

本文由博客一文多發平臺 OpenWrite 發佈!

相關文章
相關標籤/搜索