Golang中配置信息处理框架Viper有什么用

技术Golang中配置信息处理框架Viper有什么用这篇文章主要介绍Golang中配置信息处理框架Viper有什么用,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Viper项目地址:https:/

本文主要介绍配置信息处理框架Viper在Golang中的使用。很详细,有一定的参考价值。感兴趣的朋友一定要看!

Viper

项目地址:https://github.com/spf13/viper

这篇文章是从这个项目的README.md文件的内容翻译过来的。

有许多Go语言项目使用Viper框架,例如:

雨果

EMC RexRay

伊姆古尔的现任者

纳米盒/纳米包

码头公证人

BloomApi

doctl

克莱尔特尔

什么是Viper

Viper是Go语言应用程序处理配置信息的库。它可以处理多种格式的配置。它支持的功能:

设置默认值

从JSON、TOML、YAML、HCL和Java属性文件中读取配置数据

您可以监视配置文件的更改并重新读取它。

从环境变量中读取配置数据

从远程配置系统读取数据并监控它们(如etcd、Consul)

从命令参数读取配置

从缓冲区读取

调用函数设置配置信息

为什么要使用Viper

构建现代应用时,不用担心配置文件格式;你可以专注于构建优秀的软件。

Viper可以执行以下操作:

并以JSON、TOML、YAML、HCL或Java属性格式加载配置文件。

您可以为各种配置项目设置默认值。

您可以在命令行上指定一个配置项来覆盖配置值。

提供了别名系统,可以在不破坏现有代码的情况下重命名参数。

用户提供的命令行参数或配置文件与默认文件之间的区别很容易区分。

Viper按照优先级从高到低的顺序读取配置信息,如下所示:

显式调用集合函数

命令行参数

环境变量

配置文件

密钥/值存储系统

系统默认值

Viper的CI的键不区分大小写。

00-1010

设置值

默认值不是必需的。如果未指定配置文件、环境变量、远程配置系统、命令行参数和Set函数,默认值将生效。

示例:

毒蛇。SetDefault('ContentDir ',' content ')

毒蛇。SetDefault('LayoutDir ',' layouts ')

Viper.setdefault('分类学',map[string]string { ' tag ' : ' tags ',' category ' 3360 ' categories ' })

设置默认值

Viper支持JSON,TOML,YAML,HCL和Java属性文件。

Viper可以搜索多条路径,但目前单个Viper实例只支持单个概要文件。

默认情况下,Viper不搜索任何路径。

以下是如何使用Viper搜索和读取配置文件的示例。

路径不是必需的,但最好至少提供一个路径来查找配置文件。

viper . setconfigname(' config ')//设置配置文件名(不带后缀)

viper . addconfigpath('/etc/appname/')//第一个搜索路径

毒蛇。addconfigpath ('$ home/。appname’)//添加路径可以多次调用。

毒蛇。AddConfigPath(' . ')n

bsp;      // 比如添加当前目录
err := viper.ReadInConfig() // 搜索路径,并读取配置数据
if err != nil {
    panic(fmt.Errorf("Fatal error config file: %s \n", err))
}

监视配置文件,重新读取配置数据

Viper支持让您的应用程序在运行时拥有读取配置文件的能力。
需要重新启动服务器以使配置生效的日子已经一去不复返了,由viper驱动的应用程序可以在运行时读取已更新的配置文件,并且不会错过任何节拍。
只需要调用viper实例的WatchConfig函数,你也可以指定一个回调函数来获得变动的通知。

viper.WatchConfig()
viper.OnConfigChange(func(e fsnotify.Event) {
    fmt.Println("Config file changed:", e.Name)
})

从 io.Reader 中读取配置

Viper预先定义了许多配置源,例如文件、环境变量、命令行参数和远程K / V存储系统,但您并未受其约束。
您也可以实现自己的配置源,并提供给viper。

viper.SetConfigType("yaml") // or viper.SetConfigType("YAML")
// any approach to require this configuration into your program.
var yamlExample = []byte(`
Hacker: true
name: steve
hobbies:
- skateboarding
- snowboarding
- go
clothing:
  jacket: leather
  trousers: denim
age: 35
eyes : brown
beard: true
`)
viper.ReadConfig(bytes.NewBuffer(yamlExample))
viper.Get("name") // 返回 "steve"

Set 调用

viper.Set("Verbose", true)
viper.Set("LogFile", LogFile)

注册并使用别名

别名可以实现多个key引用单个值。

viper.RegisterAlias("loud", "Verbose")
viper.Set("verbose", true) 
viper.Set("loud", true)   // 这两句设置的都是同一个值
viper.GetBool("loud") // true
viper.GetBool("verbose") // true

从环境变量中读取

Viper 完全支持环境变量,这是的应用程序可以开箱即用。
有四个和环境变量有关的方法:

  • AutomaticEnv()

  • BindEnv(string...) : error

  • SetEnvPrefix(string)

  • SetEnvKeyReplacer(string...) *strings.Replacer

注意,环境变量时区分大小写的。

Viper提供了一种机制来确保Env变量是唯一的。通过SetEnvPrefix,在从环境变量读取时会添加设置的前缀。BindEnv和AutomaticEnv都会使用到这个前缀。

BindEnv需要一个或两个参数。第一个参数是键名,第二个参数是环境变量的名称。环境变量的名称区分大小写。如果未提供ENV变量名称,则Viper会自动假定该键名称与ENV变量名称匹配,并且ENV变量为全部大写。当您显式提供ENV变量名称时,它不会自动添加前缀。

使用ENV变量时要注意,当关联后,每次访问时都会读取该ENV值。Viper在BindEnv调用时不读取ENV值。

AutomaticEnv与SetEnvPrefix结合将会特别有用。当AutomaticEnv被调用时,任何viper.Get请求都会去获取环境变量。环境变量名为SetEnvPrefix设置的前缀,加上对应名称的大写。

SetEnvKeyReplacer允许你使用一个strings.Replacer对象来将配置名重写为Env名。如果你想在Get()中使用包含-的配置名 ,但希望对应的环境变量名包含_分隔符,就可以使用该方法。使用它的一个例子可以在项目中viper_test.go文件里找到。
例子:

SetEnvPrefix("spf") // 将会自动转为大写
BindEnv("id")
os.Setenv("SPF_ID", "13") // 通常通过系统环境变量来设置
id := Get("id") // 13

绑定命令行参数

Viper支持绑定pflags参数。
和BindEnv一样,当绑定方法被调用时,该值没有被获取,而是在被访问时获取。这意味着应该尽早进行绑定,甚至是在init()函数中绑定。

利用BindPFlag()方法可以绑定单个flag。
例子:

serverCmd.Flags().Int("port", 1138, "Port to run Application server on")
viper.BindPFlag("port", serverCmd.Flags().Lookup("port"))

你也可以绑定已存在的pflag集合 (pflag.FlagSet):

pflag.Int("flagname", 1234, "help message for flagname")
pflag.Parse()
viper.BindPFlags(pflag.CommandLine)
i := viper.GetInt("flagname") // 通过viper从pflag中获取值

使用pflag并不影响其他库使用标准库中的flag。通过导入,pflag可以接管通过标准库的flag定义的参数。这是通过调用pflag包中的AddGoFlagSet()方法实现的。
例子:

package main
import (
    "flag"
    "github.com/spf13/pflag"
)
func main() {
    // using standard library "flag" package
    flag.Int("flagname", 1234, "help message for flagname")
    pflag.CommandLine.AddGoFlagSet(flag.CommandLine)
    pflag.Parse()
    viper.BindPFlags(pflag.CommandLine)
    i := viper.GetInt("flagname") // retrieve value from viper
    ...
}

Flag 接口

如果你不想使用pflag,Viper 提供了两个接口来实现绑定其他的flag系统。
使用 FlagValue 接口代表单个flag。下面是实现了该接口的简单的例子:

type myFlag struct {}
func (f myFlag) HasChanged() bool { return false }
func (f myFlag) Name() string { return "my-flag-name" }
func (f myFlag) ValueString() string { return "my-flag-value" }
func (f myFlag) ValueType() string { return "string" }

一旦你实现了该接口,就可以绑定它:

viper.BindFlagValue("my-flag-name", myFlag{})

使用 FlagValueSet 接口代表一组flag。下面是实现了该接口的简单的例子:

type myFlagSet struct {
    flags []myFlag
}
func (f myFlagSet) VisitAll(fn func(FlagValue)) {
    for _, flag := range flags {
        fn(flag)
    }
}

一旦你实现了该接口,就可以绑定它:

fSet := myFlagSet{
    flags: []myFlag{myFlag{}, myFlag{}},
}
viper.BindFlagValues("my-flags", fSet)

支持远程 Key/Value 存储

启用该功能,需要导入viper/remot包:

import _ "github.com/spf13/viper/remote"

Viper 可以从例如etcd、Consul的远程Key/Value存储系统的一个路径上,读取一个配置字符串(JSON, TOML, YAML或HCL格式)。
这些值优先于默认值,但会被从磁盘文件、命令行flag、环境变量的配置所覆盖。

Viper 使用 crypt 来从  K/V 存储系统里读取配置,这意味着你可以加密储存你的配置信息,并且可以自动解密配置信息。加密是可选的。

您可以将远程配置与本地配置结合使用,也可以独立使用。

crypt 有一个命令行工具可以帮助你存储配置信息到K/V存储系统,crypt默认使用 http://127.0.0.1:4001 上的etcd。

$ go get github.com/xordataexchange/crypt/bin/crypt
$ crypt set -plaintext /config/hugo.json /Users/hugo/settings/config.json

确认你的值被设置:

$ crypt get -plaintext /config/hugo.json

有关crypt如何设置加密值或如何使用Consul的示例,请参阅文档。

远程 Key/Value 存储例子 - 未加密的

viper.AddRemoteProvider("etcd", "http://127.0.0.1:4001","/config/hugo.json")
viper.SetConfigType("json") // 因为不知道格式,所以需要指定,支持的格式有"json"、"toml"、"yaml"、"yml"、"properties"、"props"、"prop"
err := viper.ReadRemoteConfig()

远程 Key/Value 存储例子 - 加密的

viper.AddSecureRemoteProvider("etcd","http://127.0.0.1:4001","/config/hugo.json","/etc/secrets/mykeyring.gpg")
viper.SetConfigType("json") // 因为不知道格式,所以需要指定,支持的格式有"json"、"toml"、"yaml"、"yml"、"properties"、"props"、"prop"
err := viper.ReadRemoteConfig()

监视etcd的变化 - 未加密的

// 您可以创建一个新的viper实例
var runtime_viper = viper.New()
runtime_viper.AddRemoteProvider("etcd", "http://127.0.0.1:4001", "/config/hugo.yml")
runtime_viper.SetConfigType("yaml") // 因为不知道格式,所以需要指定,支持的格式有"json"、"toml"、"yaml"、"yml"、"properties"、"props"、"prop"
// 从远程读取配置
err := runtime_viper.ReadRemoteConfig()
// 解析配置到runtime_conf中
runtime_viper.Unmarshal(&runtime_conf)
// 通过一个goroutine远程的配置变化
go func(){
    for {
        time.Sleep(time.Second * 5) // delay after each request
        // currently, only tested with etcd support
        err := runtime_viper.WatchRemoteConfig()
        if err != nil {
            log.Errorf("unable to read remote config: %v", err)
            continue
        }
            // 解析新的配置到一个结构体变量中,你也可以使用channel实现一个信号通知的方式
        runtime_viper.Unmarshal(&runtime_conf)
    }
}()

获取值

在Viper中,有一些根据值的类型获取值的方法。存在一下方法:

  • Get(key string) : interface{}

  • GetBool(key string) : bool

  • GetFloat64(key string) : float64

  • GetInt(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

如果Get函数未找到值,则返回对应类型的一个零值。可以通过 IsSet() 方法来检测一个健是否存在。
例子:

viper.GetString("logfile") // Setting & Getting 不区分大小写
if viper.GetBool("verbose") {
    fmt.Println("verbose enabled")
}

访问嵌套键

访问方法也接受嵌套的键。例如,如果加载了以下JSON文件:

{
    "host": {
        "address": "localhost",
        "port": 5799
    },
    "datastore": {
        "metric": {
            "host": "127.0.0.1",
            "port": 3099
        },
        "warehouse": {
            "host": "198.0.0.1",
            "port": 2112
        }
    }
}

Viper可以通过.分隔符来访问嵌套的字段:

GetString("datastore.metric.host") // (returns "127.0.0.1")

这遵守前面确立的优先规则; 会搜索路径中所有配置,直到找到为止。
例如,上面的文件,datastore.metric.host和 datastore.metric.port都已经定义(并且可能被覆盖)。如果另外 datastore.metric.protocol的默认值,Viper也会找到它。

但是,如果datastore.metric值被覆盖(通过标志,环境变量,Set方法,...),则所有datastore.metric的子键将会未定义,它们被优先级更高的配置值所“遮蔽”。

最后,如果存在相匹配的嵌套键,则其值将被返回。例如:

{
    "datastore.metric.host": "0.0.0.0",
    "host": {
        "address": "localhost",
        "port": 5799
    },
    "datastore": {
        "metric": {
            "host": "127.0.0.1",
            "port": 3099
        },
        "warehouse": {
            "host": "198.0.0.1",
            "port": 2112
        }
    }
}
GetString("datastore.metric.host") // returns "0.0.0.0"

提取子树配置

可以从viper中提取子树。例如, viper配置为:

app:
  cache1:
    max-items: 100
    item-size: 64
  cache2:
    max-items: 200
    item-size: 80

执行后:

subv := viper.Sub("app.cache1")

subv 就代表:

max-items: 100
item-size: 64

假如我们有如下函数:

func NewCache(cfg *Viper) *Cache {...}

它的功能是根据配置信息创建缓存缓存。现在很容易分别创建这两个缓存:

cfg1 := viper.Sub("app.cache1")
cache1 := NewCache(cfg1)
cfg2 := viper.Sub("app.cache2")
cache2 := NewCache(cfg2)

解析配置

您还可以选择将所有或特定值解析到struct、map等。
有两个方法可以做到这一点:

  • Unmarshal(rawVal interface{}) : error

  • UnmarshalKey(key string, rawVal interface{}) : error

例如:

type config struct {
    Port int
    Name string
    PathMap string `mapstructure:"path_map"`
}
var C config
err := Unmarshal(&C)
if err != nil {
    t.Fatalf("unable to decode into struct, %v", err)
}

使用单个viper还是多个viper

Viper随时准备使用开箱即用。没有任何配置或初始化也可以使用Viper。由于大多数应用程序都希望使用单个存储中心进行配置,因此viper包提供了此功能。它类似于一个单例模式。

在上面的所有示例中,他们都演示了如何使用viper的单例风格的方式。

使用多个viper实例

您还可以创建多不同的viper实例以供您的应用程序使用。每实例都有自己独立的设置和配置值。每个实例可以从不同的配置文件,K/V存储系统等读取。viper包支持的所有函数也都有对应的viper实例方法。
例子:

x := viper.New()
y := viper.New()
x.SetDefault("ContentDir", "content")
y.SetDefault("ContentDir", "foobar")
//...

当使用多个viper实例时,用户需要自己管理每个实例。

以上是“Golang中配置信息处理框架Viper有什么用”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注行业资讯频道!

内容来源网络,如有侵权,联系删除,本文地址:https://www.230890.com/zhan/102534.html

(0)

相关推荐

  • 如何在Action类中获得HttpServletResponse对象

    技术如何在Action类中获得HttpServletResponse对象这篇文章将为大家详细讲解有关如何在Action类中获得HttpServletResponse对象,文章内容质量较高,因此小编分享给大家做个参考,希望

    攻略 2021年11月12日
  • 如何用配置文件来管理多个Node.js进程

    技术如何用配置文件来管理多个Node.js进程这篇文章主要介绍“如何用配置文件来管理多个Node.js进程”,在日常操作中,相信很多人在如何用配置文件来管理多个Node.js进程问题上存在疑惑,小编查阅了各式资料,整理出

    攻略 2021年11月29日
  • 哈利波特一共几部,《哈利波特》电影版共出过几部

    技术哈利波特一共几部,《哈利波特》电影版共出过几部一共七部,顺序如下:1哈利波特一共几部、哈利·波特与魔法石(Harry Potter and the Sorcerers Stone)——2001年2、哈利·波特与密室(

    生活 2021年10月19日
  • 怎样才是好代码

    技术怎样才是好代码 怎样才是好代码这是我一直都在思考的问题,或者说是我对怎么样才是一个好的程序员的思考。
    场景
    任何问题以及工具的使用,不讨论其应用场景,无疑是耍流氓。我这里讨论的对象主要是从事互联网开

    礼包 2021年11月1日
  • Kerberos命令怎么使用

    技术Kerberos命令怎么使用Kerberos命令怎么使用,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1.登录 kinit admin/admin@

    攻略 2021年11月11日
  • sparkmllib算法实例(sparkmllib系统推荐)

    技术如何进行SparkMllib主题模型案例的分析如何进行SparkMllib主题模型案例的分析,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获

    2021年12月17日