-
Notifications
You must be signed in to change notification settings - Fork 0
/
viper.go
146 lines (129 loc) · 3.18 KB
/
viper.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
package viper
import (
"github.com/hopeio/cherry/utils/log"
"github.com/spf13/viper"
_ "github.com/spf13/viper/remote"
"gopkg.in/ini.v1"
"os"
)
// 全局变量,只一个实例,只提供config
type Config struct {
Debug bool
Watch bool
ConfigName string
ConfigFile string
ConfigType string
ConfigPermissions os.FileMode
EnvPrefix string
RemoteProvider []RemoteProvider
AllowEmptyEnv bool
IniLoadOptions ini.LoadOptions
EnvVars []string
}
type RemoteProvider struct {
Provider string
Endpoint string
Path string
SecretKeyring string
}
func (c *Config) InitBeforeInject() {
}
func (c *Config) Init() {
if c.ConfigType == "" {
c.ConfigType = "toml"
}
}
func (c *Config) InitAfterInject() {
c.Init()
c.build(viper.GetViper())
}
func (c *Config) Build() *viper.Viper {
c.Init()
var runtimeViper = viper.New()
c.build(runtimeViper)
return runtimeViper
}
func (c *Config) build(runtimeViper *viper.Viper) {
if c.Debug {
runtimeViper.Debug()
}
runtimeViper.SetConfigType(c.ConfigType) // because there is no file extension in a stream of bytes, supported extensions are "json", "toml", "yaml", "yml", "properties", "props", "prop", "Env", "dotenv"
if len(c.RemoteProvider) > 0 {
var err error
for _, conf := range c.RemoteProvider {
if conf.SecretKeyring != "" {
err = runtimeViper.AddSecureRemoteProvider(conf.Provider, conf.Endpoint, conf.Path, conf.SecretKeyring)
} else {
err = runtimeViper.AddRemoteProvider(conf.Provider, conf.Endpoint, conf.Path)
}
if err != nil {
log.Fatal(err)
}
}
// read from remote Config the first time.
err = runtimeViper.ReadRemoteConfig()
if err != nil {
log.Error(err)
}
if c.Watch {
err = runtimeViper.WatchRemoteConfig()
if err != nil {
log.Fatal(err)
}
}
} else {
runtimeViper.SetConfigFile(c.ConfigFile)
if c.ConfigPermissions > 0 {
runtimeViper.SetConfigPermissions(c.ConfigPermissions)
}
err := runtimeViper.ReadInConfig()
if err != nil {
log.Fatal(err)
}
if c.Watch {
runtimeViper.WatchConfig()
}
}
runtimeViper.AllowEmptyEnv(c.AllowEmptyEnv)
runtimeViper.SetEnvPrefix(c.EnvPrefix)
if len(c.EnvVars) > 0 {
err := runtimeViper.BindEnv(c.EnvVars...)
if err != nil {
log.Error(err)
}
}
// open a goroutine to watch remote changes forever
//这段实现不够优雅
/* 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
}
vconf :=runtime_viper.AllSettings()
log.Debug(vconf)
// unmarshal new Config into our runtime Config struct. you can also use channel
// to implement a signal to notify the system of the changes
runtime_viper.Unmarshal(cCopy)
refresh(cCopy, dCopy)
log.Debug(cCopy)
}
}()*/
}
// 不建议使用,请使用viper全局变量
type Viper struct {
*viper.Viper
Conf Config
}
func (v *Viper) Config() any {
return &v.Conf
}
func (v *Viper) SetEntity() {
v.Viper = v.Conf.Build()
}
func (v *Viper) Close() error {
return nil
}