mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-03 22:25:35 +00:00
60 lines
1.7 KiB
Go
60 lines
1.7 KiB
Go
|
package yqlib
|
||
|
|
||
|
import (
|
||
|
"container/list"
|
||
|
"fmt"
|
||
|
)
|
||
|
|
||
|
func reduceOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
||
|
log.Debugf("-- reduceOp")
|
||
|
//.a as $var reduce (0; . + $var)
|
||
|
//lhs is the assignment operator
|
||
|
//rhs is the reduce block
|
||
|
// '.' refers to the current accumulator, initialised to 0
|
||
|
// $var references a single element from the .a
|
||
|
|
||
|
//ensure lhs is actually an assignment
|
||
|
//and rhs is a block (empty)
|
||
|
if expressionNode.Lhs.Operation.OperationType != assignVariableOpType {
|
||
|
return Context{}, fmt.Errorf("reduce must be given a variables assignment, got %v instead", expressionNode.Lhs.Operation.OperationType.Type)
|
||
|
} else if expressionNode.Rhs.Operation.OperationType != blockOpType {
|
||
|
return Context{}, fmt.Errorf("reduce must be given a block, got %v instead", expressionNode.Rhs.Operation.OperationType.Type)
|
||
|
}
|
||
|
|
||
|
arrayExpNode := expressionNode.Lhs.Lhs
|
||
|
array, err := d.GetMatchingNodes(context, arrayExpNode)
|
||
|
|
||
|
log.Debugf("array of %v things", array.MatchingNodes.Len())
|
||
|
|
||
|
if err != nil {
|
||
|
return Context{}, err
|
||
|
}
|
||
|
|
||
|
variableName := expressionNode.Lhs.Rhs.Operation.StringValue
|
||
|
|
||
|
initExp := expressionNode.Rhs.Lhs
|
||
|
|
||
|
accum, err := d.GetMatchingNodes(context, initExp)
|
||
|
if err != nil {
|
||
|
return Context{}, err
|
||
|
}
|
||
|
|
||
|
log.Debugf("with variable %v", variableName)
|
||
|
|
||
|
blockExp := expressionNode.Rhs.Rhs
|
||
|
for el := array.MatchingNodes.Front(); el != nil; el = el.Next() {
|
||
|
candidate := el.Value.(*CandidateNode)
|
||
|
log.Debugf("REDUCING WITH %v", NodeToString(candidate))
|
||
|
l := list.New()
|
||
|
l.PushBack(candidate)
|
||
|
accum.SetVariable(variableName, l)
|
||
|
|
||
|
accum, err = d.GetMatchingNodes(accum, blockExp)
|
||
|
if err != nil {
|
||
|
return Context{}, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return accum, nil
|
||
|
}
|