go-zero/core/conf/config.go

364 lines
8.4 KiB
Go
Raw Normal View History

2020-07-26 17:09:05 +08:00
package conf
import (
"fmt"
"log"
2020-12-25 11:42:19 +08:00
"os"
2020-07-26 17:09:05 +08:00
"path"
"reflect"
2022-05-13 23:17:43 +08:00
"strings"
2020-07-26 17:09:05 +08:00
"github.com/zeromicro/go-zero/core/jsonx"
"github.com/zeromicro/go-zero/core/mapping"
"github.com/zeromicro/go-zero/internal/encoding"
2020-07-26 17:09:05 +08:00
)
2023-03-06 23:04:19 +08:00
const (
jsonTagKey = "json"
jsonTagSep = ','
)
2023-03-02 17:20:09 +08:00
var (
fillDefaultUnmarshaler = mapping.NewUnmarshaler(jsonTagKey, mapping.WithDefault())
loaders = map[string]func([]byte, any) error{
".json": LoadFromJsonBytes,
".toml": LoadFromTomlBytes,
".yaml": LoadFromYamlBytes,
".yml": LoadFromYamlBytes,
}
)
2020-07-26 17:09:05 +08:00
2023-03-01 21:36:48 +08:00
// children and mapField should not be both filled.
// named fields and map cannot be bound to the same field name.
type fieldInfo struct {
2023-03-01 21:36:48 +08:00
children map[string]*fieldInfo
mapField *fieldInfo
}
2023-03-02 21:47:07 +08:00
// FillDefault fills the default values for the given v,
2023-03-02 21:53:05 +08:00
// and the premise is that the value of v must be guaranteed to be empty.
2023-03-02 17:20:09 +08:00
func FillDefault(v any) error {
return fillDefaultUnmarshaler.Unmarshal(map[string]any{}, v)
}
// Load loads config into v from file, .json, .yaml and .yml are acceptable.
func Load(file string, v any, opts ...Option) error {
content, err := os.ReadFile(file)
if err != nil {
2020-07-26 17:09:05 +08:00
return err
}
2022-05-13 23:17:43 +08:00
loader, ok := loaders[strings.ToLower(path.Ext(file))]
if !ok {
2021-03-29 23:35:49 +08:00
return fmt.Errorf("unrecognized file type: %s", file)
}
var opt options
for _, o := range opts {
o(&opt)
}
if opt.env {
2020-12-25 11:42:19 +08:00
return loader([]byte(os.ExpandEnv(string(content))), v)
2020-07-26 17:09:05 +08:00
}
2021-02-09 13:50:21 +08:00
return loader(content, v)
2020-07-26 17:09:05 +08:00
}
// LoadConfig loads config into v from file, .json, .yaml and .yml are acceptable.
// Deprecated: use Load instead.
func LoadConfig(file string, v any, opts ...Option) error {
return Load(file, v, opts...)
}
// LoadFromJsonBytes loads config into v from content json bytes.
func LoadFromJsonBytes(content []byte, v any) error {
info, err := buildFieldsInfo(reflect.TypeOf(v), "")
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
2023-03-01 21:36:48 +08:00
var m map[string]any
2023-03-06 20:27:20 +08:00
if err = jsonx.Unmarshal(content, &m); err != nil {
2023-03-01 11:11:45 +08:00
return err
}
2023-03-02 10:13:31 +08:00
lowerCaseKeyMap := toLowerCaseKeyMap(m, info)
return mapping.UnmarshalJsonMap(lowerCaseKeyMap, v, mapping.WithCanonicalKeyFunc(toLowerCase))
}
2021-02-18 15:56:19 +08:00
// LoadConfigFromJsonBytes loads config into v from content json bytes.
// Deprecated: use LoadFromJsonBytes instead.
func LoadConfigFromJsonBytes(content []byte, v any) error {
return LoadFromJsonBytes(content, v)
}
2022-05-13 23:17:43 +08:00
// LoadFromTomlBytes loads config into v from content toml bytes.
func LoadFromTomlBytes(content []byte, v any) error {
b, err := encoding.TomlToJson(content)
if err != nil {
return err
}
return LoadFromJsonBytes(b, v)
2022-05-13 23:17:43 +08:00
}
// LoadFromYamlBytes loads config into v from content yaml bytes.
func LoadFromYamlBytes(content []byte, v any) error {
b, err := encoding.YamlToJson(content)
if err != nil {
return err
}
return LoadFromJsonBytes(b, v)
2020-07-26 17:09:05 +08:00
}
2021-02-18 15:56:19 +08:00
// LoadConfigFromYamlBytes loads config into v from content yaml bytes.
// Deprecated: use LoadFromYamlBytes instead.
func LoadConfigFromYamlBytes(content []byte, v any) error {
return LoadFromYamlBytes(content, v)
2020-07-26 17:09:05 +08:00
}
2021-02-18 15:56:19 +08:00
// MustLoad loads config into v from path, exits on error.
func MustLoad(path string, v any, opts ...Option) {
2022-05-13 23:17:43 +08:00
if err := Load(path, v, opts...); err != nil {
2020-07-26 17:09:05 +08:00
log.Fatalf("error: config file %s, %s", path, err.Error())
}
}
func addOrMergeFields(info *fieldInfo, key string, child *fieldInfo, fullName string) error {
if prev, ok := info.children[key]; ok {
2023-03-01 21:36:48 +08:00
if child.mapField != nil {
return newConflictKeyError(fullName)
2023-03-01 11:11:45 +08:00
}
2024-03-08 22:35:17 +08:00
if err := mergeFields(prev, child.children, fullName); err != nil {
2023-03-01 21:36:48 +08:00
return err
}
} else {
info.children[key] = child
}
2023-03-01 11:11:45 +08:00
return nil
}
func buildAnonymousFieldInfo(info *fieldInfo, lowerCaseName string, ft reflect.Type, fullName string) error {
2023-03-01 21:36:48 +08:00
switch ft.Kind() {
case reflect.Struct:
fields, err := buildFieldsInfo(ft, fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
for k, v := range fields.children {
if err = addOrMergeFields(info, k, v, fullName); err != nil {
2023-03-01 21:36:48 +08:00
return err
}
}
case reflect.Map:
elemField, err := buildFieldsInfo(mapping.Deref(ft.Elem()), fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
if _, ok := info.children[lowerCaseName]; ok {
return newConflictKeyError(fullName)
2023-03-01 21:36:48 +08:00
}
info.children[lowerCaseName] = &fieldInfo{
children: make(map[string]*fieldInfo),
mapField: elemField,
}
default:
if _, ok := info.children[lowerCaseName]; ok {
return newConflictKeyError(fullName)
2023-03-01 21:36:48 +08:00
}
info.children[lowerCaseName] = &fieldInfo{
children: make(map[string]*fieldInfo),
}
}
return nil
}
func buildFieldsInfo(tp reflect.Type, fullName string) (*fieldInfo, error) {
tp = mapping.Deref(tp)
switch tp.Kind() {
case reflect.Struct:
return buildStructFieldsInfo(tp, fullName)
case reflect.Array, reflect.Slice, reflect.Map:
return buildFieldsInfo(mapping.Deref(tp.Elem()), fullName)
2023-03-02 09:57:18 +08:00
case reflect.Chan, reflect.Func:
2025-01-14 23:16:13 +08:00
return nil, fmt.Errorf("unsupported type: %s, fullName: %s", tp.Kind(), fullName)
default:
2023-03-01 21:36:48 +08:00
return &fieldInfo{
children: make(map[string]*fieldInfo),
}, nil
}
}
func buildNamedFieldInfo(info *fieldInfo, lowerCaseName string, ft reflect.Type, fullName string) error {
2023-03-01 21:36:48 +08:00
var finfo *fieldInfo
var err error
switch ft.Kind() {
case reflect.Struct:
finfo, err = buildFieldsInfo(ft, fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
case reflect.Array, reflect.Slice:
finfo, err = buildFieldsInfo(ft.Elem(), fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
case reflect.Map:
elemInfo, err := buildFieldsInfo(mapping.Deref(ft.Elem()), fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
finfo = &fieldInfo{
children: make(map[string]*fieldInfo),
mapField: elemInfo,
}
default:
finfo, err = buildFieldsInfo(ft, fullName)
2023-03-01 21:36:48 +08:00
if err != nil {
return err
}
}
2023-03-01 21:36:48 +08:00
return addOrMergeFields(info, lowerCaseName, finfo, fullName)
}
func buildStructFieldsInfo(tp reflect.Type, fullName string) (*fieldInfo, error) {
2023-03-01 21:36:48 +08:00
info := &fieldInfo{
children: make(map[string]*fieldInfo),
}
for i := 0; i < tp.NumField(); i++ {
field := tp.Field(i)
if !field.IsExported() {
continue
}
2023-03-06 20:27:20 +08:00
name := getTagName(field)
lowerCaseName := toLowerCase(name)
ft := mapping.Deref(field.Type)
// flatten anonymous fields
if field.Anonymous {
if err := buildAnonymousFieldInfo(info, lowerCaseName, ft,
getFullName(fullName, lowerCaseName)); err != nil {
2023-03-01 21:36:48 +08:00
return nil, err
}
} else if err := buildNamedFieldInfo(info, lowerCaseName, ft,
getFullName(fullName, lowerCaseName)); err != nil {
2023-03-01 21:36:48 +08:00
return nil, err
}
2023-03-01 21:36:48 +08:00
}
2023-03-01 21:36:48 +08:00
return info, nil
}
2023-03-06 23:29:26 +08:00
// getTagName get the tag name of the given field, if no tag name, use file.Name.
// field.Name is returned on tags like `json:""` and `json:",optional"`.
2023-03-06 20:27:20 +08:00
func getTagName(field reflect.StructField) string {
if tag, ok := field.Tag.Lookup(jsonTagKey); ok {
2023-03-06 23:04:19 +08:00
if pos := strings.IndexByte(tag, jsonTagSep); pos >= 0 {
tag = tag[:pos]
}
2023-03-06 23:29:26 +08:00
tag = strings.TrimSpace(tag)
2023-03-06 23:04:19 +08:00
if len(tag) > 0 {
return tag
}
2023-03-06 20:27:20 +08:00
}
return field.Name
}
2024-03-08 22:35:17 +08:00
func mergeFields(prev *fieldInfo, children map[string]*fieldInfo, fullName string) error {
2023-03-02 09:57:18 +08:00
if len(prev.children) == 0 || len(children) == 0 {
return newConflictKeyError(fullName)
2023-03-01 21:36:48 +08:00
}
2023-03-01 21:36:48 +08:00
// merge fields
for k, v := range children {
if _, ok := prev.children[k]; ok {
return newConflictKeyError(fullName)
2023-03-01 11:11:45 +08:00
}
2023-03-01 21:36:48 +08:00
prev.children[k] = v
}
2023-03-01 21:36:48 +08:00
return nil
}
func toLowerCase(s string) string {
return strings.ToLower(s)
}
2023-03-01 21:36:48 +08:00
func toLowerCaseInterface(v any, info *fieldInfo) any {
switch vv := v.(type) {
case map[string]any:
return toLowerCaseKeyMap(vv, info)
case []any:
var arr []any
for _, vvv := range vv {
arr = append(arr, toLowerCaseInterface(vvv, info))
}
return arr
default:
return v
}
}
2023-03-01 21:36:48 +08:00
func toLowerCaseKeyMap(m map[string]any, info *fieldInfo) map[string]any {
res := make(map[string]any)
for k, v := range m {
ti, ok := info.children[k]
if ok {
res[k] = toLowerCaseInterface(v, ti)
continue
}
lk := toLowerCase(k)
if ti, ok = info.children[lk]; ok {
res[lk] = toLowerCaseInterface(v, ti)
} else if info.mapField != nil {
2023-03-01 21:36:48 +08:00
res[k] = toLowerCaseInterface(v, info.mapField)
} else if vv, ok := v.(map[string]any); ok {
res[k] = toLowerCaseKeyMap(vv, info)
} else {
res[k] = v
}
}
return res
}
2023-03-01 11:11:45 +08:00
type conflictKeyError struct {
2023-03-01 11:11:45 +08:00
key string
}
func newConflictKeyError(key string) conflictKeyError {
return conflictKeyError{key: key}
2023-03-01 11:11:45 +08:00
}
func (e conflictKeyError) Error() string {
return fmt.Sprintf("conflict key %s, pay attention to anonymous fields", e.key)
2023-03-01 11:11:45 +08:00
}
func getFullName(parent, child string) string {
2023-04-24 12:34:52 +08:00
if len(parent) == 0 {
return child
}
return strings.Join([]string{parent, child}, ".")
}