上一篇文章介绍 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 发布!