mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-24 14:45:39 +00:00
132 lines
3.9 KiB
Go
132 lines
3.9 KiB
Go
package yqlib
|
|
|
|
import (
|
|
"container/list"
|
|
"fmt"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
)
|
|
|
|
func sortOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
selfExpression := &ExpressionNode{Operation: &Operation{OperationType: selfReferenceOpType}}
|
|
expressionNode.RHS = selfExpression
|
|
return sortByOperator(d, context, expressionNode)
|
|
}
|
|
|
|
// context represents the current matching nodes in the expression pipeline
|
|
//expressionNode is your current expression (sort_by)
|
|
func sortByOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
|
|
results := list.New()
|
|
|
|
for el := context.MatchingNodes.Front(); el != nil; el = el.Next() {
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
candidateNode := unwrapDoc(candidate.Node)
|
|
|
|
if candidateNode.Kind != yaml.SequenceNode {
|
|
return context, fmt.Errorf("node at path [%v] is not an array (it's a %v)", candidate.GetNicePath(), candidate.GetNiceTag())
|
|
}
|
|
|
|
sortableArray := make(sortableNodeArray, len(candidateNode.Content))
|
|
|
|
for i, originalNode := range candidateNode.Content {
|
|
|
|
childCandidate := candidate.CreateChildInArray(i, originalNode)
|
|
compareContext, err := d.GetMatchingNodes(context.SingleReadonlyChildContext(childCandidate), expressionNode.RHS)
|
|
if err != nil {
|
|
return Context{}, err
|
|
}
|
|
|
|
nodeToCompare := &yaml.Node{Kind: yaml.ScalarNode, Tag: "!!null"}
|
|
if compareContext.MatchingNodes.Len() > 0 {
|
|
nodeToCompare = compareContext.MatchingNodes.Front().Value.(*CandidateNode).Node
|
|
}
|
|
|
|
log.Debug("going to compare %v by %v", NodeToString(candidate.CreateReplacement(originalNode)), NodeToString(candidate.CreateReplacement(nodeToCompare)))
|
|
|
|
sortableArray[i] = sortableNode{Node: originalNode, NodeToCompare: nodeToCompare}
|
|
|
|
if nodeToCompare.Kind != yaml.ScalarNode {
|
|
return Context{}, fmt.Errorf("sort only works for scalars, got %v", nodeToCompare.Tag)
|
|
}
|
|
|
|
}
|
|
|
|
sort.Stable(sortableArray)
|
|
|
|
sortedList := &yaml.Node{Kind: yaml.SequenceNode, Tag: "!!seq", Style: candidateNode.Style}
|
|
sortedList.Content = make([]*yaml.Node, len(candidateNode.Content))
|
|
|
|
for i, sortedNode := range sortableArray {
|
|
sortedList.Content[i] = sortedNode.Node
|
|
}
|
|
results.PushBack(candidate.CreateReplacement(sortedList))
|
|
}
|
|
return context.ChildContext(results), nil
|
|
}
|
|
|
|
type sortableNode struct {
|
|
Node *yaml.Node
|
|
NodeToCompare *yaml.Node
|
|
}
|
|
|
|
type sortableNodeArray []sortableNode
|
|
|
|
func (a sortableNodeArray) Len() int { return len(a) }
|
|
func (a sortableNodeArray) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
|
|
func (a sortableNodeArray) Less(i, j int) bool {
|
|
lhs := a[i].NodeToCompare
|
|
rhs := a[j].NodeToCompare
|
|
|
|
if lhs.Tag == "!!null" && rhs.Tag != "!!null" {
|
|
return true
|
|
} else if lhs.Tag != "!!null" && rhs.Tag == "!!null" {
|
|
return false
|
|
} else if lhs.Tag == "!!bool" && rhs.Tag != "!!bool" {
|
|
return true
|
|
} else if lhs.Tag != "!!bool" && rhs.Tag == "!!bool" {
|
|
return false
|
|
} else if lhs.Tag == "!!bool" && rhs.Tag == "!!bool" {
|
|
lhsTruthy, err := isTruthyNode(lhs)
|
|
if err != nil {
|
|
panic(fmt.Errorf("could not parse %v as boolean: %w", lhs.Value, err))
|
|
}
|
|
|
|
rhsTruthy, err := isTruthyNode(rhs)
|
|
if err != nil {
|
|
panic(fmt.Errorf("could not parse %v as boolean: %w", rhs.Value, err))
|
|
}
|
|
|
|
return !lhsTruthy && rhsTruthy
|
|
} else if lhs.Tag != rhs.Tag || lhs.Tag == "!!str" {
|
|
return strings.Compare(lhs.Value, rhs.Value) < 0
|
|
} else if lhs.Tag == "!!int" && rhs.Tag == "!!int" {
|
|
_, lhsNum, err := parseInt(lhs.Value)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
_, rhsNum, err := parseInt(rhs.Value)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return lhsNum < rhsNum
|
|
} else if (lhs.Tag == "!!int" || lhs.Tag == "!!float") && (rhs.Tag == "!!int" || rhs.Tag == "!!float") {
|
|
lhsNum, err := strconv.ParseFloat(lhs.Value, 64)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
rhsNum, err := strconv.ParseFloat(rhs.Value, 64)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return lhsNum < rhsNum
|
|
}
|
|
|
|
return true
|
|
}
|