2020-11-03 23:48:43 +00:00
|
|
|
package yqlib
|
2020-10-17 11:10:47 +00:00
|
|
|
|
|
|
|
import (
|
2020-10-21 01:54:58 +00:00
|
|
|
"container/list"
|
|
|
|
|
2020-11-20 04:29:53 +00:00
|
|
|
yaml "gopkg.in/yaml.v3"
|
2020-10-17 11:10:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func isTruthy(c *CandidateNode) (bool, error) {
|
2020-10-27 05:45:16 +00:00
|
|
|
node := UnwrapDoc(c.Node)
|
2020-10-17 11:10:47 +00:00
|
|
|
value := true
|
2020-10-27 05:45:16 +00:00
|
|
|
|
2020-10-20 05:27:30 +00:00
|
|
|
if node.Tag == "!!null" {
|
|
|
|
return false, nil
|
|
|
|
}
|
2020-10-17 11:10:47 +00:00
|
|
|
if node.Kind == yaml.ScalarNode && node.Tag == "!!bool" {
|
|
|
|
errDecoding := node.Decode(&value)
|
|
|
|
if errDecoding != nil {
|
|
|
|
return false, errDecoding
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type boolOp func(bool, bool) bool
|
|
|
|
|
2020-12-21 00:42:35 +00:00
|
|
|
func performBoolOp(op boolOp) func(d *dataTreeNavigator, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
return func(d *dataTreeNavigator, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
|
|
|
lhs.Node = UnwrapDoc(lhs.Node)
|
|
|
|
rhs.Node = UnwrapDoc(rhs.Node)
|
2020-10-17 11:10:47 +00:00
|
|
|
|
2020-12-21 00:42:35 +00:00
|
|
|
lhsTrue, errDecoding := isTruthy(lhs)
|
|
|
|
if errDecoding != nil {
|
|
|
|
return nil, errDecoding
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
|
|
|
|
2020-12-21 00:42:35 +00:00
|
|
|
rhsTrue, errDecoding := isTruthy(rhs)
|
|
|
|
if errDecoding != nil {
|
|
|
|
return nil, errDecoding
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
|
|
|
|
2020-12-21 00:42:35 +00:00
|
|
|
return createBooleanCandidate(lhs, op(lhsTrue, rhsTrue)), nil
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-21 01:54:58 +00:00
|
|
|
func OrOperator(d *dataTreeNavigator, matchingNodes *list.List, pathNode *PathTreeNode) (*list.List, error) {
|
2020-10-17 11:10:47 +00:00
|
|
|
log.Debugf("-- orOp")
|
2020-12-21 00:42:35 +00:00
|
|
|
return crossFunction(d, matchingNodes, pathNode, performBoolOp(
|
|
|
|
func(b1 bool, b2 bool) bool {
|
|
|
|
return b1 || b2
|
|
|
|
}))
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
|
|
|
|
2020-10-21 01:54:58 +00:00
|
|
|
func AndOperator(d *dataTreeNavigator, matchingNodes *list.List, pathNode *PathTreeNode) (*list.List, error) {
|
2020-10-17 11:10:47 +00:00
|
|
|
log.Debugf("-- AndOp")
|
2020-12-21 00:42:35 +00:00
|
|
|
return crossFunction(d, matchingNodes, pathNode, performBoolOp(
|
|
|
|
func(b1 bool, b2 bool) bool {
|
|
|
|
return b1 && b2
|
|
|
|
}))
|
2020-10-17 11:10:47 +00:00
|
|
|
}
|
2020-11-22 02:16:54 +00:00
|
|
|
|
|
|
|
func NotOperator(d *dataTreeNavigator, matchMap *list.List, pathNode *PathTreeNode) (*list.List, error) {
|
|
|
|
log.Debugf("-- notOperation")
|
|
|
|
var results = list.New()
|
|
|
|
|
|
|
|
for el := matchMap.Front(); el != nil; el = el.Next() {
|
|
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
log.Debug("notOperation checking %v", candidate)
|
|
|
|
truthy, errDecoding := isTruthy(candidate)
|
|
|
|
if errDecoding != nil {
|
|
|
|
return nil, errDecoding
|
|
|
|
}
|
|
|
|
result := createBooleanCandidate(candidate, !truthy)
|
|
|
|
results.PushBack(result)
|
|
|
|
}
|
|
|
|
return results, nil
|
|
|
|
}
|