This commit is contained in:
Mike Farah 2019-12-08 15:59:24 +11:00
parent d97f1d8be2
commit 8da9a81702

View File

@ -17,6 +17,12 @@ type navigator struct {
log *logging.Logger log *logging.Logger
} }
type VisitorFn func(*yaml.Node) (*yaml.Node, error)
func identityVisitor(value *yaml.Node) (*yaml.Node, error) {
return value, nil
}
func NewDataNavigator(l *logging.Logger) DataNavigator { func NewDataNavigator(l *logging.Logger) DataNavigator {
return &navigator{ return &navigator{
log: l, log: l,
@ -24,6 +30,29 @@ func NewDataNavigator(l *logging.Logger) DataNavigator {
} }
func (n *navigator) Get(value *yaml.Node, path []string) (*yaml.Node, error) { func (n *navigator) Get(value *yaml.Node, path []string) (*yaml.Node, error) {
return n.Visit(value, path, identityVisitor)
}
func (n *navigator) Update(value *yaml.Node, path []string, changesToApply yaml.Node) error {
_, errorVisiting := n.Visit(value, path, func(nodeToUpdate *yaml.Node) (*yaml.Node, error) {
n.log.Debug("going to update")
n.debugNode(nodeToUpdate)
n.log.Debug("with")
n.debugNode(&changesToApply)
nodeToUpdate.Value = changesToApply.Value
nodeToUpdate.Tag = changesToApply.Tag
nodeToUpdate.Kind = changesToApply.Kind
nodeToUpdate.Style = changesToApply.Style
nodeToUpdate.Content = changesToApply.Content
nodeToUpdate.HeadComment = changesToApply.HeadComment
nodeToUpdate.LineComment = changesToApply.LineComment
nodeToUpdate.FootComment = changesToApply.FootComment
return nodeToUpdate, nil
})
return errorVisiting
}
func (n *navigator) Visit(value *yaml.Node, path []string, visitor VisitorFn) (*yaml.Node, error) {
realValue := value realValue := value
if realValue.Kind == yaml.DocumentNode { if realValue.Kind == yaml.DocumentNode {
realValue = value.Content[0] realValue = value.Content[0]
@ -31,9 +60,9 @@ func (n *navigator) Get(value *yaml.Node, path []string) (*yaml.Node, error) {
if len(path) > 0 { if len(path) > 0 {
n.log.Debugf("diving into %v", path[0]) n.log.Debugf("diving into %v", path[0])
n.debugNode(value) n.debugNode(value)
return n.recurse(realValue, path[0], path[1:]) return n.recurse(realValue, path[0], path[1:], visitor)
} }
return realValue, nil return visitor(realValue)
} }
func (n *navigator) guessKind(tail []string) yaml.Kind { func (n *navigator) guessKind(tail []string) yaml.Kind {
@ -72,7 +101,7 @@ func (n *navigator) debugNode(value *yaml.Node) {
} }
} }
func (n *navigator) recurse(value *yaml.Node, head string, tail []string) (*yaml.Node, error) { func (n *navigator) recurse(value *yaml.Node, head string, tail []string, visitor VisitorFn) (*yaml.Node, error) {
switch value.Kind { switch value.Kind {
case yaml.MappingNode: case yaml.MappingNode:
n.log.Debug("its a map with %v entries", len(value.Content)/2) n.log.Debug("its a map with %v entries", len(value.Content)/2)
@ -83,38 +112,38 @@ func (n *navigator) recurse(value *yaml.Node, head string, tail []string) (*yaml
continue continue
} }
value.Content[index+1] = n.getOrReplace(value.Content[index+1], n.guessKind(tail)) value.Content[index+1] = n.getOrReplace(value.Content[index+1], n.guessKind(tail))
return n.Get(value.Content[index+1], tail) return n.Visit(value.Content[index+1], tail, visitor)
} }
value.Content = append(value.Content, &yaml.Node{Value: head, Kind: yaml.ScalarNode}) value.Content = append(value.Content, &yaml.Node{Value: head, Kind: yaml.ScalarNode})
mapEntryValue := yaml.Node{Kind: n.guessKind(tail)} mapEntryValue := yaml.Node{Kind: n.guessKind(tail)}
value.Content = append(value.Content, &mapEntryValue) value.Content = append(value.Content, &mapEntryValue)
n.log.Debug("adding new node %v", value.Content) n.log.Debug("adding new node %v", value.Content)
return n.Get(&mapEntryValue, tail) return n.Visit(&mapEntryValue, tail, visitor)
case yaml.SequenceNode: case yaml.SequenceNode:
n.log.Debug("its a sequence of %v things!, %v", len(value.Content)) n.log.Debug("its a sequence of %v things!, %v", len(value.Content))
if head == "*" { if head == "*" {
originalContent := value.Content var newNode = yaml.Node{Kind: yaml.SequenceNode, Style: value.Style}
value.Content = make([]*yaml.Node, len(value.Content)) newNode.Content = make([]*yaml.Node, len(value.Content))
for index, childValue := range originalContent { for index, childValue := range value.Content {
n.log.Debug("processing") n.log.Debug("processing")
n.debugNode(childValue) n.debugNode(childValue)
childValue = n.getOrReplace(childValue, n.guessKind(tail)) childValue = n.getOrReplace(childValue, n.guessKind(tail))
var nestedValue, err = n.Get(childValue, tail) var nestedValue, err = n.Visit(childValue, tail, visitor)
n.log.Debug("nestedValue") n.log.Debug("nestedValue")
n.debugNode(nestedValue) n.debugNode(nestedValue)
if err != nil { if err != nil {
return nil, err return nil, err
} }
value.Content[index] = nestedValue newNode.Content[index] = nestedValue
} }
return value, nil return &newNode, nil
} else if head == "+" { } else if head == "+" {
var newNode = yaml.Node{Kind: n.guessKind(tail)} var newNode = yaml.Node{Kind: n.guessKind(tail)}
value.Content = append(value.Content, &newNode) value.Content = append(value.Content, &newNode)
n.log.Debug("appending a new node, %v", value.Content) n.log.Debug("appending a new node, %v", value.Content)
return n.Get(&newNode, tail) return n.Visit(&newNode, tail, visitor)
} }
var index, err = strconv.ParseInt(head, 10, 64) // nolint var index, err = strconv.ParseInt(head, 10, 64) // nolint
if err != nil { if err != nil {
@ -124,28 +153,12 @@ func (n *navigator) recurse(value *yaml.Node, head string, tail []string) (*yaml
return nil, nil return nil, nil
} }
value.Content[index] = n.getOrReplace(value.Content[index], n.guessKind(tail)) value.Content[index] = n.getOrReplace(value.Content[index], n.guessKind(tail))
return n.Get(value.Content[index], tail) return n.Visit(value.Content[index], tail, visitor)
default: default:
return nil, nil return nil, nil
} }
} }
func (n *navigator) Update(dataBucket *yaml.Node, remainingPath []string, changesToApply yaml.Node) error {
nodeToUpdate, errorRecursing := n.Get(dataBucket, remainingPath)
if errorRecursing != nil {
return errorRecursing
}
nodeToUpdate.Value = changesToApply.Value
nodeToUpdate.Tag = changesToApply.Tag
nodeToUpdate.Kind = changesToApply.Kind
nodeToUpdate.Style = changesToApply.Style
nodeToUpdate.Content = changesToApply.Content
nodeToUpdate.HeadComment = changesToApply.HeadComment
nodeToUpdate.LineComment = changesToApply.LineComment
nodeToUpdate.FootComment = changesToApply.FootComment
return nil
}
// func matchesKey(key string, actual interface{}) bool { // func matchesKey(key string, actual interface{}) bool {
// var actualString = fmt.Sprintf("%v", actual) // var actualString = fmt.Sprintf("%v", actual)
// var prefixMatch = strings.TrimSuffix(key, "*") // var prefixMatch = strings.TrimSuffix(key, "*")