mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-12 19:25:37 +00:00
180 lines
4.7 KiB
Go
180 lines
4.7 KiB
Go
package yqlib
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
)
|
|
|
|
func createAddOp(lhs *ExpressionNode, rhs *ExpressionNode) *ExpressionNode {
|
|
return &ExpressionNode{Operation: &Operation{OperationType: addOpType},
|
|
Lhs: lhs,
|
|
Rhs: rhs}
|
|
}
|
|
|
|
func addAssignOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
return compoundAssignFunction(d, context, expressionNode, createAddOp)
|
|
}
|
|
|
|
func toNodes(candidate *CandidateNode) []*yaml.Node {
|
|
if candidate.Node.Tag == "!!null" {
|
|
return []*yaml.Node{}
|
|
}
|
|
|
|
switch candidate.Node.Kind {
|
|
case yaml.SequenceNode:
|
|
return candidate.Node.Content
|
|
default:
|
|
return []*yaml.Node{candidate.Node}
|
|
}
|
|
|
|
}
|
|
|
|
func addOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
log.Debugf("Add operator")
|
|
|
|
return crossFunction(d, context.ReadOnlyClone(), expressionNode, add, false)
|
|
}
|
|
|
|
func add(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
lhs.Node = unwrapDoc(lhs.Node)
|
|
rhs.Node = unwrapDoc(rhs.Node)
|
|
|
|
lhsNode := lhs.Node
|
|
|
|
if lhsNode.Tag == "!!null" {
|
|
return lhs.CreateReplacement(rhs.Node), nil
|
|
}
|
|
|
|
target := lhs.CreateReplacement(&yaml.Node{})
|
|
|
|
switch lhsNode.Kind {
|
|
case yaml.MappingNode:
|
|
addMaps(target, lhs, rhs)
|
|
case yaml.SequenceNode:
|
|
addSequences(target, lhs, rhs)
|
|
case yaml.ScalarNode:
|
|
if rhs.Node.Kind != yaml.ScalarNode {
|
|
return nil, fmt.Errorf("%v (%v) cannot be added to a %v", rhs.Node.Tag, rhs.Path, lhsNode.Tag)
|
|
}
|
|
target.Node.Kind = yaml.ScalarNode
|
|
target.Node.Style = lhsNode.Style
|
|
if err := addScalars(target, lhsNode, rhs.Node); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return target, nil
|
|
}
|
|
|
|
func guessTagFromCustomType(node *yaml.Node) string {
|
|
if node.Value == "" {
|
|
log.Warning("node has no value to guess the type with")
|
|
return node.Tag
|
|
}
|
|
|
|
decoder := NewYamlDecoder()
|
|
decoder.Init(strings.NewReader(node.Value))
|
|
var dataBucket yaml.Node
|
|
errorReading := decoder.Decode(&dataBucket)
|
|
if errorReading != nil {
|
|
log.Warning("could not guess underlying tag type %v", errorReading)
|
|
return node.Tag
|
|
}
|
|
guessedTag := unwrapDoc(&dataBucket).Tag
|
|
log.Info("im guessing the tag %v is a %v", node.Tag, guessedTag)
|
|
return guessedTag
|
|
}
|
|
|
|
func addScalars(target *CandidateNode, lhs *yaml.Node, rhs *yaml.Node) error {
|
|
lhsTag := lhs.Tag
|
|
rhsTag := rhs.Tag
|
|
lhsIsCustom := false
|
|
if !strings.HasPrefix(lhsTag, "!!") {
|
|
// custom tag - we have to have a guess
|
|
lhsTag = guessTagFromCustomType(lhs)
|
|
lhsIsCustom = true
|
|
}
|
|
|
|
if !strings.HasPrefix(rhsTag, "!!") {
|
|
// custom tag - we have to have a guess
|
|
rhsTag = guessTagFromCustomType(rhs)
|
|
}
|
|
|
|
if lhsTag == "!!str" {
|
|
target.Node.Tag = lhs.Tag
|
|
target.Node.Value = lhs.Value + rhs.Value
|
|
} else if lhsTag == "!!int" && rhsTag == "!!int" {
|
|
format, lhsNum, err := parseInt(lhs.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, rhsNum, err := parseInt(rhs.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sum := lhsNum + rhsNum
|
|
target.Node.Tag = lhs.Tag
|
|
target.Node.Value = fmt.Sprintf(format, sum)
|
|
} else if (lhsTag == "!!int" || lhsTag == "!!float") && (rhsTag == "!!int" || rhsTag == "!!float") {
|
|
lhsNum, err := strconv.ParseFloat(lhs.Value, 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rhsNum, err := strconv.ParseFloat(rhs.Value, 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sum := lhsNum + rhsNum
|
|
if lhsIsCustom {
|
|
target.Node.Tag = lhs.Tag
|
|
} else {
|
|
target.Node.Tag = "!!float"
|
|
}
|
|
target.Node.Value = fmt.Sprintf("%v", sum)
|
|
} else {
|
|
return fmt.Errorf("%v cannot be added to %v", lhsTag, rhsTag)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func addSequences(target *CandidateNode, lhs *CandidateNode, rhs *CandidateNode) {
|
|
target.Node.Kind = yaml.SequenceNode
|
|
if len(lhs.Node.Content) > 0 {
|
|
target.Node.Style = lhs.Node.Style
|
|
}
|
|
target.Node.Tag = lhs.Node.Tag
|
|
target.Node.Content = make([]*yaml.Node, len(lhs.Node.Content))
|
|
copy(target.Node.Content, lhs.Node.Content)
|
|
target.Node.Content = append(target.Node.Content, toNodes(rhs)...)
|
|
}
|
|
|
|
func addMaps(target *CandidateNode, lhsC *CandidateNode, rhsC *CandidateNode) {
|
|
lhs := lhsC.Node
|
|
rhs := rhsC.Node
|
|
|
|
target.Node.Content = make([]*yaml.Node, len(lhs.Content))
|
|
copy(target.Node.Content, lhs.Content)
|
|
|
|
for index := 0; index < len(rhs.Content); index = index + 2 {
|
|
key := rhs.Content[index]
|
|
value := rhs.Content[index+1]
|
|
log.Debug("finding %v", key.Value)
|
|
indexInLhs := findInArray(target.Node, key)
|
|
log.Debug("indexInLhs %v", indexInLhs)
|
|
if indexInLhs < 0 {
|
|
// not in there, append it
|
|
target.Node.Content = append(target.Node.Content, key, value)
|
|
} else {
|
|
// it's there, replace it
|
|
target.Node.Content[indexInLhs+1] = value
|
|
}
|
|
}
|
|
target.Node.Kind = yaml.MappingNode
|
|
if len(lhs.Content) > 0 {
|
|
target.Node.Style = lhs.Style
|
|
}
|
|
target.Node.Tag = lhs.Tag
|
|
}
|