mirror of
https://github.com/mikefarah/yq.git
synced 2024-11-14 15:18:06 +00:00
109 lines
2.7 KiB
Go
109 lines
2.7 KiB
Go
package yqlib
|
|
|
|
import (
|
|
"container/list"
|
|
"fmt"
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
)
|
|
|
|
func DeleteChildOperator(d *dataTreeNavigator, matchingNodes *list.List, pathNode *PathTreeNode) (*list.List, error) {
|
|
|
|
nodesToDelete, err := d.GetMatchingNodes(matchingNodes, pathNode.Rhs)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for el := nodesToDelete.Front(); el != nil; el = el.Next() {
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
deleteImmediateChildOp := &Operation{
|
|
OperationType: DeleteImmediateChild,
|
|
Value: candidate.Path[len(candidate.Path)-1],
|
|
}
|
|
|
|
deleteImmediateChildOpNode := &PathTreeNode{
|
|
Operation: deleteImmediateChildOp,
|
|
Rhs: createTraversalTree(candidate.Path[0 : len(candidate.Path)-1]),
|
|
}
|
|
|
|
_, err := d.GetMatchingNodes(matchingNodes, deleteImmediateChildOpNode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return matchingNodes, nil
|
|
}
|
|
|
|
func DeleteImmediateChildOperator(d *dataTreeNavigator, matchingNodes *list.List, pathNode *PathTreeNode) (*list.List, error) {
|
|
parents, err := d.GetMatchingNodes(matchingNodes, pathNode.Rhs)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
childPath := pathNode.Operation.Value
|
|
|
|
log.Debug("childPath to remove %v", childPath)
|
|
|
|
for el := parents.Front(); el != nil; el = el.Next() {
|
|
parent := el.Value.(*CandidateNode)
|
|
parentNode := UnwrapDoc(parent.Node)
|
|
if parentNode.Kind == yaml.MappingNode {
|
|
deleteFromMap(parent, childPath)
|
|
} else if parentNode.Kind == yaml.SequenceNode {
|
|
deleteFromArray(parent, childPath)
|
|
} else {
|
|
return nil, fmt.Errorf("Cannot delete nodes from parent of tag %v", parentNode.Tag)
|
|
}
|
|
|
|
}
|
|
return matchingNodes, nil
|
|
}
|
|
|
|
func deleteFromMap(candidate *CandidateNode, childPath interface{}) {
|
|
log.Debug("deleteFromMap")
|
|
node := UnwrapDoc(candidate.Node)
|
|
contents := node.Content
|
|
newContents := make([]*yaml.Node, 0)
|
|
|
|
for index := 0; index < len(contents); index = index + 2 {
|
|
key := contents[index]
|
|
value := contents[index+1]
|
|
|
|
childCandidate := &CandidateNode{
|
|
Node: value,
|
|
Document: candidate.Document,
|
|
Path: append(candidate.Path, key.Value),
|
|
}
|
|
|
|
shouldDelete := key.Value == childPath
|
|
|
|
log.Debugf("shouldDelete %v ? %v", childCandidate.GetKey(), shouldDelete)
|
|
|
|
if !shouldDelete {
|
|
newContents = append(newContents, key, value)
|
|
}
|
|
}
|
|
node.Content = newContents
|
|
}
|
|
|
|
func deleteFromArray(candidate *CandidateNode, childPath interface{}) {
|
|
log.Debug("deleteFromArray")
|
|
node := UnwrapDoc(candidate.Node)
|
|
contents := node.Content
|
|
newContents := make([]*yaml.Node, 0)
|
|
|
|
for index := 0; index < len(contents); index = index + 1 {
|
|
value := contents[index]
|
|
|
|
shouldDelete := fmt.Sprintf("%v", index) == fmt.Sprintf("%v", childPath)
|
|
|
|
if !shouldDelete {
|
|
newContents = append(newContents, value)
|
|
}
|
|
}
|
|
node.Content = newContents
|
|
}
|