3
0
corteza/pkg/expr/parser.go

111 lines
2.0 KiB
Go

package expr
import (
"context"
"fmt"
"github.com/PaesslerAG/gval"
)
type (
Parsable interface {
Parse(string) (Evaluable, error)
ParseEvaluators(ee ...Evaluator) error
}
Evaluable interface {
Eval(context.Context, *Vars) (interface{}, error)
Test(context.Context, *Vars) (bool, error)
}
Evaluator interface {
GetExpr() string
SetEval(eval Evaluable)
}
gvalParser struct {
lang gval.Language
}
gvalEval struct {
evaluable gval.Evaluable
}
)
func NewParser(ee ...gval.Language) Parsable {
return NewGvalParser(ee...)
}
func NewGvalParser(ee ...gval.Language) *gvalParser {
ext := AllFunctions()
ext = append(ext, ee...)
return &gvalParser{lang: gval.Full(ext...)}
}
func (p *gvalParser) Parse(expr string) (Evaluable, error) {
var (
ge = &gvalEval{}
err error
)
if ge.evaluable, err = p.lang.NewEvaluable(expr); err != nil {
return nil, err
}
return ge, err
}
func (p *gvalParser) ParseEvaluators(ee ...Evaluator) error {
for _, e := range ee {
evaluable, err := p.Parse(e.GetExpr())
if err != nil {
return err
}
e.SetEval(evaluable)
}
return nil
}
func (e *gvalEval) Eval(ctx context.Context, scope *Vars) (interface{}, error) {
return e.evaluable(ctx, scope)
}
func (e *gvalEval) Test(ctx context.Context, scope *Vars) (bool, error) {
r, err := e.evaluable(ctx, scope)
if err != nil {
return false, err
}
return !isEmpty(r), nil
}
func Parser(ll ...gval.Language) gval.Language {
return gval.Full(append(AllFunctions(), ll...)...)
}
func AllFunctions() []gval.Language {
ff := make([]gval.Language, 0, 100)
ff = append(ff, GenericFunctions()...)
ff = append(ff, StringFunctions()...)
ff = append(ff, JsonFunctions()...)
ff = append(ff, NumericFunctions()...)
ff = append(ff, TimeFunctions()...)
ff = append(ff, ArrayFunctions()...)
ff = append(ff, KvFunctions()...)
return ff
}
// utility function for examples
func eval(e string, p interface{}) {
result, err := Parser().Evaluate(e, p)
if err != nil {
fmt.Printf("error: %v", err)
} else {
fmt.Printf("%+v", UntypedValue(result))
}
}