Files
mgmt/lang/funcs/structs/forkv.go
James Shubin 1df28c1d00 lang: ast, funcs: Start plumbing through the textarea
We need to get these everywhere and this is a start.
2025-06-06 03:11:06 -04:00

317 lines
10 KiB
Go

// Mgmt
// Copyright (C) James Shubin and the project contributors
// Written by James Shubin <james@shubin.ca> and the project contributors
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
//
// Additional permission under GNU GPL version 3 section 7
//
// If you modify this program, or any covered work, by linking or combining it
// with embedded mcl code and modules (and that the embedded mcl code and
// modules which link with this program, contain a copy of their source code in
// the authoritative form) containing parts covered by the terms of any other
// license, the licensors of this program grant you additional permission to
// convey the resulting work. Furthermore, the licensors of this program grant
// the original author, James Shubin, additional permission to update this
// additional permission if he deems it necessary to achieve the goals of this
// additional permission.
package structs
import (
"context"
"fmt"
"github.com/purpleidea/mgmt/lang/interfaces"
"github.com/purpleidea/mgmt/lang/types"
"github.com/purpleidea/mgmt/util/errwrap"
)
const (
// ForKVFuncName is the unique name identifier for this function.
ForKVFuncName = "forkv"
// ForKVFuncArgNameMap is the name for the edge which connects the input
// map to CallFunc.
ForKVFuncArgNameMap = "map"
)
// ForKVFunc receives a map from upstream. We iterate over the received map to
// build a subgraph that processes each key and val, and in doing so we get a
// larger function graph. This is rebuilt as necessary if the input map changes.
type ForKVFunc struct {
interfaces.Textarea
KeyType *types.Type
ValType *types.Type
EdgeName string // name of the edge used
SetOnIterBody func(innerTxn interfaces.Txn, ptr types.Value, key, val interfaces.Func) error
ClearIterBody func(length int)
init *interfaces.Init
lastForKVMap types.Value // remember the last value
lastInputMapLength int // remember the last input map length
}
// String returns a simple name for this function. This is needed so this struct
// can satisfy the pgraph.Vertex interface.
func (obj *ForKVFunc) String() string {
return ForKVFuncName
}
// Validate makes sure we've built our struct properly.
func (obj *ForKVFunc) Validate() error {
if obj.KeyType == nil {
return fmt.Errorf("must specify a type")
}
if obj.ValType == nil {
return fmt.Errorf("must specify a type")
}
// TODO: maybe we can remove this if we use this for core functions...
if obj.EdgeName == "" {
return fmt.Errorf("must specify an edge name")
}
return nil
}
// Info returns some static info about itself.
func (obj *ForKVFunc) Info() *interfaces.Info {
var typ *types.Type
if obj.KeyType != nil && obj.ValType != nil { // don't panic if called speculatively
// XXX: Improve function engine so it can return no value?
//typ = types.NewType(fmt.Sprintf("func(%s map{%s: %s})", obj.EdgeName, obj.KeyType, obj.ValType)) // returns nothing
// XXX: Temporary float type to prove we're dropping the output since we don't use it.
typ = types.NewType(fmt.Sprintf("func(%s map{%s: %s}) float", obj.EdgeName, obj.KeyType, obj.ValType))
}
return &interfaces.Info{
Pure: true,
Memo: false, // TODO: ???
Sig: typ,
Err: obj.Validate(),
}
}
// Init runs some startup code for this composite function.
func (obj *ForKVFunc) Init(init *interfaces.Init) error {
obj.init = init
obj.lastForKVMap = nil
obj.lastInputMapLength = -1
return nil
}
// Stream takes an input struct in the format as described in the Func and Graph
// methods of the Expr, and returns the actual expected value as a stream based
// on the changing inputs to that value.
func (obj *ForKVFunc) Stream(ctx context.Context) error {
defer close(obj.init.Output) // the sender closes
// A Func to send input maps to the subgraph. The Txn.Erase() call
// ensures that this Func is not removed when the subgraph is recreated,
// so that the function graph can propagate the last map we received to
// the subgraph.
inputChan := make(chan types.Value)
subgraphInput := &ChannelBasedSourceFunc{
Name: "subgraphInput",
Source: obj,
Chan: inputChan,
Type: obj.mapType(),
}
obj.init.Txn.AddVertex(subgraphInput)
if err := obj.init.Txn.Commit(); err != nil {
return errwrap.Wrapf(err, "commit error in Stream")
}
obj.init.Txn.Erase() // prevent the next Reverse() from removing subgraphInput
defer func() {
close(inputChan)
obj.init.Txn.Reverse()
obj.init.Txn.DeleteVertex(subgraphInput)
obj.init.Txn.Commit()
}()
for {
select {
case input, ok := <-obj.init.Input:
if !ok {
obj.init.Input = nil // block looping back here
//canReceiveMoreMapValues = false
// We don't ever shutdown here, since even if we
// don't get more maps, that last map value is
// still propagating inside of the subgraph and
// so we don't want to shutdown since that would
// reverse the txn which we only do at the very
// end on graph shutdown.
continue
}
forKVMap, exists := input.Struct()[obj.EdgeName]
if !exists {
return fmt.Errorf("programming error, can't find edge")
}
// It's important to have this compare step to avoid
// redundant graph replacements which slow things down,
// but also cause the engine to lock, which can preempt
// the process scheduler, which can cause duplicate or
// unnecessary re-sending of values here, which causes
// the whole process to repeat ad-nauseum.
n := len(forKVMap.Map())
// If the keys are the same, that's enough! We don't
// need to rebuild the graph unless any of the keys
// change, since those are our unique identifiers into
// the whole loop. As a result, we don't compare between
// the entire two map, since while we could rebuild the
// graph on any change, it's easier to leave it as is
// and simply push new values down the already built
// graph if any value changes.
if obj.lastInputMapLength != n || obj.cmpMapKeys(forKVMap) != nil {
// TODO: Technically we only need to save keys!
obj.lastForKVMap = forKVMap
obj.lastInputMapLength = n
// replaceSubGraph uses the above two values
if err := obj.replaceSubGraph(subgraphInput); err != nil {
return errwrap.Wrapf(err, "could not replace subgraph")
}
}
// send the new input map to the subgraph
select {
case inputChan <- forKVMap:
case <-ctx.Done():
return nil
}
case <-ctx.Done():
return nil
}
select {
case obj.init.Output <- &types.FloatValue{
V: 42.0, // XXX: temporary
}:
case <-ctx.Done():
return nil
}
}
}
func (obj *ForKVFunc) replaceSubGraph(subgraphInput interfaces.Func) error {
// delete the old subgraph
if err := obj.init.Txn.Reverse(); err != nil {
return errwrap.Wrapf(err, "could not Reverse")
}
obj.ClearIterBody(obj.lastInputMapLength) // XXX: pass in size?
forKVMap := obj.lastForKVMap.Map()
// XXX: Should we loop in a deterministic order?
// XXX: Should our type support the new iterator pattern?
for k := range forKVMap {
ptr := k
argNameKey := "forkvInputMapKey"
argNameVal := "forkvInputMapVal"
// the key
inputElemFuncKey := SimpleFnToDirectFunc(
fmt.Sprintf("forkvInputElemKey[%v]", ptr),
&types.FuncValue{
V: func(_ context.Context, args []types.Value) (types.Value, error) {
if len(args) != 1 {
return nil, fmt.Errorf("inputElemFuncKey: expected a single argument")
}
//arg := args[0]
//m, ok := arg.(*types.MapValue)
//if !ok {
// return nil, fmt.Errorf("inputElemFuncKey: expected a MapValue argument")
//}
// XXX: If we had some sort of index fn?
//return m.Map().Index(?), nil
return k, nil
},
T: types.NewType(fmt.Sprintf("func(%s %s) %s", argNameKey, obj.mapType(), obj.KeyType)),
},
)
obj.init.Txn.AddVertex(inputElemFuncKey)
obj.init.Txn.AddEdge(subgraphInput, inputElemFuncKey, &interfaces.FuncEdge{
Args: []string{argNameKey},
})
// the val
inputElemFuncVal := SimpleFnToDirectFunc(
fmt.Sprintf("forkvInputElemVal[%v]", ptr),
&types.FuncValue{
V: func(_ context.Context, args []types.Value) (types.Value, error) {
if len(args) != 1 {
return nil, fmt.Errorf("inputElemFuncVal: expected a single argument")
}
//return v, nil // If we always rebuild the map.
arg := args[0]
m, ok := arg.(*types.MapValue)
if !ok {
return nil, fmt.Errorf("inputElemFuncVal: expected a MapValue argument")
}
return m.Map()[ptr], nil
},
T: types.NewType(fmt.Sprintf("func(%s %s) %s", argNameVal, obj.mapType(), obj.ValType)),
},
)
obj.init.Txn.AddVertex(inputElemFuncVal)
obj.init.Txn.AddEdge(subgraphInput, inputElemFuncVal, &interfaces.FuncEdge{
Args: []string{argNameVal},
})
if err := obj.SetOnIterBody(obj.init.Txn, ptr, inputElemFuncKey, inputElemFuncVal); err != nil {
return errwrap.Wrapf(err, "could not call SetOnIterBody()")
}
}
return obj.init.Txn.Commit()
}
func (obj *ForKVFunc) mapType() *types.Type {
return types.NewType(fmt.Sprintf("map{%s: %s}", obj.KeyType, obj.ValType))
}
// cmpMapKeys compares the input map with the cached private lastForKVMap field.
// If either are nil, or if the keys of the maps are not identical, then this
// errors.
func (obj *ForKVFunc) cmpMapKeys(m types.Value) error {
if obj.lastForKVMap == nil || m == nil {
return fmt.Errorf("got a nil map")
}
m1 := obj.lastForKVMap.Map()
m2 := m.(*types.MapValue) // must not panic!
if len(m1) != len(m.Map()) {
return fmt.Errorf("lengths differ")
}
for k := range m1 {
if _, exists := m2.Lookup(k); !exists {
return fmt.Errorf("key not found")
}
}
return nil
}