2021-11-28 02:25:22 +00:00
|
|
|
package yqlib
|
|
|
|
|
|
|
|
import (
|
|
|
|
"container/list"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2022-11-04 01:21:12 +00:00
|
|
|
"time"
|
2021-11-28 02:25:22 +00:00
|
|
|
)
|
|
|
|
|
2021-12-04 02:54:12 +00:00
|
|
|
func sortOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
|
|
selfExpression := &ExpressionNode{Operation: &Operation{OperationType: selfReferenceOpType}}
|
2022-02-07 00:55:55 +00:00
|
|
|
expressionNode.RHS = selfExpression
|
2021-12-04 02:54:12 +00:00
|
|
|
return sortByOperator(d, context, expressionNode)
|
|
|
|
}
|
|
|
|
|
2021-11-28 02:25:22 +00:00
|
|
|
// context represents the current matching nodes in the expression pipeline
|
2022-08-29 04:13:15 +00:00
|
|
|
// expressionNode is your current expression (sort_by)
|
2021-11-28 02:25:22 +00:00
|
|
|
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)
|
|
|
|
|
2025-01-21 22:55:30 +00:00
|
|
|
var sortableArray sortableNodeArray
|
|
|
|
|
|
|
|
if candidate.CanVisitValues() {
|
|
|
|
sortableArray = make(sortableNodeArray, 0)
|
|
|
|
visitor := func(valueNode *CandidateNode) error {
|
|
|
|
compareContext, err := d.GetMatchingNodes(context.SingleReadonlyChildContext(valueNode), expressionNode.RHS)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
sortableNode := sortableNode{Node: valueNode, CompareContext: compareContext, dateTimeLayout: context.GetDateTimeLayout()}
|
|
|
|
sortableArray = append(sortableArray, sortableNode)
|
|
|
|
return nil
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
2025-01-21 22:55:30 +00:00
|
|
|
if err := candidate.VisitValues(visitor); err != nil {
|
|
|
|
return context, err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return context, fmt.Errorf("node at path [%v] is not an array or map (it's a %v)", candidate.GetNicePath(), candidate.Tag)
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
|
|
|
|
2021-12-04 02:54:12 +00:00
|
|
|
sort.Stable(sortableArray)
|
2021-11-28 02:25:22 +00:00
|
|
|
|
2025-01-21 22:55:30 +00:00
|
|
|
sortedList := candidate.CopyWithoutContent()
|
|
|
|
if candidate.Kind == MappingNode {
|
|
|
|
for _, sortedNode := range sortableArray {
|
|
|
|
sortedList.AddKeyValueChild(sortedNode.Node.Key, sortedNode.Node)
|
|
|
|
}
|
|
|
|
} else if candidate.Kind == SequenceNode {
|
|
|
|
for _, sortedNode := range sortableArray {
|
|
|
|
sortedList.AddChild(sortedNode.Node)
|
|
|
|
}
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
2025-01-21 22:55:30 +00:00
|
|
|
|
|
|
|
// convert array of value nodes back to map
|
2023-10-18 01:11:53 +00:00
|
|
|
results.PushBack(sortedList)
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
|
|
|
return context.ChildContext(results), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type sortableNode struct {
|
2023-10-18 01:11:53 +00:00
|
|
|
Node *CandidateNode
|
2023-02-10 17:44:15 +00:00
|
|
|
CompareContext Context
|
2022-11-04 01:21:12 +00:00
|
|
|
dateTimeLayout string
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2023-02-10 17:44:15 +00:00
|
|
|
lhsContext := a[i].CompareContext
|
|
|
|
rhsContext := a[j].CompareContext
|
|
|
|
|
|
|
|
rhsEl := rhsContext.MatchingNodes.Front()
|
2024-11-16 01:27:00 +00:00
|
|
|
|
2023-02-10 17:44:15 +00:00
|
|
|
for lhsEl := lhsContext.MatchingNodes.Front(); lhsEl != nil && rhsEl != nil; lhsEl = lhsEl.Next() {
|
|
|
|
lhs := lhsEl.Value.(*CandidateNode)
|
|
|
|
rhs := rhsEl.Value.(*CandidateNode)
|
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
result := a.compare(lhs, rhs, a[i].dateTimeLayout)
|
2023-02-10 17:44:15 +00:00
|
|
|
|
|
|
|
if result < 0 {
|
|
|
|
return true
|
|
|
|
} else if result > 0 {
|
|
|
|
return false
|
|
|
|
}
|
2021-11-28 02:25:22 +00:00
|
|
|
|
2023-02-10 17:44:15 +00:00
|
|
|
rhsEl = rhsEl.Next()
|
|
|
|
}
|
2024-11-16 01:27:00 +00:00
|
|
|
return lhsContext.MatchingNodes.Len() < rhsContext.MatchingNodes.Len()
|
2023-02-10 17:44:15 +00:00
|
|
|
}
|
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
func (a sortableNodeArray) compare(lhs *CandidateNode, rhs *CandidateNode, dateTimeLayout string) int {
|
2022-11-04 01:21:12 +00:00
|
|
|
lhsTag := lhs.Tag
|
|
|
|
rhsTag := rhs.Tag
|
|
|
|
|
|
|
|
if !strings.HasPrefix(lhsTag, "!!") {
|
|
|
|
// custom tag - we have to have a guess
|
2023-10-18 01:11:53 +00:00
|
|
|
lhsTag = lhs.guessTagFromCustomType()
|
2022-11-04 01:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.HasPrefix(rhsTag, "!!") {
|
|
|
|
// custom tag - we have to have a guess
|
2023-10-18 01:11:53 +00:00
|
|
|
rhsTag = rhs.guessTagFromCustomType()
|
2022-11-04 01:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
isDateTime := lhsTag == "!!timestamp" && rhsTag == "!!timestamp"
|
2023-02-10 17:44:15 +00:00
|
|
|
layout := dateTimeLayout
|
2022-11-04 01:21:12 +00:00
|
|
|
// if the lhs is a string, it might be a timestamp in a custom format.
|
|
|
|
if lhsTag == "!!str" && layout != time.RFC3339 {
|
|
|
|
_, errLhs := parseDateTime(layout, lhs.Value)
|
|
|
|
_, errRhs := parseDateTime(layout, rhs.Value)
|
|
|
|
isDateTime = errLhs == nil && errRhs == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if lhsTag == "!!null" && rhsTag != "!!null" {
|
2023-02-10 17:44:15 +00:00
|
|
|
return -1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if lhsTag != "!!null" && rhsTag == "!!null" {
|
2023-02-10 17:44:15 +00:00
|
|
|
return 1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if lhsTag == "!!bool" && rhsTag != "!!bool" {
|
2023-02-10 17:44:15 +00:00
|
|
|
return -1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if lhsTag != "!!bool" && rhsTag == "!!bool" {
|
2023-02-10 17:44:15 +00:00
|
|
|
return 1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if lhsTag == "!!bool" && rhsTag == "!!bool" {
|
2023-10-18 01:11:53 +00:00
|
|
|
lhsTruthy := isTruthyNode(lhs)
|
2021-12-04 02:54:12 +00:00
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
rhsTruthy := isTruthyNode(rhs)
|
2023-02-10 17:44:15 +00:00
|
|
|
if lhsTruthy == rhsTruthy {
|
|
|
|
return 0
|
|
|
|
} else if lhsTruthy {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
return -1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if isDateTime {
|
|
|
|
lhsTime, err := parseDateTime(layout, lhs.Value)
|
|
|
|
if err != nil {
|
|
|
|
log.Warningf("Could not parse time %v with layout %v for sort, sorting by string instead: %w", lhs.Value, layout, err)
|
2023-02-10 17:44:15 +00:00
|
|
|
return strings.Compare(lhs.Value, rhs.Value)
|
2022-11-04 01:21:12 +00:00
|
|
|
}
|
|
|
|
rhsTime, err := parseDateTime(layout, rhs.Value)
|
|
|
|
if err != nil {
|
|
|
|
log.Warningf("Could not parse time %v with layout %v for sort, sorting by string instead: %w", rhs.Value, layout, err)
|
2023-02-10 17:44:15 +00:00
|
|
|
return strings.Compare(lhs.Value, rhs.Value)
|
2022-11-04 01:21:12 +00:00
|
|
|
}
|
2023-02-10 17:44:15 +00:00
|
|
|
if lhsTime.Equal(rhsTime) {
|
|
|
|
return 0
|
|
|
|
} else if lhsTime.Before(rhsTime) {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if lhsTag == "!!int" && rhsTag == "!!int" {
|
2022-05-06 03:46:14 +00:00
|
|
|
_, lhsNum, err := parseInt64(lhs.Value)
|
2021-11-28 02:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2022-05-06 03:46:14 +00:00
|
|
|
_, rhsNum, err := parseInt64(rhs.Value)
|
2021-11-28 02:25:22 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2023-02-10 17:44:15 +00:00
|
|
|
return int(lhsNum - rhsNum)
|
2022-11-04 01:21:12 +00:00
|
|
|
} else if (lhsTag == "!!int" || lhsTag == "!!float") && (rhsTag == "!!int" || rhsTag == "!!float") {
|
2021-11-28 02:25:22 +00:00
|
|
|
lhsNum, err := strconv.ParseFloat(lhs.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
rhsNum, err := strconv.ParseFloat(rhs.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2023-02-10 17:44:15 +00:00
|
|
|
if lhsNum == rhsNum {
|
|
|
|
return 0
|
|
|
|
} else if lhsNum < rhsNum {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|
|
|
|
|
2023-02-10 17:44:15 +00:00
|
|
|
return strings.Compare(lhs.Value, rhs.Value)
|
2021-11-28 02:25:22 +00:00
|
|
|
}
|