// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com). // All rights reserved. // helpers.go package expr import ( "fmt" "io" "os" "strings" ) func EvalString(ctx ExprContext, source string) (result any, err error) { var tree *ast r := strings.NewReader(source) scanner := NewScanner(r, DefaultTranslations()) parser := NewParser(ctx) if tree, err = parser.Parse(scanner); err == nil { result, err = tree.eval(ctx, true) } return } type Arg struct { Name string Value any } func EvalStringA(source string, args ...Arg) (result any, err error) { return EvalStringV(source, args) } func EvalStringV(source string, args []Arg) (result any, err error) { ctx := NewSimpleStore() for _, arg := range args { if isFunc(arg.Value) { if f, ok := arg.Value.(FuncTemplate); ok { functor := newSimpleFunctor(f) // ctx.RegisterFunc(arg.Name, functor, 0, -1) ctx.RegisterFunc2(arg.Name, functor, typeAny, []ExprFuncParam{ newFuncParamFlagDef(paramValue, pfOptional|pfRepeat, 0), }) } else { err = fmt.Errorf("invalid function specification: %q", arg.Name) } } else if integer, ok := anyInteger(arg.Value); ok { ctx.SetVar(arg.Name, integer) } else if float, ok := anyFloat(arg.Value); ok { ctx.SetVar(arg.Name, float) } else if _, ok := arg.Value.(string); ok { ctx.SetVar(arg.Name, arg.Value) } else if _, ok := arg.Value.(bool); ok { ctx.SetVar(arg.Name, arg.Value) } else { err = fmt.Errorf("unsupported type %T specified for item %q", arg.Value, arg.Name) } } if err == nil { result, err = EvalString(ctx, source) } return } func EvalStream(ctx ExprContext, r io.Reader) (result any, err error) { var tree *ast scanner := NewScanner(r, DefaultTranslations()) parser := NewParser(ctx) if tree, err = parser.Parse(scanner); err == nil { result, err = tree.Eval(ctx) } return } func EvalFile(ctx ExprContext, filePath string) (result any, err error) { var fh *os.File if fh, err = os.Open(filePath); err != nil { return nil, err } defer fh.Close() result, err = EvalStream(ctx, fh) return }