文章博客地址:golang 应用 viper 加载配置 主动反序列化到构造
- golang应用 viper 无需设置 mapstructure tag 依据配置文件后缀 主动返序列化到构造
- 解决构造有下划线的字段解析不胜利问题
viper 失常加载配置文件
golang viper 其中能够用来 查找、加载和反序列化JSON、TOML、YAML、HCL、INI、envfile和格局的配置文件
配置文件 test_toml.toml
http_addr = ":8082"grpc_addr = ":8083"jaeger_url= "http://localhost:14268/api/traces"tracing= true
golang代码
type ConfigTest struct { HttpAddr string `json:"http_addr" toml:"http_addr" yaml:"http_addr"` GrpcAddr string `json:"grpc_addr" toml:"grpc_addr" yaml:"grpc_addr"` JaegerUrl string `json:"jaeger_url" toml:"jaeger_url" yaml:"jaeger_url" mapstructure:"jaeger_url"` Tracing bool `toml:"tracing" json:"tracing" yaml:"tracing" ` // opentelemetry tracing}// jaeger 加载配置文件func TestSourceFile_Unmarshal(t *testing.T) { filePath := "./test_toml.toml" viper.SetConfigFile(filePath) if err := viper.ReadInConfig(); err != nil { t.Error(err) } c := &ConfigTest{} if err := viper.Unmarshal(c); err != nil { t.Error(err) } logger.Infow("Unmarshal file sucess", "v", c)}
打印返序列化的配置构造
{"level":"info","ts":"2023-08-27T21:35:27.041+0800","caller":"config/source_file_test.go:31","msg":"Unmarshal file sucess","v":{"http_addr":"","grpc_addr":"","jaeger_url":"http://localhost:14268/api/traces","tracing":true}}
能够看到带下划线的字段,不加 mapstructure 标签,是不会反序列化
不加 mapstructure tag实现主动反序列化
查看viper Unmarshal 代码
func (v *Viper) Unmarshal(rawVal interface{}, opts ...DecoderConfigOption) error { return decode(v.AllSettings(), defaultDecoderConfig(rawVal, opts...))}func decode(input interface{}, config *mapstructure.DecoderConfig) error { decoder, err := mapstructure.NewDecoder(config) if err != nil { return err } return decoder.Decode(input)}func NewDecoder(config *DecoderConfig) (*Decoder, error) { if config.TagName == "" { config.TagName = "mapstructure" } // ...}
- 从代码看出 Viper应用的是 github.com/mitchellh/mapstructure来解析值
- mapstructure 用于将通用的map[string]interface{}解码到对应的 Go 构造体中
- 默认状况下,mapstructure 应用构造体中字段的名称做这个映射,不辨别大小写,比方 Name 字段能够映射到 name、NAME、NaMe 等等
- 如果没有指定 tagName ,则默认为 mapstructure,这也是为什么带下划线的字段不加 mapstructure 标签无奈解析的起因
- viper 中Unmarshal的第二个参数是能够指定 DecoderConfigOption ,从而能够指定 tagName
viper依据文类型件主动解码到构造
- 读取文件后缀比方 toml
- 依据后缀设置 tagName
- 调用 viper.Unmarshal解析
func TestSourceFile_Unmarshal1(t *testing.T) { filePath := "./test_toml.toml" c := &ConfigTest{} if err := viperUnmarshal(c, filePath); err != nil { t.Error(err) } logger.Infow("Unmarshal file sucess", "v", c)}func viperUnmarshal(v interface{}, configPath string) error { var tagName string ext := filepath.Ext(configPath) if len(ext) > 1 { tagName = ext[1:] } // set decode tag_name, default is mapstructure decoderConfigOption := func(c *mapstructure.DecoderConfig) { c.TagName = tagName } cViper := viper.New() cViper.SetConfigFile(configPath) if err := cViper.ReadInConfig(); err != nil { return err } return cViper.Unmarshal(v, decoderConfigOption)}
{"level":"info","ts":"2023-08-27T21:35:34.553+0800","caller":"config/source_file_test.go:40","msg":"Unmarshal file sucess","v":{"http_addr":":8082","grpc_addr":":8083","jaeger_url":"http://localhost:14268/api/traces","tracing":true}}
我已将viper加载配置集成进本人的我的项目,残缺example 代码能够查看 source_file_test.go