mirror of
https://github.com/opencloud-eu/opencloud.git
synced 2026-01-06 12:19:37 -06:00
Bumps [github.com/gookit/config/v2](https://github.com/gookit/config) from 2.1.8 to 2.2.2. - [Release notes](https://github.com/gookit/config/releases) - [Commits](https://github.com/gookit/config/compare/v2.1.8...v2.2.2) --- updated-dependencies: - dependency-name: github.com/gookit/config/v2 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com>
472 lines
11 KiB
Go
472 lines
11 KiB
Go
package config
|
|
|
|
import (
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"io/fs"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/gookit/goutil/errorx"
|
|
"github.com/gookit/goutil/fsutil"
|
|
"github.com/imdario/mergo"
|
|
)
|
|
|
|
// LoadFiles load one or multi files, will fire OnLoadData event
|
|
//
|
|
// Usage:
|
|
//
|
|
// config.LoadFiles(file1, file2, ...)
|
|
func LoadFiles(sourceFiles ...string) error { return dc.LoadFiles(sourceFiles...) }
|
|
|
|
// LoadFiles load and parse config files, will fire OnLoadData event
|
|
func (c *Config) LoadFiles(sourceFiles ...string) (err error) {
|
|
for _, file := range sourceFiles {
|
|
if err = c.loadFile(file, false, ""); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadExists load one or multi files, will ignore not exist
|
|
//
|
|
// Usage:
|
|
//
|
|
// config.LoadExists(file1, file2, ...)
|
|
func LoadExists(sourceFiles ...string) error { return dc.LoadExists(sourceFiles...) }
|
|
|
|
// LoadExists load and parse config files, but will ignore not exists file.
|
|
func (c *Config) LoadExists(sourceFiles ...string) (err error) {
|
|
for _, file := range sourceFiles {
|
|
if file == "" {
|
|
continue
|
|
}
|
|
|
|
if err = c.loadFile(file, true, ""); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadRemote load config data from remote URL.
|
|
func LoadRemote(format, url string) error { return dc.LoadRemote(format, url) }
|
|
|
|
// LoadRemote load config data from remote URL.
|
|
//
|
|
// Usage:
|
|
//
|
|
// c.LoadRemote(config.JSON, "http://abc.com/api-config.json")
|
|
func (c *Config) LoadRemote(format, url string) (err error) {
|
|
// create http client
|
|
client := http.Client{Timeout: 300 * time.Second}
|
|
resp, err := client.Get(url)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
//noinspection GoUnhandledErrorResult
|
|
defer resp.Body.Close()
|
|
if resp.StatusCode != 200 {
|
|
return fmt.Errorf("fetch remote config error, reply status code is %d", resp.StatusCode)
|
|
}
|
|
|
|
// read response content
|
|
bts, err := io.ReadAll(resp.Body)
|
|
if err == nil {
|
|
if err = c.parseSourceCode(format, bts); err != nil {
|
|
return
|
|
}
|
|
c.loadedUrls = append(c.loadedUrls, url)
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadOSEnv load data from OS ENV
|
|
//
|
|
// Deprecated: please use LoadOSEnvs()
|
|
func LoadOSEnv(keys []string, keyToLower bool) { dc.LoadOSEnv(keys, keyToLower) }
|
|
|
|
// LoadOSEnv load data from os ENV
|
|
//
|
|
// Deprecated: please use Config.LoadOSEnvs()
|
|
func (c *Config) LoadOSEnv(keys []string, keyToLower bool) {
|
|
for _, key := range keys {
|
|
// NOTICE: if is Windows os, os.Getenv() Key is not case-sensitive
|
|
val := os.Getenv(key)
|
|
if keyToLower {
|
|
key = strings.ToLower(key)
|
|
}
|
|
|
|
_ = c.Set(key, val)
|
|
}
|
|
|
|
c.fireHook(OnLoadData)
|
|
}
|
|
|
|
// LoadOSEnvs load data from OS ENVs. format: {ENV_NAME: config_key}
|
|
func LoadOSEnvs(nameToKeyMap map[string]string) { dc.LoadOSEnvs(nameToKeyMap) }
|
|
|
|
// LoadOSEnvs load data from os ENVs. format: {ENV_NAME: config_key}
|
|
func (c *Config) LoadOSEnvs(nameToKeyMap map[string]string) {
|
|
for name, key := range nameToKeyMap {
|
|
if val := os.Getenv(name); val != "" {
|
|
if key == "" {
|
|
key = strings.ToLower(name)
|
|
}
|
|
|
|
_ = c.Set(key, val)
|
|
}
|
|
}
|
|
|
|
c.fireHook(OnLoadData)
|
|
}
|
|
|
|
// support bound types for CLI flags vars
|
|
var validTypes = map[string]int{
|
|
"int": 1,
|
|
"uint": 1,
|
|
"bool": 1,
|
|
// string is default
|
|
"string": 1,
|
|
}
|
|
|
|
// LoadFlags load data from cli flags
|
|
func LoadFlags(keys []string) error { return dc.LoadFlags(keys) }
|
|
|
|
// LoadFlags parse command line arguments, based on provide keys.
|
|
//
|
|
// Usage:
|
|
//
|
|
// // 'debug' flag is bool type
|
|
// c.LoadFlags([]string{"env", "debug:bool"})
|
|
func (c *Config) LoadFlags(keys []string) (err error) {
|
|
hash := map[string]int8{}
|
|
|
|
// bind vars
|
|
for _, key := range keys {
|
|
key, typ := parseVarNameAndType(key)
|
|
desc := "config flag " + key
|
|
|
|
switch typ {
|
|
case "int":
|
|
ptr := new(int)
|
|
flag.IntVar(ptr, key, c.Int(key), desc)
|
|
hash[key] = 0
|
|
case "uint":
|
|
ptr := new(uint)
|
|
flag.UintVar(ptr, key, c.Uint(key), desc)
|
|
hash[key] = 0
|
|
case "bool":
|
|
ptr := new(bool)
|
|
flag.BoolVar(ptr, key, c.Bool(key), desc)
|
|
hash[key] = 0
|
|
default: // as string
|
|
ptr := new(string)
|
|
flag.StringVar(ptr, key, c.String(key), desc)
|
|
hash[key] = 0
|
|
}
|
|
}
|
|
|
|
// parse and collect
|
|
flag.Parse()
|
|
flag.Visit(func(f *flag.Flag) {
|
|
name := f.Name
|
|
// only get name in the keys.
|
|
if _, ok := hash[name]; !ok {
|
|
return
|
|
}
|
|
|
|
_ = c.Set(name, f.Value.String()) // ignore error
|
|
})
|
|
|
|
c.fireHook(OnLoadData)
|
|
return
|
|
}
|
|
|
|
// LoadData load one or multi data
|
|
func LoadData(dataSource ...any) error { return dc.LoadData(dataSource...) }
|
|
|
|
// LoadData load data from map OR struct
|
|
//
|
|
// The dataSources can be:
|
|
// - map[string]any
|
|
// - map[string]string
|
|
func (c *Config) LoadData(dataSources ...any) (err error) {
|
|
if c.opts.Delimiter == 0 {
|
|
c.opts.Delimiter = defaultDelimiter
|
|
}
|
|
|
|
for _, ds := range dataSources {
|
|
if smp, ok := ds.(map[string]string); ok {
|
|
c.LoadSMap(smp)
|
|
continue
|
|
}
|
|
|
|
err = mergo.Merge(&c.data, ds, mergo.WithOverride)
|
|
if err != nil {
|
|
return errorx.WithStack(err)
|
|
}
|
|
}
|
|
|
|
c.fireHook(OnLoadData)
|
|
return
|
|
}
|
|
|
|
// LoadSMap to config
|
|
func (c *Config) LoadSMap(smp map[string]string) {
|
|
for k, v := range smp {
|
|
c.data[k] = v
|
|
}
|
|
c.fireHook(OnLoadData)
|
|
}
|
|
|
|
// LoadSources load one or multi byte data
|
|
func LoadSources(format string, src []byte, more ...[]byte) error {
|
|
return dc.LoadSources(format, src, more...)
|
|
}
|
|
|
|
// LoadSources load data from byte content.
|
|
//
|
|
// Usage:
|
|
//
|
|
// config.LoadSources(config.Yaml, []byte(`
|
|
// name: blog
|
|
// arr:
|
|
// key: val
|
|
//
|
|
// `))
|
|
func (c *Config) LoadSources(format string, src []byte, more ...[]byte) (err error) {
|
|
err = c.parseSourceCode(format, src)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, sc := range more {
|
|
err = c.parseSourceCode(format, sc)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadStrings load one or multi string
|
|
func LoadStrings(format string, str string, more ...string) error {
|
|
return dc.LoadStrings(format, str, more...)
|
|
}
|
|
|
|
// LoadStrings load data from source string content.
|
|
func (c *Config) LoadStrings(format string, str string, more ...string) (err error) {
|
|
err = c.parseSourceCode(format, []byte(str))
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, s := range more {
|
|
err = c.parseSourceCode(format, []byte(s))
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadFilesByFormat load one or multi config files by give format, will fire OnLoadData event
|
|
func LoadFilesByFormat(format string, configFiles ...string) error {
|
|
return dc.LoadFilesByFormat(format, configFiles...)
|
|
}
|
|
|
|
// LoadFilesByFormat load one or multi files by give format, will fire OnLoadData event
|
|
func (c *Config) LoadFilesByFormat(format string, configFiles ...string) (err error) {
|
|
for _, file := range configFiles {
|
|
if err = c.loadFile(file, false, format); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadExistsByFormat load one or multi files by give format, will fire OnLoadData event
|
|
func LoadExistsByFormat(format string, configFiles ...string) error {
|
|
return dc.LoadExistsByFormat(format, configFiles...)
|
|
}
|
|
|
|
// LoadExistsByFormat load one or multi files by give format, will fire OnLoadData event
|
|
func (c *Config) LoadExistsByFormat(format string, configFiles ...string) (err error) {
|
|
for _, file := range configFiles {
|
|
if err = c.loadFile(file, true, format); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoadFromDir Load custom format files from the given directory, the file name will be used as the key.
|
|
//
|
|
// Example:
|
|
//
|
|
// // file: /somedir/task.json
|
|
// LoadFromDir("/somedir", "json")
|
|
//
|
|
// // after load
|
|
// Config.data = map[string]any{"task": file data}
|
|
func LoadFromDir(dirPath, format string) error {
|
|
return dc.LoadFromDir(dirPath, format)
|
|
}
|
|
|
|
// LoadFromDir Load custom format files from the given directory, the file name will be used as the key.
|
|
//
|
|
// Example:
|
|
//
|
|
// // file: /somedir/task.json
|
|
// Config.LoadFromDir("/somedir", "json")
|
|
//
|
|
// // after load
|
|
// Config.data = map[string]any{"task": file data}
|
|
func (c *Config) LoadFromDir(dirPath, format string) (err error) {
|
|
extName := "." + format
|
|
extLen := len(extName)
|
|
|
|
return fsutil.FindInDir(dirPath, func(fPath string, ent fs.DirEntry) error {
|
|
baseName := ent.Name()
|
|
if strings.HasSuffix(baseName, extName) {
|
|
data, err := c.parseSourceToMap(format, fsutil.MustReadFile(fPath))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
onlyName := baseName[:len(baseName)-extLen]
|
|
err = c.loadDataMap(map[string]any{onlyName: data})
|
|
|
|
// use file name as key, it cannot be reloaded. SO, cannot append to loadedFiles
|
|
// if err == nil {
|
|
// c.loadedFiles = append(c.loadedFiles, fPath)
|
|
// }
|
|
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
// ReloadFiles reload config data use loaded files
|
|
func ReloadFiles() error { return dc.ReloadFiles() }
|
|
|
|
// ReloadFiles reload config data use loaded files. use on watching loaded files change
|
|
func (c *Config) ReloadFiles() (err error) {
|
|
files := c.loadedFiles
|
|
if len(files) == 0 {
|
|
return
|
|
}
|
|
|
|
data := c.Data()
|
|
c.reloading = true
|
|
c.ClearCaches()
|
|
|
|
defer func() {
|
|
// revert to back up data on error
|
|
if err != nil {
|
|
c.data = data
|
|
}
|
|
|
|
c.lock.Unlock()
|
|
c.reloading = false
|
|
|
|
if err == nil {
|
|
c.fireHook(OnReloadData)
|
|
}
|
|
}()
|
|
|
|
// with lock
|
|
c.lock.Lock()
|
|
|
|
// reload config files
|
|
return c.LoadFiles(files...)
|
|
}
|
|
|
|
// load config file, will fire OnLoadData event
|
|
func (c *Config) loadFile(file string, loadExist bool, format string) (err error) {
|
|
fd, err := os.Open(file)
|
|
if err != nil {
|
|
// skip not exist file
|
|
if os.IsNotExist(err) && loadExist {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
//noinspection GoUnhandledErrorResult
|
|
defer fd.Close()
|
|
|
|
// read file content
|
|
bts, err := io.ReadAll(fd)
|
|
if err == nil {
|
|
// get format for file ext
|
|
if format == "" {
|
|
format = strings.Trim(filepath.Ext(file), ".")
|
|
}
|
|
|
|
// parse file content
|
|
if err = c.parseSourceCode(format, bts); err != nil {
|
|
return
|
|
}
|
|
|
|
if !c.reloading {
|
|
c.loadedFiles = append(c.loadedFiles, file)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// parse config source code to Config.
|
|
func (c *Config) parseSourceCode(format string, blob []byte) (err error) {
|
|
data, err := c.parseSourceToMap(format, blob)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return c.loadDataMap(data)
|
|
}
|
|
|
|
func (c *Config) loadDataMap(data map[string]any) (err error) {
|
|
// first: init config data
|
|
if len(c.data) == 0 {
|
|
c.data = data
|
|
} else {
|
|
// again ... will merge data
|
|
err = mergo.Merge(&c.data, data, mergo.WithOverride, mergo.WithTypeCheck)
|
|
}
|
|
|
|
if !c.reloading && err == nil {
|
|
c.fireHook(OnLoadData)
|
|
}
|
|
return err
|
|
}
|
|
|
|
// parse config source code to Config.
|
|
func (c *Config) parseSourceToMap(format string, blob []byte) (map[string]any, error) {
|
|
format = c.resolveFormat(format)
|
|
decode := c.decoders[format]
|
|
if decode == nil {
|
|
return nil, errors.New("not register decoder for the format: " + format)
|
|
}
|
|
|
|
if c.opts.Delimiter == 0 {
|
|
c.opts.Delimiter = defaultDelimiter
|
|
}
|
|
|
|
// decode content to data
|
|
data := make(map[string]any)
|
|
|
|
if err := decode(blob, &data); err != nil {
|
|
return nil, err
|
|
}
|
|
return data, nil
|
|
}
|