dae/config/parser.go

237 lines
7.4 KiB
Go
Raw Normal View History

2023-01-28 00:50:21 +07:00
/*
* SPDX-License-Identifier: AGPL-3.0-only
2023-03-14 14:01:55 +07:00
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
2023-01-28 00:50:21 +07:00
*/
package config
import (
"fmt"
2023-03-14 14:01:55 +07:00
"github.com/daeuniverse/dae/common"
"github.com/daeuniverse/dae/pkg/config_parser"
2023-01-28 00:50:21 +07:00
"reflect"
"strings"
2023-01-28 00:50:21 +07:00
)
func StringListParser(to reflect.Value, section *config_parser.Section) error {
if to.Kind() != reflect.Pointer {
return fmt.Errorf("StringListParser can only unmarshal section to *[]string")
}
to = to.Elem()
if to.Type() != reflect.TypeOf([]string{}) &&
!(to.Kind() == reflect.Slice && to.Type().Elem().Kind() == reflect.String) {
2023-01-28 00:50:21 +07:00
return fmt.Errorf("StringListParser can only unmarshal section to *[]string")
}
for _, item := range section.Items {
switch itemVal := item.Value.(type) {
case *config_parser.Param:
to.Set(reflect.Append(to, reflect.ValueOf(itemVal.String(true, false)).Convert(to.Type().Elem())))
2023-01-28 00:50:21 +07:00
default:
return fmt.Errorf("section %v does not support type %v: %v", section.Name, item.Type.String(), item.String(false, false))
2023-01-28 00:50:21 +07:00
}
}
return nil
}
2023-02-25 01:38:21 +07:00
func ParamParser(to reflect.Value, section *config_parser.Section, ignoreType []reflect.Type) error {
2023-01-28 00:50:21 +07:00
if to.Kind() != reflect.Pointer {
return fmt.Errorf("ParamParser can only unmarshal section to *struct")
}
to = to.Elem()
if to.Kind() != reflect.Struct {
return fmt.Errorf("ParamParser can only unmarshal section to struct")
}
// keyToField is for further parsing use.
type Field struct {
Val reflect.Value
Index int
Set bool
}
var keyToField = make(map[string]*Field)
tot := to.Type()
for i := 0; i < to.NumField(); i++ {
field := to.Field(i)
structField := tot.Field(i)
// Set up key to field mapping.
key, ok := structField.Tag.Lookup("mapstructure")
if !ok {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("field \"%v\" has no mapstructure tag", structField.Name)
2023-01-28 00:50:21 +07:00
}
if key == "_" {
// omit
continue
}
keyToField[key] = &Field{Val: field, Index: i}
// Fill in default value before parsing section.
defaultValue, ok := structField.Tag.Lookup("default")
if ok {
// Can we assign?
if field.Kind() == reflect.Interface ||
field.Type() == reflect.TypeOf(defaultValue) {
field.Set(reflect.ValueOf(defaultValue))
// Can we fuzzy decode?
} else if !common.FuzzyDecode(field.Addr().Interface(), defaultValue) {
2023-01-28 00:50:21 +07:00
return fmt.Errorf(`failed to decode default value of "%v"`, structField.Name)
}
}
}
// Convert ignoreType from list to set.
ignoreTypeSet := make(map[reflect.Type]struct{})
for _, typ := range ignoreType {
ignoreTypeSet[typ] = struct{}{}
}
// Parse section.
for _, item := range section.Items {
switch itemVal := item.Value.(type) {
case *config_parser.Param:
if itemVal.Key == "" {
return fmt.Errorf("unsupported text without a key: %v", itemVal.String(true, false))
2023-01-28 00:50:21 +07:00
}
field, ok := keyToField[itemVal.Key]
if !ok {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("unexpected key: %v", itemVal.Key)
2023-01-28 00:50:21 +07:00
}
if itemVal.AndFunctions != nil {
// AndFunctions.
// If field is interface{} or types equal, we can assign.
if field.Val.Kind() == reflect.Interface ||
field.Val.Type() == reflect.TypeOf(itemVal.AndFunctions) {
field.Val.Set(reflect.ValueOf(itemVal.AndFunctions))
} else {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("failed to parse \"%v\": value \"%v\" cannot be convert to %v", itemVal.Key, itemVal.Val, field.Val.Type().String())
2023-01-28 00:50:21 +07:00
}
} else {
// String value.
2023-02-01 11:30:26 +07:00
switch field.Val.Kind() {
case reflect.Interface:
2023-01-28 00:50:21 +07:00
// Field is interface{}, we can assign.
field.Val.Set(reflect.ValueOf(itemVal.Val))
2023-02-01 11:30:26 +07:00
case reflect.Slice:
// Field is not interface{}, we can decode.
values := strings.Split(itemVal.Val, ",")
for _, value := range values {
vPointerNew := reflect.New(field.Val.Type().Elem())
if !common.FuzzyDecode(vPointerNew.Interface(), value) {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("failed to parse \"%v\": value \"%v\" cannot be convert to %v", itemVal.Key, itemVal.Val, field.Val.Type().Elem().String())
}
field.Val.Set(reflect.Append(field.Val, vPointerNew.Elem()))
2023-02-01 11:30:26 +07:00
}
default:
2023-01-28 00:50:21 +07:00
// Field is not interface{}, we can decode.
if !common.FuzzyDecode(field.Val.Addr().Interface(), itemVal.Val) {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("failed to parse \"%v\": value \"%v\" cannot be convert to %v", itemVal.Key, itemVal.Val, field.Val.Type().String())
2023-01-28 00:50:21 +07:00
}
}
}
field.Set = true
2023-02-25 01:38:21 +07:00
case *config_parser.Section:
// Named section config item.
field, ok := keyToField[itemVal.Name]
if !ok {
return fmt.Errorf("unexpected key: %v", itemVal.Name)
}
if err := SectionParser(field.Val.Addr(), itemVal); err != nil {
return fmt.Errorf("failed to parse %v: %w", itemVal.Name, err)
}
field.Set = true
case *config_parser.RoutingRule:
// Assign. "to" should have field "Rules".
structField, ok := to.Type().FieldByName("Rules")
if !ok || structField.Type != reflect.TypeOf([]*config_parser.RoutingRule{}) {
return fmt.Errorf("unexpected type: \"routing rule\": %v", itemVal.String(true, false, false))
2023-02-25 01:38:21 +07:00
}
if structField.Tag.Get("mapstructure") != "_" {
return fmt.Errorf("a []*RoutingRule field \"Rules\" with mapstructure:\"_\" is required in struct %v to parse section", to.Type().String())
}
field := to.FieldByName("Rules")
field.Set(reflect.Append(field, reflect.ValueOf(itemVal)))
2023-01-28 00:50:21 +07:00
default:
if _, ignore := ignoreTypeSet[reflect.TypeOf(itemVal)]; !ignore {
return fmt.Errorf("unexpected type %v: %v", item.Type.String(), item.String(false, false))
2023-01-28 00:50:21 +07:00
}
}
}
// Check required.
for key, field := range keyToField {
if field.Set {
continue
}
t := to.Type().Field(field.Index)
_, required := t.Tag.Lookup("required")
if required {
return fmt.Errorf(`section "%v" requires param "%v" but not found`, section.Name, key)
}
}
return nil
}
2023-02-25 01:38:21 +07:00
func SectionParser(to reflect.Value, section *config_parser.Section) error {
2023-01-28 00:50:21 +07:00
if to.Kind() != reflect.Pointer {
2023-02-25 01:38:21 +07:00
return fmt.Errorf("SectionParser can only unmarshal section to a pointer")
2023-01-28 00:50:21 +07:00
}
to = to.Elem()
2023-02-25 01:38:21 +07:00
switch to.Kind() {
case reflect.Slice:
elemType := to.Type().Elem()
switch elemType.Kind() {
case reflect.String:
return StringListParser(to.Addr(), section)
case reflect.Struct:
// "to" is a section list (sections in section).
/**
to {
field1 {
...
}
field2 {
...
}
}
should be parsed to:
to []struct {
Name string `mapstructure: "_"`
...
}
*/
// The struct should contain Name.
nameStructField, ok := elemType.FieldByName("Name")
if !ok || nameStructField.Type.Kind() != reflect.String || nameStructField.Tag.Get("mapstructure") != "_" {
return fmt.Errorf("a string field \"Name\" with mapstructure:\"_\" is required in struct %v to parse section", to.Type().Elem().String())
2023-01-28 00:50:21 +07:00
}
2023-02-25 01:38:21 +07:00
// Scan sections.
for _, item := range section.Items {
elem := reflect.New(elemType).Elem()
switch itemVal := item.Value.(type) {
case *config_parser.Section:
elem.FieldByName("Name").SetString(itemVal.Name)
if err := SectionParser(elem.Addr(), itemVal); err != nil {
return fmt.Errorf("error when parse \"%v\": %w", itemVal.Name, err)
}
to.Set(reflect.Append(to, elem))
default:
return fmt.Errorf("unmatched type: %v -> %v", item.Type.String(), elemType)
}
2023-01-28 00:50:21 +07:00
}
2023-02-25 01:38:21 +07:00
return nil
2023-01-28 00:50:21 +07:00
default:
2023-02-25 01:38:21 +07:00
goto unsupported
2023-01-28 00:50:21 +07:00
}
2023-02-25 01:38:21 +07:00
case reflect.Struct:
// Section.
return ParamParser(to.Addr(), section, nil)
default:
goto unsupported
2023-01-28 00:50:21 +07:00
}
2023-02-25 01:38:21 +07:00
panic("code should not reach here")
2023-01-28 00:50:21 +07:00
2023-02-25 01:38:21 +07:00
unsupported:
return fmt.Errorf("unsupported section type %v", to.Type())
2023-01-28 00:50:21 +07:00
}