2024-04-26 04:45:42 +02:00
|
|
|
// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com).
|
|
|
|
// All rights reserved.
|
|
|
|
|
|
|
|
// operand-iterator.go
|
|
|
|
package expr
|
|
|
|
|
|
|
|
import (
|
2024-04-28 06:45:20 +02:00
|
|
|
"slices"
|
|
|
|
"strings"
|
2024-04-26 04:45:42 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// -------- iterator term
|
|
|
|
|
2024-07-07 16:10:43 +02:00
|
|
|
// func newDsIteratorTerm(tk *Token, dsTerm *term, args []*term) *term {
|
|
|
|
// tk.Sym = SymIterator
|
|
|
|
|
|
|
|
// children := make([]*term, 0, 1+len(args))
|
|
|
|
// children = append(children, dsTerm)
|
|
|
|
// children = append(children, args...)
|
|
|
|
// return &term{
|
|
|
|
// tk: *tk,
|
|
|
|
// parent: nil,
|
|
|
|
// children: children,
|
|
|
|
// position: posLeaf,
|
|
|
|
// priority: priValue,
|
|
|
|
// evalFunc: evalIterator,
|
|
|
|
// }
|
|
|
|
// }
|
2024-04-26 04:45:42 +02:00
|
|
|
|
2024-05-04 00:51:15 +02:00
|
|
|
func newIteratorTerm(tk *Token, args []*term) *term {
|
|
|
|
tk.Sym = SymIterator
|
|
|
|
return &term{
|
|
|
|
tk: *tk,
|
|
|
|
parent: nil,
|
|
|
|
children: args,
|
|
|
|
position: posLeaf,
|
|
|
|
priority: priValue,
|
|
|
|
evalFunc: evalIterator,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-26 04:45:42 +02:00
|
|
|
// -------- eval iterator
|
|
|
|
|
2024-07-07 16:10:43 +02:00
|
|
|
func evalTermArray(ctx ExprContext, terms []*term) (values []any, err error) {
|
|
|
|
values = make([]any, len(terms))
|
|
|
|
for i, t := range terms {
|
2024-04-26 04:45:42 +02:00
|
|
|
var value any
|
|
|
|
if value, err = t.compute(ctx); err == nil {
|
|
|
|
values[i] = value
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-07 16:10:43 +02:00
|
|
|
func evalFirstChild(ctx ExprContext, iteratorTerm *term) (value any, err error) {
|
|
|
|
if len(iteratorTerm.children) < 1 || iteratorTerm.children[0] == nil {
|
|
|
|
err = iteratorTerm.Errorf("missing the data-source parameter")
|
2024-04-26 04:45:42 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-07 16:10:43 +02:00
|
|
|
value, err = iteratorTerm.children[0].compute(ctx)
|
2024-05-04 00:51:15 +02:00
|
|
|
return
|
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
|
2024-07-07 16:10:43 +02:00
|
|
|
func getDataSourceDict(iteratorTerm *term, firstChildValue any) (ds map[string]Functor, err error) {
|
2024-05-19 01:47:06 +02:00
|
|
|
if dictAny, ok := firstChildValue.(*DictType); ok {
|
2024-07-18 07:27:02 +02:00
|
|
|
requiredFields := []string{CurrentName, NextName}
|
2024-05-04 00:51:15 +02:00
|
|
|
fieldsMask := 0b11
|
|
|
|
foundFields := 0
|
2024-04-26 04:45:42 +02:00
|
|
|
ds = make(map[string]Functor)
|
2024-05-19 01:47:06 +02:00
|
|
|
for keyAny, item := range *dictAny {
|
2024-04-28 06:45:20 +02:00
|
|
|
if key, ok := keyAny.(string); ok {
|
2024-05-24 06:28:48 +02:00
|
|
|
if functor, ok := item.(Functor); ok {
|
2024-04-28 06:45:20 +02:00
|
|
|
ds[key] = functor
|
|
|
|
if index := slices.Index(requiredFields, key); index >= 0 {
|
|
|
|
foundFields |= 1 << index
|
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
}
|
2024-04-27 22:32:57 +02:00
|
|
|
}
|
|
|
|
}
|
2024-04-28 06:45:20 +02:00
|
|
|
// check required functions
|
|
|
|
if foundFields != fieldsMask {
|
|
|
|
missingFields := make([]string, 0, len(requiredFields))
|
|
|
|
for index, field := range requiredFields {
|
|
|
|
if (foundFields & (1 << index)) == 0 {
|
|
|
|
missingFields = append(missingFields, field)
|
2024-04-27 22:32:57 +02:00
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
}
|
2024-07-07 16:10:43 +02:00
|
|
|
// err = fmt.Errorf("the data-source must provide a non-nil %q operator(s)", strings.Join(missingFields, ", "))
|
|
|
|
err = iteratorTerm.children[0].Errorf("the data-source must provide a non-nil %q operator(s)", strings.Join(missingFields, ", "))
|
2024-04-26 04:45:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-09 07:50:06 +02:00
|
|
|
func evalIterator(ctx ExprContext, opTerm *term) (v any, err error) {
|
2024-05-04 00:51:15 +02:00
|
|
|
var firstChildValue any
|
2024-04-26 04:45:42 +02:00
|
|
|
var ds map[string]Functor
|
|
|
|
|
2024-07-09 07:50:06 +02:00
|
|
|
if firstChildValue, err = evalFirstChild(ctx, opTerm); err != nil {
|
2024-05-04 00:51:15 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-09 07:50:06 +02:00
|
|
|
if ds, err = getDataSourceDict(opTerm, firstChildValue); err != nil {
|
2024-04-26 04:45:42 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-05-04 00:51:15 +02:00
|
|
|
if ds != nil {
|
2024-07-18 07:27:02 +02:00
|
|
|
dc := NewDataCursor(ctx, ds)
|
|
|
|
if initFunc, exists := ds[InitName]; exists && initFunc != nil {
|
2024-05-04 00:51:15 +02:00
|
|
|
var args []any
|
2024-07-09 07:50:06 +02:00
|
|
|
if len(opTerm.children) > 1 {
|
|
|
|
if args, err = evalTermArray(ctx, opTerm.children[1:]); err != nil {
|
2024-05-04 00:51:15 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
args = []any{}
|
|
|
|
}
|
|
|
|
|
|
|
|
initCtx := dc.ctx.Clone()
|
2024-07-18 07:27:02 +02:00
|
|
|
if dc.resource, err = initFunc.Invoke(initCtx, InitName, args); err != nil {
|
2024-04-26 04:45:42 +02:00
|
|
|
return
|
|
|
|
}
|
2024-05-04 00:51:15 +02:00
|
|
|
exportObjects(dc.ctx, initCtx)
|
2024-04-26 04:45:42 +02:00
|
|
|
}
|
2024-04-26 21:03:22 +02:00
|
|
|
|
2024-07-18 07:27:02 +02:00
|
|
|
dc.nextFunc = ds[NextName]
|
|
|
|
dc.currentFunc = ds[CurrentName]
|
|
|
|
dc.cleanFunc = ds[CleanName]
|
|
|
|
dc.resetFunc = ds[ResetName]
|
2024-05-04 00:51:15 +02:00
|
|
|
|
|
|
|
v = dc
|
|
|
|
} else if list, ok := firstChildValue.(*ListType); ok {
|
2024-05-04 08:07:49 +02:00
|
|
|
var args []any
|
2024-07-09 07:50:06 +02:00
|
|
|
if args, err = evalSibling(ctx, opTerm.children, nil); err == nil {
|
2024-05-04 08:07:49 +02:00
|
|
|
v = NewListIterator(list, args)
|
|
|
|
}
|
2024-05-04 00:51:15 +02:00
|
|
|
} else {
|
2024-05-04 08:07:49 +02:00
|
|
|
var list []any
|
2024-07-09 07:50:06 +02:00
|
|
|
if list, err = evalSibling(ctx, opTerm.children, firstChildValue); err == nil {
|
2024-05-04 08:07:49 +02:00
|
|
|
v = NewArrayIterator(list)
|
2024-04-26 04:45:42 +02:00
|
|
|
}
|
|
|
|
}
|
2024-05-04 00:51:15 +02:00
|
|
|
return
|
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
|
2024-05-04 08:07:49 +02:00
|
|
|
func evalSibling(ctx ExprContext, terms []*term, firstChildValue any) (list []any, err error) {
|
|
|
|
items := make([]any, 0, len(terms))
|
2024-05-04 00:51:15 +02:00
|
|
|
for i, tree := range terms {
|
|
|
|
var param any
|
2024-05-04 08:07:49 +02:00
|
|
|
if i == 0 {
|
|
|
|
if firstChildValue == nil {
|
|
|
|
continue
|
|
|
|
}
|
2024-05-04 00:51:15 +02:00
|
|
|
param = firstChildValue
|
|
|
|
} else if param, err = tree.compute(ctx); err != nil {
|
|
|
|
break
|
|
|
|
}
|
2024-05-04 08:07:49 +02:00
|
|
|
items = append(items, param)
|
2024-05-04 00:51:15 +02:00
|
|
|
}
|
|
|
|
if err == nil {
|
2024-05-04 08:07:49 +02:00
|
|
|
list = items
|
2024-05-04 00:51:15 +02:00
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
return
|
|
|
|
}
|