mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 14:07:38 +03:00
Use config.Map for global config directives
This commit is contained in:
parent
a72eb5d968
commit
18f71d8fc8
2 changed files with 58 additions and 63 deletions
|
@ -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
|
||||||
|
|
29
maddy.go
29
maddy.go
|
@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue