Use config.Map for global config directives

This commit is contained in:
fox.cpp 2019-04-06 02:38:32 +03:00 committed by Simon Ser
parent a72eb5d968
commit 18f71d8fc8
2 changed files with 58 additions and 63 deletions

View file

@ -13,15 +13,10 @@ type matcher struct {
inheritGlobal bool inheritGlobal bool
defaultVal func() (interface{}, error) defaultVal func() (interface{}, error)
mapper func(*Map, *Node) (interface{}, error) mapper func(*Map, *Node) (interface{}, error)
store reflect.Value store *reflect.Value
} }
func (m *matcher) match(map_ *Map, node *Node) error { func (m *matcher) assign(val interface{}) {
val, err := m.mapper(map_, node)
if err != nil {
return err
}
valRefl := reflect.ValueOf(val) valRefl := reflect.ValueOf(val)
// Convert untyped nil into typed nil. Otherwise it will panic. // Convert untyped nil into typed nil. Otherwise it will panic.
if !valRefl.IsValid() { if !valRefl.IsValid() {
@ -29,7 +24,6 @@ func (m *matcher) match(map_ *Map, node *Node) error {
} }
m.store.Set(valRefl) m.store.Set(valRefl)
return nil
} }
// Map structure implements reflection-based conversion between configuration // Map structure implements reflection-based conversion between configuration
@ -41,14 +35,19 @@ type Map struct {
// called. // called.
curNode *Node curNode *Node
// All values saved by Map during processing.
Values map[string]interface{}
entries map[string]matcher entries map[string]matcher
GlobalCfg map[string]Node // Values used by Process as default values if inheritGlobal is true.
Block *Node Globals map[string]interface{}
// Config block used by Process.
Block *Node
} }
func NewMap(globalCfg map[string]Node, block *Node) *Map { func NewMap(globals map[string]interface{}, block *Node) *Map {
return &Map{GlobalCfg: globalCfg, Block: block} return &Map{Globals: globals, Block: block}
} }
// MatchErr returns error with formatted message, if called from defaultVal or // MatchErr returns error with formatted message, if called from defaultVal or
@ -307,27 +306,34 @@ func (m *Map) Float(name string, inheritGlobal, required bool, defaultVal float6
// mapper is a function that should convert configuration directive arguments // mapper is a function that should convert configuration directive arguments
// into variable value. Both functions may fail with errors, configuration // into variable value. Both functions may fail with errors, configuration
// processing will stop immediately then. // processing will stop immediately then.
//
// 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{}) { 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 { if m.entries == nil {
m.entries = make(map[string]matcher) m.entries = make(map[string]matcher)
} }
val := reflect.ValueOf(store).Elem()
if !val.CanSet() {
panic("Map.Custom: store argument must be settable (a pointer)")
}
if _, ok := m.entries[name]; ok { if _, ok := m.entries[name]; ok {
panic("Map.Custom: duplicate matcher") 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{ m.entries[name] = matcher{
name: name, name: name,
inheritGlobal: inheritGlobal, inheritGlobal: inheritGlobal,
required: required, required: required,
defaultVal: defaultVal, defaultVal: defaultVal,
mapper: mapper, mapper: mapper,
store: val, store: target,
} }
} }
@ -335,33 +341,39 @@ func (m *Map) Custom(name string, inheritGlobal, required bool, defaultVal func(
// //
// If Map instance was not created using NewMap - Process panics. // If Map instance was not created using NewMap - Process panics.
func (m *Map) Process() (unmatched []Node, err error) { func (m *Map) Process() (unmatched []Node, err error) {
return m.ProcessWith(m.GlobalCfg, m.Block) return m.ProcessWith(m.Globals, m.Block.Children)
} }
// Process maps variables from global configuration and block passedin arguments. // Process maps variables from global configuration and block passed in arguments.
func (m *Map) ProcessWith(globalCfg map[string]Node, block *Node) (unmatched []Node, err error) { func (m *Map) ProcessWith(globalCfg map[string]interface{}, block []Node) (unmatched []Node, err error) {
unmatched = make([]Node, 0, len(block.Children)) unmatched = make([]Node, 0, len(block))
matched := make(map[string]bool) matched := make(map[string]bool)
m.Values = make(map[string]interface{})
for _, subnode := range block.Children { for _, subnode := range block {
m.curNode = &subnode m.curNode = &subnode
if matched[subnode.Name] { if matched[subnode.Name] {
return nil, m.MatchErr("duplicate directive") return nil, m.MatchErr("duplicate directive: %s", subnode.Name)
} }
matcher, ok := m.entries[subnode.Name] matcher, ok := m.entries[subnode.Name]
if !ok { if !ok {
if !m.allowUnknown { if !m.allowUnknown {
return nil, m.MatchErr("unexpected directive") return nil, m.MatchErr("unexpected directive: %s", subnode.Name)
} }
unmatched = append(unmatched, subnode) unmatched = append(unmatched, subnode)
continue continue
} }
if err := matcher.match(m, m.curNode); err != nil { val, err := matcher.mapper(m, m.curNode)
if err != nil {
return nil, err return nil, err
} }
m.Values[matcher.name] = val
if matcher.store != nil {
matcher.assign(val)
}
matched[subnode.Name] = true matched[subnode.Name] = true
} }
m.curNode = nil m.curNode = nil
@ -371,38 +383,30 @@ func (m *Map) ProcessWith(globalCfg map[string]Node, block *Node) (unmatched []N
continue continue
} }
globalNode, ok := globalCfg[matcher.name] var val interface{}
globalVal, ok := globalCfg[matcher.name]
if matcher.inheritGlobal && ok { if matcher.inheritGlobal && ok {
m.curNode = &globalNode val = globalVal
if err := matcher.match(m, m.curNode); err != nil {
m.curNode = nil
return nil, err
}
m.curNode = nil
} else if !matcher.required { } else if !matcher.required {
if matcher.defaultVal == nil { if matcher.defaultVal == nil {
continue continue
} }
val, err := matcher.defaultVal() val, err = matcher.defaultVal()
if err != nil { if err != nil {
return nil, err return nil, err
} }
if val == nil { if val == nil {
return nil, m.MatchErr("missing required directive: %s", matcher.name) return nil, m.MatchErr("missing required directive: %s", matcher.name)
} }
valRefl := reflect.ValueOf(val)
// Convert untyped nil into typed nil. Otherwise it will panic.
if !valRefl.IsValid() {
valRefl = reflect.Zero(matcher.store.Type())
}
matcher.store.Set(valRefl)
continue
} else { } else {
return nil, m.MatchErr("missing required directive: %s", matcher.name) 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 return unmatched, nil

View file

@ -19,26 +19,17 @@ type modInfo struct {
func Start(cfg []config.Node) error { func Start(cfg []config.Node) error {
instances := make(map[string]modInfo) instances := make(map[string]modInfo)
globalCfg := make(map[string]config.Node) globals := config.NewMap(nil, &config.Node{Children: cfg})
globals.String("hostname", false, false, "", nil)
for _, block := range cfg { globals.Custom("tls", false, true, nil, tlsDirective, nil)
switch block.Name { globals.Bool("debug", false, &log.DefaultLogger.Debug)
case "tls", "hostname", "debug": globals.AllowUnknown()
globalCfg[block.Name] = block unmatched, err := globals.Process()
continue if err != nil {
} return err
} }
if _, ok := globalCfg["debug"]; ok { for _, block := range unmatched {
log.DefaultLogger.Debug = true
}
for _, block := range cfg {
switch block.Name {
case "hostname", "tls", "debug":
continue
}
var instName string var instName string
if len(block.Args) == 0 { if len(block.Args) == 0 {
instName = block.Name instName = block.Name
@ -72,7 +63,7 @@ func Start(cfg []config.Node) error {
for _, inst := range instances { for _, inst := range instances {
log.Debugln("module init", inst.instance.Name(), inst.instance.InstanceName()) log.Debugln("module init", inst.instance.Name(), inst.instance.InstanceName())
if err := inst.instance.Init(config.NewMap(globalCfg, &inst.cfg)); err != nil { if err := inst.instance.Init(config.NewMap(globals.Values, &inst.cfg)); err != nil {
return err return err
} }
} }