mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-06-30 01:46:23 +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
143 lines
3.6 KiB
Go
143 lines
3.6 KiB
Go
// Copyright 2019 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// Package merge contains libraries for merging fields from one RNode to another
|
|
// RNode
|
|
package merge2
|
|
|
|
import (
|
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
"sigs.k8s.io/kustomize/kyaml/yaml/walk"
|
|
)
|
|
|
|
// Merge merges fields from src into dest.
|
|
func Merge(src, dest *yaml.RNode) (*yaml.RNode, error) {
|
|
return walk.Walker{Sources: []*yaml.RNode{dest, src}, Visitor: Merger{}}.Walk()
|
|
}
|
|
|
|
// Merge parses the arguments, and merges fields from srcStr into destStr.
|
|
func MergeStrings(srcStr, destStr string) (string, error) {
|
|
src, err := yaml.Parse(srcStr)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
dest, err := yaml.Parse(destStr)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
result, err := Merge(src, dest)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return result.String()
|
|
}
|
|
|
|
type Merger struct {
|
|
// for forwards compatibility when new functions are added to the interface
|
|
}
|
|
|
|
var _ walk.Visitor = Merger{}
|
|
|
|
func (m Merger) VisitMap(nodes walk.Sources) (*yaml.RNode, error) {
|
|
if err := m.SetComments(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := m.SetStyle(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
if yaml.IsEmpty(nodes.Dest()) {
|
|
// Add
|
|
return nodes.Origin(), nil
|
|
}
|
|
if yaml.IsNull(nodes.Origin()) {
|
|
// clear the value
|
|
return walk.ClearNode, nil
|
|
}
|
|
// Recursively Merge dest
|
|
return nodes.Dest(), nil
|
|
}
|
|
|
|
func (m Merger) VisitScalar(nodes walk.Sources) (*yaml.RNode, error) {
|
|
if err := m.SetComments(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := m.SetStyle(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
// Override value
|
|
if nodes.Origin() != nil {
|
|
return nodes.Origin(), nil
|
|
}
|
|
// Keep
|
|
return nodes.Dest(), nil
|
|
}
|
|
|
|
func (m Merger) VisitList(nodes walk.Sources, kind walk.ListKind) (*yaml.RNode, error) {
|
|
if err := m.SetComments(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := m.SetStyle(nodes); err != nil {
|
|
return nil, err
|
|
}
|
|
if kind == walk.NonAssociateList {
|
|
// Override value
|
|
if nodes.Origin() != nil {
|
|
return nodes.Origin(), nil
|
|
}
|
|
// Keep
|
|
return nodes.Dest(), nil
|
|
}
|
|
|
|
// Add
|
|
if yaml.IsEmpty(nodes.Dest()) {
|
|
return nodes.Origin(), nil
|
|
}
|
|
// Clear
|
|
if yaml.IsNull(nodes.Origin()) {
|
|
return walk.ClearNode, nil
|
|
}
|
|
// Recursively Merge dest
|
|
return nodes.Dest(), nil
|
|
}
|
|
|
|
func (m Merger) SetStyle(sources walk.Sources) error {
|
|
source := sources.Origin()
|
|
dest := sources.Dest()
|
|
if dest == nil || dest.YNode() == nil || source == nil || source.YNode() == nil {
|
|
// avoid panic
|
|
return nil
|
|
}
|
|
|
|
// copy the style from the source.
|
|
// special case: if the dest was an empty map or seq, then it probably had
|
|
// folded style applied, but we actually want to keep the style of the origin
|
|
// in this case (even if it was the default). otherwise the merged elements
|
|
// will get folded even though this probably isn't what is desired.
|
|
if dest.YNode().Kind != yaml.ScalarNode && len(dest.YNode().Content) == 0 {
|
|
dest.YNode().Style = source.YNode().Style
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SetComments copies the dest comments to the source comments if they are present
|
|
// on the source.
|
|
func (m Merger) SetComments(sources walk.Sources) error {
|
|
source := sources.Origin()
|
|
dest := sources.Dest()
|
|
if dest == nil || dest.YNode() == nil || source == nil || source.YNode() == nil {
|
|
// avoid panic
|
|
return nil
|
|
}
|
|
if source != nil && source.YNode().FootComment != "" {
|
|
dest.YNode().FootComment = source.YNode().FootComment
|
|
}
|
|
if source != nil && source.YNode().HeadComment != "" {
|
|
dest.YNode().HeadComment = source.YNode().HeadComment
|
|
}
|
|
if source != nil && source.YNode().LineComment != "" {
|
|
dest.YNode().LineComment = source.YNode().LineComment
|
|
}
|
|
return nil
|
|
}
|