mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-04 21:47:40 +03:00
It causes config.Map state to be corrupted which make some directives "disappear". Closes #84.
408 lines
12 KiB
Go
408 lines
12 KiB
Go
package config
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"strconv"
|
|
)
|
|
|
|
type matcher struct {
|
|
name string
|
|
required bool
|
|
inheritGlobal bool
|
|
defaultVal func() (interface{}, error)
|
|
mapper func(*Map, *Node) (interface{}, error)
|
|
store *reflect.Value
|
|
}
|
|
|
|
func (m *matcher) assign(val interface{}) {
|
|
valRefl := reflect.ValueOf(val)
|
|
// Convert untyped nil into typed nil. Otherwise it will panic.
|
|
if !valRefl.IsValid() {
|
|
valRefl = reflect.Zero(m.store.Type())
|
|
}
|
|
|
|
m.store.Set(valRefl)
|
|
}
|
|
|
|
// Map structure implements reflection-based conversion between configuration
|
|
// directives and Go variables.
|
|
type Map struct {
|
|
allowUnknown bool
|
|
|
|
// Set to currently processed node when defaultVal or mapper functions are
|
|
// called.
|
|
curNode *Node
|
|
|
|
// All values saved by Map during processing.
|
|
Values map[string]interface{}
|
|
|
|
entries map[string]matcher
|
|
|
|
// Values used by Process as default values if inheritGlobal is true.
|
|
Globals map[string]interface{}
|
|
// Config block used by Process.
|
|
Block *Node
|
|
}
|
|
|
|
func NewMap(globals map[string]interface{}, block *Node) *Map {
|
|
return &Map{Globals: globals, Block: block}
|
|
}
|
|
|
|
// MatchErr returns error with formatted message, if called from defaultVal or
|
|
// mapper functions - message will be prepended with information about
|
|
// processed config node.
|
|
func (m *Map) MatchErr(format string, args ...interface{}) error {
|
|
if m.curNode != nil {
|
|
return NodeErr(m.curNode, format, args...)
|
|
} else {
|
|
return fmt.Errorf(format, args...)
|
|
}
|
|
}
|
|
|
|
// AllowUnknown makes config.Map skip unknown configuration directives instead
|
|
// of failing.
|
|
func (m *Map) AllowUnknown() {
|
|
m.allowUnknown = true
|
|
}
|
|
|
|
// Bool maps presence of some configuration directive to a boolean variable.
|
|
//
|
|
// I.e. if directive 'io_debug' exists in processed configuration block or in
|
|
// the global configuration (if inheritGlobal is true) then Process will store
|
|
// true in target variable.
|
|
func (m *Map) Bool(name string, inheritGlobal bool, store *bool) {
|
|
m.Custom(name, inheritGlobal, false, func() (interface{}, error) {
|
|
return false, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 0 {
|
|
return nil, m.MatchErr("unexpected arguments")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
return true, nil
|
|
}, store)
|
|
}
|
|
|
|
// String maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name arbitrary_string'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) String(name string, inheritGlobal, required bool, defaultVal string, store *string) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
return node.Args[0], nil
|
|
}, store)
|
|
}
|
|
|
|
// Int maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) Int(name string, inheritGlobal, required bool, defaultVal int, store *int) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.Atoi(node.Args[0])
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return i, nil
|
|
}, store)
|
|
}
|
|
|
|
// UInt maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) UInt(name string, inheritGlobal, required bool, defaultVal uint, store *uint) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.ParseUint(node.Args[0], 10, 32)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return uint(i), nil
|
|
}, store)
|
|
}
|
|
|
|
// Int32 maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) Int32(name string, inheritGlobal, required bool, defaultVal int32, store *int32) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.ParseInt(node.Args[0], 10, 32)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return int32(i), nil
|
|
}, store)
|
|
}
|
|
|
|
// UInt32 maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) UInt32(name string, inheritGlobal, required bool, defaultVal uint32, store *uint32) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.ParseUint(node.Args[0], 10, 32)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return uint32(i), nil
|
|
}, store)
|
|
}
|
|
|
|
// Int64 maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) Int64(name string, inheritGlobal, required bool, defaultVal int64, store *int64) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.ParseInt(node.Args[0], 10, 64)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return i, nil
|
|
}, store)
|
|
}
|
|
|
|
// UInt64 maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) UInt64(name string, inheritGlobal, required bool, defaultVal uint64, store *uint64) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
if len(node.Children) != 0 {
|
|
return nil, m.MatchErr("can't declare block here")
|
|
}
|
|
|
|
i, err := strconv.ParseUint(node.Args[0], 10, 64)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid integer: %s", node.Args[0])
|
|
}
|
|
return i, nil
|
|
}, store)
|
|
}
|
|
|
|
// Float maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// Configuration directive must be in form 'name 123.55'.
|
|
//
|
|
// See Custom function for details about inheritGlobal, required and
|
|
// defaultVal.
|
|
func (m *Map) Float(name string, inheritGlobal, required bool, defaultVal float64, store *float64) {
|
|
m.Custom(name, inheritGlobal, required, func() (interface{}, error) {
|
|
return defaultVal, nil
|
|
}, func(m *Map, node *Node) (interface{}, error) {
|
|
if len(node.Args) != 1 {
|
|
return nil, m.MatchErr("expected 1 argument")
|
|
}
|
|
|
|
f, err := strconv.ParseFloat(node.Args[0], 64)
|
|
if err != nil {
|
|
return nil, m.MatchErr("invalid float: %s", node.Args[0])
|
|
}
|
|
return f, nil
|
|
}, store)
|
|
}
|
|
|
|
// Custom maps configuration directive with the specified name to variable
|
|
// referenced by 'store' pointer.
|
|
//
|
|
// If inheritGlobal is true - Map will try to use a value from globalCfg if
|
|
// none is set in a processed configuration block.
|
|
//
|
|
// If required is true - Map will fail if no value is set in the configuration,
|
|
// both global (if inheritGlobal is true) and in the processed block.
|
|
//
|
|
// defaultVal is a factory function that should return the default value for
|
|
// the variable. It will be used if no value is set in the config. It can be
|
|
// nil if required is true.
|
|
//
|
|
// mapper is a function that should convert configuration directive arguments
|
|
// into variable value. Both functions may fail with errors, configuration
|
|
// processing will stop immediately then.
|
|
// Note: mapper function should not modify passed values.
|
|
//
|
|
// store is where the value returned by mapper should be stored. Can be nil
|
|
// (value will be saved only in Map.Values).
|
|
func (m *Map) Custom(name string, inheritGlobal, required bool, defaultVal func() (interface{}, error), mapper func(*Map, *Node) (interface{}, error), store interface{}) {
|
|
if m.entries == nil {
|
|
m.entries = make(map[string]matcher)
|
|
}
|
|
if _, ok := m.entries[name]; ok {
|
|
panic("Map.Custom: duplicate matcher")
|
|
}
|
|
|
|
var target *reflect.Value
|
|
ptr := reflect.ValueOf(store)
|
|
if ptr.IsValid() && !ptr.IsNil() {
|
|
val := ptr.Elem()
|
|
if !val.CanSet() {
|
|
panic("Map.Custom: store argument must be settable (a pointer)")
|
|
}
|
|
target = &val
|
|
}
|
|
|
|
m.entries[name] = matcher{
|
|
name: name,
|
|
inheritGlobal: inheritGlobal,
|
|
required: required,
|
|
defaultVal: defaultVal,
|
|
mapper: mapper,
|
|
store: target,
|
|
}
|
|
}
|
|
|
|
// Process maps variables from global configuration and block passed in NewMap.
|
|
//
|
|
// If Map instance was not created using NewMap - Process panics.
|
|
func (m *Map) Process() (unmatched []Node, err error) {
|
|
return m.ProcessWith(m.Globals, m.Block)
|
|
}
|
|
|
|
// Process maps variables from global configuration and block passed in arguments.
|
|
func (m *Map) ProcessWith(globalCfg map[string]interface{}, block *Node) (unmatched []Node, err error) {
|
|
unmatched = make([]Node, 0, len(block.Children))
|
|
matched := make(map[string]bool)
|
|
m.Values = make(map[string]interface{})
|
|
|
|
for _, subnode := range block.Children {
|
|
m.curNode = &subnode
|
|
|
|
if matched[subnode.Name] {
|
|
return nil, m.MatchErr("duplicate directive: %s", subnode.Name)
|
|
}
|
|
|
|
matcher, ok := m.entries[subnode.Name]
|
|
if !ok {
|
|
if !m.allowUnknown {
|
|
return nil, m.MatchErr("unexpected directive: %s", subnode.Name)
|
|
}
|
|
unmatched = append(unmatched, subnode)
|
|
continue
|
|
}
|
|
|
|
val, err := matcher.mapper(m, m.curNode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m.Values[matcher.name] = val
|
|
if matcher.store != nil {
|
|
matcher.assign(val)
|
|
}
|
|
matched[subnode.Name] = true
|
|
}
|
|
m.curNode = block
|
|
|
|
for _, matcher := range m.entries {
|
|
if matched[matcher.name] {
|
|
continue
|
|
}
|
|
|
|
var val interface{}
|
|
globalVal, ok := globalCfg[matcher.name]
|
|
if matcher.inheritGlobal && ok {
|
|
val = globalVal
|
|
} else if !matcher.required {
|
|
if matcher.defaultVal == nil {
|
|
continue
|
|
}
|
|
|
|
val, err = matcher.defaultVal()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
return nil, m.MatchErr("missing required directive: %s", matcher.name)
|
|
}
|
|
|
|
m.Values[matcher.name] = val
|
|
if matcher.store != nil {
|
|
matcher.assign(val)
|
|
}
|
|
}
|
|
|
|
return unmatched, nil
|
|
}
|