2019-11-23 03:52:29 +00:00
|
|
|
package yqlib
|
2015-10-03 05:10:29 +00:00
|
|
|
|
2019-12-27 21:51:54 +00:00
|
|
|
import (
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
2019-12-01 19:44:44 +00:00
|
|
|
type PathParser interface {
|
|
|
|
ParsePath(path string) []string
|
2019-12-27 21:51:54 +00:00
|
|
|
MatchesNextPathElement(node *yaml.Node, head string, tail []string, pathStack []interface{}, lastBit string) bool
|
2015-10-03 05:10:29 +00:00
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
type pathParser struct{}
|
2019-12-01 19:44:44 +00:00
|
|
|
|
|
|
|
func NewPathParser() PathParser {
|
2019-12-03 04:50:32 +00:00
|
|
|
return &pathParser{}
|
2019-12-01 19:44:44 +00:00
|
|
|
}
|
|
|
|
|
2019-12-27 21:51:54 +00:00
|
|
|
/**
|
|
|
|
* node: node that we may traverse/visit
|
|
|
|
* head: path element expression to match against
|
|
|
|
* tail: remaining path element expressions
|
|
|
|
* pathStack: stack of actual paths we've matched to get to node
|
|
|
|
* lastBit: actual value of this nodes 'key' or index.
|
|
|
|
*/
|
|
|
|
func (p *pathParser) MatchesNextPathElement(node *yaml.Node, head string, tail []string, pathStack []interface{}, lastBit string) bool {
|
|
|
|
var prefixMatch = strings.TrimSuffix(head, "*")
|
|
|
|
if prefixMatch != head {
|
|
|
|
log.Debug("prefix match, %v", strings.HasPrefix(lastBit, prefixMatch))
|
|
|
|
return strings.HasPrefix(lastBit, prefixMatch)
|
|
|
|
}
|
|
|
|
return lastBit == head
|
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
func (p *pathParser) ParsePath(path string) []string {
|
2019-12-09 02:44:53 +00:00
|
|
|
if path == "" {
|
|
|
|
return []string{}
|
|
|
|
}
|
2019-12-01 19:44:44 +00:00
|
|
|
return p.parsePathAccum([]string{}, path)
|
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
func (p *pathParser) parsePathAccum(paths []string, remaining string) []string {
|
2019-12-01 19:44:44 +00:00
|
|
|
head, tail := p.nextYamlPath(remaining)
|
2015-10-03 05:10:29 +00:00
|
|
|
if tail == "" {
|
|
|
|
return append(paths, head)
|
|
|
|
}
|
2019-12-01 19:44:44 +00:00
|
|
|
return p.parsePathAccum(append(paths, head), tail)
|
2015-10-03 05:10:29 +00:00
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
func (p *pathParser) nextYamlPath(path string) (pathElement string, remaining string) {
|
2015-10-03 05:10:29 +00:00
|
|
|
switch path[0] {
|
|
|
|
case '[':
|
|
|
|
// e.g [0].blah.cat -> we need to return "0" and "blah.cat"
|
2019-12-01 19:44:44 +00:00
|
|
|
return p.search(path[1:], []uint8{']'}, true)
|
2015-10-03 05:10:29 +00:00
|
|
|
case '"':
|
|
|
|
// e.g "a.b".blah.cat -> we need to return "a.b" and "blah.cat"
|
2019-12-01 19:44:44 +00:00
|
|
|
return p.search(path[1:], []uint8{'"'}, true)
|
2015-10-03 05:10:29 +00:00
|
|
|
default:
|
|
|
|
// e.g "a.blah.cat" -> return "a" and "blah.cat"
|
2019-12-01 19:44:44 +00:00
|
|
|
return p.search(path[0:], []uint8{'.', '['}, false)
|
2015-10-03 05:10:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
func (p *pathParser) search(path string, matchingChars []uint8, skipNext bool) (pathElement string, remaining string) {
|
2015-10-03 05:10:29 +00:00
|
|
|
for i := 0; i < len(path); i++ {
|
|
|
|
var char = path[i]
|
2019-12-01 19:44:44 +00:00
|
|
|
if p.contains(matchingChars, char) {
|
2015-10-03 05:10:29 +00:00
|
|
|
var remainingStart = i + 1
|
|
|
|
if skipNext {
|
|
|
|
remainingStart = remainingStart + 1
|
|
|
|
} else if !skipNext && char != '.' {
|
|
|
|
remainingStart = i
|
|
|
|
}
|
|
|
|
if remainingStart > len(path) {
|
|
|
|
remainingStart = len(path)
|
|
|
|
}
|
2017-09-20 23:40:33 +00:00
|
|
|
return path[0:i], path[remainingStart:]
|
2015-10-03 05:10:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return path, ""
|
|
|
|
}
|
|
|
|
|
2019-12-03 04:50:32 +00:00
|
|
|
func (p *pathParser) contains(matchingChars []uint8, candidate uint8) bool {
|
2015-10-03 05:10:29 +00:00
|
|
|
for _, a := range matchingChars {
|
|
|
|
if a == candidate {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|