-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathexpr-func.go
More file actions
82 lines (69 loc) · 1.75 KB
/
expr-func.go
File metadata and controls
82 lines (69 loc) · 1.75 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
package ex
import (
elutils "github.com/rosbit/go-embedding-utils"
"github.com/expr-lang/expr"
"reflect"
"fmt"
)
func (e *XExpr) bindFunc(funcName string, funcVarPtr interface{}) (err error) {
helper, er := elutils.NewEmbeddingFuncHelper(funcVarPtr)
if er != nil {
err = er
return
}
wf := e.wrapFunc(funcName, helper)
if wf == nil {
err = fmt.Errorf("func %s not found")
return
}
helper.BindEmbeddingFunc(wf)
return
}
func (e *XExpr) wrapFunc(funcName string, helper *elutils.EmbeddingFuncHelper) elutils.FnGoFunc {
funcMeta, err := e.getFunc(funcName)
if err != nil {
return nil
}
return func(args []reflect.Value) (results []reflect.Value) {
itArgs := helper.MakeGoFuncArgs(args)
res, err := e.evalWithArgs(funcMeta, itArgs)
// convert result to golang
if res != nil {
resKind := reflect.TypeOf(res).Kind()
isArray := resKind == reflect.Slice || resKind == reflect.Array
results = helper.ToGolangResults(res, isArray, err)
} else {
results = helper.ToGolangResults(nil, false, err)
}
return
}
}
func (e *XExpr) callFunc(funcMeta *fn, args ...interface{}) (res interface{}, err error) {
// copy global env
exArgs := e.makeEnvs(nil)
// set args
for i, arg := range args {
if i < len(funcMeta.Params) {
exArgs[funcMeta.Params[i]] = arg
}
}
return funcMeta.eval(exArgs)
}
func (e *XExpr) evalWithArgs(f *fn, args <-chan interface{}) (res interface{}, err error) {
// copy global env
exArgs := e.makeEnvs(nil)
// make expr args
i := 0
for arg := range args {
if i < len(f.Params) {
exArgs[f.Params[i]] = arg
i += 1
}
}
return f.eval(exArgs)
}
func (f *fn) eval(envs map[string]interface{}) (res interface{}, err error) {
// res, err = expr.Eval(f.Expr, envs)
res, err = expr.Run(f.prg, envs)
return
}