alrm/config/parser.go

263 lines
4.9 KiB
Go
Raw Normal View History

package config
2020-08-20 04:34:22 -08:00
import (
"fmt"
2021-02-03 05:59:31 -09:00
"git.binarythought.com/cdramey/alrm/alarm"
"git.binarythought.com/cdramey/alrm/check"
2020-08-20 04:34:22 -08:00
"strings"
)
const (
2021-01-18 07:29:38 -09:00
PR_NONE = iota
PR_SET
PR_MONITOR
PR_GROUP
PR_HOST
PR_CHECK
PR_ALARM
2020-08-20 04:34:22 -08:00
)
type Parser struct {
DebugLevel int
states []int
lastHost *Host
lastGroup *Group
lastCheck check.Check
lastAlarm alarm.Alarm
lastAlarmName string
2020-08-20 04:34:22 -08:00
}
func (p *Parser) Parse(fn string) (*Config, error) {
config := NewConfig()
config.Path = fn
tok, err := NewTokenizer(fn)
2020-08-20 04:34:22 -08:00
if err != nil {
return nil, err
2020-08-20 04:34:22 -08:00
}
defer tok.Close()
2020-08-20 04:34:22 -08:00
for tok.Scan() {
tk := tok.Text()
2020-08-20 04:34:22 -08:00
stateswitch:
switch p.state() {
2021-01-18 07:29:38 -09:00
case PR_NONE:
2020-08-20 04:34:22 -08:00
switch strings.ToLower(tk) {
case "monitor":
2021-01-18 07:29:38 -09:00
p.setState(PR_MONITOR)
2020-08-20 04:34:22 -08:00
case "set":
2021-01-18 07:29:38 -09:00
p.setState(PR_SET)
case "alarm":
2021-01-18 07:29:38 -09:00
p.setState(PR_ALARM)
2020-08-20 04:34:22 -08:00
default:
return nil, fmt.Errorf("invalid token in %s, line %d: \"%s\"",
fn, tok.Line(), tk)
2020-08-20 04:34:22 -08:00
}
2021-01-18 07:29:38 -09:00
case PR_SET:
2020-08-20 04:34:22 -08:00
key := strings.ToLower(tk)
if !tok.Scan() {
2020-08-20 08:44:56 -08:00
return nil, fmt.Errorf("empty value name for set in %s, line %d",
fn, tok.Line())
2020-08-20 08:44:56 -08:00
}
value := tok.Text()
2020-08-20 08:44:56 -08:00
switch key {
case "interval":
err := config.SetInterval(value)
if err != nil {
return nil, fmt.Errorf(
"invalid duration for interval in %s, line %d: \"%s\"",
fn, tok.Line(), value,
)
}
case "listen":
config.Listen = value
2020-08-20 08:44:56 -08:00
default:
return nil, fmt.Errorf("unknown key for set in %s, line %d: \"%s\"",
fn, tok.Line(), tk,
2020-08-20 08:44:56 -08:00
)
2020-08-20 04:34:22 -08:00
}
p.prevState()
2021-01-18 07:29:38 -09:00
case PR_MONITOR:
2020-08-20 04:34:22 -08:00
switch strings.ToLower(tk) {
case "host":
2021-01-18 07:29:38 -09:00
p.setState(PR_HOST)
2020-08-20 04:34:22 -08:00
case "group":
2021-01-18 07:29:38 -09:00
p.setState(PR_GROUP)
2020-08-20 04:34:22 -08:00
default:
p.prevState()
goto stateswitch
}
2021-01-18 07:29:38 -09:00
case PR_GROUP:
if p.lastGroup == nil {
p.lastGroup, err = config.NewGroup(tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line(),
)
}
continue
2020-08-20 04:34:22 -08:00
}
switch strings.ToLower(tk) {
case "host":
2021-01-18 07:29:38 -09:00
p.setState(PR_HOST)
2020-08-20 04:34:22 -08:00
default:
p.prevState()
goto stateswitch
}
2021-01-18 07:29:38 -09:00
case PR_HOST:
// If a host has no group, inherit the host name
if p.lastGroup == nil {
p.lastGroup, err = config.NewGroup(tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line(),
)
}
2020-08-20 04:34:22 -08:00
}
if p.lastHost == nil {
p.lastHost, err = p.lastGroup.NewHost(tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line(),
)
}
2020-08-20 04:34:22 -08:00
continue
}
switch strings.ToLower(tk) {
case "address":
if !tok.Scan() {
2020-08-20 08:44:56 -08:00
return nil, fmt.Errorf("empty address for host in %s, line %d",
fn, tok.Line())
2020-08-20 04:34:22 -08:00
}
p.lastHost.Address = tok.Text()
2020-08-20 08:44:56 -08:00
case "check":
2021-01-18 07:29:38 -09:00
p.setState(PR_CHECK)
2020-08-20 04:34:22 -08:00
default:
p.prevState()
goto stateswitch
}
2021-01-18 07:29:38 -09:00
case PR_CHECK:
if p.lastCheck == nil {
p.lastCheck, err = p.lastHost.NewCheck(tk)
2020-08-20 08:44:56 -08:00
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line())
2020-08-20 08:44:56 -08:00
}
continue
}
cont, err := p.lastCheck.Parse(tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line())
}
if !cont {
p.lastCheck = nil
p.prevState()
goto stateswitch
}
2021-01-18 07:29:38 -09:00
case PR_ALARM:
if p.lastAlarm == nil {
if p.lastAlarmName == "" {
p.lastAlarmName = tk
continue
}
p.lastAlarm, err = config.NewAlarm(p.lastAlarmName, tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line())
}
p.lastAlarmName = ""
continue
}
cont, err := p.lastAlarm.Parse(tk)
if err != nil {
return nil, fmt.Errorf("%s in %s, line %d",
err.Error(), fn, tok.Line())
}
if !cont {
p.lastAlarm = nil
p.prevState()
goto stateswitch
}
2020-08-20 08:44:56 -08:00
2020-08-20 04:34:22 -08:00
default:
return nil, fmt.Errorf("unknown parser state: %d", p.state())
}
}
if err := tok.Err(); err != nil {
2020-08-20 04:34:22 -08:00
return nil, err
}
return config, nil
}
func (p *Parser) state() int {
if len(p.states) < 1 {
2021-01-18 07:29:38 -09:00
return PR_NONE
2020-08-20 04:34:22 -08:00
}
return p.states[len(p.states)-1]
}
func (p *Parser) setState(state int) {
switch state {
2021-01-18 07:29:38 -09:00
case PR_SET, PR_MONITOR:
fallthrough
2021-01-18 07:29:38 -09:00
case PR_GROUP:
p.lastGroup = nil
fallthrough
2021-01-18 07:29:38 -09:00
case PR_HOST:
p.lastHost = nil
p.lastCheck = nil
}
if p.DebugLevel > 1 {
fmt.Printf("Parser state: %s", p.stateName())
}
2020-08-20 04:34:22 -08:00
p.states = append(p.states, state)
if p.DebugLevel > 1 {
fmt.Printf(" -> %s\n", p.stateName())
}
2020-08-20 04:34:22 -08:00
}
func (p *Parser) prevState() int {
if len(p.states) > 0 {
p.states = p.states[:len(p.states)-1]
}
return p.state()
}
func (p *Parser) stateName() string {
switch p.state() {
2021-01-18 07:29:38 -09:00
case PR_NONE:
return "PR_NONE"
case PR_SET:
return "PR_SET"
case PR_MONITOR:
return "PR_MONITOR"
case PR_GROUP:
return "PR_GROUP"
case PR_HOST:
return "PR_HOST"
case PR_CHECK:
return "PR_CHECK"
case PR_ALARM:
return "PR_ALARM"
2020-08-20 04:34:22 -08:00
default:
return "UNKNOWN"
}
}