2022-03-17 03:08:08 +00:00
|
|
|
package yqlib
|
|
|
|
|
|
|
|
import (
|
2024-03-30 02:34:36 +00:00
|
|
|
"container/list"
|
2022-03-17 03:08:08 +00:00
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
)
|
|
|
|
|
|
|
|
type compareTypePref struct {
|
|
|
|
OrEqual bool
|
|
|
|
Greater bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func compareOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
2024-02-15 22:41:33 +00:00
|
|
|
log.Debugf("compareOperator")
|
2022-03-17 03:08:08 +00:00
|
|
|
prefs := expressionNode.Operation.Preferences.(compareTypePref)
|
2022-11-13 00:58:21 +00:00
|
|
|
return crossFunction(d, context, expressionNode, compare(prefs), true)
|
2022-03-17 03:08:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func compare(prefs compareTypePref) func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
return func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
2024-02-15 22:41:33 +00:00
|
|
|
log.Debugf("compare cross function")
|
2022-03-17 03:08:08 +00:00
|
|
|
if lhs == nil && rhs == nil {
|
|
|
|
owner := &CandidateNode{}
|
|
|
|
return createBooleanCandidate(owner, prefs.OrEqual), nil
|
|
|
|
} else if lhs == nil {
|
|
|
|
log.Debugf("lhs nil, but rhs is not")
|
|
|
|
return createBooleanCandidate(rhs, false), nil
|
|
|
|
} else if rhs == nil {
|
|
|
|
log.Debugf("rhs nil, but rhs is not")
|
|
|
|
return createBooleanCandidate(lhs, false), nil
|
|
|
|
}
|
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
switch lhs.Kind {
|
|
|
|
case MappingNode:
|
2022-03-17 03:08:08 +00:00
|
|
|
return nil, fmt.Errorf("maps not yet supported for comparison")
|
2023-10-18 01:11:53 +00:00
|
|
|
case SequenceNode:
|
2022-03-17 03:08:08 +00:00
|
|
|
return nil, fmt.Errorf("arrays not yet supported for comparison")
|
|
|
|
default:
|
2023-10-18 01:11:53 +00:00
|
|
|
if rhs.Kind != ScalarNode {
|
|
|
|
return nil, fmt.Errorf("%v (%v) cannot be subtracted from %v", rhs.Tag, rhs.GetNicePath(), lhs.Tag)
|
2022-03-17 03:08:08 +00:00
|
|
|
}
|
2023-10-18 01:11:53 +00:00
|
|
|
target := lhs.CopyWithoutContent()
|
|
|
|
boolV, err := compareScalars(context, prefs, lhs, rhs)
|
2022-03-17 03:08:08 +00:00
|
|
|
|
|
|
|
return createBooleanCandidate(target, boolV), err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
func compareDateTime(layout string, prefs compareTypePref, lhs *CandidateNode, rhs *CandidateNode) (bool, error) {
|
2022-11-04 01:21:12 +00:00
|
|
|
lhsTime, err := parseDateTime(layout, lhs.Value)
|
2022-03-17 03:08:08 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2022-11-04 01:21:12 +00:00
|
|
|
rhsTime, err := parseDateTime(layout, rhs.Value)
|
2022-03-17 03:08:08 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if prefs.OrEqual && lhsTime.Equal(rhsTime) {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if prefs.Greater {
|
|
|
|
return lhsTime.After(rhsTime), nil
|
|
|
|
}
|
|
|
|
return lhsTime.Before(rhsTime), nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-10-18 01:11:53 +00:00
|
|
|
func compareScalars(context Context, prefs compareTypePref, lhs *CandidateNode, rhs *CandidateNode) (bool, error) {
|
|
|
|
lhsTag := lhs.guessTagFromCustomType()
|
|
|
|
rhsTag := rhs.guessTagFromCustomType()
|
2022-03-17 03:08:08 +00:00
|
|
|
|
|
|
|
isDateTime := lhs.Tag == "!!timestamp"
|
|
|
|
// if the lhs is a string, it might be a timestamp in a custom format.
|
2023-02-02 02:30:48 +00:00
|
|
|
if lhsTag == "!!str" {
|
2022-11-04 01:21:12 +00:00
|
|
|
_, err := parseDateTime(context.GetDateTimeLayout(), lhs.Value)
|
2022-03-17 03:08:08 +00:00
|
|
|
isDateTime = err == nil
|
|
|
|
}
|
|
|
|
if isDateTime {
|
|
|
|
return compareDateTime(context.GetDateTimeLayout(), prefs, lhs, rhs)
|
|
|
|
} else if lhsTag == "!!int" && rhsTag == "!!int" {
|
2022-05-06 03:46:14 +00:00
|
|
|
_, lhsNum, err := parseInt64(lhs.Value)
|
2022-03-17 03:08:08 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2022-05-06 03:46:14 +00:00
|
|
|
_, rhsNum, err := parseInt64(rhs.Value)
|
2022-03-17 03:08:08 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if prefs.OrEqual && lhsNum == rhsNum {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if prefs.Greater {
|
|
|
|
return lhsNum > rhsNum, nil
|
|
|
|
}
|
|
|
|
return lhsNum < rhsNum, nil
|
|
|
|
} else if (lhsTag == "!!int" || lhsTag == "!!float") && (rhsTag == "!!int" || rhsTag == "!!float") {
|
|
|
|
lhsNum, err := strconv.ParseFloat(lhs.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
rhsNum, err := strconv.ParseFloat(rhs.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if prefs.OrEqual && lhsNum == rhsNum {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if prefs.Greater {
|
|
|
|
return lhsNum > rhsNum, nil
|
|
|
|
}
|
|
|
|
return lhsNum < rhsNum, nil
|
|
|
|
} else if lhsTag == "!!str" && rhsTag == "!!str" {
|
|
|
|
if prefs.OrEqual && lhs.Value == rhs.Value {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if prefs.Greater {
|
|
|
|
return lhs.Value > rhs.Value, nil
|
|
|
|
}
|
|
|
|
return lhs.Value < rhs.Value, nil
|
2022-11-13 00:58:21 +00:00
|
|
|
} else if lhsTag == "!!null" && rhsTag == "!!null" && prefs.OrEqual {
|
|
|
|
return true, nil
|
|
|
|
} else if lhsTag == "!!null" || rhsTag == "!!null" {
|
|
|
|
return false, nil
|
2022-03-17 03:08:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false, fmt.Errorf("%v not yet supported for comparison", lhs.Tag)
|
|
|
|
}
|
2024-03-30 02:34:36 +00:00
|
|
|
|
|
|
|
func superlativeByComparison(d *dataTreeNavigator, context Context, prefs compareTypePref) (Context, error) {
|
|
|
|
fn := compare(prefs)
|
|
|
|
|
|
|
|
var results = list.New()
|
|
|
|
|
|
|
|
for seq := context.MatchingNodes.Front(); seq != nil; seq = seq.Next() {
|
|
|
|
splatted, err := splat(context.SingleChildContext(seq.Value.(*CandidateNode)), traversePreferences{})
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
result := splatted.MatchingNodes.Front()
|
|
|
|
if result != nil {
|
|
|
|
for el := result.Next(); el != nil; el = el.Next() {
|
|
|
|
cmp, err := fn(d, context, el.Value.(*CandidateNode), result.Value.(*CandidateNode))
|
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
if isTruthyNode(cmp) {
|
|
|
|
result = el
|
|
|
|
}
|
|
|
|
}
|
|
|
|
results.PushBack(result.Value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return context.ChildContext(results), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func minOperator(d *dataTreeNavigator, context Context, _ *ExpressionNode) (Context, error) {
|
|
|
|
log.Debug(("Min"))
|
|
|
|
return superlativeByComparison(d, context, compareTypePref{Greater: false})
|
|
|
|
}
|
|
|
|
|
|
|
|
func maxOperator(d *dataTreeNavigator, context Context, _ *ExpressionNode) (Context, error) {
|
|
|
|
log.Debug(("Max"))
|
|
|
|
return superlativeByComparison(d, context, compareTypePref{Greater: true})
|
|
|
|
}
|