欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go項目配置管理神器之viper的介紹與使用詳解

 更新時間:2023年02月14日 10:48:30   作者:CK持續(xù)成長  
viper是一個完整的?Go應用程序的配置解決方案,它被設計為在應用程序中工作,并能處理所有類型的配置需求和格式,下面這篇文章主要給大家介紹了關(guān)于Go項目配置管理神器之viper的介紹與使用,需要的朋友可以參考下

1. viper的介紹

viper是go一個強大的流行的配置解決方案的庫。viper是spf13的另外一個重量級庫。有大量項目都使用該庫,比如hugo, docker等。 它基本上可以處理所有類型的配置需求和格式, viper支持功能

  • 設置默認配置
  • 支持各種配置文件,如JSON,TOML, YAML, HCL, envfile和Java屬性配置文件
  • 支持監(jiān)聽文件變化以及重新讀取配置
  • 支持從環(huán)境變量讀取配置
  • 支持從遠程配置系統(tǒng)(etcd或Consul)讀取配置,并能監(jiān)聽遠程配置修改
  • 支持從命令行標志Flag讀取配置,比如搭配cobra使用
  • 支持讀取緩沖區(qū)數(shù)據(jù)

Viper主要為我們做以下工作:

  • 查找、加載和解組JSON、TOML、YAML、HCL、INI、envfile或Java屬性格式的配置文件。
  • 提供一種機制來為不同的配置選項設置默認值。
  • 提供一種機制來為通過命令行標志指定的選項設置覆蓋值。
  • 提供別名系統(tǒng),以便在不破壞現(xiàn)有代碼的情況下輕松重命名參數(shù)。
  • 當用戶提供了與默認值相同的命令行或配置文件時,很容易區(qū)分它們。

viepr的安裝很簡單,直接再工程中使用go get命令安裝即可

$ go get github.com/spf13/viper

2. viper的使用

2.1  Viper對象的創(chuàng)建

Viper的是viper庫的主要實現(xiàn)對象, viper提供了下面的方法可以獲取Viper實例:

func GetViper() *Viper
func New() *Viper
func NewWithOptions(opts ...Option) *Viper
func Sub(key string) *Viper

使用viper.GetViper()獲取的為全局的Viper實例對象,默認使用viper包使用也是該全局Viper實例。查看viper的源碼,可以看到viper默認提供了一個全局的Viper實例:

var v *Viper
 
func init() {
	v = New()
}
 
// New returns an initialized Viper instance.
func New() *Viper {
	v := new(Viper)
	v.keyDelim = "."
	v.configName = "config"
	v.configPermissions = os.FileMode(0o644)
	v.fs = afero.NewOsFs()
	v.config = make(map[string]interface{})
	v.override = make(map[string]interface{})
	v.defaults = make(map[string]interface{})
	v.kvstore = make(map[string]interface{})
	v.pflags = make(map[string]FlagValue)
	v.env = make(map[string][]string)
	v.aliases = make(map[string]string)
	v.typeByDefValue = false
	v.logger = jwwLogger{}
 
	v.resetEncoding()
 
	return v
}

New和NewWithOptions為我們提供了創(chuàng)建實例的方法

func New1() *viper.Viper {
	return viper.New()
}
 
func New2() *viper.Viper {
	return viper.NewWithOptions()
}

Sub為我們讀取子配置項提供了一個新的實例Viper

v := viper.Sub("db")
url := v.Get("url")
log.Printf("mysql url:%s\n", url)

2.2 預設一些默認配置

viper.SetDefault("ContentDir", "content")
viper.SetDefault("Taxonomies", map[string]string{"tag": "tags", "category": "categories"})
 
viper.SetDefault("redis.port", 6379)
viper.SetDefault("mysql.url", "root:root@tcp(127.0.0.1:3306)/stock?charset=utf8mb4&parseTime=True&loc=Local")

2.3 從命令行工具的選項參數(shù)Flags讀取

viper主要提供了以下四個方法,可以綁定行參數(shù)的輸出的選項值:

func (v *Viper) BindFlagValue(key string, flag FlagValue) error
func (v *Viper) BindFlagValues(flags FlagValueSet) (err error)
func (v *Viper) BindPFlag(key string, flag *pflag.Flag) error
func (v *Viper) BindPFlags(flags *pflag.FlagSet) error

這里我們主要結(jié)合之前講的cobra庫中的pflag來講解一下viper對Flags選項參數(shù)的綁定。
在cobra中,我們主要通過cobra.Command來組織不同的命令和子命令,這里我們我通過在root根命令來做測試。代碼如下:

func init(){
	rootCmd.Flags().String("author", "YOUR NAME", "Author name for copyright attribution")
	rootCmd.Flags().String("email", "YOUR EMAIL", "Author email for contact")
 
	// 綁定多個key-value值
	viper.BindPFlags(rootCmd.Flags())
 
  // 單個綁定不同的key
	viper.BindPFlag("author", rootCmd.Flags().Lookup("author"))
	viper.BindPFlag("email", rootCmd.Flags().Lookup("email"))
 
	rootCmd.AddCommand(version.VersionCmd)
}

在cobra的命令的run回調(diào)方法中,我們通過viper的來獲取輸入的選項值

func run(){
	fmt.Println("go root cmd run")
 
	fmt.Println(viper.GetString("author"))
 
	fmt.Println(viper.GetString("email"))
}

啟動飲用,傳入?yún)?shù)測試一下:

go run main.go --author ckeen --email ck@gmail.com

查看一下打印結(jié)果,可以看到從viper成功獲取到以flag傳入的參數(shù)值:

?  cli git:(master) ? go run main.go --author keen --email ck@gmail.com
go root cmd run
ckeen
ck@gmail.com

2.4 從環(huán)境變量讀取

viper支持環(huán)境變量的函數(shù):

func (v *Viper) AutomaticEnv()			// 開啟綁定環(huán)境變量
func (v *Viper) BindEnv(input ...string) error		// 綁定系統(tǒng)中某個環(huán)境變量
func (v *Viper) SetEnvKeyReplacer(r *strings.Replacer)
func (v *Viper) SetEnvPrefix(in string)

使用方法:

使用AutomaticEnv()開啟綁定環(huán)境變量,沒開啟的時候不會從環(huán)境變量獲取,開啟后可以獲取環(huán)境變量的值。如果不想開啟所有環(huán)境變量值,可以使用BindEnv(input …string)方法綁定單個環(huán)境變量的綁定, 那么只有該綁定的環(huán)境變量的key才能獲取到值

綁定環(huán)境變量后,可以使用正常的Get方法來獲取變量值,示例代碼如下:

func testEnv(){
	v := New1()
 
	os.Setenv("CK_HOME","123")
	os.Setenv("CK_NAME","ckeen")
 
	v.AutomaticEnv()
	//v.BindEnv("SHELL")
	v.AllowEmptyEnv(true)
 
	log.Printf("os env:%+v\n", os.Environ())
 
	log.Printf("env: %+v\n", v.Get("HOME"))
	log.Printf("env: %+v\n", v.Get("SHELL"))
 
	v.SetEnvPrefix("CK")
 
	log.Printf("ck-home: %+v\n", v.Get("HOME"))
	log.Printf("ck-email: %+v\n", v.Get("NAME"))
 
}

還可以通過SetEnvPrefix()方法設置環(huán)境變量前綴, 前綴和Key之間用下劃線分割

2.5 從配置文件讀取

下面我們看一下操作實例, 先看我們的配置文件app.yml文件:

app:
  name: viper-test
  mode: dev
 
db:
  mysql:
    url: "root:root@tcp(127.0.0.1:3306)/stock?charset=utf8mb4&parseTime=True&loc=Local"
  redis:
    host: 127.0.0.1
    port:  6067
    db: 0
    passwd: 123456

初始化配置

func InitConfig() (*viper.Viper, error) {
  v := viper.New()
  v.AddConfigPath(".")					// 添加配置文件搜索路徑,點號為當前目錄
  v.AddConfigPath("./configs")		// 添加多個搜索目錄
  v.SetConfigType("yaml")				// 如果配置文件沒有后綴,可以不用配置
  v.SetConfigName("app.yml")			// 文件名,沒有后綴
	
	// v.SetConfigFile("configs/app.yml")
  
  // 讀取配置文件
  if err := v.ReadInConfig(); err == nil {
		log.Printf("use config file -> %s\n", v.ConfigFileUsed())
	} else {
		return nil,err
	}
  return v, nil
}

首先這里我們添加一個配置文件搜索路徑,點號表示當前路徑,搜索路徑可以添加多個然后設置了配置文件類型,這里我們設置文件類型為yaml,

接著我們設置了配置文件名稱,這個文件可以從配置的搜索路徑從查找。

最后我們通過提供的ReadInConfig()函數(shù)讀取配置文件

讀取配置文件

// 通過.號來區(qū)分不同層級,來獲取配置值
log.Printf("app.mode=%s\n", v.Get("app.mode"))
log.Printf("db.mysql.url=%s\n", v.Get("db.mysql.url"))
log.Printf("db.redis.host=%s\n", v.GetString("db.redis.host"))
log.Printf("db.redis.port=%d\n", v.GetInt("db.redis.port"))
 
// 使用Sub獲取子配置,然后獲取配置值
v2 := v.Sub("db")
log.Printf("db.mysql.url:%s\n", v2.Sub("mysql").GetString("url"))
log.Printf("db.redis.host:%s\n", v2.Sub("redis").GetString("host"))
log.Printf("db.redis.port:%s\n", v2.Sub("redis").GetInt("port"))

viper還提供了如下獲取類型獲取配置項值:

 注: 其中重要的一個函數(shù)IsSet可以用來判斷某個key是否被設置

2.6 從遠程key/value存儲讀取

在Viper中啟用遠程支持,需要在代碼中匿名導入viper/remote這個包。

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

Viper將讀取從Key/Value存儲中的路徑檢索到的配置字符串(如JSONTOML、YAML格式)。viper目前支持Consul/Etcd/firestore三種Key/Value的存儲系統(tǒng)。下面我來演示從etcd讀取配置:

首先我們安裝crypt的工具

go get github.com/bketelsen/crypt/bin/crypt

使用crypt的命令,將app.yml的文件添加到detcd

crypt set --endpoint=http://127.0.0.1:2379 -plaintext /config/app.yml /Users/ckeen/Documents/code/gosource/go-awesome/go-samples/viper/configs/app.yml

添加viper的操作遠程資源的配置

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

實現(xiàn)從遠程讀取配置

func InitConfigFromRemote() (*viper.Viper,error) {
	v := viper.New()
	// 遠程配置
	v.AddRemoteProvider("etcd","http://127.0.0.1:2379","config/app.yml")
	//v.SetConfigType("json")
	v.SetConfigFile("app.yml")
	v.SetConfigType("yml")
 
	if err := v.ReadRemoteConfig(); err == nil {
		log.Printf("use config file -> %s\n", v.ConfigFileUsed())
	} else {
		return nil, err
	}
	return v, nil
}
 
func main(){
  
  v, err := InitConfigFromRemote()
	if err != nil {
		log.Printf("read remote error:%+v\n")
	}
 
	log.Printf("remote read app.mode=%+v\n", v.GetString("app.mode"))
	log.Printf("remote read db.mysql.url=%+v\n", v.GetString("db.mysql.url"))
 
}

測試打印結(jié)果

2.7 監(jiān)聽配置變化

viper提供如下兩種監(jiān)聽配置的函數(shù),一個是本地的監(jiān)聽和一個遠程監(jiān)聽的:

func (v *Viper) WatchConfig()
func (v *Viper) WatchRemoteConfig() error
func (v *Viper) WatchRemoteConfigOnChannel() error

我們主要看一下監(jiān)聽本地文件變更的示例

v, err := InitConfig()
if err != nil {
log.Fatalf("viper讀取失敗, error:%+v\n",err)
}
 
// 監(jiān)聽到文件變化后的回調(diào)
v.OnConfigChange(func(e fsnotify.Event) {
  fmt.Println("Config file changed:", e.Name)
  fmt.Println(v.Get("db.redis.passwd"))
})
 
v.WatchConfig()
 
// 阻塞進程退出
time.Sleep(time.Duration(1000000) * time.Second)

我們使用前面的InitConfig()方法來初始化本地文件讀取配置,然后設定了監(jiān)聽函數(shù),最后使用WatchConfig()開啟本地文件監(jiān)聽。

當我們修改本地配置configs/app.yml的db.redis.passwd的值,然后保存后,我們可以看到控制臺有打印最新修改后的值,不要我們重新去獲取。

2.8 寫入配置到文件

viper提供了如下四個寫入配置文件發(fā)方法

func (v *Viper) SafeWriteConfig() error
func (v *Viper) SafeWriteConfigAs(filename string) error
func (v *Viper) WriteConfig() error
func (v *Viper) WriteConfigAs(filename string) error

使用SafeWriteConfig()和WriteConfig()時,可以先設定SetConfigFile()設定配置文件的路徑。配置寫入示例:

v := New1()
v.SetConfigFile("./hello.yml")
 
log.Printf("config path:%+v\n", v.ConfigFileUsed())
 
v.SetDefault("author","CKeen")
v.SetDefault("email", "ck@gmail.com")
 
v.Set("hello", "foo")
 
v.Set("slice", []string {"slice1","slice2","slice3"})
 
v.SetDefault("test.web", "https://ckeen.cn")
 
v.WriteConfig()
 
//v.WriteConfigAs("./hello.yml")

如果使用SafeWriteConfigAs()或者WriteConfigAs()方法,則直接傳入配置文件路徑即可。

3. 源碼分析--配置讀取的順序

通過上面的示例我們知道,viper讀取配置主要通過一系列Get方法來實現(xiàn),我們從Get方法跳轉(zhuǎn)到源碼可以發(fā)現(xiàn), 主要獲取的配置值的為find方法, 方法實現(xiàn)如下:

func (v *Viper) find(lcaseKey string, flagDefault bool) interface{} {
	var (
		val    interface{}
		exists bool
		path   = strings.Split(lcaseKey, v.keyDelim)
		nested = len(path) > 1
	)
 
	// compute the path through the nested maps to the nested value
	if nested && v.isPathShadowedInDeepMap(path, castMapStringToMapInterface(v.aliases)) != "" {
		return nil
	}
 
	// if the requested key is an alias, then return the proper key
	lcaseKey = v.realKey(lcaseKey)
	path = strings.Split(lcaseKey, v.keyDelim)
	nested = len(path) > 1
 
	// Set() override first
	val = v.searchMap(v.override, path)
	if val != nil {
		return val
	}
	if nested && v.isPathShadowedInDeepMap(path, v.override) != "" {
		return nil
	}
 
	// PFlag override next
	flag, exists := v.pflags[lcaseKey]
	if exists && flag.HasChanged() {
		switch flag.ValueType() {
		case "int", "int8", "int16", "int32", "int64":
			return cast.ToInt(flag.ValueString())
		case "bool":
			return cast.ToBool(flag.ValueString())
		case "stringSlice", "stringArray":
			s := strings.TrimPrefix(flag.ValueString(), "[")
			s = strings.TrimSuffix(s, "]")
			res, _ := readAsCSV(s)
			return res
		case "intSlice":
			s := strings.TrimPrefix(flag.ValueString(), "[")
			s = strings.TrimSuffix(s, "]")
			res, _ := readAsCSV(s)
			return cast.ToIntSlice(res)
		case "stringToString":
			return stringToStringConv(flag.ValueString())
		default:
			return flag.ValueString()
		}
	}
	if nested && v.isPathShadowedInFlatMap(path, v.pflags) != "" {
		return nil
	}
 
	// Env override next
	if v.automaticEnvApplied {
		// even if it hasn't been registered, if automaticEnv is used,
		// check any Get request
		if val, ok := v.getEnv(v.mergeWithEnvPrefix(lcaseKey)); ok {
			return val
		}
		if nested && v.isPathShadowedInAutoEnv(path) != "" {
			return nil
		}
	}
	envkeys, exists := v.env[lcaseKey]
	if exists {
		for _, envkey := range envkeys {
			if val, ok := v.getEnv(envkey); ok {
				return val
			}
		}
	}
	if nested && v.isPathShadowedInFlatMap(path, v.env) != "" {
		return nil
	}
 
	// Config file next
	val = v.searchIndexableWithPathPrefixes(v.config, path)
	if val != nil {
		return val
	}
	if nested && v.isPathShadowedInDeepMap(path, v.config) != "" {
		return nil
	}
 
	// K/V store next
	val = v.searchMap(v.kvstore, path)
	if val != nil {
		return val
	}
	if nested && v.isPathShadowedInDeepMap(path, v.kvstore) != "" {
		return nil
	}
 
	// Default next
	val = v.searchMap(v.defaults, path)
	if val != nil {
		return val
	}
	if nested && v.isPathShadowedInDeepMap(path, v.defaults) != "" {
		return nil
	}
 
	if flagDefault {
		// last chance: if no value is found and a flag does exist for the key,
		// get the flag's default value even if the flag's value has not been set.
		if flag, exists := v.pflags[lcaseKey]; exists {
			switch flag.ValueType() {
			case "int", "int8", "int16", "int32", "int64":
				return cast.ToInt(flag.ValueString())
			case "bool":
				return cast.ToBool(flag.ValueString())
			case "stringSlice", "stringArray":
				s := strings.TrimPrefix(flag.ValueString(), "[")
				s = strings.TrimSuffix(s, "]")
				res, _ := readAsCSV(s)
				return res
			case "intSlice":
				s := strings.TrimPrefix(flag.ValueString(), "[")
				s = strings.TrimSuffix(s, "]")
				res, _ := readAsCSV(s)
				return cast.ToIntSlice(res)
			case "stringToString":
				return stringToStringConv(flag.ValueString())
			default:
				return flag.ValueString()
			}
		}
		// last item, no need to check shadowing
	}
 
	return nil
}

通過源碼,我們可以知道viper讀取配置的優(yōu)先級順序:alias別名 > 調(diào)用Set設置 > flag > env > config > key/value store > default

還有一個注意點:viper配置鍵不區(qū)分大小寫,因為viper內(nèi)部對key統(tǒng)一轉(zhuǎn)為了小寫。

4. 參考資料

viper的包地址:viper package - github.com/spf13/viper - Go Packages

viper的github地址: GitHub - spf13/viper: Go configuration with fangs

總結(jié)

到此這篇關(guān)于Go項目配置管理神器之viper的介紹與使用的文章就介紹到這了,更多相關(guān)Go配置管理神器viper使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • GoLang協(xié)程庫libtask學習筆記

    GoLang協(xié)程庫libtask學習筆記

    libtask一個C語言的協(xié)程庫,是go語言的前身很早期的原型. 測試機器是我的mac air 安裝的centos虛擬機(只有一個核), 代碼沒有采用任何優(yōu)化,只是使用默認配置
    2022-12-12
  • Golang實現(xiàn)多存儲驅(qū)動設計SDK案例

    Golang實現(xiàn)多存儲驅(qū)動設計SDK案例

    這篇文章主要介紹了Golang實現(xiàn)多存儲驅(qū)動設計SDK案例,Gocache是一個基于Go語言編寫的多存儲驅(qū)動的緩存擴展組件,更多具體內(nèi)容感興趣的小伙伴可以參考一下
    2022-09-09
  • Go語言中Map的神奇操作小結(jié)

    Go語言中Map的神奇操作小結(jié)

    Map是一個強大而又有趣的工具,它可以幫助我們高效地存儲和操作鍵值對數(shù)據(jù),本文主要介紹了Go語言中Map的各種操作,包括增加、查找、刪除、遍歷等,具有一定的參考價值,感興趣的可以了解一下
    2023-08-08
  • Golang中下劃線(_)的不錯用法分享

    Golang中下劃線(_)的不錯用法分享

    golang中的下劃線表示忽略變量的意思,也沒有產(chǎn)生新的變量,但是后面的表達式依然會被執(zhí)行,本文為大家整理了golang中下劃線的一些不錯的用法,需要的可以參考下
    2023-05-05
  • GoFrame框架garray對比PHP的array優(yōu)勢

    GoFrame框架garray對比PHP的array優(yōu)勢

    這篇文章主要為大家介紹了GoFrame框架garray對比PHP的array優(yōu)勢詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06
  • golang中的空接口使用詳解

    golang中的空接口使用詳解

    這篇文章主要介紹了golang中的空接口使用,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • Golang?HTTP服務超時控制實現(xiàn)原理分析

    Golang?HTTP服務超時控制實現(xiàn)原理分析

    這篇文章主要介紹了Golang?HTTP服務超時控制實現(xiàn)原理,HTTP服務的超時控制是保障服務高可用性的重要措施之一,由于HTTP服務可能會遇到網(wǎng)絡延遲,資源瓶頸等問題,因此需要對請求進行超時控制,以避免服務雪崩等問題,需要的朋友可以參考下
    2023-05-05
  • 一文教你如何在Golang中用好泛型

    一文教你如何在Golang中用好泛型

    golang的泛型已經(jīng)出來了一年多了,從提案被接受開始我就在關(guān)注泛型了,好用是好用,但問題也很多,所以本文就來教大家如何在Golang中用好泛型吧
    2023-07-07
  • Go語言提升開發(fā)效率的語法糖技巧分享

    Go語言提升開發(fā)效率的語法糖技巧分享

    每門語言都有自己的語法糖,像java的語法糖就有方法變長參數(shù)、拆箱與裝箱、枚舉、for-each等等,Go語言也不例外。本文就來介紹一些Go語言的語法糖,需要的可以參考一下
    2022-07-07
  • Go語言中Timer計時器的使用技巧詳解

    Go語言中Timer計時器的使用技巧詳解

    Go語言中的time包里有個Timer計時器的功能,這篇文章主要就是來和大家介紹一下Timer計時器的使用技巧,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-07-07

最新評論