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"
|
2024-05-04 22:35:03 +02:00
|
|
|
"fmt"
|
2024-04-02 04:36:03 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// -------- function call term
|
|
|
|
func newFuncCallTerm(tk *Token, args []*term) *term {
|
2024-03-26 07:00:53 +01:00
|
|
|
return &term{
|
2024-04-26 04:36:03 +02:00
|
|
|
tk: *tk,
|
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-06-01 19:56:40 +02:00
|
|
|
func checkFunctionCall(ctx ExprContext, name string, varParams *[]any) (err error) {
|
2024-05-11 10:45:38 +02:00
|
|
|
if info, exists, owner := GetFuncInfo(ctx, name); exists {
|
2024-06-01 19:56:40 +02:00
|
|
|
passedCount := len(*varParams)
|
|
|
|
if info.MinArgs() > passedCount {
|
|
|
|
err = errTooFewParams(name, info.MinArgs(), info.MaxArgs(), passedCount)
|
2024-05-04 22:35:03 +02:00
|
|
|
}
|
2024-06-01 19:56:40 +02:00
|
|
|
for i, p := range info.Params() {
|
|
|
|
if i >= passedCount {
|
|
|
|
if !p.IsOptional() {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
*varParams = append(*varParams, p.DefaultValue())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err == nil && info.MaxArgs() >= 0 && info.MaxArgs() < len(*varParams) {
|
|
|
|
err = errTooMuchParams(name, info.MaxArgs(), len(*varParams))
|
2024-05-04 22:35:03 +02:00
|
|
|
}
|
2024-05-11 10:45:38 +02:00
|
|
|
if err == nil && owner != ctx {
|
2024-05-22 20:52:44 +02:00
|
|
|
// ctx.RegisterFunc(name, info.Functor(), info.MinArgs(), info.MaxArgs())
|
|
|
|
ctx.RegisterFuncInfo(info)
|
2024-05-11 10:45:38 +02:00
|
|
|
}
|
2024-05-04 22:35:03 +02:00
|
|
|
} else {
|
|
|
|
err = fmt.Errorf("unknown function %s()", name)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func evalFuncCall(parentCtx ExprContext, self *term) (v any, err error) {
|
2024-04-26 20:12:56 +02:00
|
|
|
ctx := cloneContext(parentCtx)
|
2024-03-26 07:00:53 +01:00
|
|
|
name, _ := self.tk.Value.(string)
|
2024-05-01 05:57:08 +02:00
|
|
|
// fmt.Printf("Call %s(), context: %p\n", name, ctx)
|
2024-06-01 19:56:40 +02:00
|
|
|
params := make([]any, len(self.children), len(self.children)+5)
|
2024-03-26 07:00:53 +01:00
|
|
|
for i, tree := range self.children {
|
|
|
|
var param any
|
|
|
|
if param, err = tree.compute(ctx); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
params[i] = param
|
|
|
|
}
|
2024-06-01 19:56:40 +02:00
|
|
|
|
2024-03-26 07:00:53 +01:00
|
|
|
if err == nil {
|
2024-06-01 19:56:40 +02:00
|
|
|
if err = checkFunctionCall(ctx, name, ¶ms); err == nil {
|
2024-05-04 22:35:03 +02:00
|
|
|
if v, err = ctx.Call(name, params); err == nil {
|
|
|
|
exportObjects(parentCtx, ctx)
|
|
|
|
}
|
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
|
|
|
|
|
|
|
// -------- function definition term
|
|
|
|
func newFuncDefTerm(tk *Token, args []*term) *term {
|
|
|
|
return &term{
|
2024-05-04 22:35:03 +02:00
|
|
|
tk: *tk, // value is the expression body
|
2024-04-02 04:36:03 +02:00
|
|
|
parent: nil,
|
2024-05-04 22:35:03 +02:00
|
|
|
children: args, // function params
|
2024-04-02 04:36:03 +02:00
|
|
|
position: posLeaf,
|
|
|
|
priority: priValue,
|
|
|
|
evalFunc: evalFuncDef,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------- eval func def
|
2024-06-01 19:56:40 +02:00
|
|
|
// func _evalFuncDef(ctx ExprContext, self *term) (v any, err error) {
|
|
|
|
// bodySpec := self.value()
|
|
|
|
// if expr, ok := bodySpec.(*ast); ok {
|
|
|
|
// paramList := make([]string, 0, len(self.children))
|
|
|
|
// for _, param := range self.children {
|
|
|
|
// paramList = append(paramList, param.source())
|
|
|
|
// }
|
|
|
|
// v = newExprFunctor(expr, paramList, ctx)
|
|
|
|
// } else {
|
|
|
|
// err = errors.New("invalid function definition: the body specification must be an expression")
|
|
|
|
// }
|
|
|
|
// 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 {
|
2024-06-01 19:56:40 +02:00
|
|
|
paramList := make([]ExprFuncParam, 0, len(self.children))
|
2024-04-04 12:54:26 +02:00
|
|
|
for _, param := range self.children {
|
2024-06-01 19:56:40 +02:00
|
|
|
var defValue any
|
|
|
|
flags := paramFlags(0)
|
|
|
|
if len(param.children) > 0 {
|
|
|
|
flags |= pfOptional
|
|
|
|
if defValue, err = param.children[0].compute(ctx); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info := newFuncParamFlagDef(param.source(), flags, defValue)
|
|
|
|
paramList = append(paramList, info)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
2024-05-30 07:13:26 +02:00
|
|
|
v = newExprFunctor(expr, paramList, ctx)
|
2024-04-02 04:36:03 +02:00
|
|
|
} else {
|
|
|
|
err = errors.New("invalid function definition: the body specification must be an expression")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|