2019-12-23 23:35:57 +00:00
|
|
|
package yqlib
|
|
|
|
|
|
|
|
import (
|
2019-12-27 08:06:08 +00:00
|
|
|
"fmt"
|
2019-12-23 23:35:57 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
2019-12-25 01:11:04 +00:00
|
|
|
type VisitedNode struct {
|
|
|
|
Node *yaml.Node
|
|
|
|
Head string
|
|
|
|
Tail []string
|
|
|
|
PathStack []interface{}
|
|
|
|
}
|
|
|
|
|
2019-12-23 23:35:57 +00:00
|
|
|
type NavigationSettings interface {
|
|
|
|
FollowAlias(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool
|
|
|
|
AutoCreateMap(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool
|
2019-12-25 01:11:04 +00:00
|
|
|
Visit(node *yaml.Node, head string, tail []string, pathStack []interface{}) error
|
2019-12-27 08:06:08 +00:00
|
|
|
ShouldTraverse(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool
|
2019-12-25 01:11:04 +00:00
|
|
|
GetVisitedNodes() []*VisitedNode
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type NavigationSettingsImpl struct {
|
|
|
|
followAlias func(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool
|
|
|
|
autoCreateMap func(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool
|
2019-12-25 01:11:04 +00:00
|
|
|
visit func(node *yaml.Node, head string, tail []string, pathStack []interface{}) error
|
|
|
|
visitedNodes []*VisitedNode
|
|
|
|
}
|
|
|
|
|
|
|
|
func matches(node *yaml.Node, head string) bool {
|
|
|
|
var prefixMatch = strings.TrimSuffix(head, "*")
|
|
|
|
if prefixMatch != head {
|
|
|
|
log.Debug("prefix match, %v", strings.HasPrefix(node.Value, prefixMatch))
|
|
|
|
return strings.HasPrefix(node.Value, prefixMatch)
|
|
|
|
}
|
|
|
|
log.Debug("equals match, %v", node.Value == head)
|
|
|
|
return node.Value == head
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ns *NavigationSettingsImpl) GetVisitedNodes() []*VisitedNode {
|
|
|
|
return ns.visitedNodes
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
|
|
|
|
2019-12-25 01:11:04 +00:00
|
|
|
func (ns *NavigationSettingsImpl) FollowAlias(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool {
|
2019-12-23 23:35:57 +00:00
|
|
|
return ns.followAlias(node, head, tail, pathStack)
|
|
|
|
}
|
|
|
|
|
2019-12-25 01:11:04 +00:00
|
|
|
func (ns *NavigationSettingsImpl) AutoCreateMap(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool {
|
2019-12-23 23:35:57 +00:00
|
|
|
return ns.autoCreateMap(node, head, tail, pathStack)
|
|
|
|
}
|
|
|
|
|
2019-12-27 08:06:08 +00:00
|
|
|
func (ns *NavigationSettingsImpl) matchesNextPath(path string, candidate string) bool {
|
|
|
|
var prefixMatch = strings.TrimSuffix(path, "*")
|
|
|
|
if prefixMatch != path {
|
|
|
|
log.Debug("prefix match, %v", strings.HasPrefix(candidate, prefixMatch))
|
|
|
|
return strings.HasPrefix(candidate, prefixMatch)
|
|
|
|
}
|
|
|
|
return candidate == path
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
|
|
|
|
2019-12-27 08:06:08 +00:00
|
|
|
func (ns *NavigationSettingsImpl) ShouldTraverse(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool {
|
|
|
|
// we should traverse aliases (if enabled), but not visit them :/
|
|
|
|
if len(pathStack) == 0 {
|
|
|
|
return true
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
2019-12-27 08:06:08 +00:00
|
|
|
|
|
|
|
if ns.alreadyVisited(pathStack) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
lastBit := fmt.Sprintf("%v", pathStack[len(pathStack)-1])
|
|
|
|
|
|
|
|
return (lastBit == "<<" && ns.FollowAlias(node, head, tail, pathStack)) || (lastBit != "<<" && ns.matchesNextPath(head, lastBit))
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
|
|
|
|
2019-12-27 08:06:08 +00:00
|
|
|
func (ns *NavigationSettingsImpl) shouldVisit(node *yaml.Node, head string, tail []string, pathStack []interface{}) bool {
|
|
|
|
// we should traverse aliases (if enabled), but not visit them :/
|
|
|
|
if len(pathStack) == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if ns.alreadyVisited(pathStack) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
lastBit := fmt.Sprintf("%v", pathStack[len(pathStack)-1])
|
|
|
|
// only visit aliases if its an exact match
|
|
|
|
return (lastBit == "<<" && head == "<<") || (lastBit != "<<" && ns.matchesNextPath(head, lastBit))
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ns *NavigationSettingsImpl) Visit(node *yaml.Node, head string, tail []string, pathStack []interface{}) error {
|
|
|
|
if ns.shouldVisit(node, head, tail, pathStack) {
|
|
|
|
ns.visitedNodes = append(ns.visitedNodes, &VisitedNode{node, head, tail, pathStack})
|
|
|
|
log.Debug("adding to visited nodes, %v", head)
|
|
|
|
return ns.visit(node, head, tail, pathStack)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ns *NavigationSettingsImpl) alreadyVisited(pathStack []interface{}) bool {
|
|
|
|
log.Debug("looking for pathStack")
|
|
|
|
for _, val := range pathStack {
|
|
|
|
log.Debug("\t %v", val)
|
|
|
|
}
|
2019-12-25 01:11:04 +00:00
|
|
|
for _, candidate := range ns.visitedNodes {
|
2019-12-27 08:06:08 +00:00
|
|
|
candidatePathStack := candidate.PathStack
|
|
|
|
if patchStacksMatch(candidatePathStack, pathStack) {
|
|
|
|
log.Debug("paths match, already seen it")
|
2019-12-23 23:35:57 +00:00
|
|
|
return true
|
2019-12-25 01:11:04 +00:00
|
|
|
}
|
2019-12-27 08:06:08 +00:00
|
|
|
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
2019-12-27 08:06:08 +00:00
|
|
|
log.Debug("never seen it before!")
|
2019-12-25 01:11:04 +00:00
|
|
|
return false
|
2019-12-23 23:35:57 +00:00
|
|
|
}
|
2019-12-27 08:06:08 +00:00
|
|
|
|
|
|
|
func patchStacksMatch(path1 []interface{}, path2 []interface{}) bool {
|
|
|
|
log.Debug("checking against path")
|
|
|
|
for _, val := range path1 {
|
|
|
|
log.Debug("\t %v", val)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(path1) != len(path2) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for index, p1Value := range path1 {
|
|
|
|
|
|
|
|
p2Value := path2[index]
|
|
|
|
if p1Value != p2Value {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
|
|
|
|
}
|