2021-01-07 01:17:03 +08:00
|
|
|
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
2017-12-29 16:03:30 +08:00
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
|
|
// If a copy of the MIT was not distributed with this file,
|
2019-02-02 16:18:25 +08:00
|
|
|
// You can obtain one at https://github.com/gogf/gf.
|
2017-12-29 16:03:30 +08:00
|
|
|
|
2019-05-08 22:04:36 +08:00
|
|
|
// Package gcfg provides reading, caching and managing for configuration.
|
2017-12-30 23:49:55 +08:00
|
|
|
package gcfg
|
2017-12-13 16:45:00 +08:00
|
|
|
|
|
|
|
import (
|
2021-06-26 16:23:54 +08:00
|
|
|
"context"
|
2021-09-19 10:01:09 +08:00
|
|
|
"fmt"
|
2021-10-11 21:41:56 +08:00
|
|
|
"github.com/gogf/gf/v2/container/gmap"
|
|
|
|
"github.com/gogf/gf/v2/container/gvar"
|
|
|
|
"github.com/gogf/gf/v2/internal/intlog"
|
2017-12-14 17:32:51 +08:00
|
|
|
)
|
2017-12-13 16:45:00 +08:00
|
|
|
|
2021-09-27 21:27:24 +08:00
|
|
|
// Config is the configuration management object.
|
2017-12-14 17:32:51 +08:00
|
|
|
type Config struct {
|
2021-09-19 10:01:09 +08:00
|
|
|
adapter Adapter
|
|
|
|
dataMap *gmap.StrAnyMap
|
2017-12-13 16:45:00 +08:00
|
|
|
}
|
|
|
|
|
2021-05-17 00:07:06 +08:00
|
|
|
const (
|
2021-09-19 10:01:09 +08:00
|
|
|
DefaultName = "config" // DefaultName is the default group name for instance usage.
|
2021-05-17 00:07:06 +08:00
|
|
|
)
|
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// New creates and returns a Config object with default adapter of AdapterFile.
|
|
|
|
func New() (*Config, error) {
|
|
|
|
adapterFile, err := NewAdapterFile()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-06-19 09:06:52 +08:00
|
|
|
}
|
2021-09-19 10:01:09 +08:00
|
|
|
return &Config{
|
|
|
|
adapter: adapterFile,
|
|
|
|
dataMap: gmap.NewStrAnyMap(true),
|
|
|
|
}, nil
|
2021-03-16 14:39:01 +08:00
|
|
|
}
|
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// NewWithAdapter creates and returns a Config object with given adapter.
|
|
|
|
func NewWithAdapter(adapter Adapter) *Config {
|
|
|
|
return &Config{
|
|
|
|
adapter: adapter,
|
|
|
|
dataMap: gmap.NewStrAnyMap(true),
|
2021-03-16 14:39:01 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// Instance returns an instance of Config with default settings.
|
|
|
|
// The parameter `name` is the name for the instance. But very note that, if the file "name.toml"
|
|
|
|
// exists in the configuration directory, it then sets it as the default configuration file. The
|
|
|
|
// toml file type is the default configuration file type.
|
|
|
|
func Instance(name ...string) *Config {
|
|
|
|
key := DefaultName
|
|
|
|
if len(name) > 0 && name[0] != "" {
|
|
|
|
key = name[0]
|
2019-06-19 09:06:52 +08:00
|
|
|
}
|
2021-09-19 10:01:09 +08:00
|
|
|
return localInstances.GetOrSetFuncLock(key, func() interface{} {
|
|
|
|
adapter, err := NewAdapterFile()
|
|
|
|
if err != nil {
|
|
|
|
intlog.Error(context.Background(), err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// If it's not using default configuration or its configuration file is not available,
|
|
|
|
// it searches the possible configuration file according to the name and all supported
|
|
|
|
// file types.
|
|
|
|
if key != DefaultName || !adapter.Available() {
|
|
|
|
for _, fileType := range supportedFileTypes {
|
|
|
|
if file := fmt.Sprintf(`%s.%s`, key, fileType); adapter.Available(file) {
|
|
|
|
adapter.SetFileName(file)
|
|
|
|
break
|
|
|
|
}
|
2021-03-03 14:29:01 +08:00
|
|
|
}
|
2021-01-27 00:55:45 +08:00
|
|
|
}
|
2021-09-19 10:01:09 +08:00
|
|
|
return NewWithAdapter(adapter)
|
|
|
|
}).(*Config)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetAdapter sets the adapter of current Config object.
|
|
|
|
func (c *Config) SetAdapter(adapter Adapter) {
|
|
|
|
c.adapter = adapter
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAdapter returns the adapter of current Config object.
|
|
|
|
func (c *Config) GetAdapter() Adapter {
|
|
|
|
return c.adapter
|
|
|
|
}
|
2019-11-28 23:19:37 +08:00
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// Set sets value with specified `pattern`.
|
|
|
|
// It supports hierarchical data access by char separator, which is '.' in default.
|
|
|
|
// It is commonly used for updates certain configuration value in runtime.
|
|
|
|
func (c *Config) Set(ctx context.Context, pattern string, value interface{}) {
|
|
|
|
c.dataMap.Set(pattern, value)
|
2021-03-29 18:05:47 +08:00
|
|
|
}
|
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// Get retrieves and returns value by specified `pattern`.
|
|
|
|
// It returns all values of current Json object if `pattern` is given empty or string ".".
|
|
|
|
// It returns nil if no value found by `pattern`.
|
|
|
|
//
|
|
|
|
// It returns a default value specified by `def` if value for `pattern` is not found.
|
|
|
|
func (c *Config) Get(ctx context.Context, pattern string, def ...interface{}) (*gvar.Var, error) {
|
|
|
|
var (
|
|
|
|
err error
|
|
|
|
value interface{}
|
|
|
|
)
|
|
|
|
if value = c.dataMap.Get(pattern); value == nil {
|
|
|
|
value, err = c.adapter.Get(ctx, pattern)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if value == nil && len(def) > 0 {
|
|
|
|
return gvar.New(def[0]), nil
|
2019-06-19 09:06:52 +08:00
|
|
|
}
|
2021-09-19 10:01:09 +08:00
|
|
|
}
|
|
|
|
return gvar.New(value), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Data retrieves and returns all configuration data as map type.
|
|
|
|
func (c *Config) Data(ctx context.Context) (data map[string]interface{}, err error) {
|
|
|
|
adapterData, err := c.adapter.Data(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
data = make(map[string]interface{})
|
|
|
|
for k, v := range adapterData {
|
|
|
|
data[k] = v
|
|
|
|
}
|
|
|
|
c.dataMap.Iterator(func(k string, v interface{}) bool {
|
|
|
|
data[k] = v
|
|
|
|
return true
|
2019-06-19 09:06:52 +08:00
|
|
|
})
|
2021-09-19 10:01:09 +08:00
|
|
|
return
|
|
|
|
}
|
2021-05-17 00:07:06 +08:00
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// MustGet acts as function Get, but it panics if error occurs.
|
|
|
|
func (c *Config) MustGet(ctx context.Context, pattern string, def ...interface{}) *gvar.Var {
|
|
|
|
v, err := c.Get(ctx, pattern, def...)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return gvar.New(v)
|
2021-05-17 00:07:06 +08:00
|
|
|
}
|
|
|
|
|
2021-09-19 10:01:09 +08:00
|
|
|
// MustData acts as function Data, but it panics if error occurs.
|
|
|
|
func (c *Config) MustData(ctx context.Context) map[string]interface{} {
|
|
|
|
v, err := c.Data(ctx)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return v
|
2018-10-27 16:05:36 +08:00
|
|
|
}
|