2024-05-24 06:28:48 +02:00
|
|
|
// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com).
|
|
|
|
// All rights reserved.
|
|
|
|
|
|
|
|
// function.go
|
|
|
|
package expr
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ---- Function template
|
2024-07-28 18:49:08 +02:00
|
|
|
type FuncTemplate func(ctx ExprContext, name string, args map[string]any) (result any, err error)
|
2024-05-24 06:28:48 +02:00
|
|
|
|
|
|
|
// ---- Common functor definition
|
|
|
|
type baseFunctor struct {
|
|
|
|
info ExprFunc
|
|
|
|
}
|
|
|
|
|
|
|
|
func (functor *baseFunctor) ToString(opt FmtOpt) (s string) {
|
|
|
|
if functor.info != nil {
|
|
|
|
s = functor.info.ToString(opt)
|
|
|
|
} else {
|
2024-06-24 07:20:17 +02:00
|
|
|
s = "func(){}"
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2024-06-01 16:31:50 +02:00
|
|
|
func (functor *baseFunctor) GetParams() (params []ExprFuncParam) {
|
|
|
|
if functor.info != nil {
|
|
|
|
return functor.info.Params()
|
|
|
|
} else {
|
|
|
|
return []ExprFuncParam{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-24 06:28:48 +02:00
|
|
|
func (functor *baseFunctor) SetFunc(info ExprFunc) {
|
|
|
|
functor.info = info
|
|
|
|
}
|
|
|
|
|
|
|
|
func (functor *baseFunctor) GetFunc() ExprFunc {
|
|
|
|
return functor.info
|
|
|
|
}
|
|
|
|
|
2024-06-24 07:20:17 +02:00
|
|
|
func (functor *baseFunctor) GetDefinitionContext() ExprContext {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-05-24 06:28:48 +02:00
|
|
|
// ---- Function Parameters
|
|
|
|
type paramFlags uint16
|
|
|
|
|
|
|
|
const (
|
2024-06-07 09:45:02 +02:00
|
|
|
PfDefault paramFlags = 1 << iota
|
|
|
|
PfOptional
|
2024-06-06 05:31:35 +02:00
|
|
|
PfRepeat
|
2024-05-24 06:28:48 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type funcParamInfo struct {
|
|
|
|
name string
|
|
|
|
flags paramFlags
|
|
|
|
defaultValue any
|
|
|
|
}
|
|
|
|
|
2024-06-06 05:31:35 +02:00
|
|
|
func NewFuncParam(name string) ExprFuncParam {
|
2024-05-24 06:28:48 +02:00
|
|
|
return &funcParamInfo{name: name}
|
|
|
|
}
|
|
|
|
|
2024-06-06 05:31:35 +02:00
|
|
|
func NewFuncParamFlag(name string, flags paramFlags) ExprFuncParam {
|
2024-05-24 06:28:48 +02:00
|
|
|
return &funcParamInfo{name: name, flags: flags}
|
|
|
|
}
|
|
|
|
|
2024-06-06 05:31:35 +02:00
|
|
|
func NewFuncParamFlagDef(name string, flags paramFlags, defValue any) *funcParamInfo {
|
2024-05-24 06:28:48 +02:00
|
|
|
return &funcParamInfo{name: name, flags: flags, defaultValue: defValue}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) Name() string {
|
|
|
|
return param.name
|
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) Type() string {
|
2024-06-07 09:45:02 +02:00
|
|
|
return TypeAny
|
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) IsDefault() bool {
|
|
|
|
return (param.flags & PfDefault) != 0
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) IsOptional() bool {
|
2024-06-06 05:31:35 +02:00
|
|
|
return (param.flags & PfOptional) != 0
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) IsRepeat() bool {
|
2024-06-06 05:31:35 +02:00
|
|
|
return (param.flags & PfRepeat) != 0
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (param *funcParamInfo) DefaultValue() any {
|
|
|
|
return param.defaultValue
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- Functions
|
2024-06-24 07:20:17 +02:00
|
|
|
|
|
|
|
// funcInfo implements ExprFunc
|
2024-05-24 06:28:48 +02:00
|
|
|
type funcInfo struct {
|
2024-06-24 07:20:17 +02:00
|
|
|
name string
|
|
|
|
minArgs int
|
|
|
|
maxArgs int
|
|
|
|
functor Functor
|
|
|
|
formalParams []ExprFuncParam
|
|
|
|
returnType string
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func newFuncInfo(name string, functor Functor, returnType string, params []ExprFuncParam) (info *funcInfo, err error) {
|
|
|
|
var minArgs = 0
|
|
|
|
var maxArgs = 0
|
2024-07-07 15:58:29 +02:00
|
|
|
for _, p := range params {
|
|
|
|
if maxArgs == -1 {
|
|
|
|
return nil, fmt.Errorf("no more params can be specified after the ellipsis symbol: %q", p.Name())
|
|
|
|
}
|
|
|
|
if p.IsDefault() || p.IsOptional() {
|
|
|
|
maxArgs++
|
|
|
|
} else if maxArgs == minArgs {
|
|
|
|
minArgs++
|
|
|
|
maxArgs++
|
|
|
|
} else {
|
|
|
|
return nil, fmt.Errorf("can't specify non-optional param after optional ones: %q", p.Name())
|
|
|
|
}
|
|
|
|
if p.IsRepeat() {
|
|
|
|
minArgs--
|
|
|
|
maxArgs = -1
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
}
|
2024-07-07 15:58:29 +02:00
|
|
|
|
2024-05-24 06:28:48 +02:00
|
|
|
info = &funcInfo{
|
2024-06-24 07:20:17 +02:00
|
|
|
name: name, minArgs: minArgs, maxArgs: maxArgs, functor: functor, returnType: returnType, formalParams: params,
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
functor.SetFunc(info)
|
|
|
|
return info, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) Params() []ExprFuncParam {
|
2024-06-24 07:20:17 +02:00
|
|
|
return info.formalParams
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) ReturnType() string {
|
|
|
|
return info.returnType
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) ToString(opt FmtOpt) string {
|
|
|
|
var sb strings.Builder
|
|
|
|
if len(info.Name()) == 0 {
|
|
|
|
sb.WriteString("func")
|
|
|
|
} else {
|
|
|
|
sb.WriteString(info.Name())
|
|
|
|
}
|
|
|
|
sb.WriteByte('(')
|
2024-06-24 07:20:17 +02:00
|
|
|
if info.formalParams != nil {
|
|
|
|
for i, p := range info.formalParams {
|
2024-05-24 06:28:48 +02:00
|
|
|
if i > 0 {
|
|
|
|
sb.WriteString(", ")
|
|
|
|
}
|
|
|
|
sb.WriteString(p.Name())
|
|
|
|
|
2024-06-07 09:45:02 +02:00
|
|
|
if p.IsDefault() {
|
2024-05-24 06:28:48 +02:00
|
|
|
sb.WriteByte('=')
|
|
|
|
if s, ok := p.DefaultValue().(string); ok {
|
|
|
|
sb.WriteByte('"')
|
|
|
|
sb.WriteString(s)
|
|
|
|
sb.WriteByte('"')
|
|
|
|
} else {
|
|
|
|
sb.WriteString(fmt.Sprintf("%v", p.DefaultValue()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if info.maxArgs < 0 {
|
|
|
|
sb.WriteString(" ...")
|
|
|
|
}
|
2024-06-02 11:30:24 +02:00
|
|
|
sb.WriteString("):")
|
2024-05-24 06:28:48 +02:00
|
|
|
if len(info.returnType) > 0 {
|
|
|
|
sb.WriteString(info.returnType)
|
|
|
|
} else {
|
2024-06-06 05:31:35 +02:00
|
|
|
sb.WriteString(TypeAny)
|
2024-05-24 06:28:48 +02:00
|
|
|
}
|
2024-06-24 07:20:17 +02:00
|
|
|
sb.WriteString("{}")
|
2024-05-24 06:28:48 +02:00
|
|
|
return sb.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) Name() string {
|
|
|
|
return info.name
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) MinArgs() int {
|
|
|
|
return info.minArgs
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) MaxArgs() int {
|
|
|
|
return info.maxArgs
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) Functor() Functor {
|
|
|
|
return info.functor
|
|
|
|
}
|
2024-06-17 06:54:50 +02:00
|
|
|
|
2024-06-24 07:20:17 +02:00
|
|
|
func (info *funcInfo) AllocContext(parentCtx ExprContext) (ctx ExprContext) {
|
|
|
|
if defCtx := info.functor.GetDefinitionContext(); defCtx != nil {
|
|
|
|
ctx = defCtx.Clone()
|
|
|
|
ctx.SetParent(defCtx)
|
|
|
|
} else {
|
|
|
|
ctx = parentCtx.Clone()
|
|
|
|
ctx.SetParent(parentCtx)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2024-06-17 06:54:50 +02:00
|
|
|
|
2024-07-28 18:49:08 +02:00
|
|
|
func (info *funcInfo) checkExistingParam(paramName string) (exists bool) {
|
|
|
|
exists = false
|
|
|
|
for _, spec := range info.formalParams {
|
|
|
|
if spec.Name() == paramName {
|
|
|
|
exists = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) initActualParams(ctx ExprContext, name string, callTerm *term) (params map[string]any, err error) {
|
|
|
|
var varArgs []any
|
|
|
|
var varName string
|
|
|
|
|
|
|
|
namedParamsStarted := false
|
|
|
|
|
|
|
|
actualParams := make(map[string]any, len(info.formalParams))
|
|
|
|
for i, tree := range callTerm.children {
|
|
|
|
var paramValue any
|
|
|
|
paramCtx := ctx.Clone()
|
|
|
|
if paramValue, err = tree.compute(paramCtx); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if paramName, namedParam := getAssignVarName(tree); namedParam {
|
|
|
|
if !info.checkExistingParam(paramName) {
|
|
|
|
err = fmt.Errorf("%s(): unknown param %q", name, paramName)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
actualParams[paramName] = paramValue
|
|
|
|
namedParamsStarted = true
|
|
|
|
} else if !namedParamsStarted {
|
|
|
|
if varArgs != nil {
|
|
|
|
varArgs = append(varArgs, paramValue)
|
|
|
|
} else if i < len(info.formalParams) {
|
|
|
|
spec := info.formalParams[i]
|
|
|
|
if spec.IsRepeat() {
|
|
|
|
varArgs = make([]any, 0, len(callTerm.children)-i)
|
|
|
|
varArgs = append(varArgs, paramValue)
|
|
|
|
varName = spec.Name()
|
|
|
|
} else {
|
|
|
|
actualParams[spec.Name()] = paramValue
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = ErrTooManyParams(name, len(info.formalParams), len(callTerm.children))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = fmt.Errorf("%s(): positional param nr %d not allowed after named params", name, i+1)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
if varArgs != nil {
|
|
|
|
actualParams[varName] = varArgs
|
|
|
|
}
|
|
|
|
params = actualParams
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info *funcInfo) PrepareCall(parentCtx ExprContext, name string, callTerm *term) (ctx ExprContext, actualParams map[string]any, err error) {
|
|
|
|
if actualParams, err = info.initActualParams(parentCtx, name, callTerm); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
passedCount := len(actualParams)
|
2024-06-24 07:20:17 +02:00
|
|
|
if info.MinArgs() > passedCount {
|
|
|
|
err = ErrTooFewParams(name, info.MinArgs(), info.MaxArgs(), passedCount)
|
|
|
|
}
|
|
|
|
|
2024-07-28 18:49:08 +02:00
|
|
|
if passedCount < len(info.formalParams) {
|
|
|
|
for _, p := range info.formalParams {
|
|
|
|
if _, exists := actualParams[p.Name()]; !exists {
|
|
|
|
if !p.IsDefault() {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if p.IsRepeat() {
|
|
|
|
varArgs := make([]any, 1)
|
|
|
|
varArgs[0] = p.DefaultValue()
|
|
|
|
actualParams[p.Name()] = varArgs
|
|
|
|
} else {
|
|
|
|
actualParams[p.Name()] = p.DefaultValue()
|
|
|
|
}
|
|
|
|
}
|
2024-06-17 06:54:50 +02:00
|
|
|
}
|
2024-06-24 07:20:17 +02:00
|
|
|
}
|
2024-07-28 18:49:08 +02:00
|
|
|
|
|
|
|
if err == nil && info.MaxArgs() >= 0 && info.MaxArgs() < len(actualParams) {
|
|
|
|
err = ErrTooManyParams(name, info.MaxArgs(), len(actualParams))
|
2024-06-24 07:20:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
ctx = info.AllocContext(parentCtx)
|
2024-06-17 06:54:50 +02:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-06-24 07:20:17 +02:00
|
|
|
// ----- Call a function ---
|
2024-06-17 06:54:50 +02:00
|
|
|
|
2024-07-28 18:49:08 +02:00
|
|
|
func getAssignVarName(t *term) (name string, ok bool) {
|
|
|
|
if ok = t.symbol() == SymEqual; ok {
|
|
|
|
name = t.children[0].source()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func CallFunction3(parentCtx ExprContext, name string, callTerm *term) (result any, err error) {
|
|
|
|
var actualParams map[string]any
|
|
|
|
if info, exists := GetFuncInfo(parentCtx, name); exists {
|
2024-06-24 07:20:17 +02:00
|
|
|
var ctx ExprContext
|
2024-07-28 18:49:08 +02:00
|
|
|
if ctx, actualParams, err = info.PrepareCall(parentCtx, name, callTerm); err == nil {
|
2024-06-24 07:20:17 +02:00
|
|
|
functor := info.Functor()
|
2024-07-28 18:49:08 +02:00
|
|
|
result, err = functor.InvokeNamed(ctx, name, actualParams)
|
2024-06-24 07:20:17 +02:00
|
|
|
exportObjectsToParent(ctx)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = fmt.Errorf("unknown function %s()", name)
|
2024-06-17 06:54:50 +02:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|