2020-11-03 23:48:43 +00:00
|
|
|
package yqlib
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-01-18 02:28:40 +00:00
|
|
|
"strconv"
|
2020-10-19 05:14:29 +00:00
|
|
|
|
2020-10-21 01:54:58 +00:00
|
|
|
"container/list"
|
|
|
|
|
2020-11-20 11:57:32 +00:00
|
|
|
yaml "gopkg.in/yaml.v3"
|
2020-10-19 05:14:29 +00:00
|
|
|
)
|
|
|
|
|
2021-01-11 06:13:48 +00:00
|
|
|
type multiplyPreferences struct {
|
2021-02-18 00:16:54 +00:00
|
|
|
AppendArrays bool
|
|
|
|
DeepMergeArrays bool
|
|
|
|
TraversePrefs traversePreferences
|
2020-11-27 23:41:09 +00:00
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func multiplyOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
2020-10-20 05:27:30 +00:00
|
|
|
log.Debugf("-- MultiplyOperator")
|
2021-02-02 07:17:59 +00:00
|
|
|
return crossFunction(d, context, expressionNode, multiply(expressionNode.Operation.Preferences.(multiplyPreferences)))
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func multiply(preferences multiplyPreferences) func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
return func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
2021-01-12 23:00:51 +00:00
|
|
|
lhs.Node = unwrapDoc(lhs.Node)
|
|
|
|
rhs.Node = unwrapDoc(rhs.Node)
|
2020-11-27 23:41:09 +00:00
|
|
|
log.Debugf("Multipling LHS: %v", lhs.Node.Tag)
|
|
|
|
log.Debugf("- RHS: %v", rhs.Node.Tag)
|
2020-10-27 05:45:16 +00:00
|
|
|
|
2020-11-27 23:41:09 +00:00
|
|
|
if lhs.Node.Kind == yaml.MappingNode && rhs.Node.Kind == yaml.MappingNode ||
|
|
|
|
(lhs.Node.Kind == yaml.SequenceNode && rhs.Node.Kind == yaml.SequenceNode) {
|
|
|
|
|
2021-01-12 08:36:28 +00:00
|
|
|
var newBlank = lhs.CreateChild(nil, &yaml.Node{})
|
2021-03-19 01:54:03 +00:00
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
var newThing, err = mergeObjects(d, context, newBlank, lhs, multiplyPreferences{})
|
2020-11-27 23:41:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-02 07:17:59 +00:00
|
|
|
return mergeObjects(d, context, newThing, rhs, preferences)
|
2021-01-18 02:28:40 +00:00
|
|
|
} else if lhs.Node.Tag == "!!int" && rhs.Node.Tag == "!!int" {
|
|
|
|
return multiplyIntegers(lhs, rhs)
|
2021-02-18 00:16:54 +00:00
|
|
|
} else if (lhs.Node.Tag == "!!int" || lhs.Node.Tag == "!!float") && (rhs.Node.Tag == "!!int" || rhs.Node.Tag == "!!float") {
|
|
|
|
return multiplyFloats(lhs, rhs)
|
2020-11-27 23:41:09 +00:00
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("Cannot multiply %v with %v", lhs.Node.Tag, rhs.Node.Tag)
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-18 00:16:54 +00:00
|
|
|
func multiplyFloats(lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
target := lhs.CreateChild(nil, &yaml.Node{})
|
|
|
|
target.Node.Kind = yaml.ScalarNode
|
|
|
|
target.Node.Style = lhs.Node.Style
|
|
|
|
target.Node.Tag = "!!float"
|
|
|
|
|
|
|
|
lhsNum, err := strconv.ParseFloat(lhs.Node.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
rhsNum, err := strconv.ParseFloat(rhs.Node.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
target.Node.Value = fmt.Sprintf("%v", lhsNum*rhsNum)
|
|
|
|
return target, nil
|
|
|
|
}
|
|
|
|
|
2021-01-18 02:28:40 +00:00
|
|
|
func multiplyIntegers(lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
target := lhs.CreateChild(nil, &yaml.Node{})
|
|
|
|
target.Node.Kind = yaml.ScalarNode
|
|
|
|
target.Node.Style = lhs.Node.Style
|
|
|
|
target.Node.Tag = "!!int"
|
|
|
|
|
|
|
|
lhsNum, err := strconv.Atoi(lhs.Node.Value)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
rhsNum, err := strconv.Atoi(rhs.Node.Value)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
target.Node.Value = fmt.Sprintf("%v", lhsNum*rhsNum)
|
|
|
|
return target, nil
|
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func mergeObjects(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode, preferences multiplyPreferences) (*CandidateNode, error) {
|
2021-01-13 05:54:28 +00:00
|
|
|
shouldAppendArrays := preferences.AppendArrays
|
2020-10-28 02:00:26 +00:00
|
|
|
var results = list.New()
|
2020-11-27 23:41:09 +00:00
|
|
|
|
|
|
|
// shouldn't recurse arrays if appending
|
2021-01-13 06:00:03 +00:00
|
|
|
prefs := recursiveDescentPreferences{RecurseArray: !shouldAppendArrays,
|
2021-02-08 02:58:46 +00:00
|
|
|
TraversePreferences: traversePreferences{DontFollowAlias: true, IncludeMapKeys: true}}
|
2021-02-02 07:17:59 +00:00
|
|
|
err := recursiveDecent(d, results, context.SingleChildContext(rhs), prefs)
|
2020-11-13 03:07:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-10-28 02:00:26 +00:00
|
|
|
|
|
|
|
var pathIndexToStartFrom int = 0
|
|
|
|
if results.Front() != nil {
|
|
|
|
pathIndexToStartFrom = len(results.Front().Value.(*CandidateNode).Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
for el := results.Front(); el != nil; el = el.Next() {
|
2021-02-08 02:58:46 +00:00
|
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
if candidate.Node.Tag == "!!merge" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
err := applyAssignment(d, context, pathIndexToStartFrom, lhs, candidate, preferences)
|
2020-10-28 02:00:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return lhs, nil
|
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func applyAssignment(d *dataTreeNavigator, context Context, pathIndexToStartFrom int, lhs *CandidateNode, rhs *CandidateNode, preferences multiplyPreferences) error {
|
2021-01-13 05:54:28 +00:00
|
|
|
shouldAppendArrays := preferences.AppendArrays
|
2021-02-08 02:58:46 +00:00
|
|
|
log.Debugf("merge - applyAssignment lhs %v, rhs: %v", lhs.GetKey(), rhs.GetKey())
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
lhsPath := rhs.Path[pathIndexToStartFrom:]
|
|
|
|
|
2021-01-11 06:13:48 +00:00
|
|
|
assignmentOp := &Operation{OperationType: assignAttributesOpType}
|
2021-02-18 00:16:54 +00:00
|
|
|
if shouldAppendArrays && rhs.Node.Kind == yaml.SequenceNode {
|
|
|
|
assignmentOp.OperationType = addAssignOpType
|
|
|
|
} else if !preferences.DeepMergeArrays && rhs.Node.Kind == yaml.SequenceNode ||
|
|
|
|
(rhs.Node.Kind == yaml.ScalarNode || rhs.Node.Kind == yaml.AliasNode) {
|
2021-01-11 06:13:48 +00:00
|
|
|
assignmentOp.OperationType = assignOpType
|
2021-01-06 09:22:50 +00:00
|
|
|
assignmentOp.UpdateAssign = false
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
2021-01-11 06:13:48 +00:00
|
|
|
rhsOp := &Operation{OperationType: valueOpType, CandidateNode: rhs}
|
2020-10-19 05:14:29 +00:00
|
|
|
|
2021-02-08 02:58:46 +00:00
|
|
|
assignmentOpNode := &ExpressionNode{Operation: assignmentOp, Lhs: createTraversalTree(lhsPath, preferences.TraversePrefs, rhs.IsMapKey), Rhs: &ExpressionNode{Operation: rhsOp}}
|
2020-10-19 05:14:29 +00:00
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
_, err := d.GetMatchingNodes(context.SingleChildContext(lhs), assignmentOpNode)
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|