2020-11-03 23:48:43 +00:00
|
|
|
package yqlib
|
2020-10-10 04:00:39 +00:00
|
|
|
|
2020-10-11 00:45:20 +00:00
|
|
|
import (
|
2020-10-21 01:54:58 +00:00
|
|
|
"container/list"
|
2020-12-22 00:45:51 +00:00
|
|
|
"fmt"
|
2020-10-12 01:24:59 +00:00
|
|
|
|
2021-02-08 02:58:46 +00:00
|
|
|
"github.com/jinzhu/copier"
|
2020-10-11 00:45:20 +00:00
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
)
|
2020-10-10 04:00:39 +00:00
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
type operatorHandler func(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error)
|
2020-10-10 04:00:39 +00:00
|
|
|
|
2021-07-07 10:00:46 +00:00
|
|
|
type compoundCalculation func(lhs *ExpressionNode, rhs *ExpressionNode) *ExpressionNode
|
|
|
|
|
|
|
|
func compoundAssignFunction(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode, calculation compoundCalculation) (Context, error) {
|
|
|
|
lhs, err := d.GetMatchingNodes(context, expressionNode.Lhs)
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
|
2022-01-15 04:48:34 +00:00
|
|
|
assignmentOp := &Operation{OperationType: assignOpType, Preferences: expressionNode.Operation.Preferences}
|
2021-07-07 10:00:46 +00:00
|
|
|
valueOp := &Operation{OperationType: valueOpType}
|
|
|
|
|
|
|
|
for el := lhs.MatchingNodes.Front(); el != nil; el = el.Next() {
|
|
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
valueOp.CandidateNode = candidate
|
|
|
|
valueExpression := &ExpressionNode{Operation: valueOp}
|
|
|
|
|
|
|
|
assignmentOpNode := &ExpressionNode{Operation: assignmentOp, Lhs: valueExpression, Rhs: calculation(valueExpression, expressionNode.Rhs)}
|
|
|
|
|
|
|
|
_, err = d.GetMatchingNodes(context, assignmentOpNode)
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return context, nil
|
|
|
|
}
|
|
|
|
|
2021-01-12 23:00:51 +00:00
|
|
|
func unwrapDoc(node *yaml.Node) *yaml.Node {
|
2020-10-27 05:45:16 +00:00
|
|
|
if node.Kind == yaml.DocumentNode {
|
|
|
|
return node.Content[0]
|
|
|
|
}
|
|
|
|
return node
|
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func emptyOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
|
|
context.MatchingNodes = list.New()
|
|
|
|
return context, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type crossFunctionCalculation func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error)
|
|
|
|
|
2021-05-16 02:16:10 +00:00
|
|
|
func resultsForRhs(d *dataTreeNavigator, context Context, lhsCandidate *CandidateNode, rhs Context, calculation crossFunctionCalculation, results *list.List, calcWhenEmpty bool) error {
|
|
|
|
|
|
|
|
if calcWhenEmpty && rhs.MatchingNodes.Len() == 0 {
|
|
|
|
resultCandidate, err := calculation(d, context, lhsCandidate, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-28 06:59:02 +00:00
|
|
|
if resultCandidate != nil {
|
|
|
|
results.PushBack(resultCandidate)
|
|
|
|
}
|
2021-05-16 02:16:10 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-04-13 00:42:20 +00:00
|
|
|
for rightEl := rhs.MatchingNodes.Front(); rightEl != nil; rightEl = rightEl.Next() {
|
|
|
|
log.Debugf("Applying calc")
|
|
|
|
rhsCandidate := rightEl.Value.(*CandidateNode)
|
|
|
|
resultCandidate, err := calculation(d, context, lhsCandidate, rhsCandidate)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-28 06:59:02 +00:00
|
|
|
if resultCandidate != nil {
|
|
|
|
results.PushBack(resultCandidate)
|
|
|
|
}
|
2021-04-13 00:42:20 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func doCrossFunc(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode, calculation crossFunctionCalculation, calcWhenEmpty bool) (Context, error) {
|
2021-02-02 07:17:59 +00:00
|
|
|
var results = list.New()
|
|
|
|
lhs, err := d.GetMatchingNodes(context, expressionNode.Lhs)
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
log.Debugf("crossFunction LHS len: %v", lhs.MatchingNodes.Len())
|
|
|
|
|
|
|
|
rhs, err := d.GetMatchingNodes(context, expressionNode.Rhs)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
|
2021-04-13 00:42:20 +00:00
|
|
|
if calcWhenEmpty && lhs.MatchingNodes.Len() == 0 {
|
2021-05-16 02:16:10 +00:00
|
|
|
err := resultsForRhs(d, context, nil, rhs, calculation, results, calcWhenEmpty)
|
2021-04-13 00:42:20 +00:00
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for el := lhs.MatchingNodes.Front(); el != nil; el = el.Next() {
|
|
|
|
lhsCandidate := el.Value.(*CandidateNode)
|
|
|
|
|
2021-05-16 02:16:10 +00:00
|
|
|
err := resultsForRhs(d, context, lhsCandidate, rhs, calculation, results, calcWhenEmpty)
|
2021-04-13 00:42:20 +00:00
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
2021-02-02 07:17:59 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
return context.ChildContext(results), nil
|
|
|
|
}
|
|
|
|
|
2021-04-13 00:42:20 +00:00
|
|
|
func crossFunction(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode, calculation crossFunctionCalculation, calcWhenEmpty bool) (Context, error) {
|
2021-02-02 07:17:59 +00:00
|
|
|
var results = list.New()
|
|
|
|
|
|
|
|
var evaluateAllTogether = true
|
|
|
|
for matchEl := context.MatchingNodes.Front(); matchEl != nil; matchEl = matchEl.Next() {
|
|
|
|
evaluateAllTogether = evaluateAllTogether && matchEl.Value.(*CandidateNode).EvaluateTogether
|
|
|
|
if !evaluateAllTogether {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if evaluateAllTogether {
|
2021-07-07 09:22:51 +00:00
|
|
|
log.Debug("crossFunction evaluateAllTogether!")
|
2021-04-13 00:42:20 +00:00
|
|
|
return doCrossFunc(d, context, expressionNode, calculation, calcWhenEmpty)
|
2021-02-02 07:17:59 +00:00
|
|
|
}
|
|
|
|
|
2021-07-07 09:22:51 +00:00
|
|
|
log.Debug("crossFunction evaluate apart!")
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
for matchEl := context.MatchingNodes.Front(); matchEl != nil; matchEl = matchEl.Next() {
|
2021-04-13 00:42:20 +00:00
|
|
|
innerResults, err := doCrossFunc(d, context.SingleChildContext(matchEl.Value.(*CandidateNode)), expressionNode, calculation, calcWhenEmpty)
|
2021-02-02 07:17:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
results.PushBackList(innerResults.MatchingNodes)
|
|
|
|
}
|
|
|
|
|
|
|
|
return context.ChildContext(results), nil
|
2020-11-13 02:19:54 +00:00
|
|
|
}
|
|
|
|
|
2020-10-17 11:10:47 +00:00
|
|
|
func createBooleanCandidate(owner *CandidateNode, value bool) *CandidateNode {
|
|
|
|
valString := "true"
|
|
|
|
if !value {
|
|
|
|
valString = "false"
|
|
|
|
}
|
|
|
|
node := &yaml.Node{Kind: yaml.ScalarNode, Value: valString, Tag: "!!bool"}
|
2021-11-23 22:57:35 +00:00
|
|
|
return owner.CreateReplacement(node)
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
|
|
|
|
2021-02-08 02:58:46 +00:00
|
|
|
func createTraversalTree(path []interface{}, traversePrefs traversePreferences, targetKey bool) *ExpressionNode {
|
2020-12-22 00:45:51 +00:00
|
|
|
if len(path) == 0 {
|
2021-01-12 23:18:53 +00:00
|
|
|
return &ExpressionNode{Operation: &Operation{OperationType: selfReferenceOpType}}
|
2020-12-22 00:45:51 +00:00
|
|
|
} else if len(path) == 1 {
|
2021-02-08 02:58:46 +00:00
|
|
|
lastPrefs := traversePrefs
|
|
|
|
if targetKey {
|
|
|
|
err := copier.Copy(&lastPrefs, traversePrefs)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
lastPrefs.IncludeMapKeys = true
|
|
|
|
lastPrefs.DontIncludeMapValues = true
|
|
|
|
}
|
|
|
|
return &ExpressionNode{Operation: &Operation{OperationType: traversePathOpType, Preferences: lastPrefs, Value: path[0], StringValue: fmt.Sprintf("%v", path[0])}}
|
2020-12-22 00:45:51 +00:00
|
|
|
}
|
2021-01-13 05:54:28 +00:00
|
|
|
|
2021-01-12 23:18:53 +00:00
|
|
|
return &ExpressionNode{
|
2021-01-11 06:13:48 +00:00
|
|
|
Operation: &Operation{OperationType: shortPipeOpType},
|
2021-02-08 02:58:46 +00:00
|
|
|
Lhs: createTraversalTree(path[0:1], traversePrefs, false),
|
|
|
|
Rhs: createTraversalTree(path[1:], traversePrefs, targetKey),
|
2021-01-13 05:54:28 +00:00
|
|
|
}
|
2020-12-22 00:45:51 +00:00
|
|
|
}
|