mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-06-10 08:20:59 +00:00
dropping the node style creates a compatibility issue where quotes around "on" are dropped because yaml.v3 interprets it as a string. other yaml parsers interpret on as a bool value, and parse it as a bool rather than string. fix: retain the original style so it is kept as quoted. - fmt: don't drop the styles - merge2: keep the style when merging elements - setting a field: if changing the value of a scalar field, retain its style by default
616 lines
17 KiB
Go
616 lines
17 KiB
Go
// Copyright 2019 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package yaml
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
"gopkg.in/yaml.v3"
|
|
"sigs.k8s.io/kustomize/kyaml/errors"
|
|
)
|
|
|
|
// Append creates an ElementAppender
|
|
func Append(elements ...*yaml.Node) ElementAppender {
|
|
return ElementAppender{Elements: elements}
|
|
}
|
|
|
|
// ElementAppender adds all element to a SequenceNode's Content.
|
|
// Returns Elements[0] if len(Elements) == 1, otherwise returns nil.
|
|
type ElementAppender struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Elem is the value to append.
|
|
Elements []*yaml.Node `yaml:"elements,omitempty"`
|
|
}
|
|
|
|
func (a ElementAppender) Filter(rn *RNode) (*RNode, error) {
|
|
if err := ErrorIfInvalid(rn, yaml.SequenceNode); err != nil {
|
|
return nil, err
|
|
}
|
|
for i := range a.Elements {
|
|
rn.YNode().Content = append(rn.Content(), a.Elements[i])
|
|
}
|
|
if len(a.Elements) == 1 {
|
|
return NewRNode(a.Elements[0]), nil
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// ElementSetter sets the value for an Element in an associative list. ElementSetter
|
|
// will remove any elements which are empty.
|
|
type ElementSetter struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Element is the new value to set -- remove the existing element if nil
|
|
Element *Node
|
|
|
|
// Key is a field on the elements. It is used to find the matching element to
|
|
// update / delete.
|
|
Key string `yaml:"key,omitempty"`
|
|
|
|
// Value is a field value on the elements. It is used to find matching elements to
|
|
// update / delete.
|
|
Value string `yaml:"value,omitempty"`
|
|
}
|
|
|
|
func (e ElementSetter) Filter(rn *RNode) (*RNode, error) {
|
|
if err := ErrorIfInvalid(rn, SequenceNode); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// build the new Content slice
|
|
var newContent []*yaml.Node
|
|
matchingElementFound := false
|
|
for i := range rn.YNode().Content {
|
|
elem := rn.Content()[i]
|
|
|
|
// empty elements are not valid -- they at least need an associative key
|
|
if IsEmpty(NewRNode(elem)) {
|
|
continue
|
|
}
|
|
|
|
// check if this is the element we are matching
|
|
val, err := NewRNode(elem).Pipe(FieldMatcher{Name: e.Key, StringValue: e.Value})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if val == nil {
|
|
// not the element we are looking for, keep it in the Content
|
|
newContent = append(newContent, elem)
|
|
continue
|
|
}
|
|
matchingElementFound = true
|
|
|
|
// deletion operation -- remove the element from the new Content
|
|
if e.Element == nil {
|
|
continue
|
|
}
|
|
// replace operation -- replace the element in the Content
|
|
newContent = append(newContent, e.Element)
|
|
}
|
|
rn.YNode().Content = newContent
|
|
|
|
// deletion operation -- return nil
|
|
if IsMissingOrNull(NewRNode(e.Element)) {
|
|
return nil, nil
|
|
}
|
|
|
|
// append operation -- add the element to the Content
|
|
if !matchingElementFound {
|
|
rn.YNode().Content = append(rn.YNode().Content, e.Element)
|
|
}
|
|
|
|
return NewRNode(e.Element), nil
|
|
}
|
|
|
|
// Clear returns a FieldClearer
|
|
func Clear(name string) FieldClearer {
|
|
return FieldClearer{Name: name}
|
|
}
|
|
|
|
// FieldClearer removes the field or map key.
|
|
// Returns a RNode with the removed field or map entry.
|
|
type FieldClearer struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Name is the name of the field or key in the map.
|
|
Name string `yaml:"name,omitempty"`
|
|
|
|
IfEmpty bool `yaml:"ifEmpty,omitempty"`
|
|
}
|
|
|
|
func (c FieldClearer) Filter(rn *RNode) (*RNode, error) {
|
|
if err := ErrorIfInvalid(rn, yaml.MappingNode); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for i := 0; i < len(rn.Content()); i += 2 {
|
|
// if name matches, remove these 2 elements from the list because
|
|
// they are treated as a fieldName/fieldValue pair.
|
|
if rn.Content()[i].Value == c.Name {
|
|
if c.IfEmpty {
|
|
if len(rn.Content()[i+1].Content) > 0 {
|
|
continue
|
|
}
|
|
}
|
|
|
|
// save the item we are about to remove
|
|
removed := NewRNode(rn.Content()[i+1])
|
|
if len(rn.YNode().Content) > i+2 {
|
|
l := len(rn.YNode().Content)
|
|
// remove from the middle of the list
|
|
rn.YNode().Content = rn.Content()[:i]
|
|
rn.YNode().Content = append(
|
|
rn.YNode().Content,
|
|
rn.Content()[i+2:l]...)
|
|
} else {
|
|
// remove from the end of the list
|
|
rn.YNode().Content = rn.Content()[:i]
|
|
}
|
|
|
|
// return the removed field name and value
|
|
return removed, nil
|
|
}
|
|
}
|
|
// nothing removed
|
|
return nil, nil
|
|
}
|
|
|
|
func MatchElement(field, value string) ElementMatcher {
|
|
return ElementMatcher{FieldName: field, FieldValue: value}
|
|
}
|
|
|
|
// ElementMatcher returns the first element from a Sequence matching the
|
|
// specified field's value.
|
|
type ElementMatcher struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// FieldName will attempt to match this field in each list element.
|
|
// Optional. Leave empty for lists of primitives (ScalarNode).
|
|
FieldName string `yaml:"name,omitempty"`
|
|
|
|
// FieldValue will attempt to match each element field to this value.
|
|
// For lists of primitives, this will be used to match the primitive value.
|
|
FieldValue string `yaml:"value,omitempty"`
|
|
|
|
// Create will create the Element if it is not found
|
|
Create *RNode `yaml:"create,omitempty"`
|
|
}
|
|
|
|
func (e ElementMatcher) Filter(rn *RNode) (*RNode, error) {
|
|
if err := ErrorIfInvalid(rn, yaml.SequenceNode); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// SequenceNode Content is a slice of ScalarNodes. Each ScalarNode has a
|
|
// YNode containing the primitive data.
|
|
if len(e.FieldName) == 0 {
|
|
for i := range rn.Content() {
|
|
if rn.Content()[i].Value == e.FieldValue {
|
|
return &RNode{value: rn.Content()[i]}, nil
|
|
}
|
|
}
|
|
if e.Create != nil {
|
|
return rn.Pipe(Append(e.Create.YNode()))
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// SequenceNode Content is a slice of MappingNodes. Each MappingNode has Content
|
|
// with a slice of key-value pairs containing the fields.
|
|
for i := range rn.Content() {
|
|
// cast the entry to a RNode so we can operate on it
|
|
elem := NewRNode(rn.Content()[i])
|
|
|
|
field, err := elem.Pipe(MatchField(e.FieldName, e.FieldValue))
|
|
if IsFoundOrError(field, err) {
|
|
return elem, err
|
|
}
|
|
}
|
|
|
|
// create the element
|
|
if e.Create != nil {
|
|
return rn.Pipe(Append(e.Create.YNode()))
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
func Get(name string) FieldMatcher {
|
|
return FieldMatcher{Name: name}
|
|
}
|
|
|
|
func MatchField(name, value string) FieldMatcher {
|
|
return FieldMatcher{Name: name, Value: NewScalarRNode(value)}
|
|
}
|
|
|
|
func Match(value string) FieldMatcher {
|
|
return FieldMatcher{Value: NewScalarRNode(value)}
|
|
}
|
|
|
|
// FieldMatcher returns the value of a named field or map entry.
|
|
type FieldMatcher struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Name of the field to return
|
|
Name string `yaml:"name,omitempty"`
|
|
|
|
// YNode of the field to return.
|
|
// Optional. Will only need to match field name if unset.
|
|
Value *RNode `yaml:"value,omitempty"`
|
|
|
|
StringValue string `yaml:"stringValue,omitempty"`
|
|
|
|
StringRegexValue string `yaml:"stringRegexValue,omitempty"`
|
|
|
|
// Create will cause the field to be created with this value
|
|
// if it is set.
|
|
Create *RNode `yaml:"create,omitempty"`
|
|
}
|
|
|
|
func (f FieldMatcher) Filter(rn *RNode) (*RNode, error) {
|
|
if f.StringValue != "" && f.Value == nil {
|
|
f.Value = NewScalarRNode(f.StringValue)
|
|
}
|
|
|
|
// never match nil or null fields
|
|
if IsMissingOrNull(rn) {
|
|
return nil, nil
|
|
}
|
|
|
|
if f.Name == "" {
|
|
if err := ErrorIfInvalid(rn, yaml.ScalarNode); err != nil {
|
|
return nil, err
|
|
}
|
|
switch {
|
|
case f.StringRegexValue != "":
|
|
// TODO(pwittrock): pre-compile this when unmarshalling and cache to a field
|
|
rg, err := regexp.Compile(f.StringRegexValue)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if match := rg.MatchString(rn.value.Value); match {
|
|
return rn, nil
|
|
}
|
|
return nil, nil
|
|
case rn.value.Value == f.Value.YNode().Value:
|
|
return rn, nil
|
|
default:
|
|
return nil, nil
|
|
}
|
|
}
|
|
|
|
if err := ErrorIfInvalid(rn, yaml.MappingNode); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for i := 0; i < len(rn.Content()); i = IncrementFieldIndex(i) {
|
|
isMatchingField := rn.Content()[i].Value == f.Name
|
|
if isMatchingField {
|
|
requireMatchFieldValue := f.Value != nil
|
|
if !requireMatchFieldValue || rn.Content()[i+1].Value == f.Value.YNode().Value {
|
|
return NewRNode(rn.Content()[i+1]), nil
|
|
}
|
|
}
|
|
}
|
|
|
|
if f.Create != nil {
|
|
return rn.Pipe(SetField(f.Name, f.Create))
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
// Lookup returns a PathGetter to lookup a field by its path.
|
|
func Lookup(path ...string) PathGetter {
|
|
return PathGetter{Path: path}
|
|
}
|
|
|
|
// Lookup returns a PathGetter to lookup a field by its path and create it if it doesn't already
|
|
// exist.
|
|
func LookupCreate(kind yaml.Kind, path ...string) PathGetter {
|
|
return PathGetter{Path: path, Create: kind}
|
|
}
|
|
|
|
// PathGetter returns the RNode under Path.
|
|
type PathGetter struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Path is a slice of parts leading to the RNode to lookup.
|
|
// Each path part may be one of:
|
|
// * FieldMatcher -- e.g. "spec"
|
|
// * Map Key -- e.g. "app.k8s.io/version"
|
|
// * List Entry -- e.g. "[name=nginx]" or "[=-jar]"
|
|
//
|
|
// Map Keys and Fields are equivalent.
|
|
// See FieldMatcher for more on Fields and Map Keys.
|
|
//
|
|
// List Entries are specified as map entry to match [fieldName=fieldValue].
|
|
// See Elem for more on List Entries.
|
|
//
|
|
// Examples:
|
|
// * spec.template.spec.container with matching name: [name=nginx]
|
|
// * spec.template.spec.container.argument matching a value: [=-jar]
|
|
Path []string `yaml:"path,omitempty"`
|
|
|
|
// Create will cause missing path parts to be created as they are walked.
|
|
//
|
|
// * The leaf Node (final path) will be created with a Kind matching Create
|
|
// * Intermediary Nodes will be created as either a MappingNodes or
|
|
// SequenceNodes as appropriate for each's Path location.
|
|
Create yaml.Kind `yaml:"create,omitempty"`
|
|
|
|
// Style is the style to apply to created value Nodes.
|
|
// Created key Nodes keep an unspecified Style.
|
|
Style yaml.Style `yaml:"style,omitempty"`
|
|
}
|
|
|
|
func (l PathGetter) Filter(rn *RNode) (*RNode, error) {
|
|
var err error
|
|
fieldPath := append([]string{}, rn.FieldPath()...)
|
|
match := rn
|
|
|
|
// iterate over path until encountering an error or missing value
|
|
l.Path = cleanPath(l.Path)
|
|
for i := range l.Path {
|
|
var part, nextPart string
|
|
part = l.Path[i]
|
|
if len(l.Path) > i+1 {
|
|
nextPart = l.Path[i+1]
|
|
}
|
|
if IsListIndex(part) {
|
|
match, err = l.doElem(match, part)
|
|
} else {
|
|
fieldPath = append(fieldPath, part)
|
|
match, err = l.doField(match, part, l.getKind(nextPart))
|
|
}
|
|
if IsMissingOrError(match, err) {
|
|
return nil, err
|
|
}
|
|
match.AppendToFieldPath(fieldPath...)
|
|
}
|
|
return match, nil
|
|
}
|
|
|
|
func (l PathGetter) doElem(rn *RNode, part string) (*RNode, error) {
|
|
var match *RNode
|
|
name, value, err := SplitIndexNameValue(part)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err)
|
|
}
|
|
if !IsCreate(l.Create) {
|
|
return rn.Pipe(MatchElement(name, value))
|
|
}
|
|
|
|
var elem *RNode
|
|
primitiveElement := len(name) == 0
|
|
if primitiveElement {
|
|
// append a ScalarNode
|
|
elem = NewScalarRNode(value)
|
|
elem.YNode().Style = l.Style
|
|
match = elem
|
|
} else {
|
|
// append a MappingNode
|
|
match = NewRNode(&yaml.Node{Kind: yaml.ScalarNode, Value: value, Style: l.Style})
|
|
elem = NewRNode(&yaml.Node{
|
|
Kind: yaml.MappingNode,
|
|
Content: []*yaml.Node{{Kind: yaml.ScalarNode, Value: name}, match.YNode()},
|
|
Style: l.Style,
|
|
})
|
|
}
|
|
// Append the Node
|
|
return rn.Pipe(ElementMatcher{FieldName: name, FieldValue: value, Create: elem})
|
|
}
|
|
|
|
func (l PathGetter) doField(
|
|
rn *RNode, name string, kind yaml.Kind) (*RNode, error) {
|
|
if !IsCreate(l.Create) {
|
|
return rn.Pipe(Get(name))
|
|
}
|
|
return rn.Pipe(FieldMatcher{Name: name, Create: &RNode{value: &yaml.Node{Kind: kind, Style: l.Style}}})
|
|
}
|
|
|
|
func (l PathGetter) getKind(nextPart string) yaml.Kind {
|
|
if IsListIndex(nextPart) {
|
|
// if nextPart is of the form [a=b], then it is an index into a Sequence
|
|
// so the current part must be a SequenceNode
|
|
return yaml.SequenceNode
|
|
}
|
|
if nextPart == "" {
|
|
// final name in the path, use the l.Create defined Kind
|
|
return l.Create
|
|
}
|
|
|
|
// non-sequence intermediate Node
|
|
return yaml.MappingNode
|
|
}
|
|
|
|
func SetField(name string, value *RNode) FieldSetter {
|
|
return FieldSetter{Name: name, Value: value}
|
|
}
|
|
|
|
func Set(value *RNode) FieldSetter {
|
|
return FieldSetter{Value: value}
|
|
}
|
|
|
|
// FieldSetter sets a field or map entry to a value.
|
|
type FieldSetter struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Name is the name of the field or key to lookup in a MappingNode.
|
|
// If Name is unspecified, and the input is a ScalarNode, FieldSetter will set the
|
|
// value on the ScalarNode.
|
|
Name string `yaml:"name,omitempty"`
|
|
|
|
// Value is the value to set.
|
|
// Optional if Kind is set.
|
|
Value *RNode `yaml:"value,omitempty"`
|
|
|
|
StringValue string `yaml:"stringValue,omitempty"`
|
|
|
|
// OverrideStyle can be set to override the style of the existing node
|
|
// when setting it. Otherwise, if an existing node is found, the style is
|
|
// retained.
|
|
OverrideStyle bool `yaml:"overrideStyle,omitempty"`
|
|
}
|
|
|
|
func (s FieldSetter) Filter(rn *RNode) (*RNode, error) {
|
|
if s.StringValue != "" && s.Value == nil {
|
|
s.Value = NewScalarRNode(s.StringValue)
|
|
}
|
|
|
|
if s.Name == "" {
|
|
if err := ErrorIfInvalid(rn, yaml.ScalarNode); err != nil {
|
|
return rn, err
|
|
}
|
|
// only apply the style if there is not an existing style
|
|
// or we want to override it
|
|
if !s.OverrideStyle || s.Value.YNode().Style == 0 {
|
|
// keep the original style if it exists
|
|
s.Value.YNode().Style = rn.YNode().Style
|
|
}
|
|
rn.SetYNode(s.Value.YNode())
|
|
return rn, nil
|
|
}
|
|
|
|
// Clear the field if it is empty, or explicitly null
|
|
if s.Value == nil || IsNull(s.Value) {
|
|
return rn.Pipe(Clear(s.Name))
|
|
}
|
|
|
|
field, err := rn.Pipe(FieldMatcher{Name: s.Name})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if field != nil {
|
|
// only apply the style if there is not an existing style
|
|
// or we want to override it
|
|
if !s.OverrideStyle || field.YNode().Style == 0 {
|
|
// keep the original style if it exists
|
|
s.Value.YNode().Style = field.YNode().Style
|
|
}
|
|
// need to def ref the Node since field is ephemeral
|
|
field.SetYNode(s.Value.YNode())
|
|
return field, nil
|
|
}
|
|
|
|
// create the field
|
|
rn.YNode().Content = append(rn.YNode().Content,
|
|
&yaml.Node{Kind: yaml.ScalarNode, Value: s.Name},
|
|
s.Value.YNode())
|
|
return s.Value, nil
|
|
}
|
|
|
|
// Tee calls the provided Filters, and returns its argument rather than the result
|
|
// of the filters.
|
|
// May be used to fork sub-filters from a call.
|
|
// e.g. locate field, set value; locate another field, set another value
|
|
func Tee(filters ...Filter) Filter {
|
|
return TeePiper{Filters: filters}
|
|
}
|
|
|
|
// TeePiper Calls a slice of Filters and returns its input.
|
|
// May be used to fork sub-filters from a call.
|
|
// e.g. locate field, set value; locate another field, set another value
|
|
type TeePiper struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// Filters are the set of Filters run by TeePiper.
|
|
Filters []Filter `yaml:"filters,omitempty"`
|
|
}
|
|
|
|
func (t TeePiper) Filter(rn *RNode) (*RNode, error) {
|
|
_, err := rn.Pipe(t.Filters...)
|
|
return rn, err
|
|
}
|
|
|
|
// IsCreate returns true if kind is specified
|
|
func IsCreate(kind yaml.Kind) bool {
|
|
return kind != 0
|
|
}
|
|
|
|
// IsMissingOrError returns true if rn is NOT found or err is non-nil
|
|
func IsMissingOrError(rn *RNode, err error) bool {
|
|
return rn == nil || err != nil
|
|
}
|
|
|
|
// IsFoundOrError returns true if rn is found or err is non-nil
|
|
func IsFoundOrError(rn *RNode, err error) bool {
|
|
return rn != nil || err != nil
|
|
}
|
|
|
|
func ErrorIfAnyInvalidAndNonNull(kind yaml.Kind, rn ...*RNode) error {
|
|
for i := range rn {
|
|
if IsEmpty(rn[i]) {
|
|
continue
|
|
}
|
|
if err := ErrorIfInvalid(rn[i], kind); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
var nodeTypeIndex = map[yaml.Kind]string{
|
|
yaml.SequenceNode: "SequenceNode",
|
|
yaml.MappingNode: "MappingNode",
|
|
yaml.ScalarNode: "ScalarNode",
|
|
yaml.DocumentNode: "DocumentNode",
|
|
yaml.AliasNode: "AliasNode",
|
|
}
|
|
|
|
func ErrorIfInvalid(rn *RNode, kind yaml.Kind) error {
|
|
if rn == nil || rn.YNode() == nil || IsNull(rn) {
|
|
// node has no type, pass validation
|
|
return nil
|
|
}
|
|
|
|
if rn.YNode().Kind != kind {
|
|
s, _ := rn.String()
|
|
return errors.Errorf(
|
|
"wrong Node Kind for %s expected: %v was %v: value: {%s}",
|
|
strings.Join(rn.FieldPath(), "."),
|
|
nodeTypeIndex[kind], nodeTypeIndex[rn.YNode().Kind], strings.TrimSpace(s))
|
|
}
|
|
|
|
if kind == yaml.MappingNode {
|
|
if len(rn.YNode().Content)%2 != 0 {
|
|
return errors.Errorf(
|
|
"yaml MappingNodes must have even length contents: %v", spew.Sdump(rn))
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// IsListIndex returns true if p is an index into a Val.
|
|
// e.g. [fieldName=fieldValue]
|
|
// e.g. [=primitiveValue]
|
|
func IsListIndex(p string) bool {
|
|
return strings.HasPrefix(p, "[") && strings.HasSuffix(p, "]")
|
|
}
|
|
|
|
// SplitIndexNameValue splits a lookup part Val index into the field name
|
|
// and field value to match.
|
|
// e.g. splits [name=nginx] into (name, nginx)
|
|
// e.g. splits [=-jar] into ("", jar)
|
|
func SplitIndexNameValue(p string) (string, string, error) {
|
|
elem := strings.TrimSuffix(p, "]")
|
|
elem = strings.TrimPrefix(elem, "[")
|
|
parts := strings.SplitN(elem, "=", 2)
|
|
if len(parts) == 1 {
|
|
return "", "", fmt.Errorf("list path element must contain fieldName=fieldValue for element to match")
|
|
}
|
|
return parts[0], parts[1], nil
|
|
}
|
|
|
|
// IncrementFieldIndex increments i to point to the next field name element in
|
|
// a slice of Contents.
|
|
func IncrementFieldIndex(i int) int {
|
|
return i + 2
|
|
}
|