2024-03-26 08:45:18 +01:00
|
|
|
// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com).
|
|
|
|
// All rights reserved.
|
|
|
|
|
2024-03-26 07:00:53 +01:00
|
|
|
// operand-func.go
|
|
|
|
package expr
|
|
|
|
|
2024-04-02 04:36:03 +02:00
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// -------- function call term
|
|
|
|
func newFuncCallTerm(tk *Token, args []*term) *term {
|
2024-03-26 07:00:53 +01:00
|
|
|
return &term{
|
2024-04-09 05:32:50 +02:00
|
|
|
tk: *tk,
|
|
|
|
// class: classVar,
|
|
|
|
// kind: kindUnknown,
|
2024-03-26 07:00:53 +01:00
|
|
|
parent: nil,
|
|
|
|
children: args,
|
|
|
|
position: posLeaf,
|
|
|
|
priority: priValue,
|
2024-04-02 04:36:03 +02:00
|
|
|
evalFunc: evalFuncCall,
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-02 04:36:03 +02:00
|
|
|
// -------- eval func call
|
2024-04-08 23:17:56 +02:00
|
|
|
func evalFuncCall(parentCtx ExprContext, self *term) (v any, err error) {
|
2024-04-04 12:54:26 +02:00
|
|
|
ctx := parentCtx.Clone()
|
2024-03-26 07:00:53 +01:00
|
|
|
name, _ := self.tk.Value.(string)
|
|
|
|
params := make([]any, len(self.children))
|
|
|
|
for i, tree := range self.children {
|
|
|
|
var param any
|
|
|
|
if param, err = tree.compute(ctx); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
params[i] = param
|
|
|
|
}
|
|
|
|
if err == nil {
|
2024-04-04 12:54:26 +02:00
|
|
|
if v, err = ctx.Call(name, params); err == nil {
|
2024-04-06 03:06:07 +02:00
|
|
|
exportAll := isEnabled(ctx, control_export_all)
|
2024-04-06 01:00:29 +02:00
|
|
|
// Export variables
|
2024-04-06 03:06:07 +02:00
|
|
|
for _, refName := range ctx.EnumVars(func(name string) bool { return exportAll || name[0] == '@' }) {
|
2024-04-04 12:54:26 +02:00
|
|
|
refValue, _ := ctx.GetVar(refName)
|
2024-04-06 03:06:07 +02:00
|
|
|
exportVar(parentCtx, refName, refValue)
|
2024-04-04 12:54:26 +02:00
|
|
|
}
|
2024-04-06 01:00:29 +02:00
|
|
|
// Export functions
|
2024-04-06 03:06:07 +02:00
|
|
|
for _, refName := range ctx.EnumFuncs(func(name string) bool { return exportAll || name[0] == '@' }) {
|
2024-04-06 01:00:29 +02:00
|
|
|
if info := ctx.GetFuncInfo(refName); info != nil {
|
2024-04-06 03:06:07 +02:00
|
|
|
exportFunc(parentCtx, refName, info)
|
2024-04-06 01:00:29 +02:00
|
|
|
}
|
|
|
|
}
|
2024-04-04 12:54:26 +02:00
|
|
|
}
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func exportVar(ctx ExprContext, name string, value any) {
|
2024-04-06 03:06:07 +02:00
|
|
|
if name[0] == '@' {
|
|
|
|
name = name[1:]
|
|
|
|
}
|
2024-04-09 07:12:22 +02:00
|
|
|
ctx.setVar(name, value)
|
2024-04-06 03:06:07 +02:00
|
|
|
}
|
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func exportFunc(ctx ExprContext, name string, info ExprFunc) {
|
2024-04-06 03:06:07 +02:00
|
|
|
if name[0] == '@' {
|
|
|
|
name = name[1:]
|
|
|
|
}
|
|
|
|
ctx.RegisterFunc(name, info.Functor(), info.MinArgs(), info.MaxArgs())
|
|
|
|
}
|
|
|
|
|
2024-04-02 04:36:03 +02:00
|
|
|
// -------- function definition term
|
|
|
|
func newFuncDefTerm(tk *Token, args []*term) *term {
|
|
|
|
return &term{
|
|
|
|
tk: *tk,
|
|
|
|
parent: nil,
|
|
|
|
children: args, // arg[0]=formal-param-list, arg[1]=*ast
|
|
|
|
position: posLeaf,
|
|
|
|
priority: priValue,
|
|
|
|
evalFunc: evalFuncDef,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------- eval func def
|
|
|
|
// TODO
|
|
|
|
type funcDefFunctor struct {
|
|
|
|
params []string
|
|
|
|
expr Expr
|
|
|
|
}
|
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func (functor *funcDefFunctor) Invoke(ctx ExprContext, name string, args []any) (result any, err error) {
|
2024-04-02 04:36:03 +02:00
|
|
|
for i, p := range functor.params {
|
|
|
|
if i < len(args) {
|
2024-04-09 07:12:22 +02:00
|
|
|
ctx.setVar(p, args[i])
|
2024-04-02 04:36:03 +02:00
|
|
|
} else {
|
2024-04-09 07:12:22 +02:00
|
|
|
ctx.setVar(p, nil)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
}
|
2024-04-08 23:17:56 +02:00
|
|
|
result, err = functor.expr.eval(ctx, false)
|
2024-04-02 04:36:03 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func evalFuncDef(ctx ExprContext, self *term) (v any, err error) {
|
2024-04-02 04:36:03 +02:00
|
|
|
bodySpec := self.value()
|
|
|
|
if expr, ok := bodySpec.(*ast); ok {
|
|
|
|
paramList := make([]string, 0, len(self.children))
|
2024-04-04 12:54:26 +02:00
|
|
|
for _, param := range self.children {
|
|
|
|
paramList = append(paramList, param.source())
|
|
|
|
// if paramName, ok := param.value().(string); ok {
|
|
|
|
// paramList = append(paramList, paramName)
|
|
|
|
// } else {
|
|
|
|
// err = fmt.Errorf("invalid function definition: formal param nr %d must be an identifier", i+1)
|
|
|
|
// break
|
|
|
|
// }
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
v = &funcDefFunctor{
|
|
|
|
params: paramList,
|
|
|
|
expr: expr,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = errors.New("invalid function definition: the body specification must be an expression")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|