mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-06-30 01:46:23 +00:00
Compare commits
1 Commits
api/v0.7.3
...
bugfix-kya
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7432392a51 |
63
Makefile
63
Makefile
@@ -4,7 +4,7 @@
|
|||||||
# Makefile for kustomize CLI and API.
|
# Makefile for kustomize CLI and API.
|
||||||
|
|
||||||
MYGOBIN := $(shell go env GOPATH)/bin
|
MYGOBIN := $(shell go env GOPATH)/bin
|
||||||
SHELL := /usr/bin/env bash
|
SHELL := /bin/bash
|
||||||
export PATH := $(MYGOBIN):$(PATH)
|
export PATH := $(MYGOBIN):$(PATH)
|
||||||
MODULES := '"cmd/config" "api/" "kustomize/" "kyaml/"'
|
MODULES := '"cmd/config" "api/" "kustomize/" "kyaml/"'
|
||||||
|
|
||||||
@@ -26,21 +26,22 @@ verify-kustomize: \
|
|||||||
lint-kustomize \
|
lint-kustomize \
|
||||||
test-unit-kustomize-all \
|
test-unit-kustomize-all \
|
||||||
test-examples-kustomize-against-HEAD \
|
test-examples-kustomize-against-HEAD \
|
||||||
test-examples-kustomize-against-3.9.2 \
|
test-examples-kustomize-against-3.8.6
|
||||||
test-examples-kustomize-against-3.8.9
|
|
||||||
|
|
||||||
# The following target referenced by a file in
|
# The following target referenced by a file in
|
||||||
# https://github.com/kubernetes/test-infra/tree/master/config/jobs/kubernetes-sigs/kustomize
|
# https://github.com/kubernetes/test-infra/tree/master/config/jobs/kubernetes-sigs/kustomize
|
||||||
.PHONY: prow-presubmit-check
|
.PHONY: prow-presubmit-check
|
||||||
prow-presubmit-check: \
|
prow-presubmit-check: \
|
||||||
lint-kustomize \
|
lint-kustomize \
|
||||||
test-multi-module \
|
|
||||||
test-unit-kustomize-all \
|
test-unit-kustomize-all \
|
||||||
test-unit-cmd-all \
|
test-unit-cmd-all \
|
||||||
test-go-mod \
|
test-go-mod \
|
||||||
test-examples-kustomize-against-HEAD \
|
test-examples-kustomize-against-HEAD \
|
||||||
test-examples-kustomize-against-3.9.2 \
|
test-examples-kustomize-against-3.8.6
|
||||||
test-examples-kustomize-against-3.8.9
|
|
||||||
|
# test-multi-module \
|
||||||
|
# Temporarily removed from prow-presubmit-check
|
||||||
|
# See https://github.com/kubernetes-sigs/kustomize/issues/3191
|
||||||
|
|
||||||
.PHONY: verify-kustomize-e2e
|
.PHONY: verify-kustomize-e2e
|
||||||
verify-kustomize-e2e: test-examples-e2e-kustomize
|
verify-kustomize-e2e: test-examples-e2e-kustomize
|
||||||
@@ -192,27 +193,24 @@ $(pGen)/%.go: $(MYGOBIN)/pluginator
|
|||||||
.PHONY: generate-kustomize-builtin-plugins
|
.PHONY: generate-kustomize-builtin-plugins
|
||||||
generate-kustomize-builtin-plugins: $(builtinplugins)
|
generate-kustomize-builtin-plugins: $(builtinplugins)
|
||||||
|
|
||||||
.PHONY: build-kustomize-external-go-plugin
|
.PHONY: kustomize-external-go-plugin-build
|
||||||
build-kustomize-external-go-plugin:
|
kustomize-external-go-plugin-build:
|
||||||
./hack/buildExternalGoPlugins.sh ./plugin
|
./hack/buildExternalGoPlugins.sh ./plugin
|
||||||
|
|
||||||
.PHONY: clean-kustomize-external-go-plugin
|
.PHONY: kustomize-external-go-plugin-clean
|
||||||
clean-kustomize-external-go-plugin:
|
kustomize-external-go-plugin-clean:
|
||||||
./hack/buildExternalGoPlugins.sh ./plugin clean
|
./hack/buildExternalGoPlugins.sh ./plugin clean
|
||||||
|
|
||||||
### End kustomize plugin rules.
|
### End kustomize plugin rules.
|
||||||
|
|
||||||
.PHONY: lint-kustomize
|
.PHONY: lint-kustomize
|
||||||
lint-kustomize: install-tools $(builtinplugins)
|
lint-kustomize: install-tools $(builtinplugins)
|
||||||
cd api; $(MYGOBIN)/golangci-lint-kustomize \
|
cd api; \
|
||||||
-c ../.golangci-kustomize.yml \
|
$(MYGOBIN)/golangci-lint-kustomize -c ../.golangci-kustomize.yml run ./...
|
||||||
run ./...
|
cd kustomize; \
|
||||||
cd kustomize; $(MYGOBIN)/golangci-lint-kustomize \
|
$(MYGOBIN)/golangci-lint-kustomize -c ../.golangci-kustomize.yml run ./...
|
||||||
-c ../.golangci-kustomize.yml \
|
cd cmd/pluginator; \
|
||||||
run ./...
|
$(MYGOBIN)/golangci-lint-kustomize -c ../../.golangci-kustomize.yml run ./...
|
||||||
cd cmd/pluginator; $(MYGOBIN)/golangci-lint-kustomize \
|
|
||||||
-c ../../.golangci-kustomize.yml \
|
|
||||||
run ./...
|
|
||||||
|
|
||||||
# Used to add non-default compilation flags when experimenting with
|
# Used to add non-default compilation flags when experimenting with
|
||||||
# plugin-to-api compatibility checks.
|
# plugin-to-api compatibility checks.
|
||||||
@@ -220,10 +218,6 @@ lint-kustomize: install-tools $(builtinplugins)
|
|||||||
build-kustomize-api: $(builtinplugins)
|
build-kustomize-api: $(builtinplugins)
|
||||||
cd api; go build ./...
|
cd api; go build ./...
|
||||||
|
|
||||||
.PHONY: generate-kustomize-api
|
|
||||||
generate-kustomize-api:
|
|
||||||
cd api; go generate ./...
|
|
||||||
|
|
||||||
.PHONY: test-unit-kustomize-api
|
.PHONY: test-unit-kustomize-api
|
||||||
test-unit-kustomize-api: build-kustomize-api
|
test-unit-kustomize-api: build-kustomize-api
|
||||||
cd api; go test ./... -ldflags "-X sigs.k8s.io/kustomize/api/provenance.version=v444.333.222"
|
cd api; go test ./... -ldflags "-X sigs.k8s.io/kustomize/api/provenance.version=v444.333.222"
|
||||||
@@ -276,12 +270,17 @@ test-examples-kustomize-against-HEAD: $(MYGOBIN)/kustomize $(MYGOBIN)/mdrip
|
|||||||
./hack/testExamplesAgainstKustomize.sh HEAD
|
./hack/testExamplesAgainstKustomize.sh HEAD
|
||||||
|
|
||||||
.PHONY:
|
.PHONY:
|
||||||
test-examples-kustomize-against-3.9.2: $(MYGOBIN)/mdrip
|
test-examples-kustomize-against-3.8.6: $(MYGOBIN)/mdrip
|
||||||
./hack/testExamplesAgainstKustomize.sh v3.9.2
|
( \
|
||||||
|
set -e; \
|
||||||
.PHONY:
|
tag=v3.8.6; \
|
||||||
test-examples-kustomize-against-3.8.9: $(MYGOBIN)/mdrip
|
/bin/rm -f $(MYGOBIN)/kustomize; \
|
||||||
./hack/testExamplesAgainstKustomize.sh v3.8.9
|
echo "Installing kustomize $$tag."; \
|
||||||
|
GO111MODULE=on go get sigs.k8s.io/kustomize/kustomize/v3@$${tag}; \
|
||||||
|
./hack/testExamplesAgainstKustomize.sh $$tag; \
|
||||||
|
echo "Reinstalling kustomize from HEAD."; \
|
||||||
|
cd kustomize; go install .; \
|
||||||
|
)
|
||||||
|
|
||||||
# linux only.
|
# linux only.
|
||||||
# This is for testing an example plugin that
|
# This is for testing an example plugin that
|
||||||
@@ -355,15 +354,13 @@ $(MYGOBIN)/gh:
|
|||||||
)
|
)
|
||||||
|
|
||||||
.PHONY: clean
|
.PHONY: clean
|
||||||
clean: clean-kustomize-external-go-plugin
|
clean: kustomize-external-go-plugin-clean
|
||||||
go clean --cache
|
go clean --cache
|
||||||
rm -f $(builtinplugins)
|
rm -f $(builtinplugins)
|
||||||
|
rm -f $(MYGOBIN)/pluginator
|
||||||
rm -f $(MYGOBIN)/kustomize
|
rm -f $(MYGOBIN)/kustomize
|
||||||
rm -f $(MYGOBIN)/golangci-lint-kustomize
|
rm -f $(MYGOBIN)/golangci-lint-kustomize
|
||||||
|
|
||||||
# Handle pluginator manually.
|
|
||||||
# rm -f $(MYGOBIN)/pluginator
|
|
||||||
|
|
||||||
# Nuke the site from orbit. It's the only way to be sure.
|
# Nuke the site from orbit. It's the only way to be sure.
|
||||||
.PHONY: nuke
|
.PHONY: nuke
|
||||||
nuke: clean
|
nuke: clean
|
||||||
|
|||||||
21
README.md
21
README.md
@@ -20,20 +20,15 @@ This tool is sponsored by [sig-cli] ([KEP]).
|
|||||||
|
|
||||||
## kubectl integration
|
## kubectl integration
|
||||||
|
|
||||||
The kustomize build flow at [v2.0.3] was added
|
Since [v1.14][kubectl announcement] the kustomize build system has been included in kubectl.
|
||||||
to [kubectl v1.14][kubectl announcement]. The kustomize
|
|
||||||
flow in kubectl has remained frozen at v2.0.3 while work
|
|
||||||
to extract kubectl from the k/k repo, and work to remove
|
|
||||||
kustomize's dependence on core k/k code ([#2506]) has proceeded.
|
|
||||||
The reintegration effort is tracked in [#1500] (and its blocking
|
|
||||||
issues).
|
|
||||||
|
|
||||||
[v2.0.3]: /../../tree/v2.0.3
|
| kubectl version | kustomize version |
|
||||||
[#2506]: https://github.com/kubernetes-sigs/kustomize/issues/2506
|
|---------|--------|
|
||||||
[#1500]: https://github.com/kubernetes-sigs/kustomize/issues/1500
|
| v1.16.0 | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
| v1.15.x | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
| v1.14.x | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
|
||||||
For examples and guides for using the kubectl integration please
|
For examples and guides for using the kubectl integration please see the [kubectl book] or the [kubernetes documentation].
|
||||||
see the [kubectl book] or the [kubernetes documentation].
|
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@@ -153,7 +148,7 @@ is governed by the [Kubernetes Code of Conduct].
|
|||||||
[imageBase]: docs/images/base.jpg
|
[imageBase]: docs/images/base.jpg
|
||||||
[imageOverlay]: docs/images/overlay.jpg
|
[imageOverlay]: docs/images/overlay.jpg
|
||||||
[kubectl announcement]: https://kubernetes.io/blog/2019/03/25/kubernetes-1-14-release-announcement
|
[kubectl announcement]: https://kubernetes.io/blog/2019/03/25/kubernetes-1-14-release-announcement
|
||||||
[kubectl book]: https://kubectl.docs.kubernetes.io/guides/introduction/kustomize/
|
[kubectl book]: https://kubectl.docs.kubernetes.io/pages/app_customization/introduction.html
|
||||||
[kubernetes documentation]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/kustomization/
|
[kubernetes documentation]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/kustomization/
|
||||||
[kubernetes style]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kubernetes-style-object
|
[kubernetes style]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kubernetes-style-object
|
||||||
[kustomization]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kustomization
|
[kustomization]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kustomization
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/filters/annotations"
|
"sigs.k8s.io/kustomize/api/filters/annotations"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -24,14 +25,11 @@ func (p *AnnotationsTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *AnnotationsTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *AnnotationsTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if len(p.Annotations) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
err := r.ApplyFilter(annotations.Filter{
|
err := filtersutil.ApplyToJSON(annotations.Filter{
|
||||||
Annotations: p.Annotations,
|
Annotations: p.Annotations,
|
||||||
FsSlice: p.FieldSpecs,
|
FsSlice: p.FieldSpecs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ func (p *HashTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
res.StorePreviousId()
|
|
||||||
res.SetName(fmt.Sprintf("%s-%s", res.GetName(), h))
|
res.SetName(fmt.Sprintf("%s-%s", res.GetName(), h))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,15 +6,12 @@ package builtins
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path/filepath"
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/imdario/mergo"
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
@@ -51,7 +48,7 @@ func (p *HelmChartInflationGeneratorPlugin) Config(h *resmap.PluginHelpers, conf
|
|||||||
return fmt.Errorf("chartName cannot be empty")
|
return fmt.Errorf("chartName cannot be empty")
|
||||||
}
|
}
|
||||||
if p.ChartHome == "" {
|
if p.ChartHome == "" {
|
||||||
p.ChartHome = filepath.Join(p.tmpDir, "chart")
|
p.ChartHome = path.Join(p.tmpDir, "chart")
|
||||||
}
|
}
|
||||||
if p.ChartRepoName == "" {
|
if p.ChartRepoName == "" {
|
||||||
p.ChartRepoName = "stable"
|
p.ChartRepoName = "stable"
|
||||||
@@ -60,13 +57,10 @@ func (p *HelmChartInflationGeneratorPlugin) Config(h *resmap.PluginHelpers, conf
|
|||||||
p.HelmBin = "helm"
|
p.HelmBin = "helm"
|
||||||
}
|
}
|
||||||
if p.HelmHome == "" {
|
if p.HelmHome == "" {
|
||||||
p.HelmHome = filepath.Join(p.tmpDir, ".helm")
|
p.HelmHome = path.Join(p.tmpDir, ".helm")
|
||||||
}
|
}
|
||||||
if p.Values == "" {
|
if p.Values == "" {
|
||||||
p.Values = filepath.Join(p.ChartHome, p.ChartName, "values.yaml")
|
p.Values = path.Join(p.ChartHome, p.ChartName, "values.yaml")
|
||||||
}
|
|
||||||
if p.ValuesMerge == "" {
|
|
||||||
p.ValuesMerge = "override"
|
|
||||||
}
|
}
|
||||||
// runHelmCommand will run `helm` command with args provided. Return stdout
|
// runHelmCommand will run `helm` command with args provided. Return stdout
|
||||||
// and error if there is any.
|
// and error if there is any.
|
||||||
@@ -94,94 +88,6 @@ func (p *HelmChartInflationGeneratorPlugin) Config(h *resmap.PluginHelpers, conf
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EncodeValues for writing
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) EncodeValues(w io.Writer) error {
|
|
||||||
d, err := yaml.Marshal(p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, err = w.Write(d)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// useValuesLocal process (merge) inflator config provided values with chart default values.yaml
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) useValuesLocal() error {
|
|
||||||
// not override, merge, none
|
|
||||||
if !(p.ValuesMerge == "none" || p.ValuesMerge == "no" || p.ValuesMerge == "false") {
|
|
||||||
var pValues []byte
|
|
||||||
var err error
|
|
||||||
|
|
||||||
if filepath.IsAbs(p.Values) {
|
|
||||||
pValues, err = ioutil.ReadFile(p.Values)
|
|
||||||
} else {
|
|
||||||
pValues, err = p.h.Loader().Load(p.Values)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
chValues := make(map[string]interface{})
|
|
||||||
err = yaml.Unmarshal(pValues, &chValues)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if p.ValuesMerge == "override" {
|
|
||||||
err = mergo.Merge(&chValues, p.ValuesLocal, mergo.WithOverride)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if p.ValuesMerge == "merge" {
|
|
||||||
err = mergo.Merge(&chValues, p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
p.ValuesLocal = chValues
|
|
||||||
}
|
|
||||||
b, err := yaml.Marshal(p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
path, err := p.writeValuesBytes(b)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Values = path
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// copyValues will copy the relative values file into the temp directory
|
|
||||||
// to avoid messing up with CWD.
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) copyValues() error {
|
|
||||||
// only copy when the values path is not absolute
|
|
||||||
if filepath.IsAbs(p.Values) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
// we must use use loader to read values file
|
|
||||||
b, err := p.h.Loader().Load(p.Values)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
path, err := p.writeValuesBytes(b)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Values = path
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) writeValuesBytes(b []byte) (string, error) {
|
|
||||||
path := filepath.Join(p.ChartHome, p.ChartName, "kustomize-values.yaml")
|
|
||||||
err := ioutil.WriteFile(path, b, 0644)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return path, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate implements generator
|
// Generate implements generator
|
||||||
func (p *HelmChartInflationGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
func (p *HelmChartInflationGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
||||||
// cleanup
|
// cleanup
|
||||||
@@ -198,20 +104,6 @@ func (p *HelmChartInflationGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// inflator config valuesLocal
|
|
||||||
if len(p.ValuesLocal) > 0 {
|
|
||||||
err := p.useValuesLocal()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
err := p.copyValues()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// render the charts
|
// render the charts
|
||||||
stdout, err := p.runHelmCommand(p.getTemplateCommandArgs())
|
stdout, err := p.runHelmCommand(p.getTemplateCommandArgs())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -226,14 +118,13 @@ func (p *HelmChartInflationGeneratorPlugin) getTemplateCommandArgs() []string {
|
|||||||
if p.ReleaseName != "" {
|
if p.ReleaseName != "" {
|
||||||
args = append(args, p.ReleaseName)
|
args = append(args, p.ReleaseName)
|
||||||
}
|
}
|
||||||
args = append(args, filepath.Join(p.ChartHome, p.ChartName))
|
args = append(args, path.Join(p.ChartHome, p.ChartName))
|
||||||
if p.ReleaseNamespace != "" {
|
if p.ReleaseNamespace != "" {
|
||||||
args = append(args, "--namespace", p.ReleaseNamespace)
|
args = append(args, "--namespace", p.ReleaseNamespace)
|
||||||
}
|
}
|
||||||
if p.Values != "" {
|
if p.Values != "" {
|
||||||
args = append(args, "--values", p.Values)
|
args = append(args, "--values", p.Values)
|
||||||
}
|
}
|
||||||
args = append(args, p.ExtraArgs...)
|
|
||||||
return args
|
return args
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -256,7 +147,7 @@ func (p *HelmChartInflationGeneratorPlugin) getPullCommandArgs() []string {
|
|||||||
// checkLocalChart will return true if the chart does exist in
|
// checkLocalChart will return true if the chart does exist in
|
||||||
// local chart home.
|
// local chart home.
|
||||||
func (p *HelmChartInflationGeneratorPlugin) checkLocalChart() bool {
|
func (p *HelmChartInflationGeneratorPlugin) checkLocalChart() bool {
|
||||||
path := filepath.Join(p.ChartHome, p.ChartName)
|
path := path.Join(p.ChartHome, p.ChartName)
|
||||||
s, err := os.Stat(path)
|
s, err := os.Stat(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false
|
return false
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/filters/imagetag"
|
"sigs.k8s.io/kustomize/api/filters/imagetag"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -31,17 +32,17 @@ func (p *ImageTagTransformerPlugin) Config(
|
|||||||
func (p *ImageTagTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *ImageTagTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
// traverse all fields at first
|
// traverse all fields at first
|
||||||
err := r.ApplyFilter(imagetag.LegacyFilter{
|
err := filtersutil.ApplyToJSON(imagetag.LegacyFilter{
|
||||||
ImageTag: p.ImageTag,
|
ImageTag: p.ImageTag,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// then use user specified field specs
|
// then use user specified field specs
|
||||||
err = r.ApplyFilter(imagetag.Filter{
|
err = filtersutil.ApplyToJSON(imagetag.Filter{
|
||||||
ImageTag: p.ImageTag,
|
ImageTag: p.ImageTag,
|
||||||
FsSlice: p.FieldSpecs,
|
FsSlice: p.FieldSpecs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/filters/labels"
|
"sigs.k8s.io/kustomize/api/filters/labels"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -24,14 +25,11 @@ func (p *LabelTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *LabelTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *LabelTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if len(p.Labels) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
err := r.ApplyFilter(labels.Filter{
|
err := filtersutil.ApplyToJSON(labels.Filter{
|
||||||
Labels: p.Labels,
|
Labels: p.Labels,
|
||||||
FsSlice: p.FieldSpecs,
|
FsSlice: p.FieldSpecs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/filters/namespace"
|
"sigs.k8s.io/kustomize/api/filters/namespace"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -34,11 +35,10 @@ func (p *NamespaceTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
// Don't mutate empty objects?
|
// Don't mutate empty objects?
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
r.StorePreviousId()
|
err := filtersutil.ApplyToJSON(namespace.Filter{
|
||||||
err := r.ApplyFilter(namespace.Filter{
|
|
||||||
Namespace: p.Namespace,
|
Namespace: p.Namespace,
|
||||||
FsSlice: p.FieldSpecs,
|
FsSlice: p.FieldSpecs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -78,9 +79,9 @@ func (p *PatchJson6902TransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for _, res := range resources {
|
for _, res := range resources {
|
||||||
err = res.ApplyFilter(patchjson6902.Filter{
|
err = filtersutil.ApplyToJSON(patchjson6902.Filter{
|
||||||
Patch: p.JsonOp,
|
Patch: p.JsonOp,
|
||||||
})
|
}, res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,14 +5,19 @@ package builtins
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"sigs.k8s.io/kustomize/api/filters/patchstrategicmerge"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
type PatchStrategicMergeTransformerPlugin struct {
|
type PatchStrategicMergeTransformerPlugin struct {
|
||||||
|
h *resmap.PluginHelpers
|
||||||
loadedPatches []*resource.Resource
|
loadedPatches []*resource.Resource
|
||||||
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
||||||
Patches string `json:"patches,omitempty" yaml:"patches,omitempty"`
|
Patches string `json:"patches,omitempty" yaml:"patches,omitempty"`
|
||||||
@@ -20,6 +25,7 @@ type PatchStrategicMergeTransformerPlugin struct {
|
|||||||
|
|
||||||
func (p *PatchStrategicMergeTransformerPlugin) Config(
|
func (p *PatchStrategicMergeTransformerPlugin) Config(
|
||||||
h *resmap.PluginHelpers, c []byte) (err error) {
|
h *resmap.PluginHelpers, c []byte) (err error) {
|
||||||
|
p.h = h
|
||||||
err = yaml.Unmarshal(c, p)
|
err = yaml.Unmarshal(c, p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -29,18 +35,13 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
if len(p.Paths) != 0 {
|
if len(p.Paths) != 0 {
|
||||||
for _, onePath := range p.Paths {
|
for _, onePath := range p.Paths {
|
||||||
// The following oddly attempts to interpret a path string as an
|
res, err := p.h.ResmapFactory().RF().SliceFromBytes([]byte(onePath))
|
||||||
// actual patch (instead of as a path to a file containing a patch).
|
|
||||||
// All tests pass if this code is commented out. This code should
|
|
||||||
// be deleted; the user should use the Patches field which
|
|
||||||
// exists for this purpose (inline patch declaration).
|
|
||||||
res, err := h.ResmapFactory().RF().SliceFromBytes([]byte(onePath))
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.loadedPatches = append(p.loadedPatches, res...)
|
p.loadedPatches = append(p.loadedPatches, res...)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
res, err = h.ResmapFactory().RF().SliceFromPatches(
|
res, err = p.h.ResmapFactory().RF().SliceFromPatches(
|
||||||
h.Loader(), []types.PatchStrategicMerge{onePath})
|
p.h.Loader(), []types.PatchStrategicMerge{onePath})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -48,7 +49,7 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if p.Patches != "" {
|
if p.Patches != "" {
|
||||||
res, err := h.ResmapFactory().RF().SliceFromBytes([]byte(p.Patches))
|
res, err := p.h.ResmapFactory().RF().SliceFromBytes([]byte(p.Patches))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -59,24 +60,57 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
return fmt.Errorf(
|
return fmt.Errorf(
|
||||||
"patch appears to be empty; files=%v, Patch=%s", p.Paths, p.Patches)
|
"patch appears to be empty; files=%v, Patch=%s", p.Paths, p.Patches)
|
||||||
}
|
}
|
||||||
// Merge the patches, looking for conflicts.
|
return err
|
||||||
_, err = h.ResmapFactory().ConflatePatches(p.loadedPatches)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchStrategicMergeTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *PatchStrategicMergeTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
for _, patch := range p.loadedPatches {
|
patches, err := p.h.ResmapFactory().Merge(p.loadedPatches)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, patch := range patches.Resources() {
|
||||||
target, err := m.GetById(patch.OrgId())
|
target, err := m.GetById(patch.OrgId())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err = m.ApplySmPatch(
|
patchCopy := patch.DeepCopy()
|
||||||
resource.MakeIdSet([]*resource.Resource{target}), patch); err != nil {
|
patchCopy.SetName(target.GetName())
|
||||||
|
patchCopy.SetNamespace(target.GetNamespace())
|
||||||
|
patchCopy.SetGvk(target.GetGvk())
|
||||||
|
node, err := filtersutil.GetRNode(patchCopy)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
err = filtersutil.ApplyToJSON(patchstrategicmerge.Filter{
|
||||||
|
Patch: node,
|
||||||
|
}, target)
|
||||||
|
if err != nil {
|
||||||
|
// Check for an error string from UnmarshalJSON that's indicative
|
||||||
|
// of an object that's missing basic KRM fields, and thus may have been
|
||||||
|
// entirely deleted (an acceptable outcome). This error handling should
|
||||||
|
// be deleted along with use of ResMap and apimachinery functions like
|
||||||
|
// UnmarshalJSON.
|
||||||
|
if !strings.Contains(err.Error(), "Object 'Kind' is missing") {
|
||||||
|
// Some unknown error, let it through.
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !target.IsEmpty() {
|
||||||
|
return errors.Wrapf(
|
||||||
|
err, "with unexpectedly non-empty object map of size %d",
|
||||||
|
len(target.Map()))
|
||||||
|
}
|
||||||
|
// Fall through to handle deleted object.
|
||||||
|
}
|
||||||
|
if target.IsEmpty() {
|
||||||
|
// This means all fields have been removed from the object.
|
||||||
|
// This can happen if a patch required deletion of the
|
||||||
|
// entire resource (not just a part of it). This means
|
||||||
|
// the overall resmap must shrink by one.
|
||||||
|
err = m.Remove(target.CurId())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,10 +8,13 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
jsonpatch "github.com/evanphx/json-patch"
|
||||||
|
"github.com/pkg/errors"
|
||||||
"sigs.k8s.io/kustomize/api/filters/patchjson6902"
|
"sigs.k8s.io/kustomize/api/filters/patchjson6902"
|
||||||
|
"sigs.k8s.io/kustomize/api/filters/patchstrategicmerge"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -38,6 +41,7 @@ func (p *PatchTransformerPlugin) Config(
|
|||||||
return fmt.Errorf(
|
return fmt.Errorf(
|
||||||
"patch and path can't be set at the same time\n%s", string(c))
|
"patch and path can't be set at the same time\n%s", string(c))
|
||||||
}
|
}
|
||||||
|
|
||||||
if p.Path != "" {
|
if p.Path != "" {
|
||||||
loaded, loadErr := h.Loader().Load(p.Path)
|
loaded, loadErr := h.Loader().Load(p.Path)
|
||||||
if loadErr != nil {
|
if loadErr != nil {
|
||||||
@@ -84,13 +88,66 @@ func (p *PatchTransformerPlugin) transformStrategicMerge(m resmap.ResMap, patch
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return target.ApplySmPatch(patch)
|
return p.applySMPatch(target, patch)
|
||||||
}
|
}
|
||||||
selected, err := m.Select(*p.Target)
|
|
||||||
|
resources, err := m.Select(*p.Target)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return m.ApplySmPatch(resource.MakeIdSet(selected), patch)
|
for _, res := range resources {
|
||||||
|
patchCopy := patch.DeepCopy()
|
||||||
|
patchCopy.SetName(res.GetName())
|
||||||
|
patchCopy.SetNamespace(res.GetNamespace())
|
||||||
|
patchCopy.SetGvk(res.GetGvk())
|
||||||
|
err := p.applySMPatch(res, patchCopy)
|
||||||
|
if err != nil {
|
||||||
|
// Check for an error string from UnmarshalJSON that's indicative
|
||||||
|
// of an object that's missing basic KRM fields, and thus may have been
|
||||||
|
// entirely deleted (an acceptable outcome). This error handling should
|
||||||
|
// be deleted along with use of ResMap and apimachinery functions like
|
||||||
|
// UnmarshalJSON.
|
||||||
|
if !strings.Contains(err.Error(), "Object 'Kind' is missing") {
|
||||||
|
// Some unknown error, let it through.
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !res.IsEmpty() {
|
||||||
|
return errors.Wrapf(
|
||||||
|
err, "with unexpectedly non-empty object map of size %d",
|
||||||
|
len(res.Map()))
|
||||||
|
}
|
||||||
|
// Fall through to handle deleted object.
|
||||||
|
}
|
||||||
|
if res.IsEmpty() {
|
||||||
|
// This means all fields have been removed from the object.
|
||||||
|
// This can happen if a patch required deletion of the
|
||||||
|
// entire resource (not just a part of it). This means
|
||||||
|
// the overall resmap must shrink by one.
|
||||||
|
err = m.Remove(res.CurId())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// applySMPatch applies the provided strategic merge patch to the
|
||||||
|
// given resource.
|
||||||
|
func (p *PatchTransformerPlugin) applySMPatch(resource, patch *resource.Resource) error {
|
||||||
|
node, err := filtersutil.GetRNode(patch)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
n, ns := resource.GetName(), resource.GetNamespace()
|
||||||
|
err = filtersutil.ApplyToJSON(patchstrategicmerge.Filter{
|
||||||
|
Patch: node,
|
||||||
|
}, resource)
|
||||||
|
if !resource.IsEmpty() {
|
||||||
|
resource.SetName(n)
|
||||||
|
resource.SetNamespace(ns)
|
||||||
|
}
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// transformJson6902 applies the provided json6902 patch
|
// transformJson6902 applies the provided json6902 patch
|
||||||
@@ -104,10 +161,9 @@ func (p *PatchTransformerPlugin) transformJson6902(m resmap.ResMap, patch jsonpa
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for _, res := range resources {
|
for _, res := range resources {
|
||||||
res.StorePreviousId()
|
err = filtersutil.ApplyToJSON(patchjson6902.Filter{
|
||||||
err = res.ApplyFilter(patchjson6902.Filter{
|
|
||||||
Patch: p.Patch,
|
Patch: p.Patch,
|
||||||
})
|
}, res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -66,18 +67,14 @@ func (p *PrefixSuffixTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
// this will add a prefix and a suffix
|
// this will add a prefix and a suffix
|
||||||
// to the resource even if those are
|
// to the resource even if those are
|
||||||
// empty
|
// empty
|
||||||
|
|
||||||
r.AddNamePrefix(p.Prefix)
|
r.AddNamePrefix(p.Prefix)
|
||||||
r.AddNameSuffix(p.Suffix)
|
r.AddNameSuffix(p.Suffix)
|
||||||
if p.Prefix != "" || p.Suffix != "" {
|
|
||||||
r.StorePreviousId()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
err := r.ApplyFilter(prefixsuffix.Filter{
|
err := filtersutil.ApplyToJSON(prefixsuffix.Filter{
|
||||||
Prefix: p.Prefix,
|
Prefix: p.Prefix,
|
||||||
Suffix: p.Suffix,
|
Suffix: p.Suffix,
|
||||||
FieldSpec: fs,
|
FieldSpec: fs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/replicacount"
|
"sigs.k8s.io/kustomize/api/filters/replicacount"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
@@ -31,17 +33,19 @@ func (p *ReplicaCountTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
found := false
|
found := false
|
||||||
for _, fs := range p.FieldSpecs {
|
for _, fs := range p.FieldSpecs {
|
||||||
matcher := p.createMatcher(fs)
|
matcher := p.createMatcher(fs)
|
||||||
resList := m.GetMatchingResourcesByAnyId(matcher)
|
matchOriginal := m.GetMatchingResourcesByOriginalId(matcher)
|
||||||
|
resList := append(
|
||||||
|
matchOriginal, m.GetMatchingResourcesByCurrentId(matcher)...)
|
||||||
if len(resList) > 0 {
|
if len(resList) > 0 {
|
||||||
found = true
|
found = true
|
||||||
for _, r := range resList {
|
for _, r := range resList {
|
||||||
// There are redundant checks in the filter
|
// There are redundant checks in the filter
|
||||||
// that we'll live with until resolution of
|
// that we'll live with until resolution of
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/2506
|
// https://github.com/kubernetes-sigs/kustomize/issues/2506
|
||||||
err := r.ApplyFilter(replicacount.Filter{
|
err := filtersutil.ApplyToJSON(replicacount.Filter{
|
||||||
Replica: p.Replica,
|
Replica: p.Replica,
|
||||||
FieldSpec: fs,
|
FieldSpec: fs,
|
||||||
})
|
}, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -118,15 +119,15 @@ func (p *ValueAddTransformerPlugin) Transform(m resmap.ResMap) (err error) {
|
|||||||
// TODO: consider t.NotSelector if implemented
|
// TODO: consider t.NotSelector if implemented
|
||||||
for _, res := range resources {
|
for _, res := range resources {
|
||||||
if t.FieldPath == types.MetadataNamespacePath {
|
if t.FieldPath == types.MetadataNamespacePath {
|
||||||
err = res.ApplyFilter(namespace.Filter{
|
err = filtersutil.ApplyToJSON(namespace.Filter{
|
||||||
Namespace: p.Value,
|
Namespace: p.Value,
|
||||||
})
|
}, res)
|
||||||
} else {
|
} else {
|
||||||
err = res.ApplyFilter(valueadd.Filter{
|
err = filtersutil.ApplyToJSON(valueadd.Filter{
|
||||||
Value: p.Value,
|
Value: p.Value,
|
||||||
FieldPath: t.FieldPath,
|
FieldPath: t.FieldPath,
|
||||||
FilePathPosition: t.FilePathPosition,
|
FilePathPosition: t.FilePathPosition,
|
||||||
})
|
}, res)
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
package filesys_test
|
package filesys_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
@@ -99,7 +98,6 @@ func TestNewTempConfirmDir(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(string(tmp))
|
|
||||||
|
|
||||||
delinked, err := filepath.EvalSymlinks(string(tmp))
|
delinked, err := filepath.EvalSymlinks(string(tmp))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ type Filter struct {
|
|||||||
var _ kio.Filter = Filter{}
|
var _ kio.Filter = Filter{}
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
||||||
keys := yaml.SortedMapKeys(f.Annotations)
|
keys := filtersutil.SortedMapKeys(f.Annotations)
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(
|
_, err := kio.FilterAll(yaml.FilterFunc(
|
||||||
func(node *yaml.RNode) (*yaml.RNode, error) {
|
func(node *yaml.RNode) (*yaml.RNode, error) {
|
||||||
for _, k := range keys {
|
for _, k := range keys {
|
||||||
|
|||||||
@@ -28,9 +28,7 @@ metadata:
|
|||||||
`)}},
|
`)}},
|
||||||
Filters: []kio.Filter{Filter{
|
Filters: []kio.Filter{Filter{
|
||||||
Annotations: map[string]string{
|
Annotations: map[string]string{
|
||||||
"foo": "bar",
|
"foo": "bar",
|
||||||
"booleanValue": "true",
|
|
||||||
"numberValue": "42",
|
|
||||||
},
|
},
|
||||||
FsSlice: fss,
|
FsSlice: fss,
|
||||||
}},
|
}},
|
||||||
@@ -46,16 +44,12 @@ metadata:
|
|||||||
// metadata:
|
// metadata:
|
||||||
// name: instance
|
// name: instance
|
||||||
// annotations:
|
// annotations:
|
||||||
// booleanValue: "true"
|
|
||||||
// foo: bar
|
// foo: bar
|
||||||
// numberValue: "42"
|
|
||||||
// ---
|
// ---
|
||||||
// apiVersion: example.com/v1
|
// apiVersion: example.com/v1
|
||||||
// kind: Bar
|
// kind: Bar
|
||||||
// metadata:
|
// metadata:
|
||||||
// name: instance
|
// name: instance
|
||||||
// annotations:
|
// annotations:
|
||||||
// booleanValue: "true"
|
|
||||||
// foo: bar
|
// foo: bar
|
||||||
// numberValue: "42"
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,11 +4,9 @@
|
|||||||
package fieldspec
|
package fieldspec
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
||||||
"sigs.k8s.io/kustomize/api/internal/utils"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/errors"
|
"sigs.k8s.io/kustomize/kyaml/errors"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
@@ -16,16 +14,7 @@ import (
|
|||||||
|
|
||||||
var _ yaml.Filter = Filter{}
|
var _ yaml.Filter = Filter{}
|
||||||
|
|
||||||
// Filter possibly mutates its object argument using a FieldSpec.
|
// Filter applies a single fieldSpec to a single object
|
||||||
// If the object matches the FieldSpec, and the node found
|
|
||||||
// by following the fieldSpec's path is non-null, this filter calls
|
|
||||||
// the setValue function on the node at the end of the path.
|
|
||||||
// If any part of the path doesn't exist, the filter returns
|
|
||||||
// without doing anything and without error, unless it was set
|
|
||||||
// to create the path. If set to create, it creates a tree of maps
|
|
||||||
// along the path, and the leaf node gets the setValue called on it.
|
|
||||||
// Error on GVK mismatch, empty or poorly formed path.
|
|
||||||
// Filter expect kustomize style paths, not JSON paths.
|
|
||||||
// Filter stores internal state and should not be reused
|
// Filter stores internal state and should not be reused
|
||||||
type Filter struct {
|
type Filter struct {
|
||||||
// FieldSpec contains the path to the value to set.
|
// FieldSpec contains the path to the value to set.
|
||||||
@@ -48,17 +37,15 @@ func (fltr Filter) Filter(obj *yaml.RNode) (*yaml.RNode, error) {
|
|||||||
if match, err := isMatchGVK(fltr.FieldSpec, obj); !match || err != nil {
|
if match, err := isMatchGVK(fltr.FieldSpec, obj); !match || err != nil {
|
||||||
return obj, errors.Wrap(err)
|
return obj, errors.Wrap(err)
|
||||||
}
|
}
|
||||||
fltr.path = utils.PathSplitter(fltr.FieldSpec.Path)
|
fltr.path = splitPath(fltr.FieldSpec.Path)
|
||||||
err := fltr.filter(obj)
|
if err := fltr.filter(obj); err != nil {
|
||||||
if err != nil {
|
|
||||||
s, _ := obj.String()
|
s, _ := obj.String()
|
||||||
return nil, errors.WrapPrefixf(err,
|
return nil, errors.WrapPrefixf(err,
|
||||||
"considering field '%s' of object\n%v", fltr.FieldSpec.Path, s)
|
"obj '%s' at path '%v'", s, fltr.FieldSpec.Path)
|
||||||
}
|
}
|
||||||
return obj, nil
|
return obj, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Recursively called.
|
|
||||||
func (fltr Filter) filter(obj *yaml.RNode) error {
|
func (fltr Filter) filter(obj *yaml.RNode) error {
|
||||||
if len(fltr.path) == 0 {
|
if len(fltr.path) == 0 {
|
||||||
// found the field -- set its value
|
// found the field -- set its value
|
||||||
@@ -69,30 +56,25 @@ func (fltr Filter) filter(obj *yaml.RNode) error {
|
|||||||
}
|
}
|
||||||
switch obj.YNode().Kind {
|
switch obj.YNode().Kind {
|
||||||
case yaml.SequenceNode:
|
case yaml.SequenceNode:
|
||||||
return fltr.handleSequence(obj)
|
return fltr.seq(obj)
|
||||||
case yaml.MappingNode:
|
case yaml.MappingNode:
|
||||||
return fltr.handleMap(obj)
|
return fltr.field(obj)
|
||||||
case yaml.AliasNode:
|
|
||||||
return fltr.filter(yaml.NewRNode(obj.YNode().Alias))
|
|
||||||
default:
|
default:
|
||||||
return errors.Errorf("expected sequence or mapping node")
|
return errors.Errorf("expected sequence or mapping node")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// handleMap calls filter on the map field matching the next path element
|
// field calls filter on the field matching the next path element
|
||||||
func (fltr Filter) handleMap(obj *yaml.RNode) error {
|
func (fltr Filter) field(obj *yaml.RNode) error {
|
||||||
fieldName, isSeq := isSequenceField(fltr.path[0])
|
fieldName, isSeq := isSequenceField(fltr.path[0])
|
||||||
if fieldName == "" {
|
|
||||||
return fmt.Errorf("cannot set or create an empty field name")
|
|
||||||
}
|
|
||||||
// lookup the field matching the next path element
|
// lookup the field matching the next path element
|
||||||
var operation yaml.Filter
|
var lookupField yaml.Filter
|
||||||
var kind yaml.Kind
|
var kind yaml.Kind
|
||||||
tag := yaml.NodeTagEmpty
|
tag := yaml.NodeTagEmpty
|
||||||
switch {
|
switch {
|
||||||
case !fltr.FieldSpec.CreateIfNotPresent || fltr.CreateKind == 0 || isSeq:
|
case !fltr.FieldSpec.CreateIfNotPresent || fltr.CreateKind == 0 || isSeq:
|
||||||
// don't create the field if we don't find it
|
// dont' create the field if we don't find it
|
||||||
operation = yaml.Lookup(fieldName)
|
lookupField = yaml.Lookup(fieldName)
|
||||||
if isSeq {
|
if isSeq {
|
||||||
// The query path thinks this field should be a sequence;
|
// The query path thinks this field should be a sequence;
|
||||||
// accept this hint for use later if the tag is NodeTagNull.
|
// accept this hint for use later if the tag is NodeTagNull.
|
||||||
@@ -100,25 +82,21 @@ func (fltr Filter) handleMap(obj *yaml.RNode) error {
|
|||||||
}
|
}
|
||||||
case len(fltr.path) <= 1:
|
case len(fltr.path) <= 1:
|
||||||
// create the field if it is missing: use the provided node kind
|
// create the field if it is missing: use the provided node kind
|
||||||
operation = yaml.LookupCreate(fltr.CreateKind, fieldName)
|
lookupField = yaml.LookupCreate(fltr.CreateKind, fieldName)
|
||||||
kind = fltr.CreateKind
|
kind = fltr.CreateKind
|
||||||
tag = fltr.CreateTag
|
tag = fltr.CreateTag
|
||||||
default:
|
default:
|
||||||
// create the field if it is missing: must be a mapping node
|
// create the field if it is missing: must be a mapping node
|
||||||
operation = yaml.LookupCreate(yaml.MappingNode, fieldName)
|
lookupField = yaml.LookupCreate(yaml.MappingNode, fieldName)
|
||||||
kind = yaml.MappingNode
|
kind = yaml.MappingNode
|
||||||
tag = yaml.NodeTagMap
|
tag = yaml.NodeTagMap
|
||||||
}
|
}
|
||||||
|
|
||||||
// locate (or maybe create) the field
|
// locate (or maybe create) the field
|
||||||
field, err := obj.Pipe(operation)
|
field, err := obj.Pipe(lookupField)
|
||||||
if err != nil {
|
if err != nil || field == nil {
|
||||||
return errors.WrapPrefixf(err, "fieldName: %s", fieldName)
|
return errors.WrapPrefixf(err, "fieldName: %s", fieldName)
|
||||||
}
|
}
|
||||||
if field == nil {
|
|
||||||
// No error if field not found.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the value exists, but is null and kind is set,
|
// if the value exists, but is null and kind is set,
|
||||||
// then change it to the creation type
|
// then change it to the creation type
|
||||||
@@ -136,7 +114,7 @@ func (fltr Filter) handleMap(obj *yaml.RNode) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// seq calls filter on all sequence elements
|
// seq calls filter on all sequence elements
|
||||||
func (fltr Filter) handleSequence(obj *yaml.RNode) error {
|
func (fltr Filter) seq(obj *yaml.RNode) error {
|
||||||
if err := obj.VisitElements(func(node *yaml.RNode) error {
|
if err := obj.VisitElements(func(node *yaml.RNode) error {
|
||||||
// recurse on each element -- re-allocating a Filter is
|
// recurse on each element -- re-allocating a Filter is
|
||||||
// not strictly required, but is more consistent with field
|
// not strictly required, but is more consistent with field
|
||||||
@@ -147,14 +125,16 @@ func (fltr Filter) handleSequence(obj *yaml.RNode) error {
|
|||||||
return errors.WrapPrefixf(err,
|
return errors.WrapPrefixf(err,
|
||||||
"visit traversal on path: %v", fltr.path)
|
"visit traversal on path: %v", fltr.path)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// isSequenceField returns true if the path element is for a sequence field.
|
// isSequenceField returns true if the path element is for a sequence field.
|
||||||
// isSequence also returns the path element with the '[]' suffix trimmed
|
// isSequence also returns the path element with the '[]' suffix trimmed
|
||||||
func isSequenceField(name string) (string, bool) {
|
func isSequenceField(name string) (string, bool) {
|
||||||
shorter := strings.TrimSuffix(name, "[]")
|
isSeq := strings.HasSuffix(name, "[]")
|
||||||
return shorter, shorter != name
|
name = strings.TrimSuffix(name, "[]")
|
||||||
|
return name, isSeq
|
||||||
}
|
}
|
||||||
|
|
||||||
// isMatchGVK returns true if the fs.GVK matches the obj GVK.
|
// isMatchGVK returns true if the fs.GVK matches the obj GVK.
|
||||||
@@ -183,3 +163,18 @@ func isMatchGVK(fs types.FieldSpec, obj *yaml.RNode) (bool, error) {
|
|||||||
|
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func splitPath(path string) []string {
|
||||||
|
ps := strings.Split(path, "/")
|
||||||
|
var res []string
|
||||||
|
res = append(res, ps[0])
|
||||||
|
for i := 1; i < len(ps); i++ {
|
||||||
|
lastIndex := len(res) - 1
|
||||||
|
if strings.HasSuffix(res[lastIndex], "\\") {
|
||||||
|
res[lastIndex] = strings.TrimSuffix(res[lastIndex], "\\") + "/" + ps[i]
|
||||||
|
} else {
|
||||||
|
res = append(res, ps[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,243 +15,206 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestFilter_Filter(t *testing.T) {
|
type TestCase struct {
|
||||||
testCases := map[string]struct {
|
name string
|
||||||
input string
|
input string
|
||||||
expected string
|
expected string
|
||||||
filter fieldspec.Filter
|
filter fieldspec.Filter
|
||||||
fieldSpec string
|
fieldSpec string
|
||||||
error string
|
error string
|
||||||
}{
|
}
|
||||||
"path not found": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"empty path": {
|
|
||||||
fieldSpec: `
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
error: `considering field '' of object
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
: cannot set or create an empty field name`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update": {
|
var tests = []TestCase{
|
||||||
fieldSpec: `
|
{
|
||||||
|
name: "update",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo
|
group: foo
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: e
|
b: e
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"update-kind-not-match": {
|
{
|
||||||
fieldSpec: `
|
name: "update-kind-not-match",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo
|
group: foo
|
||||||
kind: Bar1
|
kind: Bar1
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar2
|
kind: Bar2
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar2
|
kind: Bar2
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"update-group-not-match": {
|
{
|
||||||
fieldSpec: `
|
name: "update-group-not-match",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo1
|
group: foo1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo2/v1beta1
|
apiVersion: foo2/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo2/v1beta1
|
apiVersion: foo2/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"update-version-not-match": {
|
{
|
||||||
fieldSpec: `
|
name: "update-version-not-match",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo
|
group: foo
|
||||||
version: v1beta1
|
version: v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta2
|
apiVersion: foo/v1beta2
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta2
|
apiVersion: foo/v1beta2
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"bad-version": {
|
{
|
||||||
fieldSpec: `
|
name: "bad-version",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo
|
group: foo
|
||||||
version: v1beta1
|
version: v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta2/something
|
apiVersion: foo/v1beta2/something
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta2/something
|
apiVersion: foo/v1beta2/something
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"bad-meta": {
|
{
|
||||||
fieldSpec: `
|
name: "bad-meta",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: foo
|
group: foo
|
||||||
version: v1beta1
|
version: v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
a:
|
a:
|
||||||
b: c
|
b: c
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
error: "missing Resource metadata",
|
|
||||||
},
|
},
|
||||||
|
error: "missing Resource metadata",
|
||||||
|
},
|
||||||
|
|
||||||
"miss-match-type": {
|
{
|
||||||
fieldSpec: `
|
name: "miss-match-type",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b/c
|
path: a/b/c
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: a
|
b: a
|
||||||
`,
|
`,
|
||||||
error: `considering field 'a/b/c' of object
|
error: "obj 'kind: Bar\na:\n b: a\n' at path 'a/b/c': " +
|
||||||
kind: Bar
|
"expected sequence or mapping node",
|
||||||
a:
|
filter: fieldspec.Filter{
|
||||||
b: a
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
: expected sequence or mapping node`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"add": {
|
{
|
||||||
fieldSpec: `
|
name: "add",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b/c/d
|
path: a/b/c/d
|
||||||
group: foo
|
group: foo
|
||||||
create: true
|
create: true
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a: {}
|
a: {}
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a: {b: {c: {d: e}}}
|
a: {b: {c: {d: e}}}
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"update-in-sequence": {
|
{
|
||||||
fieldSpec: `
|
name: "update-in-sequence",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b[]/c/d
|
path: a/b[]/c/d
|
||||||
group: foo
|
group: foo
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
@@ -259,7 +222,7 @@ a:
|
|||||||
- c:
|
- c:
|
||||||
d: a
|
d: a
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
@@ -267,237 +230,244 @@ a:
|
|||||||
- c:
|
- c:
|
||||||
d: e
|
d: e
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
// Don't create a sequence
|
// Don't create a sequence
|
||||||
"empty-sequence-no-create": {
|
{
|
||||||
fieldSpec: `
|
name: "empty-sequence-no-create",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b[]/c/d
|
path: a/b[]/c/d
|
||||||
group: foo
|
group: foo
|
||||||
create: true
|
create: true
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a: {}
|
a: {}
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a: {}
|
a: {}
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
// Create a new field for an element in a sequence
|
// Create a new field for an element in a sequence
|
||||||
"empty-sequence-create": {
|
{
|
||||||
fieldSpec: `
|
name: "empty-sequence-create",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b[]/c/d
|
path: a/b[]/c/d
|
||||||
group: foo
|
group: foo
|
||||||
create: true
|
create: true
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b:
|
b:
|
||||||
- c: {}
|
- c: {}
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: foo/v1beta1
|
apiVersion: foo/v1beta1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b:
|
b:
|
||||||
- c: {d: e}
|
- c: {d: e}
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"group v1": {
|
{
|
||||||
fieldSpec: `
|
name: "group v1",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
group: v1
|
group: v1
|
||||||
create: true
|
create: true
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
"version v1": {
|
{
|
||||||
fieldSpec: `
|
name: "version v1",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b
|
path: a/b
|
||||||
version: v1
|
version: v1
|
||||||
create: true
|
create: true
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b: e
|
b: e
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
SetValue: filtersutil.SetScalar("e"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
"successfully set field on array entry no sequence hint": {
|
{
|
||||||
fieldSpec: `
|
name: "successfully set field on array entry no sequence hint",
|
||||||
|
fieldSpec: `
|
||||||
path: spec/containers/image
|
path: spec/containers/image
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- image: foo
|
- image: foo
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- image: bar
|
- image: bar
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
"successfully set field on array entry with sequence hint": {
|
{
|
||||||
fieldSpec: `
|
name: "successfully set field on array entry with sequence hint",
|
||||||
|
fieldSpec: `
|
||||||
path: spec/containers[]/image
|
path: spec/containers[]/image
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- image: foo
|
- image: foo
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- image: bar
|
- image: bar
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
"failure to set field on array entry with sequence hint in path": {
|
},
|
||||||
fieldSpec: `
|
{
|
||||||
|
name: "failure to set field on array entry with sequence hint in path",
|
||||||
|
fieldSpec: `
|
||||||
path: spec/containers[]/image
|
path: spec/containers[]/image
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers: []
|
containers: []
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
"failure to set field on array entry, no sequence hint in path": {
|
{
|
||||||
fieldSpec: `
|
name: "failure to set field on array entry, no sequence hint in path",
|
||||||
|
fieldSpec: `
|
||||||
path: spec/containers/image
|
path: spec/containers/image
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
"fieldname with slash '/'": {
|
},
|
||||||
fieldSpec: `
|
{
|
||||||
|
name: "filedname with slash '/'",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b\/c/d
|
path: a/b\/c/d
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b/c:
|
b/c:
|
||||||
d: foo
|
d: foo
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
b/c:
|
b/c:
|
||||||
d: bar
|
d: bar
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
"fieldname with multiple '/'": {
|
},
|
||||||
fieldSpec: `
|
{
|
||||||
|
name: "filedname with multiple '/'",
|
||||||
|
fieldSpec: `
|
||||||
path: a/b\/c/d\/e/f
|
path: a/b\/c/d\/e/f
|
||||||
version: v1
|
version: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
`,
|
`,
|
||||||
input: `
|
input: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
@@ -505,7 +475,7 @@ a:
|
|||||||
d/e:
|
d/e:
|
||||||
f: foo
|
f: foo
|
||||||
`,
|
`,
|
||||||
expected: `
|
expected: `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Bar
|
kind: Bar
|
||||||
a:
|
a:
|
||||||
@@ -513,24 +483,25 @@ a:
|
|||||||
d/e:
|
d/e:
|
||||||
f: bar
|
f: bar
|
||||||
`,
|
`,
|
||||||
filter: fieldspec.Filter{
|
filter: fieldspec.Filter{
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
SetValue: filtersutil.SetScalar("bar"),
|
||||||
CreateKind: yaml.ScalarNode,
|
CreateKind: yaml.ScalarNode,
|
||||||
},
|
|
||||||
},
|
},
|
||||||
}
|
},
|
||||||
|
}
|
||||||
|
|
||||||
for n := range testCases {
|
func TestFilter_Filter(t *testing.T) {
|
||||||
tc := testCases[n]
|
for i := range tests {
|
||||||
t.Run(n, func(t *testing.T) {
|
test := tests[i]
|
||||||
err := yaml.Unmarshal([]byte(tc.fieldSpec), &tc.filter.FieldSpec)
|
t.Run(test.name, func(t *testing.T) {
|
||||||
|
err := yaml.Unmarshal([]byte(test.fieldSpec), &test.filter.FieldSpec)
|
||||||
if !assert.NoError(t, err) {
|
if !assert.NoError(t, err) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
|
|
||||||
out := &bytes.Buffer{}
|
out := &bytes.Buffer{}
|
||||||
rw := &kio.ByteReadWriter{
|
rw := &kio.ByteReadWriter{
|
||||||
Reader: bytes.NewBufferString(tc.input),
|
Reader: bytes.NewBufferString(test.input),
|
||||||
Writer: out,
|
Writer: out,
|
||||||
OmitReaderAnnotations: true,
|
OmitReaderAnnotations: true,
|
||||||
}
|
}
|
||||||
@@ -538,11 +509,11 @@ a:
|
|||||||
// run the filter
|
// run the filter
|
||||||
err = kio.Pipeline{
|
err = kio.Pipeline{
|
||||||
Inputs: []kio.Reader{rw},
|
Inputs: []kio.Reader{rw},
|
||||||
Filters: []kio.Filter{kio.FilterAll(tc.filter)},
|
Filters: []kio.Filter{kio.FilterAll(test.filter)},
|
||||||
Outputs: []kio.Writer{rw},
|
Outputs: []kio.Writer{rw},
|
||||||
}.Execute()
|
}.Execute()
|
||||||
if tc.error != "" {
|
if test.error != "" {
|
||||||
if !assert.EqualError(t, err, tc.error) {
|
if !assert.EqualError(t, err, test.error) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
// stop rest of test
|
// stop rest of test
|
||||||
@@ -555,7 +526,7 @@ a:
|
|||||||
|
|
||||||
// check results
|
// check results
|
||||||
if !assert.Equal(t,
|
if !assert.Equal(t,
|
||||||
strings.TrimSpace(tc.expected),
|
strings.TrimSpace(test.expected),
|
||||||
strings.TrimSpace(out.String())) {
|
strings.TrimSpace(out.String())) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
|
|||||||
21
api/filters/filtersutil/filtersutil.go
Normal file
21
api/filters/filtersutil/filtersutil.go
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
package filtersutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SortedMapKeys returns a sorted slice of keys to the given map.
|
||||||
|
// Writing this function never gets old.
|
||||||
|
func SortedMapKeys(m map[string]string) []string {
|
||||||
|
keys := make([]string, len(m))
|
||||||
|
i := 0
|
||||||
|
for k := range m {
|
||||||
|
keys[i] = k
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
return keys
|
||||||
|
}
|
||||||
@@ -1,13 +1,10 @@
|
|||||||
// Copyright 2021 The Kubernetes Authors.
|
package filtersutil_test
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package yaml_test
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSortedKeys(t *testing.T) {
|
func TestSortedKeys(t *testing.T) {
|
||||||
@@ -26,10 +23,9 @@ func TestSortedKeys(t *testing.T) {
|
|||||||
expected: []string{"a", "b", "c"}},
|
expected: []string{"a", "b", "c"}},
|
||||||
}
|
}
|
||||||
for tn, tc := range testCases {
|
for tn, tc := range testCases {
|
||||||
tc := tc
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tn, func(t *testing.T) {
|
||||||
if !assert.Equal(t,
|
if !assert.Equal(t,
|
||||||
yaml.SortedMapKeys(tc.input),
|
filtersutil.SortedMapKeys(tc.input),
|
||||||
tc.expected) {
|
tc.expected) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
@@ -21,7 +21,7 @@ func TestImageTagUpdater_Filter(t *testing.T) {
|
|||||||
}{
|
}{
|
||||||
"ignore CustomResourceDefinition": {
|
"ignore CustomResourceDefinition": {
|
||||||
input: `
|
input: `
|
||||||
apiVersion: apiextensions.k8s.io/v1
|
apiVersion: apiextensions.k8s.io/v1beta1
|
||||||
kind: CustomResourceDefinition
|
kind: CustomResourceDefinition
|
||||||
metadata:
|
metadata:
|
||||||
name: whatever
|
name: whatever
|
||||||
@@ -30,7 +30,7 @@ spec:
|
|||||||
- image: whatever
|
- image: whatever
|
||||||
`,
|
`,
|
||||||
expectedOutput: `
|
expectedOutput: `
|
||||||
apiVersion: apiextensions.k8s.io/v1
|
apiVersion: apiextensions.k8s.io/v1beta1
|
||||||
kind: CustomResourceDefinition
|
kind: CustomResourceDefinition
|
||||||
metadata:
|
metadata:
|
||||||
name: whatever
|
name: whatever
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ type Filter struct {
|
|||||||
var _ kio.Filter = Filter{}
|
var _ kio.Filter = Filter{}
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
||||||
keys := yaml.SortedMapKeys(f.Labels)
|
keys := filtersutil.SortedMapKeys(f.Labels)
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(
|
_, err := kio.FilterAll(yaml.FilterFunc(
|
||||||
func(node *yaml.RNode) (*yaml.RNode, error) {
|
func(node *yaml.RNode) (*yaml.RNode, error) {
|
||||||
for _, k := range keys {
|
for _, k := range keys {
|
||||||
|
|||||||
@@ -5,189 +5,100 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
||||||
|
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
kyaml_filtersutil "sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
"sigs.k8s.io/kustomize/kyaml/kio"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Filter updates a name references.
|
// Filter will update the name reference
|
||||||
type Filter struct {
|
type Filter struct {
|
||||||
// Referrer refers to another resource X by X's name.
|
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
||||||
// E.g. A Deployment can refer to a ConfigMap.
|
Referrer *resource.Resource
|
||||||
// The Deployment is the Referrer,
|
Target resid.Gvk
|
||||||
// the ConfigMap is the ReferralTarget.
|
|
||||||
// This filter seeks to repair the reference in Deployment, given
|
|
||||||
// that the ConfigMap's name may have changed.
|
|
||||||
Referrer *resource.Resource
|
|
||||||
|
|
||||||
// NameFieldToUpdate is the field in the Referrer
|
|
||||||
// that holds the name requiring an update.
|
|
||||||
// This is the field to write.
|
|
||||||
NameFieldToUpdate types.FieldSpec
|
|
||||||
|
|
||||||
// ReferralTarget is the source of the new value for
|
|
||||||
// the name, always in the 'metadata/name' field.
|
|
||||||
// This is the field to read.
|
|
||||||
ReferralTarget resid.Gvk
|
|
||||||
|
|
||||||
// Set of resources to scan to find the ReferralTarget.
|
|
||||||
ReferralCandidates resmap.ResMap
|
ReferralCandidates resmap.ResMap
|
||||||
|
isRoleRef bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// At time of writing, in practice this is called with a slice with only
|
|
||||||
// one entry, the node also referred to be the resource in the Referrer field.
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
||||||
return kio.FilterAll(yaml.FilterFunc(f.run)).Filter(nodes)
|
return kio.FilterAll(yaml.FilterFunc(f.run)).Filter(nodes)
|
||||||
}
|
}
|
||||||
|
|
||||||
// The node passed in here is the same node as held in Referrer;
|
|
||||||
// that's how the referrer's name field is updated.
|
|
||||||
// Currently, however, this filter still needs the extra methods on Referrer
|
|
||||||
// to consult things like the resource Id, its namespace, etc.
|
|
||||||
// TODO(3455): No filter should use the Resource api; all information
|
|
||||||
// about names should come from annotations, with helper methods
|
|
||||||
// on the RNode object. Resource should get stupider, RNode smarter.
|
|
||||||
func (f Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
func (f Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
||||||
if err := f.confirmNodeMatchesReferrer(node); err != nil {
|
err := node.PipeE(fieldspec.Filter{
|
||||||
// sanity check.
|
FieldSpec: f.FieldSpec,
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := node.PipeE(fieldspec.Filter{
|
|
||||||
FieldSpec: f.NameFieldToUpdate,
|
|
||||||
SetValue: f.set,
|
SetValue: f.set,
|
||||||
}); err != nil {
|
})
|
||||||
return nil, errors.Wrapf(
|
return node, err
|
||||||
err, "updating name reference in '%s' field of '%s'",
|
|
||||||
f.NameFieldToUpdate.Path, f.Referrer.CurId().String())
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// This function is called on the node found at FieldSpec.Path.
|
|
||||||
// It's some node in the Referrer.
|
|
||||||
func (f Filter) set(node *yaml.RNode) error {
|
func (f Filter) set(node *yaml.RNode) error {
|
||||||
if yaml.IsMissingOrNull(node) {
|
if yaml.IsMissingOrNull(node) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
if strings.HasSuffix(f.FieldSpec.Path, "roleRef/name") {
|
||||||
|
f.isRoleRef = true
|
||||||
|
}
|
||||||
switch node.YNode().Kind {
|
switch node.YNode().Kind {
|
||||||
case yaml.ScalarNode:
|
case yaml.ScalarNode:
|
||||||
return f.setScalar(node)
|
return f.setScalar(node)
|
||||||
case yaml.MappingNode:
|
case yaml.MappingNode:
|
||||||
|
// Kind: ValidatingWebhookConfiguration
|
||||||
|
// FieldSpec is webhooks/clientConfig/service
|
||||||
return f.setMapping(node)
|
return f.setMapping(node)
|
||||||
case yaml.SequenceNode:
|
case yaml.SequenceNode:
|
||||||
return applyFilterToSeq(seqFilter{
|
return f.setSequence(node)
|
||||||
setScalarFn: f.setScalar,
|
|
||||||
setMappingFn: f.setMapping,
|
|
||||||
}, node)
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("node must be a scalar, sequence or map")
|
return fmt.Errorf(
|
||||||
|
"node is expected to be either a string or a slice of string or a map of string")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// This method used when NameFieldToUpdate doesn't lead to
|
func (f Filter) setSequence(node *yaml.RNode) error {
|
||||||
// one scalar field (typically called 'name'), but rather
|
return applyFilterToSeq(seqFilter{
|
||||||
// leads to a map field (called anything). In this case we
|
setScalarFn: f.setScalar,
|
||||||
// must complete the field path, looking for both a 'name'
|
setMappingFn: f.setMapping,
|
||||||
// and a 'namespace' field to help select the proper
|
}, node)
|
||||||
// ReferralTarget to read the name and namespace from.
|
}
|
||||||
|
|
||||||
func (f Filter) setMapping(node *yaml.RNode) error {
|
func (f Filter) setMapping(node *yaml.RNode) error {
|
||||||
if node.YNode().Kind != yaml.MappingNode {
|
return setNameAndNs(
|
||||||
return fmt.Errorf("expect a mapping node")
|
node,
|
||||||
}
|
f.Referrer,
|
||||||
nameNode, err := node.Pipe(yaml.FieldMatcher{Name: "name"})
|
f.Target,
|
||||||
if err != nil {
|
f.ReferralCandidates,
|
||||||
return errors.Wrap(err, "trying to match 'name' field")
|
f.isRoleRef,
|
||||||
}
|
)
|
||||||
if nameNode == nil {
|
|
||||||
// This is a _configuration_ error; the field path
|
|
||||||
// specified in NameFieldToUpdate.Path doesn't resolve
|
|
||||||
// to a map with a 'name' field, so we have no idea what
|
|
||||||
// field to update with a new name.
|
|
||||||
return fmt.Errorf("path config error; no 'name' field in node")
|
|
||||||
}
|
|
||||||
candidates, err := f.filterMapCandidatesByNamespace(node)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
oldName := nameNode.YNode().Value
|
|
||||||
referral, err := f.selectReferral(oldName, candidates)
|
|
||||||
if err != nil || referral == nil {
|
|
||||||
// Nil referral means nothing to do.
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
f.recordTheReferral(referral)
|
|
||||||
if referral.GetName() == oldName && referral.GetNamespace() == "" {
|
|
||||||
// The name has not changed, nothing to do.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if err = node.PipeE(yaml.FieldSetter{
|
|
||||||
Name: "name",
|
|
||||||
StringValue: referral.GetName(),
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if referral.GetNamespace() == "" {
|
|
||||||
// Don't write an empty string into the namespace field, as
|
|
||||||
// it should not replace the value "default". The empty
|
|
||||||
// string is handled as a wild card here, not as an implicit
|
|
||||||
// specification of the "default" k8s namespace.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return node.PipeE(yaml.FieldSetter{
|
|
||||||
Name: "namespace",
|
|
||||||
StringValue: referral.GetNamespace(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) filterMapCandidatesByNamespace(
|
|
||||||
node *yaml.RNode) ([]*resource.Resource, error) {
|
|
||||||
namespaceNode, err := node.Pipe(yaml.FieldMatcher{Name: "namespace"})
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrap(err, "trying to match 'namespace' field")
|
|
||||||
}
|
|
||||||
if namespaceNode == nil {
|
|
||||||
return f.ReferralCandidates.Resources(), nil
|
|
||||||
}
|
|
||||||
namespace := namespaceNode.YNode().Value
|
|
||||||
nsMap := f.ReferralCandidates.GroupedByOriginalNamespace()
|
|
||||||
if candidates, ok := nsMap[namespace]; ok {
|
|
||||||
return candidates, nil
|
|
||||||
}
|
|
||||||
nsMap = f.ReferralCandidates.GroupedByCurrentNamespace()
|
|
||||||
// This could be nil, or an empty list.
|
|
||||||
return nsMap[namespace], nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f Filter) setScalar(node *yaml.RNode) error {
|
func (f Filter) setScalar(node *yaml.RNode) error {
|
||||||
referral, err := f.selectReferral(
|
newValue, err := getSimpleNameField(
|
||||||
node.YNode().Value, f.ReferralCandidates.Resources())
|
node.YNode().Value,
|
||||||
if err != nil || referral == nil {
|
f.Referrer,
|
||||||
// Nil referral means nothing to do.
|
f.Target,
|
||||||
|
f.ReferralCandidates,
|
||||||
|
f.ReferralCandidates.Resources(),
|
||||||
|
f.isRoleRef,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
f.recordTheReferral(referral)
|
err = filtersutil.SetScalar(newValue)(node)
|
||||||
if referral.GetName() == node.YNode().Value {
|
if err != nil {
|
||||||
// The name has not changed, nothing to do.
|
return err
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
return node.PipeE(yaml.FieldSetter{StringValue: referral.GetName()})
|
return nil
|
||||||
}
|
|
||||||
|
|
||||||
// In the resource, make a note that it is referred to by the Referrer.
|
|
||||||
func (f Filter) recordTheReferral(referral *resource.Resource) {
|
|
||||||
referral.AppendRefBy(f.Referrer.CurId())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// getRoleRefGvk returns a Gvk in the roleRef field. Return error
|
// getRoleRefGvk returns a Gvk in the roleRef field. Return error
|
||||||
// if the roleRef, roleRef/apiGroup or roleRef/kind is missing.
|
// if the roleRef, roleRef/apiGroup or roleRef/kind is missing.
|
||||||
func getRoleRefGvk(res json.Marshaler) (*resid.Gvk, error) {
|
func getRoleRefGvk(res json.Marshaler) (*resid.Gvk, error) {
|
||||||
n, err := filtersutil.GetRNode(res)
|
n, err := kyaml_filtersutil.GetRNode(res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -203,16 +114,14 @@ func getRoleRefGvk(res json.Marshaler) (*resid.Gvk, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if apiGroup.IsNil() {
|
if apiGroup.IsNil() {
|
||||||
return nil, fmt.Errorf(
|
return nil, fmt.Errorf("apiGroup cannot be found in roleRef %s", roleRef.MustString())
|
||||||
"apiGroup cannot be found in roleRef %s", roleRef.MustString())
|
|
||||||
}
|
}
|
||||||
kind, err := roleRef.Pipe(yaml.Lookup("kind"))
|
kind, err := roleRef.Pipe(yaml.Lookup("kind"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if kind.IsNil() {
|
if kind.IsNil() {
|
||||||
return nil, fmt.Errorf(
|
return nil, fmt.Errorf("kind cannot be found in roleRef %s", roleRef.MustString())
|
||||||
"kind cannot be found in roleRef %s", roleRef.MustString())
|
|
||||||
}
|
}
|
||||||
return &resid.Gvk{
|
return &resid.Gvk{
|
||||||
Group: apiGroup.YNode().Value,
|
Group: apiGroup.YNode().Value,
|
||||||
@@ -220,183 +129,169 @@ func getRoleRefGvk(res json.Marshaler) (*resid.Gvk, error) {
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// sieveFunc returns true if the resource argument satisfies some criteria.
|
func filterReferralCandidates(
|
||||||
type sieveFunc func(*resource.Resource) bool
|
referrer *resource.Resource,
|
||||||
|
matches []*resource.Resource,
|
||||||
// doSieve uses a function to accept or ignore resources from a list.
|
target resid.Gvk,
|
||||||
// If list is nil, returns immediately.
|
) []*resource.Resource {
|
||||||
// It's a filter obviously, but that term is overloaded here.
|
var ret []*resource.Resource
|
||||||
func doSieve(list []*resource.Resource, fn sieveFunc) (s []*resource.Resource) {
|
for _, m := range matches {
|
||||||
for _, r := range list {
|
// If target kind is not ServiceAccount, we shouldn't consider condidates which
|
||||||
if fn(r) {
|
// doesn't have same namespace.
|
||||||
s = append(s, r)
|
if target.Kind != "ServiceAccount" && m.GetNamespace() != referrer.GetNamespace() {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
}
|
if !referrer.PrefixesSuffixesEquals(m) {
|
||||||
return
|
continue
|
||||||
}
|
|
||||||
|
|
||||||
func acceptAll(r *resource.Resource) bool {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func previousNameMatches(name string) sieveFunc {
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
for _, id := range r.PrevIds() {
|
|
||||||
if id.Name == name {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return false
|
ret = append(ret, m)
|
||||||
}
|
}
|
||||||
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
func previousIdSelectedByGvk(gvk *resid.Gvk) sieveFunc {
|
// selectReferral picks the referral among a subset of candidates.
|
||||||
return func(r *resource.Resource) bool {
|
// It returns the current name and namespace of the selected candidate.
|
||||||
for _, id := range r.PrevIds() {
|
// Note that the content of the referricalCandidateSubset slice is most of the time
|
||||||
if id.IsSelected(gvk) {
|
// identical to the referralCandidates resmap. Still in some cases, such
|
||||||
return true
|
// as ClusterRoleBinding, the subset only contains the resources of a specific
|
||||||
}
|
// namespace.
|
||||||
}
|
func selectReferral(
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the we are updating a 'roleRef/name' field, the 'apiGroup' and 'kind'
|
|
||||||
// fields in the same 'roleRef' map must be considered.
|
|
||||||
// If either object is cluster-scoped (!IsNamespaceableKind), there
|
|
||||||
// can be a referral.
|
|
||||||
// E.g. a RoleBinding (which exists in a namespace) can refer
|
|
||||||
// to a ClusterRole (cluster-scoped) object.
|
|
||||||
// https://kubernetes.io/docs/reference/access-authn-authz/rbac/#role-and-clusterrole
|
|
||||||
// Likewise, a ClusterRole can refer to a Secret (in a namespace).
|
|
||||||
// Objects in different namespaces generally cannot refer to other
|
|
||||||
// with some exceptions (e.g. RoleBinding and ServiceAccount are both
|
|
||||||
// namespaceable, but the former can refer to accounts in other namespaces).
|
|
||||||
func (f Filter) roleRefFilter() sieveFunc {
|
|
||||||
if !strings.HasSuffix(f.NameFieldToUpdate.Path, "roleRef/name") {
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
roleRefGvk, err := getRoleRefGvk(f.Referrer)
|
|
||||||
if err != nil {
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
return previousIdSelectedByGvk(roleRefGvk)
|
|
||||||
}
|
|
||||||
|
|
||||||
func prefixSuffixEquals(other resource.ResCtx) sieveFunc {
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
return r.PrefixesSuffixesEquals(other)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) sameCurrentNamespaceAsReferrer() sieveFunc {
|
|
||||||
referrerCurId := f.Referrer.CurId()
|
|
||||||
if !referrerCurId.IsNamespaceableKind() {
|
|
||||||
// If the referrer is cluster-scoped, let anything through.
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
if !r.CurId().IsNamespaceableKind() {
|
|
||||||
// Allow cluster-scoped through.
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if r.GetKind() == "ServiceAccount" {
|
|
||||||
// Allow service accounts through, even though they
|
|
||||||
// are in a namespace. A RoleBinding in another namespace
|
|
||||||
// can reference them.
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return referrerCurId.IsNsEquals(r.CurId())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// selectReferral picks the best referral from a list of candidates.
|
|
||||||
func (f Filter) selectReferral(
|
|
||||||
// The name referral that may need to be updated.
|
|
||||||
oldName string,
|
oldName string,
|
||||||
candidates []*resource.Resource) (*resource.Resource, error) {
|
referrer *resource.Resource,
|
||||||
candidates = doSieve(candidates, previousNameMatches(oldName))
|
target resid.Gvk,
|
||||||
candidates = doSieve(candidates, previousIdSelectedByGvk(&f.ReferralTarget))
|
referralCandidates resmap.ResMap,
|
||||||
candidates = doSieve(candidates, f.roleRefFilter())
|
referralCandidateSubset []*resource.Resource,
|
||||||
candidates = doSieve(candidates, f.sameCurrentNamespaceAsReferrer())
|
isRoleRef bool) (string, string, error) {
|
||||||
if len(candidates) == 1 {
|
var roleRefGvk *resid.Gvk
|
||||||
return candidates[0], nil
|
if isRoleRef {
|
||||||
}
|
var err error
|
||||||
candidates = doSieve(candidates, prefixSuffixEquals(f.Referrer))
|
roleRefGvk, err = getRoleRefGvk(referrer)
|
||||||
if len(candidates) == 1 {
|
if err != nil {
|
||||||
return candidates[0], nil
|
return "", "", err
|
||||||
}
|
|
||||||
if len(candidates) == 0 {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
if allNamesAreTheSame(candidates) {
|
|
||||||
// Just take the first one.
|
|
||||||
return candidates[0], nil
|
|
||||||
}
|
|
||||||
ids := getIds(candidates)
|
|
||||||
f.failureDetails(candidates)
|
|
||||||
return nil, fmt.Errorf(" found multiple possible referrals: %s", ids)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) failureDetails(resources []*resource.Resource) {
|
|
||||||
fmt.Printf(
|
|
||||||
"\n**** Too many possible referral targets to referrer:\n%s\n",
|
|
||||||
f.Referrer.MustYaml())
|
|
||||||
for i, r := range resources {
|
|
||||||
fmt.Printf(
|
|
||||||
"--- possible referral %d:\n%s", i, r.MustYaml())
|
|
||||||
fmt.Println("------")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func allNamesAreTheSame(resources []*resource.Resource) bool {
|
|
||||||
name := resources[0].GetName()
|
|
||||||
for i := 1; i < len(resources); i++ {
|
|
||||||
if name != resources[i].GetName() {
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true
|
for _, res := range referralCandidateSubset {
|
||||||
|
id := res.OrgId()
|
||||||
|
// If the we are processing a roleRef, the apiGroup and Kind in the
|
||||||
|
// roleRef are needed to be considered.
|
||||||
|
if (!isRoleRef || id.IsSelected(roleRefGvk)) &&
|
||||||
|
id.IsSelected(&target) && res.GetOriginalName() == oldName {
|
||||||
|
matches := referralCandidates.GetMatchingResourcesByOriginalId(id.Equals)
|
||||||
|
// If there's more than one match,
|
||||||
|
// filter the matches by prefix and suffix
|
||||||
|
if len(matches) > 1 {
|
||||||
|
filteredMatches := filterReferralCandidates(referrer, matches, target)
|
||||||
|
if len(filteredMatches) > 1 {
|
||||||
|
return "", "", fmt.Errorf(
|
||||||
|
"multiple matches for %s:\n %v",
|
||||||
|
id, getIds(filteredMatches))
|
||||||
|
}
|
||||||
|
// Check is the match the resource we are working on
|
||||||
|
if len(filteredMatches) == 0 || res != filteredMatches[0] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// In the resource, note that it is referenced
|
||||||
|
// by the referrer.
|
||||||
|
res.AppendRefBy(referrer.CurId())
|
||||||
|
// Return transformed name of the object,
|
||||||
|
// complete with prefixes, hashes, etc.
|
||||||
|
return res.GetName(), res.GetNamespace(), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return oldName, "", nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func getIds(rs []*resource.Resource) string {
|
// utility function to replace a simple string by the new name
|
||||||
|
func getSimpleNameField(
|
||||||
|
oldName string,
|
||||||
|
referrer *resource.Resource,
|
||||||
|
target resid.Gvk,
|
||||||
|
referralCandidates resmap.ResMap,
|
||||||
|
referralCandidateSubset []*resource.Resource,
|
||||||
|
isRoleRef bool) (string, error) {
|
||||||
|
|
||||||
|
newName, _, err := selectReferral(oldName, referrer, target,
|
||||||
|
referralCandidates, referralCandidateSubset, isRoleRef)
|
||||||
|
|
||||||
|
return newName, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func getIds(rs []*resource.Resource) []string {
|
||||||
var result []string
|
var result []string
|
||||||
for _, r := range rs {
|
for _, r := range rs {
|
||||||
result = append(result, r.CurId().String())
|
result = append(result, r.CurId().String()+"\n")
|
||||||
}
|
}
|
||||||
return strings.Join(result, ", ")
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkEqual(k, a, b string) error {
|
// utility function to replace name field within a map RNode
|
||||||
if a != b {
|
// and leverage the namespace field.
|
||||||
return fmt.Errorf(
|
func setNameAndNs(
|
||||||
"node-referrerOriginal '%s' mismatch '%s' != '%s'",
|
in *yaml.RNode,
|
||||||
k, a, b)
|
referrer *resource.Resource,
|
||||||
}
|
target resid.Gvk,
|
||||||
return nil
|
referralCandidates resmap.ResMap,
|
||||||
}
|
isRoleRef bool) error {
|
||||||
|
|
||||||
func (f Filter) confirmNodeMatchesReferrer(node *yaml.RNode) error {
|
if in.YNode().Kind != yaml.MappingNode {
|
||||||
meta, err := node.GetMeta()
|
return fmt.Errorf("expect a mapping node")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get name field
|
||||||
|
nameNode, err := in.Pipe(yaml.FieldMatcher{
|
||||||
|
Name: "name",
|
||||||
|
})
|
||||||
|
if err != nil || nameNode == nil {
|
||||||
|
return fmt.Errorf("cannot find field 'name' in node")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get namespace field
|
||||||
|
namespaceNode, err := in.Pipe(yaml.FieldMatcher{
|
||||||
|
Name: "namespace",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error when find field 'namespace'")
|
||||||
|
}
|
||||||
|
|
||||||
|
// check is namespace matched
|
||||||
|
// name will bot be updated if the namespace doesn't match
|
||||||
|
subset := referralCandidates.Resources()
|
||||||
|
if namespaceNode != nil {
|
||||||
|
namespace := namespaceNode.YNode().Value
|
||||||
|
bynamespace := referralCandidates.GroupedByOriginalNamespace()
|
||||||
|
if _, ok := bynamespace[namespace]; !ok {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
subset = bynamespace[namespace]
|
||||||
|
}
|
||||||
|
|
||||||
|
oldName := nameNode.YNode().Value
|
||||||
|
newname, newnamespace, err := selectReferral(oldName, referrer, target,
|
||||||
|
referralCandidates, subset, isRoleRef)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
gvk := f.Referrer.GetGvk()
|
|
||||||
if err = checkEqual(
|
if (newname == oldName) && (newnamespace == "") {
|
||||||
"APIVersion", meta.APIVersion, gvk.ApiVersion()); err != nil {
|
// no candidate found.
|
||||||
return err
|
return nil
|
||||||
}
|
}
|
||||||
if err = checkEqual(
|
|
||||||
"Kind", meta.Kind, gvk.Kind); err != nil {
|
// set name
|
||||||
return err
|
in.Pipe(yaml.FieldSetter{
|
||||||
}
|
Name: "name",
|
||||||
if err = checkEqual(
|
StringValue: newname,
|
||||||
"Name", meta.Name, f.Referrer.GetName()); err != nil {
|
})
|
||||||
return err
|
if newnamespace != "" {
|
||||||
}
|
// We don't want value "" to replace value "default" since
|
||||||
if err = checkEqual(
|
// the empty string is handled as a wild card here not default namespace
|
||||||
"Namespace", meta.Namespace, f.Referrer.GetNamespace()); err != nil {
|
// by kubernetes.
|
||||||
return err
|
in.Pipe(yaml.FieldSetter{
|
||||||
|
Name: "namespace",
|
||||||
|
StringValue: newnamespace,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,23 +5,24 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNamerefFilter(t *testing.T) {
|
func TestNamerefFilter(t *testing.T) {
|
||||||
testCases := map[string]struct {
|
testCases := map[string]struct {
|
||||||
referrerOriginal string
|
input string
|
||||||
candidates string
|
candidates string
|
||||||
referrerFinal string
|
expected string
|
||||||
filter Filter
|
filter Filter
|
||||||
originalNames []string
|
originalNames []string
|
||||||
}{
|
}{
|
||||||
"simple scalar": {
|
"simple scalar": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -40,8 +41,8 @@ kind: NotSecret
|
|||||||
metadata:
|
metadata:
|
||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "newName2"},
|
originalNames: []string{"oldName", ""},
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -50,8 +51,8 @@ ref:
|
|||||||
name: newName
|
name: newName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -59,7 +60,7 @@ ref:
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
"sequence": {
|
"sequence": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -79,8 +80,8 @@ kind: NotSecret
|
|||||||
metadata:
|
metadata:
|
||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName1", "newName2"},
|
originalNames: []string{"oldName1", ""},
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -90,8 +91,8 @@ seq:
|
|||||||
- oldName2
|
- oldName2
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "seq"},
|
FieldSpec: types.FieldSpec{Path: "seq"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -99,7 +100,7 @@ seq:
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
"mapping": {
|
"mapping": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -118,8 +119,8 @@ kind: NotSecret
|
|||||||
metadata:
|
metadata:
|
||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "newName2"},
|
originalNames: []string{"oldName", ""},
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -128,8 +129,8 @@ map:
|
|||||||
name: newName
|
name: newName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
FieldSpec: types.FieldSpec{Path: "map"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -137,47 +138,40 @@ map:
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
"mapping with namespace": {
|
"mapping with namespace": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
name: dep
|
name: dep
|
||||||
namespace: someNs
|
|
||||||
map:
|
map:
|
||||||
name: oldName
|
name: oldName
|
||||||
namespace: someNs
|
namespace: oldNs
|
||||||
`,
|
`,
|
||||||
candidates: `
|
candidates: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Secret
|
kind: Secret
|
||||||
metadata:
|
metadata:
|
||||||
name: newName
|
name: newName
|
||||||
namespace: someNs
|
namespace: oldNs
|
||||||
---
|
---
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: NotSecret
|
kind: NotSecret
|
||||||
metadata:
|
metadata:
|
||||||
name: newName2
|
name: newName2
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: thirdName
|
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "oldName", "oldName"},
|
originalNames: []string{"oldName", ""},
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
name: dep
|
name: dep
|
||||||
namespace: someNs
|
|
||||||
map:
|
map:
|
||||||
name: newName
|
name: newName
|
||||||
namespace: someNs
|
namespace: oldNs
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
FieldSpec: types.FieldSpec{Path: "map"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -185,7 +179,7 @@ map:
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
"null value": {
|
"null value": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -204,8 +198,8 @@ kind: NotSecret
|
|||||||
metadata:
|
metadata:
|
||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "newName2"},
|
originalNames: []string{"oldName", ""},
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -214,8 +208,8 @@ map:
|
|||||||
name: null
|
name: null
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
FieldSpec: types.FieldSpec{Path: "map"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -226,8 +220,8 @@ map:
|
|||||||
|
|
||||||
for tn, tc := range testCases {
|
for tn, tc := range testCases {
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tn, func(t *testing.T) {
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
factory := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
referrer, err := factory.FromBytes([]byte(tc.input))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@@ -243,10 +237,10 @@ map:
|
|||||||
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
||||||
tc.filter.ReferralCandidates = candidates
|
tc.filter.ReferralCandidates = candidates
|
||||||
|
|
||||||
result := filtertest_test.RunFilter(t, tc.referrerOriginal, tc.filter)
|
|
||||||
if !assert.Equal(t,
|
if !assert.Equal(t,
|
||||||
strings.TrimSpace(tc.referrerFinal),
|
strings.TrimSpace(tc.expected),
|
||||||
strings.TrimSpace(result)) {
|
strings.TrimSpace(
|
||||||
|
filtertest_test.RunFilter(t, tc.input, tc.filter))) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -255,14 +249,14 @@ map:
|
|||||||
|
|
||||||
func TestNamerefFilterUnhappy(t *testing.T) {
|
func TestNamerefFilterUnhappy(t *testing.T) {
|
||||||
testCases := map[string]struct {
|
testCases := map[string]struct {
|
||||||
referrerOriginal string
|
input string
|
||||||
candidates string
|
candidates string
|
||||||
referrerFinal string
|
expected string
|
||||||
filter Filter
|
filter Filter
|
||||||
originalNames []string
|
originalNames []string
|
||||||
}{
|
}{
|
||||||
"multiple match": {
|
"multiple match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -282,10 +276,10 @@ metadata:
|
|||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "oldName"},
|
originalNames: []string{"oldName", "oldName"},
|
||||||
referrerFinal: "",
|
expected: "",
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -293,7 +287,7 @@ metadata:
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
"no name": {
|
"no name": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -313,10 +307,10 @@ metadata:
|
|||||||
name: newName2
|
name: newName2
|
||||||
`,
|
`,
|
||||||
originalNames: []string{"oldName", "oldName"},
|
originalNames: []string{"oldName", "oldName"},
|
||||||
referrerFinal: "",
|
expected: "",
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref"},
|
FieldSpec: types.FieldSpec{Path: "ref"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -327,8 +321,8 @@ metadata:
|
|||||||
|
|
||||||
for tn, tc := range testCases {
|
for tn, tc := range testCases {
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tn, func(t *testing.T) {
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
factory := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
referrer, err := factory.FromBytes([]byte(tc.input))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@@ -344,11 +338,11 @@ metadata:
|
|||||||
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
||||||
tc.filter.ReferralCandidates = candidates
|
tc.filter.ReferralCandidates = candidates
|
||||||
|
|
||||||
_, err = filtertest_test.RunFilterE(t, tc.referrerOriginal, tc.filter)
|
_, err = filtertest_test.RunFilterE(t, tc.input, tc.filter)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatalf("expect an error")
|
t.Fatalf("expect an error")
|
||||||
}
|
}
|
||||||
if tc.referrerFinal != "" && !assert.EqualError(t, err, tc.referrerFinal) {
|
if tc.expected != "" && !assert.EqualError(t, err, tc.expected) {
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -357,19 +351,19 @@ metadata:
|
|||||||
|
|
||||||
func TestCandidatesWithDifferentPrefixSuffix(t *testing.T) {
|
func TestCandidatesWithDifferentPrefixSuffix(t *testing.T) {
|
||||||
testCases := map[string]struct {
|
testCases := map[string]struct {
|
||||||
referrerOriginal string
|
input string
|
||||||
candidates string
|
candidates string
|
||||||
referrerFinal string
|
expected string
|
||||||
filter Filter
|
filter Filter
|
||||||
originalNames []string
|
originalNames []string
|
||||||
prefix []string
|
prefix []string
|
||||||
suffix []string
|
suffix []string
|
||||||
inputPrefix string
|
inputPrefix string
|
||||||
inputSuffix string
|
inputSuffix string
|
||||||
err bool
|
err bool
|
||||||
}{
|
}{
|
||||||
"prefix match": {
|
"prefix match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -393,7 +387,7 @@ metadata:
|
|||||||
suffix: []string{"", "suffix2"},
|
suffix: []string{"", "suffix2"},
|
||||||
inputPrefix: "prefix1",
|
inputPrefix: "prefix1",
|
||||||
inputSuffix: "",
|
inputSuffix: "",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -402,8 +396,8 @@ ref:
|
|||||||
name: newName
|
name: newName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -412,7 +406,7 @@ ref:
|
|||||||
err: false,
|
err: false,
|
||||||
},
|
},
|
||||||
"suffix match": {
|
"suffix match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -436,7 +430,7 @@ metadata:
|
|||||||
suffix: []string{"suffix1", "suffix2"},
|
suffix: []string{"suffix1", "suffix2"},
|
||||||
inputPrefix: "",
|
inputPrefix: "",
|
||||||
inputSuffix: "suffix1",
|
inputSuffix: "suffix1",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -445,8 +439,8 @@ ref:
|
|||||||
name: newName
|
name: newName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -455,7 +449,7 @@ ref:
|
|||||||
err: false,
|
err: false,
|
||||||
},
|
},
|
||||||
"prefix suffix both match": {
|
"prefix suffix both match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -479,7 +473,7 @@ metadata:
|
|||||||
suffix: []string{"suffix1", "suffix2"},
|
suffix: []string{"suffix1", "suffix2"},
|
||||||
inputPrefix: "prefix1",
|
inputPrefix: "prefix1",
|
||||||
inputSuffix: "suffix1",
|
inputSuffix: "suffix1",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -488,8 +482,8 @@ ref:
|
|||||||
name: newName
|
name: newName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -498,7 +492,7 @@ ref:
|
|||||||
err: false,
|
err: false,
|
||||||
},
|
},
|
||||||
"multiple match: both": {
|
"multiple match: both": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -522,10 +516,10 @@ metadata:
|
|||||||
suffix: []string{"suffix", "suffix"},
|
suffix: []string{"suffix", "suffix"},
|
||||||
inputPrefix: "prefix",
|
inputPrefix: "prefix",
|
||||||
inputSuffix: "suffix",
|
inputSuffix: "suffix",
|
||||||
referrerFinal: "",
|
expected: "",
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -534,7 +528,7 @@ metadata:
|
|||||||
err: true,
|
err: true,
|
||||||
},
|
},
|
||||||
"multiple match: only prefix": {
|
"multiple match: only prefix": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -558,10 +552,10 @@ metadata:
|
|||||||
suffix: []string{"", ""},
|
suffix: []string{"", ""},
|
||||||
inputPrefix: "prefix",
|
inputPrefix: "prefix",
|
||||||
inputSuffix: "",
|
inputSuffix: "",
|
||||||
referrerFinal: "",
|
expected: "",
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -570,7 +564,7 @@ metadata:
|
|||||||
err: true,
|
err: true,
|
||||||
},
|
},
|
||||||
"multiple match: only suffix": {
|
"multiple match: only suffix": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -594,10 +588,10 @@ metadata:
|
|||||||
suffix: []string{"suffix", "suffix"},
|
suffix: []string{"suffix", "suffix"},
|
||||||
inputPrefix: "",
|
inputPrefix: "",
|
||||||
inputSuffix: "suffix",
|
inputSuffix: "suffix",
|
||||||
referrerFinal: "",
|
expected: "",
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -606,7 +600,7 @@ metadata:
|
|||||||
err: true,
|
err: true,
|
||||||
},
|
},
|
||||||
"no match: neither match": {
|
"no match: neither match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -630,7 +624,7 @@ metadata:
|
|||||||
suffix: []string{"suffix1", "suffix2"},
|
suffix: []string{"suffix1", "suffix2"},
|
||||||
inputPrefix: "prefix",
|
inputPrefix: "prefix",
|
||||||
inputSuffix: "suffix",
|
inputSuffix: "suffix",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -639,8 +633,8 @@ ref:
|
|||||||
name: oldName
|
name: oldName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -649,7 +643,7 @@ ref:
|
|||||||
err: false,
|
err: false,
|
||||||
},
|
},
|
||||||
"no match: prefix doesn't match": {
|
"no match: prefix doesn't match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -673,7 +667,7 @@ metadata:
|
|||||||
suffix: []string{"suffix", "suffix"},
|
suffix: []string{"suffix", "suffix"},
|
||||||
inputPrefix: "prefix",
|
inputPrefix: "prefix",
|
||||||
inputSuffix: "suffix",
|
inputSuffix: "suffix",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -682,8 +676,8 @@ ref:
|
|||||||
name: oldName
|
name: oldName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -692,7 +686,7 @@ ref:
|
|||||||
err: false,
|
err: false,
|
||||||
},
|
},
|
||||||
"no match: suffix doesn't match": {
|
"no match: suffix doesn't match": {
|
||||||
referrerOriginal: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -716,7 +710,7 @@ metadata:
|
|||||||
suffix: []string{"suffix1", "suffix2"},
|
suffix: []string{"suffix1", "suffix2"},
|
||||||
inputPrefix: "prefix",
|
inputPrefix: "prefix",
|
||||||
inputSuffix: "suffix",
|
inputSuffix: "suffix",
|
||||||
referrerFinal: `
|
expected: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
@@ -725,8 +719,8 @@ ref:
|
|||||||
name: oldName
|
name: oldName
|
||||||
`,
|
`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
FieldSpec: types.FieldSpec{Path: "ref/name"},
|
||||||
ReferralTarget: resid.Gvk{
|
Target: resid.Gvk{
|
||||||
Group: "apps",
|
Group: "apps",
|
||||||
Version: "v1",
|
Version: "v1",
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
@@ -738,8 +732,8 @@ ref:
|
|||||||
|
|
||||||
for tn, tc := range testCases {
|
for tn, tc := range testCases {
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tn, func(t *testing.T) {
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
factory := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
referrer, err := factory.FromBytes([]byte(tc.input))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@@ -771,15 +765,13 @@ ref:
|
|||||||
|
|
||||||
if !tc.err {
|
if !tc.err {
|
||||||
if !assert.Equal(t,
|
if !assert.Equal(t,
|
||||||
strings.TrimSpace(tc.referrerFinal),
|
strings.TrimSpace(tc.expected),
|
||||||
strings.TrimSpace(
|
strings.TrimSpace(
|
||||||
filtertest_test.RunFilter(
|
filtertest_test.RunFilter(t, tc.input, tc.filter))) {
|
||||||
t, tc.referrerOriginal, tc.filter))) {
|
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
_, err := filtertest_test.RunFilterE(
|
_, err := filtertest_test.RunFilterE(t, tc.input, tc.filter)
|
||||||
t, tc.referrerOriginal, tc.filter)
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatalf("an error is expected")
|
t.Fatalf("an error is expected")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
package patchstrategicmerge
|
package patchstrategicmerge
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
"sigs.k8s.io/kustomize/kyaml/kio"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml/merge2"
|
"sigs.k8s.io/kustomize/kyaml/yaml/merge2"
|
||||||
@@ -16,7 +15,6 @@ type Filter struct {
|
|||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
var _ kio.Filter = Filter{}
|
||||||
|
|
||||||
// Filter does a strategic merge patch, which can delete nodes.
|
|
||||||
func (pf Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
func (pf Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
||||||
var result []*yaml.RNode
|
var result []*yaml.RNode
|
||||||
for i := range nodes {
|
for i := range nodes {
|
||||||
@@ -29,9 +27,7 @@ func (pf Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if !konfig.FlagEnableKyamlDefaultValue || r != nil {
|
result = append(result, r)
|
||||||
result = append(result, r)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,101 +15,6 @@ func TestFilter(t *testing.T) {
|
|||||||
patch *yaml.RNode
|
patch *yaml.RNode
|
||||||
expected string
|
expected string
|
||||||
}{
|
}{
|
||||||
"simple": {
|
|
||||||
input: `apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 1
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 999
|
|
||||||
`),
|
|
||||||
expected: `apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 999
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"nullMapEntry1": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
B:
|
|
||||||
C: Z
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"nullMapEntry2": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
B:
|
|
||||||
C: Z
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"simple patch": {
|
"simple patch": {
|
||||||
input: `
|
input: `
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
@@ -672,63 +577,6 @@ spec:
|
|||||||
ports:
|
ports:
|
||||||
- containerPort: 80
|
- containerPort: 80
|
||||||
- containerPort: 8080
|
- containerPort: 8080
|
||||||
`,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Test for issue #3513
|
|
||||||
// Currently broken; when one port has only containerPort, the output
|
|
||||||
// should not merge containerPort 8301 together
|
|
||||||
// This occurs because when protocol is missing on the first port,
|
|
||||||
// the merge code uses [containerPort] as the merge key rather than
|
|
||||||
// [containerPort, protocol]
|
|
||||||
"list map keys - protocol only present on some ports": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: consul
|
|
||||||
image: "hashicorp/consul:1.9.1"
|
|
||||||
ports:
|
|
||||||
- containerPort: 8500
|
|
||||||
name: http
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "TCP"
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "UDP"
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
labels:
|
|
||||||
test: label
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
labels:
|
|
||||||
test: label
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: consul
|
|
||||||
image: "hashicorp/consul:1.9.1"
|
|
||||||
ports:
|
|
||||||
- containerPort: 8500
|
|
||||||
name: http
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "UDP"
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "UDP"
|
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
// Package refvar contains a kio.Filter implementation of the kustomize
|
// Package refvar contains a kio.Filter implementation of the kustomize
|
||||||
// refvar transformer (find and replace $(FOO) style variables in strings).
|
// refvar transformer.
|
||||||
package refvar
|
package refvar
|
||||||
|
|||||||
@@ -8,13 +8,15 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
"sigs.k8s.io/kustomize/kyaml/kio"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
|
|
||||||
|
expansion2 "sigs.k8s.io/kustomize/api/internal/accumulator/expansion"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Filter updates $(VAR) style variables with values.
|
// Filter updates $(VAR) style variables with values.
|
||||||
// The fieldSpecs are the places to look for occurrences of $(VAR).
|
// The fieldSpecs are the places to look for occurrences of $(VAR).
|
||||||
type Filter struct {
|
type Filter struct {
|
||||||
MappingFunc MappingFunc `json:"mappingFunc,omitempty" yaml:"mappingFunc,omitempty"`
|
MappingFunc func(string) interface{} `json:"mappingFunc,omitempty" yaml:"mappingFunc,omitempty"`
|
||||||
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
||||||
@@ -47,21 +49,12 @@ func (f Filter) set(node *yaml.RNode) error {
|
|||||||
|
|
||||||
func updateNodeValue(node *yaml.Node, newValue interface{}) {
|
func updateNodeValue(node *yaml.Node, newValue interface{}) {
|
||||||
switch newValue := newValue.(type) {
|
switch newValue := newValue.(type) {
|
||||||
case int:
|
|
||||||
node.Value = strconv.FormatInt(int64(newValue), 10)
|
|
||||||
node.Tag = yaml.NodeTagInt
|
|
||||||
case int32:
|
|
||||||
node.Value = strconv.FormatInt(int64(newValue), 10)
|
|
||||||
node.Tag = yaml.NodeTagInt
|
|
||||||
case int64:
|
case int64:
|
||||||
node.Value = strconv.FormatInt(newValue, 10)
|
node.Value = strconv.FormatInt(newValue, 10)
|
||||||
node.Tag = yaml.NodeTagInt
|
node.Tag = yaml.NodeTagInt
|
||||||
case bool:
|
case bool:
|
||||||
node.SetString(strconv.FormatBool(newValue))
|
node.SetString(strconv.FormatBool(newValue))
|
||||||
node.Tag = yaml.NodeTagBool
|
node.Tag = yaml.NodeTagBool
|
||||||
case float32:
|
|
||||||
node.SetString(strconv.FormatFloat(float64(newValue), 'f', -1, 32))
|
|
||||||
node.Tag = yaml.NodeTagFloat
|
|
||||||
case float64:
|
case float64:
|
||||||
node.SetString(strconv.FormatFloat(newValue, 'f', -1, 64))
|
node.SetString(strconv.FormatFloat(newValue, 'f', -1, 64))
|
||||||
node.Tag = yaml.NodeTagFloat
|
node.Tag = yaml.NodeTagFloat
|
||||||
@@ -76,7 +69,7 @@ func (f Filter) setScalar(node *yaml.RNode) error {
|
|||||||
if !yaml.IsYNodeString(node.YNode()) {
|
if !yaml.IsYNodeString(node.YNode()) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
v := DoReplacements(node.YNode().Value, f.MappingFunc)
|
v := expansion2.Expand(node.YNode().Value, f.MappingFunc)
|
||||||
updateNodeValue(node.YNode(), v)
|
updateNodeValue(node.YNode(), v)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@@ -85,14 +78,12 @@ func (f Filter) setMap(node *yaml.RNode) error {
|
|||||||
contents := node.YNode().Content
|
contents := node.YNode().Content
|
||||||
for i := 0; i < len(contents); i += 2 {
|
for i := 0; i < len(contents); i += 2 {
|
||||||
if !yaml.IsYNodeString(contents[i]) {
|
if !yaml.IsYNodeString(contents[i]) {
|
||||||
return fmt.Errorf(
|
return fmt.Errorf("invalid map key: %s, type: %s", contents[i].Value, contents[i].Tag)
|
||||||
"invalid map key: value='%s', tag='%s'",
|
|
||||||
contents[i].Value, contents[i].Tag)
|
|
||||||
}
|
}
|
||||||
if !yaml.IsYNodeString(contents[i+1]) {
|
if !yaml.IsYNodeString(contents[i+1]) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newValue := DoReplacements(contents[i+1].Value, f.MappingFunc)
|
newValue := expansion2.Expand(contents[i+1].Value, f.MappingFunc)
|
||||||
updateNodeValue(contents[i+1], newValue)
|
updateNodeValue(contents[i+1], newValue)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@@ -103,7 +94,7 @@ func (f Filter) setSeq(node *yaml.RNode) error {
|
|||||||
if !yaml.IsYNodeString(item) {
|
if !yaml.IsYNodeString(item) {
|
||||||
return fmt.Errorf("invalid value type expect a string")
|
return fmt.Errorf("invalid value type expect a string")
|
||||||
}
|
}
|
||||||
newValue := DoReplacements(item.Value, f.MappingFunc)
|
newValue := expansion2.Expand(item.Value, f.MappingFunc)
|
||||||
updateNodeValue(item, newValue)
|
updateNodeValue(item, newValue)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|||||||
@@ -1,22 +1,18 @@
|
|||||||
package refvar_test
|
package refvar
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
. "sigs.k8s.io/kustomize/api/filters/refvar"
|
expansion2 "sigs.k8s.io/kustomize/api/internal/accumulator/expansion"
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
var makeMf = func(theMap map[string]interface{}) MappingFunc {
|
|
||||||
ignored := make(map[string]int)
|
|
||||||
return MakePrimitiveReplacer(ignored, theMap)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
func TestFilter(t *testing.T) {
|
||||||
|
replacementCounts := make(map[string]int)
|
||||||
|
|
||||||
testCases := map[string]struct {
|
testCases := map[string]struct {
|
||||||
input string
|
input string
|
||||||
@@ -39,7 +35,7 @@ metadata:
|
|||||||
spec:
|
spec:
|
||||||
replicas: 5`,
|
replicas: 5`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"VAR": int64(5),
|
"VAR": int64(5),
|
||||||
}),
|
}),
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
||||||
@@ -61,7 +57,7 @@ metadata:
|
|||||||
spec:
|
spec:
|
||||||
replicas: 1`,
|
replicas: 1`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"VAR": int64(5),
|
"VAR": int64(5),
|
||||||
}),
|
}),
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
||||||
@@ -83,7 +79,7 @@ metadata:
|
|||||||
spec:
|
spec:
|
||||||
replicas: 1`,
|
replicas: 1`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"VAR": int64(5),
|
"VAR": int64(5),
|
||||||
}),
|
}),
|
||||||
FieldSpec: types.FieldSpec{Path: "a/b/c"},
|
FieldSpec: types.FieldSpec{Path: "a/b/c"},
|
||||||
@@ -115,7 +111,7 @@ data:
|
|||||||
- false
|
- false
|
||||||
- 1.23`,
|
- 1.23`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"FOO": "foo",
|
"FOO": "foo",
|
||||||
"BAR": "bar",
|
"BAR": "bar",
|
||||||
"BOOL": false,
|
"BOOL": false,
|
||||||
@@ -146,7 +142,7 @@ data:
|
|||||||
BAZ: $(BAZ)
|
BAZ: $(BAZ)
|
||||||
PLUS: foo+bar`,
|
PLUS: foo+bar`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"FOO": "foo",
|
"FOO": "foo",
|
||||||
"BAR": "bar",
|
"BAR": "bar",
|
||||||
}),
|
}),
|
||||||
@@ -185,7 +181,7 @@ data:
|
|||||||
SLICE:
|
SLICE:
|
||||||
- $(FOO)`,
|
- $(FOO)`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"FOO": "foo",
|
"FOO": "foo",
|
||||||
"BAR": "bar",
|
"BAR": "bar",
|
||||||
}),
|
}),
|
||||||
@@ -208,10 +204,8 @@ metadata:
|
|||||||
data:
|
data:
|
||||||
FOO: null`,
|
FOO: null`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{}),
|
||||||
// no replacements!
|
FieldSpec: types.FieldSpec{Path: "data/FOO"},
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data/FOO"},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -229,6 +223,8 @@ data:
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestFilterUnhappy(t *testing.T) {
|
func TestFilterUnhappy(t *testing.T) {
|
||||||
|
replacementCounts := make(map[string]int)
|
||||||
|
|
||||||
testCases := map[string]struct {
|
testCases := map[string]struct {
|
||||||
input string
|
input string
|
||||||
expectedError string
|
expectedError string
|
||||||
@@ -243,8 +239,7 @@ metadata:
|
|||||||
data:
|
data:
|
||||||
slice:
|
slice:
|
||||||
- false`,
|
- false`,
|
||||||
expectedError: `considering field 'data/slice' of object
|
expectedError: `obj 'apiVersion: apps/v1
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
name: dep
|
name: dep
|
||||||
@@ -253,9 +248,9 @@ metadata:
|
|||||||
data:
|
data:
|
||||||
slice:
|
slice:
|
||||||
- false
|
- false
|
||||||
: invalid value type expect a string`,
|
' at path 'data/slice': invalid value type expect a string`,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"VAR": int64(5),
|
"VAR": int64(5),
|
||||||
}),
|
}),
|
||||||
FieldSpec: types.FieldSpec{Path: "data/slice"},
|
FieldSpec: types.FieldSpec{Path: "data/slice"},
|
||||||
@@ -269,8 +264,7 @@ metadata:
|
|||||||
name: dep
|
name: dep
|
||||||
data:
|
data:
|
||||||
1: str`,
|
1: str`,
|
||||||
expectedError: `considering field 'data' of object
|
expectedError: `obj 'apiVersion: apps/v1
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
name: dep
|
name: dep
|
||||||
@@ -278,9 +272,9 @@ metadata:
|
|||||||
config.kubernetes.io/index: '0'
|
config.kubernetes.io/index: '0'
|
||||||
data:
|
data:
|
||||||
1: str
|
1: str
|
||||||
: invalid map key: value='1', tag='` + yaml.NodeTagInt + `'`,
|
' at path 'data': invalid map key: 1, type: ` + yaml.NodeTagInt,
|
||||||
filter: Filter{
|
filter: Filter{
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
MappingFunc: expansion2.MappingFuncFor(replacementCounts, map[string]interface{}{
|
||||||
"VAR": int64(5),
|
"VAR": int64(5),
|
||||||
}),
|
}),
|
||||||
FieldSpec: types.FieldSpec{Path: "data"},
|
FieldSpec: types.FieldSpec{Path: "data"},
|
||||||
|
|||||||
12
api/go.mod
12
api/go.mod
@@ -1,26 +1,26 @@
|
|||||||
module sigs.k8s.io/kustomize/api
|
module sigs.k8s.io/kustomize/api
|
||||||
|
|
||||||
go 1.15
|
go 1.14
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/evanphx/json-patch v4.5.0+incompatible
|
github.com/evanphx/json-patch v4.5.0+incompatible
|
||||||
github.com/go-errors/errors v1.0.1
|
|
||||||
github.com/go-openapi/spec v0.19.5
|
github.com/go-openapi/spec v0.19.5
|
||||||
github.com/golangci/golangci-lint v1.21.0
|
github.com/golangci/golangci-lint v1.21.0
|
||||||
github.com/google/go-cmp v0.3.0
|
github.com/google/go-cmp v0.3.0
|
||||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510
|
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510
|
||||||
github.com/hashicorp/go-multierror v1.1.0
|
github.com/hashicorp/go-multierror v1.1.0
|
||||||
github.com/imdario/mergo v0.3.5
|
|
||||||
github.com/pkg/errors v0.8.1
|
github.com/pkg/errors v0.8.1
|
||||||
github.com/stretchr/testify v1.4.0
|
github.com/stretchr/testify v1.4.0
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf
|
github.com/yujunz/go-getter v1.4.1-lite
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e
|
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e
|
||||||
gopkg.in/yaml.v2 v2.3.0
|
gopkg.in/yaml.v2 v2.3.0
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c
|
gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71
|
||||||
k8s.io/api v0.17.0
|
k8s.io/api v0.17.0
|
||||||
k8s.io/apimachinery v0.17.0
|
k8s.io/apimachinery v0.17.0
|
||||||
k8s.io/client-go v0.17.0
|
k8s.io/client-go v0.17.0
|
||||||
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a
|
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.7
|
sigs.k8s.io/kustomize/kyaml v0.9.3
|
||||||
sigs.k8s.io/yaml v1.2.0
|
sigs.k8s.io/yaml v1.2.0
|
||||||
)
|
)
|
||||||
|
|
||||||
|
replace sigs.k8s.io/kustomize/kyaml v0.9.3 => ../kyaml
|
||||||
|
|||||||
21
api/go.sum
21
api/go.sum
@@ -158,7 +158,6 @@ github.com/go-toolsmith/strparse v1.0.0 h1:Vcw78DnpCAKlM20kSbAyO4mPfJn/lyYA4BJUD
|
|||||||
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
||||||
github.com/go-toolsmith/typep v1.0.0 h1:zKymWyA1TRYvqYrYDrfEMZULyrhcnGY3x7LDKU2XQaA=
|
github.com/go-toolsmith/typep v1.0.0 h1:zKymWyA1TRYvqYrYDrfEMZULyrhcnGY3x7LDKU2XQaA=
|
||||||
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
||||||
github.com/gobuffalo/here v0.6.0/go.mod h1:wAG085dHOYqUpf+Ap+WOdrPTp5IYcDAs/x7PLa8Y5fM=
|
|
||||||
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
|
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
|
||||||
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
||||||
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b h1:ekuhfTjngPhisSjOJ0QWKpPQE8/rbknHaes6WVJj5Hw=
|
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b h1:ekuhfTjngPhisSjOJ0QWKpPQE8/rbknHaes6WVJj5Hw=
|
||||||
@@ -167,7 +166,6 @@ github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7a
|
|||||||
github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
|
github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
|
||||||
github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d h1:3PaI8p3seN09VjbTYC/QWlUZdZ1qS1zGjy7LH2Wt07I=
|
github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d h1:3PaI8p3seN09VjbTYC/QWlUZdZ1qS1zGjy7LH2Wt07I=
|
||||||
github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
|
github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58=
|
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
||||||
github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
||||||
github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
||||||
@@ -227,7 +225,6 @@ github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+
|
|||||||
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d h1:7XGaL1e6bYS1yIonGp9761ExpPPV1ui0SAC59Yube9k=
|
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d h1:7XGaL1e6bYS1yIonGp9761ExpPPV1ui0SAC59Yube9k=
|
||||||
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY=
|
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY=
|
||||||
github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8=
|
github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8=
|
||||||
github.com/gorilla/websocket v1.4.0 h1:WDFjx/TMzVgy9VdMMQi2K2Emtwi2QcUQsztZ/zLaH/Q=
|
|
||||||
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
|
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
|
||||||
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3 h1:JVnpOZS+qxli+rgVl98ILOXVNbW+kb5wcxeGx8ShUIw=
|
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3 h1:JVnpOZS+qxli+rgVl98ILOXVNbW+kb5wcxeGx8ShUIw=
|
||||||
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3/go.mod h1:eEOZF4jCKGi+aprrirO9e7WKB3beBRtWgqGunKl6pKE=
|
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3/go.mod h1:eEOZF4jCKGi+aprrirO9e7WKB3beBRtWgqGunKl6pKE=
|
||||||
@@ -252,7 +249,6 @@ github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
|||||||
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
||||||
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
||||||
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
github.com/imdario/mergo v0.3.5 h1:JboBksRwiiAJWvIYJVo46AfV+IAIKZpfrSzVKj42R4Q=
|
|
||||||
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
||||||
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
|
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
|
||||||
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
||||||
@@ -291,7 +287,6 @@ github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN
|
|||||||
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
||||||
github.com/mailru/easyjson v0.7.0 h1:aizVhC/NAAcKWb+5QsU1iNOZb4Yws5UO2I+aIprQITM=
|
github.com/mailru/easyjson v0.7.0 h1:aizVhC/NAAcKWb+5QsU1iNOZb4Yws5UO2I+aIprQITM=
|
||||||
github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
|
github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
|
||||||
github.com/markbates/pkger v0.17.1/go.mod h1:0JoVlrol20BSywW79rN3kdFFsE5xYM+rSCQDXbLhiuI=
|
|
||||||
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb h1:RHba4YImhrUVQDHUCe2BNSOz4tVy2yGyXhvYDvxGgeE=
|
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb h1:RHba4YImhrUVQDHUCe2BNSOz4tVy2yGyXhvYDvxGgeE=
|
||||||
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
||||||
github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
|
github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
|
||||||
@@ -370,7 +365,6 @@ github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e h1:MZM7FHLqUHYI0Y/mQAt
|
|||||||
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk=
|
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk=
|
||||||
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041 h1:llrF3Fs4018ePo4+G/HV/uQUqEI1HMDjCeOf2V6puPc=
|
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041 h1:llrF3Fs4018ePo4+G/HV/uQUqEI1HMDjCeOf2V6puPc=
|
||||||
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ=
|
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ=
|
||||||
github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo=
|
|
||||||
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
|
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
|
||||||
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
||||||
github.com/sirupsen/logrus v1.4.2 h1:SPIRibHv4MatM3XXNO2BJeFLZwZ2LvZgfQ5+UNI2im4=
|
github.com/sirupsen/logrus v1.4.2 h1:SPIRibHv4MatM3XXNO2BJeFLZwZ2LvZgfQ5+UNI2im4=
|
||||||
@@ -412,8 +406,8 @@ github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e/go.mod h1:Qimiff
|
|||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
||||||
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
||||||
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
||||||
github.com/ulikunitz/xz v0.5.8 h1:ERv8V6GKqVi23rgu5cj9pVfVzJbOqAY2Ntl88O6c2nQ=
|
github.com/ulikunitz/xz v0.5.5 h1:pFrO0lVpTBXLpYw+pnLj6TbvHuyjXMfjGeCwSqCVwok=
|
||||||
github.com/ulikunitz/xz v0.5.8/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
|
github.com/ulikunitz/xz v0.5.5/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8=
|
||||||
github.com/ultraware/funlen v0.0.2 h1:Av96YVBwwNSe4MLR7iI/BIa3VyI7/djnto/pK3Uxbdo=
|
github.com/ultraware/funlen v0.0.2 h1:Av96YVBwwNSe4MLR7iI/BIa3VyI7/djnto/pK3Uxbdo=
|
||||||
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
||||||
github.com/ultraware/whitespace v0.0.4 h1:If7Va4cM03mpgrNH9k49/VOicWpGoG70XPBFFODYDsg=
|
github.com/ultraware/whitespace v0.0.4 h1:If7Va4cM03mpgrNH9k49/VOicWpGoG70XPBFFODYDsg=
|
||||||
@@ -429,8 +423,8 @@ github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q
|
|||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca h1:1CFlNzQhALwjS9mBAUkycX616GzgsuYUOCHA5+HSlXI=
|
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca h1:1CFlNzQhALwjS9mBAUkycX616GzgsuYUOCHA5+HSlXI=
|
||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
|
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
|
||||||
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf h1:gvEmqF83GB8R5XtrMseJb6A6R0OCtNAS8f4TmZg2dGc=
|
github.com/yujunz/go-getter v1.4.1-lite h1:FhvNc94AXMZkfqUwfMKhnQEC9phkphSGdPTL7tIdhOM=
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf/go.mod h1:bL0Pr07HEdsMZ1WBqZIxXj96r5LnFsY4LgPaPEGkw1k=
|
github.com/yujunz/go-getter v1.4.1-lite/go.mod h1:sbmqxXjyLunH1PkF3n7zSlnVeMvmYUuIl9ZVs/7NyCc=
|
||||||
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
||||||
go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
||||||
go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
||||||
@@ -567,13 +561,12 @@ gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bl
|
|||||||
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
|
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
|
||||||
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo=
|
gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71 h1:Xe2gvTZUJpsvOWUnvmL/tmhVBZUmHSvLbMjRj6NUUKo=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.1-2019.2.3 h1:3JgtbtFHMiCmsznwGVTUWbgGov+pVqnlf1dEJTNAXeM=
|
honnef.co/go/tools v0.0.1-2019.2.3 h1:3JgtbtFHMiCmsznwGVTUWbgGov+pVqnlf1dEJTNAXeM=
|
||||||
@@ -599,8 +592,6 @@ mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b h1:DxJ5nJdkhDlLok9K6qO+5290kphD
|
|||||||
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
||||||
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f h1:Cq7MalBHYACRd6EesksG1Q8EoIAKOsiZviGKbOLIej4=
|
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f h1:Cq7MalBHYACRd6EesksG1Q8EoIAKOsiZviGKbOLIej4=
|
||||||
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.7 h1:r0r8UEL0bL7X56HKUmhJZ+TP+nvRNGrDHHSLO7izlcQ=
|
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.7/go.mod h1:K9yg1k/HB/6xNOf5VH3LhTo1DK9/5ykSZO5uIv+Y/1k=
|
|
||||||
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
||||||
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
||||||
sigs.k8s.io/yaml v1.2.0 h1:kr/MCeFWJWTwyaHoR9c8EjH9OumOmoF9YGiZd7lFm/Q=
|
sigs.k8s.io/yaml v1.2.0 h1:kr/MCeFWJWTwyaHoR9c8EjH9OumOmoF9YGiZd7lFm/Q=
|
||||||
|
|||||||
@@ -43,12 +43,12 @@ type Kunstructured interface {
|
|||||||
// Several uses.
|
// Several uses.
|
||||||
Copy() Kunstructured
|
Copy() Kunstructured
|
||||||
|
|
||||||
// GetAnnotations returns the k8s annotations.
|
// Used by Resource.Replace, which in turn is used in many places, e.g.
|
||||||
|
// - resource.Resource.Merge
|
||||||
|
// - resWrangler.appendReplaceOrMerge (AbsorbAll)
|
||||||
|
// - api.internal.k8sdeps.transformer.patch.conflictdetector
|
||||||
GetAnnotations() map[string]string
|
GetAnnotations() map[string]string
|
||||||
|
|
||||||
// GetData returns a top-level "data" field, as in a ConfigMap.
|
|
||||||
GetDataMap() map[string]string
|
|
||||||
|
|
||||||
// Used by ResAccumulator and ReplacementTransformer.
|
// Used by ResAccumulator and ReplacementTransformer.
|
||||||
GetFieldValue(string) (interface{}, error)
|
GetFieldValue(string) (interface{}, error)
|
||||||
|
|
||||||
@@ -58,7 +58,7 @@ type Kunstructured interface {
|
|||||||
// Used by resource.Factory.SliceFromBytes
|
// Used by resource.Factory.SliceFromBytes
|
||||||
GetKind() string
|
GetKind() string
|
||||||
|
|
||||||
// GetLabels returns the k8s labels.
|
// Used by Resource.Replace
|
||||||
GetLabels() map[string]string
|
GetLabels() map[string]string
|
||||||
|
|
||||||
// Used by Resource.CurId and resource factory.
|
// Used by Resource.CurId and resource factory.
|
||||||
@@ -84,22 +84,19 @@ type Kunstructured interface {
|
|||||||
// Used by resWrangler.Select
|
// Used by resWrangler.Select
|
||||||
MatchesLabelSelector(selector string) (bool, error)
|
MatchesLabelSelector(selector string) (bool, error)
|
||||||
|
|
||||||
// SetAnnotations replaces the k8s annotations.
|
// Used by Resource.Replace.
|
||||||
SetAnnotations(map[string]string)
|
SetAnnotations(map[string]string)
|
||||||
|
|
||||||
// SetDataMap sets a top-level "data" field, as in a ConfigMap.
|
|
||||||
SetDataMap(map[string]string)
|
|
||||||
|
|
||||||
// Used by PatchStrategicMergeTransformer.
|
// Used by PatchStrategicMergeTransformer.
|
||||||
SetGvk(resid.Gvk)
|
SetGvk(resid.Gvk)
|
||||||
|
|
||||||
// SetLabels replaces the k8s labels.
|
// Used by Resource.Replace and used to remove "validated by" labels.
|
||||||
SetLabels(map[string]string)
|
SetLabels(map[string]string)
|
||||||
|
|
||||||
// SetName changes the name.
|
// Used by Resource.Replace.
|
||||||
SetName(string)
|
SetName(string)
|
||||||
|
|
||||||
// SetNamespace changes the namespace.
|
// Used by Resource.Replace.
|
||||||
SetNamespace(string)
|
SetNamespace(string)
|
||||||
|
|
||||||
// Needed, for now, by kyaml/filtersutil.ApplyToJSON.
|
// Needed, for now, by kyaml/filtersutil.ApplyToJSON.
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
package refvar
|
// Package expansion provides functions find and replace $(FOO) style variables in strings.
|
||||||
|
package expansion
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@@ -17,64 +17,38 @@ const (
|
|||||||
|
|
||||||
// syntaxWrap returns the input string wrapped by the expansion syntax.
|
// syntaxWrap returns the input string wrapped by the expansion syntax.
|
||||||
func syntaxWrap(input string) string {
|
func syntaxWrap(input string) string {
|
||||||
var sb strings.Builder
|
return string(operator) + string(referenceOpener) + input + string(referenceCloser)
|
||||||
sb.WriteByte(operator)
|
|
||||||
sb.WriteByte(referenceOpener)
|
|
||||||
sb.WriteString(input)
|
|
||||||
sb.WriteByte(referenceCloser)
|
|
||||||
return sb.String()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MappingFunc maps a string to anything.
|
// MappingFuncFor returns a mapping function for use with Expand that
|
||||||
type MappingFunc func(string) interface{}
|
// implements the expansion semantics defined in the expansion spec; it
|
||||||
|
// returns the input string wrapped in the expansion syntax if no mapping
|
||||||
// MakePrimitiveReplacer returns a MappingFunc that uses a map to do
|
// for the input is found.
|
||||||
// replacements, and a histogram to count map hits.
|
func MappingFuncFor(
|
||||||
//
|
counts map[string]int,
|
||||||
// Func behavior:
|
context ...map[string]interface{}) func(string) interface{} {
|
||||||
//
|
return func(input string) interface{} {
|
||||||
// If the input key is NOT found in the map, the key is wrapped up as
|
for _, vars := range context {
|
||||||
// as a variable declaration string and returned, e.g. key FOO becomes $(FOO).
|
val, ok := vars[input]
|
||||||
// This string is presumably put back where it was found, and might get replaced
|
if ok {
|
||||||
// later.
|
counts[input]++
|
||||||
//
|
switch typedV := val.(type) {
|
||||||
// If the key is found in the map, the value is returned if it is a primitive
|
case string, int64, float64, bool:
|
||||||
// type (string, bool, number), and the hit is counted.
|
return typedV
|
||||||
//
|
default:
|
||||||
// If it's not a primitive type (e.g. a map, struct, func, etc.) then this
|
return syntaxWrap(input)
|
||||||
// function doesn't know what to do with it and it returns the key wrapped up
|
}
|
||||||
// again as if it had not been replaced. This should probably be an error.
|
|
||||||
func MakePrimitiveReplacer(
|
|
||||||
counts map[string]int, someMap map[string]interface{}) MappingFunc {
|
|
||||||
return func(key string) interface{} {
|
|
||||||
if value, ok := someMap[key]; ok {
|
|
||||||
switch typedV := value.(type) {
|
|
||||||
case string, int, int32, int64, float32, float64, bool:
|
|
||||||
counts[key]++
|
|
||||||
return typedV
|
|
||||||
default:
|
|
||||||
// If the value is some complicated type (e.g. a map or struct),
|
|
||||||
// this function doesn't know how to jam it into a string,
|
|
||||||
// so just pretend it was a cache miss.
|
|
||||||
// Likely this should be an error instead of a silent failure,
|
|
||||||
// since the programmer passed an impossible value.
|
|
||||||
log.Printf(
|
|
||||||
"MakePrimitiveReplacer: bad replacement type=%T val=%v",
|
|
||||||
typedV, typedV)
|
|
||||||
return syntaxWrap(key)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If unable to return the mapped variable, return it
|
return syntaxWrap(input)
|
||||||
// as it was found, and a later mapping might be able to
|
|
||||||
// replace it.
|
|
||||||
return syntaxWrap(key)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoReplacements replaces variable references in the input string
|
// Expand replaces variable references in the input string according to
|
||||||
// using the mapping function.
|
// the expansion spec using the given mapping function to resolve the
|
||||||
func DoReplacements(input string, mapping MappingFunc) interface{} {
|
// values of variables.
|
||||||
var buf strings.Builder
|
func Expand(input string, mapping func(string) interface{}) interface{} {
|
||||||
|
var buf bytes.Buffer
|
||||||
checkpoint := 0
|
checkpoint := 0
|
||||||
for cursor := 0; cursor < len(input); cursor++ {
|
for cursor := 0; cursor < len(input); cursor++ {
|
||||||
if input[cursor] == operator && cursor+1 < len(input) {
|
if input[cursor] == operator && cursor+1 < len(input) {
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
package refvar_test
|
package expansion_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
. "sigs.k8s.io/kustomize/api/internal/accumulator/expansion"
|
||||||
. "sigs.k8s.io/kustomize/api/filters/refvar"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type expected struct {
|
type expected struct {
|
||||||
@@ -16,48 +15,6 @@ type expected struct {
|
|||||||
edited string
|
edited string
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPrimitiveReplacer(t *testing.T) {
|
|
||||||
varCounts := make(map[string]int)
|
|
||||||
f := MakePrimitiveReplacer(
|
|
||||||
varCounts,
|
|
||||||
map[string]interface{}{
|
|
||||||
"FOO": "bar",
|
|
||||||
"ZOO": "$(FOO)-1",
|
|
||||||
"BLU": "$(ZOO)-2",
|
|
||||||
"EIGHT": 8,
|
|
||||||
"PI": 3.14159,
|
|
||||||
"ZINT": "$(INT)",
|
|
||||||
"BOOL": "true",
|
|
||||||
"HUGENUMBER": int64(9223372036854775807),
|
|
||||||
"CRAZYMAP": map[string]int{"crazy": 200},
|
|
||||||
"ZBOOL": "$(BOOL)",
|
|
||||||
})
|
|
||||||
assert.Equal(t, "$()", f(""))
|
|
||||||
assert.Equal(t, "$( )", f(" "))
|
|
||||||
assert.Equal(t, "$(florida)", f("florida"))
|
|
||||||
assert.Equal(t, "$(0)", f("0"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, 8, f("EIGHT"))
|
|
||||||
assert.Equal(t, 8, f("EIGHT"))
|
|
||||||
assert.Equal(t, 3.14159, f("PI"))
|
|
||||||
assert.Equal(t, "true", f("BOOL"))
|
|
||||||
assert.Equal(t, int64(9223372036854775807), f("HUGENUMBER"))
|
|
||||||
assert.Equal(t, "$(FOO)-1", f("ZOO"))
|
|
||||||
assert.Equal(t, "$(CRAZYMAP)", f("CRAZYMAP"))
|
|
||||||
assert.Equal(t,
|
|
||||||
map[string]int{
|
|
||||||
"FOO": 3,
|
|
||||||
"EIGHT": 2,
|
|
||||||
"BOOL": 1,
|
|
||||||
"PI": 1,
|
|
||||||
"ZOO": 1,
|
|
||||||
"HUGENUMBER": 1,
|
|
||||||
},
|
|
||||||
varCounts)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMapReference(t *testing.T) {
|
func TestMapReference(t *testing.T) {
|
||||||
type env struct {
|
type env struct {
|
||||||
Name string
|
Name string
|
||||||
@@ -94,7 +51,7 @@ func TestMapReference(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
varMap := map[string]interface{}{
|
declaredEnv := map[string]interface{}{
|
||||||
"FOO": "bar",
|
"FOO": "bar",
|
||||||
"ZOO": "$(FOO)-1",
|
"ZOO": "$(FOO)-1",
|
||||||
"BLU": "$(ZOO)-2",
|
"BLU": "$(ZOO)-2",
|
||||||
@@ -104,11 +61,11 @@ func TestMapReference(t *testing.T) {
|
|||||||
"ZBOOL": "$(BOOL)",
|
"ZBOOL": "$(BOOL)",
|
||||||
}
|
}
|
||||||
|
|
||||||
varCounts := make(map[string]int)
|
counts := make(map[string]int)
|
||||||
|
mapping := MappingFuncFor(counts, declaredEnv)
|
||||||
|
|
||||||
for _, env := range envs {
|
for _, env := range envs {
|
||||||
varMap[env.Name] = DoReplacements(
|
declaredEnv[env.Name] = Expand(fmt.Sprintf("%v", env.Value), mapping)
|
||||||
fmt.Sprintf("%v", env.Value),
|
|
||||||
MakePrimitiveReplacer(varCounts, varMap))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
expectedEnv := map[string]expected{
|
expectedEnv := map[string]expected{
|
||||||
@@ -122,20 +79,45 @@ func TestMapReference(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for k, v := range expectedEnv {
|
for k, v := range expectedEnv {
|
||||||
if e, a := v, varMap[k]; e.edited != a || e.count != varCounts[k] {
|
if e, a := v, declaredEnv[k]; e.edited != a || e.count != counts[k] {
|
||||||
t.Errorf("Expected %v count=%d, got %v count=%d",
|
t.Errorf("Expected %v count=%d, got %v count=%d",
|
||||||
e.edited, e.count, a, varCounts[k])
|
e.edited, e.count, a, counts[k])
|
||||||
} else {
|
} else {
|
||||||
delete(varMap, k)
|
delete(declaredEnv, k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(varMap) != 0 {
|
if len(declaredEnv) != 0 {
|
||||||
t.Errorf("Unexpected keys in declared env: %v", varMap)
|
t.Errorf("Unexpected keys in declared env: %v", declaredEnv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestMapping(t *testing.T) {
|
func TestMapping(t *testing.T) {
|
||||||
|
context := map[string]interface{}{
|
||||||
|
"VAR_A": "A",
|
||||||
|
"VAR_B": "B",
|
||||||
|
"VAR_C": "C",
|
||||||
|
"VAR_REF": "$(VAR_A)",
|
||||||
|
"VAR_EMPTY": "",
|
||||||
|
}
|
||||||
|
doExpansionTest(t, context)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMappingDual(t *testing.T) {
|
||||||
|
context := map[string]interface{}{
|
||||||
|
"VAR_A": "A",
|
||||||
|
"VAR_EMPTY": "",
|
||||||
|
}
|
||||||
|
context2 := map[string]interface{}{
|
||||||
|
"VAR_B": "B",
|
||||||
|
"VAR_C": "C",
|
||||||
|
"VAR_REF": "$(VAR_A)",
|
||||||
|
}
|
||||||
|
|
||||||
|
doExpansionTest(t, context, context2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func doExpansionTest(t *testing.T, context ...map[string]interface{}) {
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
name string
|
name string
|
||||||
input string
|
input string
|
||||||
@@ -351,17 +333,11 @@ func TestMapping(t *testing.T) {
|
|||||||
expected: "\n",
|
expected: "\n",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range cases {
|
for _, tc := range cases {
|
||||||
counts := make(map[string]int)
|
counts := make(map[string]int)
|
||||||
expanded := DoReplacements(
|
mapping := MappingFuncFor(counts, context...)
|
||||||
fmt.Sprintf("%v", tc.input),
|
expanded := Expand(fmt.Sprintf("%v", tc.input), mapping)
|
||||||
MakePrimitiveReplacer(counts, map[string]interface{}{
|
|
||||||
"VAR_A": "A",
|
|
||||||
"VAR_B": "B",
|
|
||||||
"VAR_C": "C",
|
|
||||||
"VAR_REF": "$(VAR_A)",
|
|
||||||
"VAR_EMPTY": "",
|
|
||||||
}))
|
|
||||||
if e, a := tc.expected, expanded; e != a {
|
if e, a := tc.expected, expanded; e != a {
|
||||||
t.Errorf("%v: expected %q, got %q", tc.name, e, a)
|
t.Errorf("%v: expected %q, got %q", tc.name, e, a)
|
||||||
}
|
}
|
||||||
@@ -371,7 +347,8 @@ func TestMapping(t *testing.T) {
|
|||||||
}
|
}
|
||||||
if len(tc.counts) > 0 {
|
if len(tc.counts) > 0 {
|
||||||
for k, expectedCount := range tc.counts {
|
for k, expectedCount := range tc.counts {
|
||||||
if c, ok := counts[k]; ok {
|
c, ok := counts[k]
|
||||||
|
if ok {
|
||||||
if c != expectedCount {
|
if c != expectedCount {
|
||||||
t.Errorf(
|
t.Errorf(
|
||||||
"%v: k=%s, expected count %d, got %d",
|
"%v: k=%s, expected count %d, got %d",
|
||||||
@@ -162,7 +162,7 @@ func loadCrdIntoConfig(
|
|||||||
err = theConfig.AddNamereferenceFieldSpec(
|
err = theConfig.AddNamereferenceFieldSpec(
|
||||||
builtinconfig.NameBackReferences{
|
builtinconfig.NameBackReferences{
|
||||||
Gvk: resid.Gvk{Kind: kind, Version: version},
|
Gvk: resid.Gvk{Kind: kind, Version: version},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
makeFs(theGvk, append(path, propName, nameKey))},
|
makeFs(theGvk, append(path, propName, nameKey))},
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -8,9 +8,10 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
|
"sigs.k8s.io/kustomize/api/loader"
|
||||||
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/loader"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
@@ -139,19 +140,21 @@ func TestLoadCRDs(t *testing.T) {
|
|||||||
nbrs := []builtinconfig.NameBackReferences{
|
nbrs := []builtinconfig.NameBackReferences{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "Secret", Version: "v1"},
|
Gvk: resid.Gvk{Kind: "Secret", Version: "v1"},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "MyKind"},
|
CreateIfNotPresent: false,
|
||||||
Path: "spec/secretRef/name",
|
Gvk: resid.Gvk{Kind: "MyKind"},
|
||||||
|
Path: "spec/secretRef/name",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "Bee", Version: "v1beta1"},
|
Gvk: resid.Gvk{Kind: "Bee", Version: "v1beta1"},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "MyKind"},
|
CreateIfNotPresent: false,
|
||||||
Path: "spec/beeRef/name",
|
Gvk: resid.Gvk{Kind: "MyKind"},
|
||||||
|
Path: "spec/beeRef/name",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -4,29 +4,23 @@
|
|||||||
package accumulator
|
package accumulator
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"log"
|
"log"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/nameref"
|
"sigs.k8s.io/kustomize/api/filters/nameref"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
type nameReferenceTransformer struct {
|
type nameReferenceTransformer struct {
|
||||||
backRefs []builtinconfig.NameBackReferences
|
backRefs []builtinconfig.NameBackReferences
|
||||||
}
|
}
|
||||||
|
|
||||||
const doDebug = false
|
|
||||||
|
|
||||||
var _ resmap.Transformer = &nameReferenceTransformer{}
|
var _ resmap.Transformer = &nameReferenceTransformer{}
|
||||||
|
|
||||||
type filterMap map[*resource.Resource][]nameref.Filter
|
|
||||||
|
|
||||||
// newNameReferenceTransformer constructs a nameReferenceTransformer
|
// newNameReferenceTransformer constructs a nameReferenceTransformer
|
||||||
// with a given slice of NameBackReferences.
|
// with a given slice of NameBackReferences.
|
||||||
func newNameReferenceTransformer(
|
func newNameReferenceTransformer(br []builtinconfig.NameBackReferences) resmap.Transformer {
|
||||||
br []builtinconfig.NameBackReferences) resmap.Transformer {
|
|
||||||
if br == nil {
|
if br == nil {
|
||||||
log.Fatal("backrefs not expected to be nil")
|
log.Fatal("backrefs not expected to be nil")
|
||||||
}
|
}
|
||||||
@@ -39,61 +33,13 @@ func newNameReferenceTransformer(
|
|||||||
//
|
//
|
||||||
// For example, a HorizontalPodAutoscaler (HPA)
|
// For example, a HorizontalPodAutoscaler (HPA)
|
||||||
// necessarily refers to a Deployment, the thing that
|
// necessarily refers to a Deployment, the thing that
|
||||||
// an HPA scales. In this case:
|
// the HPA scales. The Deployment name might change
|
||||||
|
// (e.g. prefix added), and the reference in the HPA
|
||||||
|
// has to be fixed.
|
||||||
//
|
//
|
||||||
// - the HPA instance is the Referrer,
|
// In the outer loop over the ResMap below, say we
|
||||||
// - the Deployment instance is the ReferralTarget.
|
// encounter a specific HPA. Then, in scanning backrefs,
|
||||||
//
|
// we encounter an entry like
|
||||||
// If the Deployment's name changes, e.g. a prefix is added,
|
|
||||||
// then the HPA's reference to the Deployment must be fixed.
|
|
||||||
//
|
|
||||||
func (t *nameReferenceTransformer) Transform(m resmap.ResMap) error {
|
|
||||||
fMap := t.determineFilters(m.Resources())
|
|
||||||
debug(fMap)
|
|
||||||
for r, fList := range fMap {
|
|
||||||
c := m.SubsetThatCouldBeReferencedByResource(r)
|
|
||||||
for _, f := range fList {
|
|
||||||
f.Referrer = r
|
|
||||||
f.ReferralCandidates = c
|
|
||||||
if err := f.Referrer.ApplyFilter(f); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func debug(fMap filterMap) {
|
|
||||||
if !doDebug {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
fmt.Printf("filterMap has %d entries:\n", len(fMap))
|
|
||||||
rCount := 0
|
|
||||||
for r, fList := range fMap {
|
|
||||||
yml, _ := r.AsYAML()
|
|
||||||
rCount++
|
|
||||||
fmt.Printf(`
|
|
||||||
---- %3d. possible referrer -------------
|
|
||||||
%s
|
|
||||||
---------`, rCount, string(yml),
|
|
||||||
)
|
|
||||||
for i, f := range fList {
|
|
||||||
fmt.Printf(`
|
|
||||||
%3d/%3d update: %s
|
|
||||||
from: %s
|
|
||||||
`, rCount, i+1, f.NameFieldToUpdate.Path, f.ReferralTarget,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Produce a map from referrer resources that might need to be fixed
|
|
||||||
// to filters that might fix them. The keys to this map are potential
|
|
||||||
// referrers, so won't include resources like ConfigMap or Secret.
|
|
||||||
//
|
|
||||||
// In the inner loop over the resources below, say we
|
|
||||||
// encounter an HPA instance. Then, in scanning the set
|
|
||||||
// of all known backrefs, we encounter an entry like
|
|
||||||
//
|
//
|
||||||
// - kind: Deployment
|
// - kind: Deployment
|
||||||
// fieldSpecs:
|
// fieldSpecs:
|
||||||
@@ -102,53 +48,54 @@ func debug(fMap filterMap) {
|
|||||||
//
|
//
|
||||||
// This entry says that an HPA, via its
|
// This entry says that an HPA, via its
|
||||||
// 'spec/scaleTargetRef/name' field, may refer to a
|
// 'spec/scaleTargetRef/name' field, may refer to a
|
||||||
// Deployment.
|
// Deployment. This match to HPA means we may need to
|
||||||
|
// modify the value in its 'spec/scaleTargetRef/name'
|
||||||
|
// field, by searching for the thing it refers to,
|
||||||
|
// and getting its new name.
|
||||||
//
|
//
|
||||||
// This means that a filter will need to hunt for the right Deployment,
|
// As a filter, and search optimization, we compute a
|
||||||
// obtain it's new name, and write that name into the HPA's
|
// subset of all resources that the HPA could refer to,
|
||||||
// 'spec/scaleTargetRef/name' field. Return a filter that can do that.
|
// by excluding objects from other namespaces, and
|
||||||
func (t *nameReferenceTransformer) determineFilters(
|
// excluding objects that don't have the same prefix-
|
||||||
resources []*resource.Resource) (fMap filterMap) {
|
// suffix mods as the HPA.
|
||||||
fMap = make(filterMap)
|
//
|
||||||
for _, backReference := range t.backRefs {
|
// We look in this subset for all Deployment objects
|
||||||
for _, referrerSpec := range backReference.Referrers {
|
// with a resId that has a Name matching the field value
|
||||||
for _, res := range resources {
|
// present in the HPA. If no match do nothing; if more
|
||||||
if res.OrgId().IsSelected(&referrerSpec.Gvk) {
|
// than one match, it's an error.
|
||||||
// If this is true, the res might be a referrer, and if
|
//
|
||||||
// so, the name reference it holds might need an update.
|
// We overwrite the HPA name field with the value found
|
||||||
if resHasField(res, referrerSpec.Path) {
|
// in the Deployment's name field (the name in the raw
|
||||||
// Optimization - the referrer has the field
|
// object - the modified name - not the unmodified name
|
||||||
// that might need updating.
|
// in the Deployment's resId).
|
||||||
fMap[res] = append(fMap[res], nameref.Filter{
|
//
|
||||||
// Name field to write in the Referrer.
|
// This process assumes that the name stored in a ResId
|
||||||
// If the path specified here isn't found in
|
// (the ResMap key) isn't modified by name transformers.
|
||||||
// the Referrer, nothing happens (no error,
|
// Name transformers should only modify the name in the
|
||||||
// no field creation).
|
// body of the resource object (the value in the ResMap).
|
||||||
NameFieldToUpdate: referrerSpec,
|
//
|
||||||
// Specification of object class to read from.
|
func (o *nameReferenceTransformer) Transform(m resmap.ResMap) error {
|
||||||
// Always read from metadata/name field.
|
// TODO: Too much looping, here and in transitive calls.
|
||||||
ReferralTarget: backReference.Gvk,
|
for _, referrer := range m.Resources() {
|
||||||
})
|
var candidates resmap.ResMap
|
||||||
|
for _, target := range o.backRefs {
|
||||||
|
for _, fSpec := range target.FieldSpecs {
|
||||||
|
if referrer.OrgId().IsSelected(&fSpec.Gvk) {
|
||||||
|
if candidates == nil {
|
||||||
|
candidates = m.SubsetThatCouldBeReferencedByResource(referrer)
|
||||||
|
}
|
||||||
|
err := filtersutil.ApplyToJSON(nameref.Filter{
|
||||||
|
FieldSpec: fSpec,
|
||||||
|
Referrer: referrer,
|
||||||
|
Target: target.Gvk,
|
||||||
|
ReferralCandidates: candidates,
|
||||||
|
}, referrer)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return fMap
|
return nil
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: check res for field existence here to avoid extra work.
|
|
||||||
// res.GetFieldValue, which uses yaml.Lookup under the hood, doesn't know
|
|
||||||
// how to parse fieldspec-style paths that make no distinction
|
|
||||||
// between maps and sequences. This means it cannot lookup commonly
|
|
||||||
// used "indeterminate" paths like
|
|
||||||
// spec/containers/env/valueFrom/configMapKeyRef/name
|
|
||||||
// ('containers' is a list, not a map).
|
|
||||||
// However, the fieldspec filter does know how to handle this;
|
|
||||||
// extract that code and call it here?
|
|
||||||
func resHasField(res *resource.Resource, path string) bool {
|
|
||||||
return true
|
|
||||||
// fld := strings.Join(utils.PathSplitter(path), ".")
|
|
||||||
// _, e := res.GetFieldValue(fld)
|
|
||||||
// return e == nil
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,17 +8,17 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
||||||
)
|
)
|
||||||
|
|
||||||
const notEqualErrFmt = "expected (self) doesn't match actual (other): %v"
|
|
||||||
|
|
||||||
func TestNameReferenceHappyRun(t *testing.T) {
|
func TestNameReferenceHappyRun(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).AddWithName(
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).AddWithName(
|
||||||
"cm1",
|
"cm1",
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -220,7 +220,6 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
"secret1",
|
"secret1",
|
||||||
"secret1",
|
"secret1",
|
||||||
"secret2",
|
"secret2",
|
||||||
"cm1",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -262,8 +261,7 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}).ResMap()
|
}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).ReplaceResource(
|
||||||
t, m.ShallowCopy()).ReplaceResource(
|
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"group": "apps",
|
"group": "apps",
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -424,7 +422,6 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
"someprefix-secret1-somehash",
|
"someprefix-secret1-somehash",
|
||||||
"someprefix-secret1-somehash",
|
"someprefix-secret1-somehash",
|
||||||
"secret2",
|
"secret2",
|
||||||
"someprefix-cm1-somehash",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -473,17 +470,19 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNameReferenceUnhappyRun(t *testing.T) {
|
func TestNameReferenceUnhappyRun(t *testing.T) {
|
||||||
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
resMap resmap.ResMap
|
resMap resmap.ResMap
|
||||||
expectedErr string
|
expectedErr string
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
resMap: resmaptest_test.NewRmBuilderDefault(t).Add(
|
resMap: resmaptest_test.NewRmBuilder(t, rf).Add(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
@@ -503,7 +502,7 @@ func TestNameReferenceUnhappyRun(t *testing.T) {
|
|||||||
}).ResMap(),
|
}).ResMap(),
|
||||||
expectedErr: "is expected to be"},
|
expectedErr: "is expected to be"},
|
||||||
{
|
{
|
||||||
resMap: resmaptest_test.NewRmBuilderDefault(t).Add(
|
resMap: resmaptest_test.NewRmBuilder(t, rf).Add(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
@@ -521,27 +520,7 @@ func TestNameReferenceUnhappyRun(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}).ResMap(),
|
}).ResMap(),
|
||||||
// TODO(#3304): DECISION - kyaml better; not a bug.
|
expectedErr: "cannot find field 'name' in node"},
|
||||||
expectedErr: konfig.IfApiMachineryElseKyaml(
|
|
||||||
`updating name reference in 'rules/resourceNames' field of `+
|
|
||||||
`'rbac.authorization.k8s.io_v1_ClusterRole|~X|cr'`+
|
|
||||||
`: considering field 'rules/resourceNames' of object
|
|
||||||
{"apiVersion": "rbac.authorization.k8s.io/v1", "kind": "ClusterRole", "metadata": {
|
|
||||||
"name": "cr"}, "rules": [{"resourceNames": {"foo": "bar"}, "resources": ["secrets"]}]}
|
|
||||||
: visit traversal on path: [resourceNames]: path config error; no 'name' field in node`,
|
|
||||||
`updating name reference in 'rules/resourceNames' field of `+
|
|
||||||
`'rbac.authorization.k8s.io_v1_ClusterRole|~X|cr'`+
|
|
||||||
`: considering field 'rules/resourceNames' of object
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRole
|
|
||||||
metadata:
|
|
||||||
name: cr
|
|
||||||
rules:
|
|
||||||
- resourceNames:
|
|
||||||
foo: bar
|
|
||||||
resources:
|
|
||||||
- secrets
|
|
||||||
: visit traversal on path: [resourceNames]: path config error; no 'name' field in node`)},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nrt := newNameReferenceTransformer(builtinconfig.MakeDefaultConfig().NameReference)
|
nrt := newNameReferenceTransformer(builtinconfig.MakeDefaultConfig().NameReference)
|
||||||
@@ -552,14 +531,15 @@ rules:
|
|||||||
}
|
}
|
||||||
|
|
||||||
if !strings.Contains(err.Error(), test.expectedErr) {
|
if !strings.Contains(err.Error(), test.expectedErr) {
|
||||||
t.Fatalf("Incorrect error.\nExpected:\n %s\nGot:\n%v",
|
t.Fatalf("Incorrect error.\nExpected: %s, but got %v",
|
||||||
test.expectedErr, err)
|
test.expectedErr, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
||||||
rf := provider.NewDefaultDepProvider().GetResourceFactory()
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
|
||||||
v1 := rf.FromMapWithName(
|
v1 := rf.FromMapWithName(
|
||||||
"volume1",
|
"volume1",
|
||||||
@@ -610,7 +590,7 @@ func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
|||||||
v2.AppendRefBy(c2.CurId())
|
v2.AppendRefBy(c2.CurId())
|
||||||
|
|
||||||
if err := m1.ErrorIfNotEqualLists(m2); err != nil {
|
if err := m1.ErrorIfNotEqualLists(m2); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -684,7 +664,9 @@ const (
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceNamespace(t *testing.T) {
|
func TestNameReferenceNamespace(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
// Add ConfigMap with the same org name in noNs, "ns1" and "ns2" namespaces
|
// Add ConfigMap with the same org name in noNs, "ns1" and "ns2" namespaces
|
||||||
AddWithName(orgname, map[string]interface{}{
|
AddWithName(orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -733,7 +715,7 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
AddWithNsAndName(ns1, orgname, deploymentMap(ns1, prefixedname, orgname, orgname)).
|
AddWithNsAndName(ns1, orgname, deploymentMap(ns1, prefixedname, orgname, orgname)).
|
||||||
AddWithNsAndName(ns2, orgname, deploymentMap(ns2, suffixedname, orgname, orgname)).ResMap()
|
AddWithNsAndName(ns2, orgname, deploymentMap(ns2, suffixedname, orgname, orgname)).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(deploymentMap(defaultNs, modifiedname, modifiedname, modifiedname)).
|
ReplaceResource(deploymentMap(defaultNs, modifiedname, modifiedname, modifiedname)).
|
||||||
ReplaceResource(deploymentMap(ns1, prefixedname, prefixedname, prefixedname)).
|
ReplaceResource(deploymentMap(ns1, prefixedname, prefixedname, prefixedname)).
|
||||||
ReplaceResource(deploymentMap(ns2, suffixedname, suffixedname, suffixedname)).ResMap()
|
ReplaceResource(deploymentMap(ns2, suffixedname, suffixedname, suffixedname)).ResMap()
|
||||||
@@ -744,9 +726,8 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -754,7 +735,9 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceClusterWide(t *testing.T) {
|
func TestNameReferenceClusterWide(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
// Add ServiceAccount with the same org name in noNs, "ns1" and "ns2" namespaces
|
// Add ServiceAccount with the same org name in noNs, "ns1" and "ns2" namespaces
|
||||||
AddWithName(orgname, map[string]interface{}{
|
AddWithName(orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -833,7 +816,7 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}}).ResMap()
|
}}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(
|
ReplaceResource(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
@@ -906,9 +889,8 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -916,7 +898,9 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
AddWithNsAndName(ns4, orgname, map[string]interface{}{
|
AddWithNsAndName(ns4, orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "Secret",
|
"kind": "Secret",
|
||||||
@@ -980,7 +964,7 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}}).ResMap()
|
}}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(
|
ReplaceResource(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
@@ -1033,9 +1017,8 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1043,7 +1026,9 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
// It validates the change done is IsSameFuzzyNamespace which
|
// It validates the change done is IsSameFuzzyNamespace which
|
||||||
// uses the IsNsEquals method instead of the simple == operator.
|
// uses the IsNsEquals method instead of the simple == operator.
|
||||||
func TestNameReferenceCandidateSelection(t *testing.T) {
|
func TestNameReferenceCandidateSelection(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
AddWithName("cm1", map[string]interface{}{
|
AddWithName("cm1", map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -1060,7 +1045,7 @@ func TestNameReferenceCandidateSelection(t *testing.T) {
|
|||||||
AddWithName("deploy1", deploymentMap("", "p1-deploy1", "cm1", "secret1")).
|
AddWithName("deploy1", deploymentMap("", "p1-deploy1", "cm1", "secret1")).
|
||||||
ResMap()
|
ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(deploymentMap("", "p1-deploy1", "p1-cm1-hash", "p1-secret1-hash")).
|
ReplaceResource(deploymentMap("", "p1-deploy1", "p1-cm1-hash", "p1-secret1-hash")).
|
||||||
ResMap()
|
ResMap()
|
||||||
|
|
||||||
@@ -1070,8 +1055,7 @@ func TestNameReferenceCandidateSelection(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,15 +4,19 @@
|
|||||||
package accumulator
|
package accumulator
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
expansion2 "sigs.k8s.io/kustomize/api/internal/accumulator/expansion"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/refvar"
|
"sigs.k8s.io/kustomize/api/filters/refvar"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
type refVarTransformer struct {
|
type refVarTransformer struct {
|
||||||
varMap map[string]interface{}
|
varMap map[string]interface{}
|
||||||
replacementCounts map[string]int
|
replacementCounts map[string]int
|
||||||
fieldSpecs []types.FieldSpec
|
fieldSpecs []types.FieldSpec
|
||||||
|
mappingFunc func(string) interface{}
|
||||||
}
|
}
|
||||||
|
|
||||||
// newRefVarTransformer returns a new refVarTransformer
|
// newRefVarTransformer returns a new refVarTransformer
|
||||||
@@ -31,7 +35,8 @@ func newRefVarTransformer(
|
|||||||
func (rv *refVarTransformer) UnusedVars() []string {
|
func (rv *refVarTransformer) UnusedVars() []string {
|
||||||
var unused []string
|
var unused []string
|
||||||
for k := range rv.varMap {
|
for k := range rv.varMap {
|
||||||
if _, ok := rv.replacementCounts[k]; !ok {
|
_, ok := rv.replacementCounts[k]
|
||||||
|
if !ok {
|
||||||
unused = append(unused, k)
|
unused = append(unused, k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -41,13 +46,14 @@ func (rv *refVarTransformer) UnusedVars() []string {
|
|||||||
// Transform replaces $(VAR) style variables with values.
|
// Transform replaces $(VAR) style variables with values.
|
||||||
func (rv *refVarTransformer) Transform(m resmap.ResMap) error {
|
func (rv *refVarTransformer) Transform(m resmap.ResMap) error {
|
||||||
rv.replacementCounts = make(map[string]int)
|
rv.replacementCounts = make(map[string]int)
|
||||||
mf := refvar.MakePrimitiveReplacer(rv.replacementCounts, rv.varMap)
|
rv.mappingFunc = expansion2.MappingFuncFor(
|
||||||
|
rv.replacementCounts, rv.varMap)
|
||||||
for _, res := range m.Resources() {
|
for _, res := range m.Resources() {
|
||||||
for _, fieldSpec := range rv.fieldSpecs {
|
for _, fieldSpec := range rv.fieldSpecs {
|
||||||
err := res.ApplyFilter(refvar.Filter{
|
err := filtersutil.ApplyToJSON(refvar.Filter{
|
||||||
MappingFunc: mf,
|
MappingFunc: rv.mappingFunc,
|
||||||
FieldSpec: fieldSpec,
|
FieldSpec: fieldSpec,
|
||||||
})
|
}, res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,9 +7,10 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
@@ -24,12 +25,14 @@ func TestRefVarTransformer(t *testing.T) {
|
|||||||
res resmap.ResMap
|
res resmap.ResMap
|
||||||
unused []string
|
unused []string
|
||||||
}
|
}
|
||||||
testCases := map[string]struct {
|
testCases := []struct {
|
||||||
given given
|
description string
|
||||||
expected expected
|
given given
|
||||||
errMessage string
|
expected expected
|
||||||
|
errMessage string
|
||||||
}{
|
}{
|
||||||
"var replacement in map[string]": {
|
{
|
||||||
|
description: "var replacement in map[string]",
|
||||||
given: given{
|
given: given{
|
||||||
varMap: map[string]interface{}{
|
varMap: map[string]interface{}{
|
||||||
"FOO": "replacementForFoo",
|
"FOO": "replacementForFoo",
|
||||||
@@ -43,7 +46,8 @@ func TestRefVarTransformer(t *testing.T) {
|
|||||||
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/interface"},
|
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/interface"},
|
||||||
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/num"},
|
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/num"},
|
||||||
},
|
},
|
||||||
res: resmaptest_test.NewRmBuilderDefault(t).
|
res: resmaptest_test.NewRmBuilder(
|
||||||
|
t, resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -73,7 +77,8 @@ func TestRefVarTransformer(t *testing.T) {
|
|||||||
}}).ResMap(),
|
}}).ResMap(),
|
||||||
},
|
},
|
||||||
expected: expected{
|
expected: expected{
|
||||||
res: resmaptest_test.NewRmBuilderDefault(t).
|
res: resmaptest_test.NewRmBuilder(
|
||||||
|
t, resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -104,13 +109,15 @@ func TestRefVarTransformer(t *testing.T) {
|
|||||||
unused: []string{"BAR"},
|
unused: []string{"BAR"},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"var replacement panic in map[string]": {
|
{
|
||||||
|
description: "var replacement panic in map[string]",
|
||||||
given: given{
|
given: given{
|
||||||
varMap: map[string]interface{}{},
|
varMap: map[string]interface{}{},
|
||||||
fs: []types.FieldSpec{
|
fs: []types.FieldSpec{
|
||||||
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/slice"},
|
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/slice"},
|
||||||
},
|
},
|
||||||
res: resmaptest_test.NewRmBuilderDefault(t).
|
res: resmaptest_test.NewRmBuilder(
|
||||||
|
t, resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -121,29 +128,18 @@ func TestRefVarTransformer(t *testing.T) {
|
|||||||
"slice": []interface{}{5}, // noticeably *not* a []string
|
"slice": []interface{}{5}, // noticeably *not* a []string
|
||||||
}}).ResMap(),
|
}}).ResMap(),
|
||||||
},
|
},
|
||||||
// TODO(#3304): DECISION - kyaml better; not a bug.
|
errMessage: `obj '{"apiVersion": "v1", "data": {"slice": [5]}, "kind": "ConfigMap", "metadata": {"name": "cm1"}}
|
||||||
errMessage: konfig.IfApiMachineryElseKyaml(
|
' at path 'data/slice': invalid value type expect a string`,
|
||||||
`considering field 'data/slice' of object
|
|
||||||
{"apiVersion": "v1", "data": {"slice": [5]}, "kind": "ConfigMap", "metadata": {"name": "cm1"}}
|
|
||||||
: invalid value type expect a string`,
|
|
||||||
`considering field 'data/slice' of object
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
slice:
|
|
||||||
- 5
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: cm1
|
|
||||||
: invalid value type expect a string`,
|
|
||||||
),
|
|
||||||
},
|
},
|
||||||
"var replacement in nil": {
|
{
|
||||||
|
description: "var replacement in nil",
|
||||||
given: given{
|
given: given{
|
||||||
varMap: map[string]interface{}{},
|
varMap: map[string]interface{}{},
|
||||||
fs: []types.FieldSpec{
|
fs: []types.FieldSpec{
|
||||||
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/nil"},
|
{Gvk: resid.Gvk{Version: "v1", Kind: "ConfigMap"}, Path: "data/nil"},
|
||||||
},
|
},
|
||||||
res: resmaptest_test.NewRmBuilderDefault(t).
|
res: resmaptest_test.NewRmBuilder(
|
||||||
|
t, resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -155,7 +151,8 @@ metadata:
|
|||||||
}}).ResMap(),
|
}}).ResMap(),
|
||||||
},
|
},
|
||||||
expected: expected{
|
expected: expected{
|
||||||
res: resmaptest_test.NewRmBuilderDefault(t).
|
res: resmaptest_test.NewRmBuilder(
|
||||||
|
t, resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -169,18 +166,20 @@ metadata:
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tc.description, func(t *testing.T) {
|
||||||
|
// arrange
|
||||||
tr := newRefVarTransformer(tc.given.varMap, tc.given.fs)
|
tr := newRefVarTransformer(tc.given.varMap, tc.given.fs)
|
||||||
|
|
||||||
|
// act
|
||||||
err := tr.Transform(tc.given.res)
|
err := tr.Transform(tc.given.res)
|
||||||
|
|
||||||
|
// assert
|
||||||
if tc.errMessage != "" {
|
if tc.errMessage != "" {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatalf("missing expected error %v", tc.errMessage)
|
t.Fatalf("missing expected error %v", tc.errMessage)
|
||||||
} else if err.Error() != tc.errMessage {
|
} else if err.Error() != tc.errMessage {
|
||||||
t.Fatalf(`actual error doesn't match expected error:
|
t.Fatalf("actual error doesn't match expected error: \nACTUAL: %v\nEXPECTED: %v", err.Error(), tc.errMessage)
|
||||||
ACTUAL: %v
|
|
||||||
EXPECTED: %v`,
|
|
||||||
err.Error(), tc.errMessage)
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -190,13 +189,7 @@ EXPECTED: %v`,
|
|||||||
a, e := tc.given.res, tc.expected.res
|
a, e := tc.given.res, tc.expected.res
|
||||||
if !reflect.DeepEqual(a, e) {
|
if !reflect.DeepEqual(a, e) {
|
||||||
err = e.ErrorIfNotEqualLists(a)
|
err = e.ErrorIfNotEqualLists(a)
|
||||||
t.Fatalf(`actual doesn't match expected:
|
t.Fatalf("actual doesn't match expected: \nACTUAL:\n%v\nEXPECTED:\n%v\nERR: %v", a, e, err)
|
||||||
ACTUAL:
|
|
||||||
%v
|
|
||||||
EXPECTED:
|
|
||||||
%v
|
|
||||||
ERR: %v`,
|
|
||||||
a, e, err)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -41,11 +41,13 @@ func (ra *ResAccumulator) Vars() []types.Var {
|
|||||||
return ra.varSet.AsSlice()
|
return ra.varSet.AsSlice()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ra *ResAccumulator) AppendAll(resources resmap.ResMap) error {
|
func (ra *ResAccumulator) AppendAll(
|
||||||
|
resources resmap.ResMap) error {
|
||||||
return ra.resMap.AppendAll(resources)
|
return ra.resMap.AppendAll(resources)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ra *ResAccumulator) AbsorbAll(resources resmap.ResMap) error {
|
func (ra *ResAccumulator) AbsorbAll(
|
||||||
|
resources resmap.ResMap) error {
|
||||||
return ra.resMap.AbsorbAll(resources)
|
return ra.resMap.AbsorbAll(resources)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,15 +61,6 @@ func (ra *ResAccumulator) GetTransformerConfig() *builtinconfig.TransformerConfi
|
|||||||
return ra.tConfig
|
return ra.tConfig
|
||||||
}
|
}
|
||||||
|
|
||||||
// MergeVars accumulates vars into ResAccumulator.
|
|
||||||
// A Var is a tuple of name, object reference and field reference.
|
|
||||||
// This func takes a list of vars from the current kustomization file and
|
|
||||||
// annotates the accumulated resources with the names of the vars that match
|
|
||||||
// those resources. E.g. if there's a var named "sam" that wants to get
|
|
||||||
// its data from a ConfigMap named "james", and the resource list contains a
|
|
||||||
// ConfigMap named "james", then that ConfigMap will be annotated with the
|
|
||||||
// var name "sam". Later this annotation is used to find the data for "sam"
|
|
||||||
// by digging into a particular fieldpath of "james".
|
|
||||||
func (ra *ResAccumulator) MergeVars(incoming []types.Var) error {
|
func (ra *ResAccumulator) MergeVars(incoming []types.Var) error {
|
||||||
for _, v := range incoming {
|
for _, v := range incoming {
|
||||||
targetId := resid.NewResIdWithNamespace(v.ObjRef.GVK(), v.ObjRef.Name, v.ObjRef.Namespace)
|
targetId := resid.NewResIdWithNamespace(v.ObjRef.GVK(), v.ObjRef.Name, v.ObjRef.Namespace)
|
||||||
@@ -77,7 +70,7 @@ func (ra *ResAccumulator) MergeVars(incoming []types.Var) error {
|
|||||||
// wildcard search on the namespace hence we still use GvknEquals
|
// wildcard search on the namespace hence we still use GvknEquals
|
||||||
idMatcher = targetId.Equals
|
idMatcher = targetId.Equals
|
||||||
}
|
}
|
||||||
matched := ra.resMap.GetMatchingResourcesByAnyId(idMatcher)
|
matched := ra.resMap.GetMatchingResourcesByOriginalId(idMatcher)
|
||||||
if len(matched) > 1 {
|
if len(matched) > 1 {
|
||||||
return fmt.Errorf(
|
return fmt.Errorf(
|
||||||
"found %d resId matches for var %s "+
|
"found %d resId matches for var %s "+
|
||||||
@@ -113,10 +106,12 @@ func (ra *ResAccumulator) findVarValueFromResources(v types.Var) (interface{}, e
|
|||||||
"field specified in var '%v' "+
|
"field specified in var '%v' "+
|
||||||
"not found in corresponding resource", v)
|
"not found in corresponding resource", v)
|
||||||
}
|
}
|
||||||
|
|
||||||
return s, nil
|
return s, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return "", fmt.Errorf(
|
return "", fmt.Errorf(
|
||||||
"var '%v' cannot be mapped to a field "+
|
"var '%v' cannot be mapped to a field "+
|
||||||
"in the set of known resources", v)
|
"in the set of known resources", v)
|
||||||
@@ -132,8 +127,10 @@ func (ra *ResAccumulator) makeVarReplacementMap() (map[string]interface{}, error
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
result[v.Name] = s
|
result[v.Name] = s
|
||||||
}
|
}
|
||||||
|
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -164,6 +161,6 @@ func (ra *ResAccumulator) FixBackReferences() (err error) {
|
|||||||
if ra.tConfig.NameReference == nil {
|
if ra.tConfig.NameReference == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return ra.Transform(
|
return ra.Transform(newNameReferenceTransformer(
|
||||||
newNameReferenceTransformer(ra.tConfig.NameReference))
|
ra.tConfig.NameReference))
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import (
|
|||||||
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
@@ -20,14 +20,16 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
func makeResAccumulator(t *testing.T) *ResAccumulator {
|
func makeResAccumulator(t *testing.T) (*ResAccumulator, *resource.Factory) {
|
||||||
ra := MakeEmptyAccumulator()
|
ra := MakeEmptyAccumulator()
|
||||||
err := ra.MergeConfig(builtinconfig.MakeDefaultConfig())
|
err := ra.MergeConfig(builtinconfig.MakeDefaultConfig())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected err: %v", err)
|
t.Fatalf("unexpected err: %v", err)
|
||||||
}
|
}
|
||||||
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
err = ra.AppendAll(
|
err = ra.AppendAll(
|
||||||
resmaptest_test.NewRmBuilderDefault(t).
|
resmaptest_test.NewRmBuilder(t, rf).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "apps/v1",
|
"apiVersion": "apps/v1",
|
||||||
"kind": "Deployment",
|
"kind": "Deployment",
|
||||||
@@ -64,11 +66,11 @@ func makeResAccumulator(t *testing.T) *ResAccumulator {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected err: %v", err)
|
t.Fatalf("unexpected err: %v", err)
|
||||||
}
|
}
|
||||||
return ra
|
return ra, rf
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsHappy(t *testing.T) {
|
func TestResolveVarsHappy(t *testing.T) {
|
||||||
ra := makeResAccumulator(t)
|
ra, _ := makeResAccumulator(t)
|
||||||
err := ra.MergeVars([]types.Var{
|
err := ra.MergeVars([]types.Var{
|
||||||
{
|
{
|
||||||
Name: "SERVICE_ONE",
|
Name: "SERVICE_ONE",
|
||||||
@@ -97,7 +99,7 @@ func TestResolveVarsHappy(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsOneUnused(t *testing.T) {
|
func TestResolveVarsOneUnused(t *testing.T) {
|
||||||
ra := makeResAccumulator(t)
|
ra, _ := makeResAccumulator(t)
|
||||||
err := ra.MergeVars([]types.Var{
|
err := ra.MergeVars([]types.Var{
|
||||||
{
|
{
|
||||||
Name: "SERVICE_ONE",
|
Name: "SERVICE_ONE",
|
||||||
@@ -138,10 +140,11 @@ func expectLog(t *testing.T, log bytes.Buffer, expect string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsVarNeedsDisambiguation(t *testing.T) {
|
func TestResolveVarsVarNeedsDisambiguation(t *testing.T) {
|
||||||
ra := makeResAccumulator(t)
|
ra, rf := makeResAccumulator(t)
|
||||||
|
|
||||||
rm0 := resmap.New()
|
rm0 := resmap.New()
|
||||||
err := rm0.Append(
|
err := rm0.Append(
|
||||||
provider.NewDefaultDepProvider().GetResourceFactory().FromMap(
|
rf.FromMap(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "Service",
|
"kind": "Service",
|
||||||
@@ -210,7 +213,8 @@ func makeVarToNamepaceAndPath(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarConflicts(t *testing.T) {
|
func TestResolveVarConflicts(t *testing.T) {
|
||||||
rf := provider.NewDefaultDepProvider().GetResourceFactory()
|
rf := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
|
||||||
// create configmaps in foo and bar namespaces with `data.provider` values.
|
// create configmaps in foo and bar namespaces with `data.provider` values.
|
||||||
fooAws := makeNamespacedConfigMapWithDataProviderValue("foo", "aws")
|
fooAws := makeNamespacedConfigMapWithDataProviderValue("foo", "aws")
|
||||||
barAws := makeNamespacedConfigMapWithDataProviderValue("bar", "aws")
|
barAws := makeNamespacedConfigMapWithDataProviderValue("bar", "aws")
|
||||||
@@ -257,7 +261,7 @@ func TestResolveVarConflicts(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsGoodResIdBadField(t *testing.T) {
|
func TestResolveVarsGoodResIdBadField(t *testing.T) {
|
||||||
ra := makeResAccumulator(t)
|
ra, _ := makeResAccumulator(t)
|
||||||
err := ra.MergeVars([]types.Var{
|
err := ra.MergeVars([]types.Var{
|
||||||
{
|
{
|
||||||
Name: "SERVICE_ONE",
|
Name: "SERVICE_ONE",
|
||||||
@@ -282,7 +286,7 @@ func TestResolveVarsGoodResIdBadField(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsUnmappableVar(t *testing.T) {
|
func TestResolveVarsUnmappableVar(t *testing.T) {
|
||||||
ra := makeResAccumulator(t)
|
ra, _ := makeResAccumulator(t)
|
||||||
err := ra.MergeVars([]types.Var{
|
err := ra.MergeVars([]types.Var{
|
||||||
{
|
{
|
||||||
Name: "SERVICE_THREE",
|
Name: "SERVICE_THREE",
|
||||||
@@ -306,7 +310,7 @@ func TestResolveVarsUnmappableVar(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestResolveVarsWithNoambiguation(t *testing.T) {
|
func TestResolveVarsWithNoambiguation(t *testing.T) {
|
||||||
ra1 := makeResAccumulator(t)
|
ra1, rf := makeResAccumulator(t)
|
||||||
err := ra1.MergeVars([]types.Var{
|
err := ra1.MergeVars([]types.Var{
|
||||||
{
|
{
|
||||||
Name: "SERVICE_ONE",
|
Name: "SERVICE_ONE",
|
||||||
@@ -323,7 +327,7 @@ func TestResolveVarsWithNoambiguation(t *testing.T) {
|
|||||||
// Create another accumulator having a resource with different prefix
|
// Create another accumulator having a resource with different prefix
|
||||||
ra2 := MakeEmptyAccumulator()
|
ra2 := MakeEmptyAccumulator()
|
||||||
|
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "apps/v1",
|
"apiVersion": "apps/v1",
|
||||||
"kind": "Deployment",
|
"kind": "Deployment",
|
||||||
@@ -344,20 +348,19 @@ func TestResolveVarsWithNoambiguation(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}}).
|
}}).
|
||||||
// Make it seem like this resource
|
|
||||||
// went through a prefix transformer.
|
|
||||||
Add(map[string]interface{}{
|
Add(map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "Service",
|
"kind": "Service",
|
||||||
"metadata": map[string]interface{}{
|
"metadata": map[string]interface{}{
|
||||||
"name": "sub-backendOne",
|
"name": "backendOne",
|
||||||
"annotations": map[string]interface{}{
|
|
||||||
"config.kubernetes.io/previousNames": "backendOne",
|
|
||||||
"config.kubernetes.io/previousNamespaces": "default",
|
|
||||||
"config.kubernetes.io/prefixes": "sub-",
|
|
||||||
},
|
|
||||||
}}).ResMap()
|
}}).ResMap()
|
||||||
|
|
||||||
|
// Make it seem like this resource
|
||||||
|
// went through a prefix transformer.
|
||||||
|
r := m.GetByIndex(1)
|
||||||
|
r.AddNamePrefix("sub-")
|
||||||
|
r.SetName("sub-backendOne") // original name remains "backendOne"
|
||||||
|
|
||||||
err = ra2.AppendAll(m)
|
err = ra2.AppendAll(m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected err: %v", err)
|
t.Fatalf("unexpected err: %v", err)
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package conflict
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
type cdFactory struct{}
|
|
||||||
|
|
||||||
var _ resource.ConflictDetectorFactory = &cdFactory{}
|
|
||||||
|
|
||||||
// NewFactory returns a new conflict detector factory.
|
|
||||||
func NewFactory() resource.ConflictDetectorFactory {
|
|
||||||
return &cdFactory{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// New returns an instance of smPatchMergeOnlyDetector.
|
|
||||||
func (c cdFactory) New(_ resid.Gvk) (resource.ConflictDetector, error) {
|
|
||||||
return &smPatchMergeOnlyDetector{}, nil
|
|
||||||
}
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package conflict
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
// smPatchMergeOnlyDetector ignores conflicts,
|
|
||||||
// but does real strategic merge patching.
|
|
||||||
// This is part of an effort to eliminate dependence on
|
|
||||||
// apimachinery package to allow kustomize integration
|
|
||||||
// into kubectl (#2506 and #1500)
|
|
||||||
type smPatchMergeOnlyDetector struct{}
|
|
||||||
|
|
||||||
var _ resource.ConflictDetector = &smPatchMergeOnlyDetector{}
|
|
||||||
|
|
||||||
func (c *smPatchMergeOnlyDetector) HasConflict(
|
|
||||||
_, _ *resource.Resource) (bool, error) {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// There's at least one case that doesn't work. Suppose one has a
|
|
||||||
// Deployment with a volume with the bizarre "emptyDir: {}" entry.
|
|
||||||
// If you want to get rid of this entry via a patch containing
|
|
||||||
// the entry "emptyDir: null", then the following won't work,
|
|
||||||
// because null entries are eliminated.
|
|
||||||
func (c *smPatchMergeOnlyDetector) MergePatches(
|
|
||||||
r, patch *resource.Resource) (*resource.Resource, error) {
|
|
||||||
err := r.ApplySmPatch(patch)
|
|
||||||
return r, err
|
|
||||||
}
|
|
||||||
@@ -7,13 +7,14 @@ import (
|
|||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var fileReader = kunstruct.NewKunstructuredFactoryImpl()
|
||||||
|
|
||||||
// This document is meant to be used at the elasticsearch document type.
|
// This document is meant to be used at the elasticsearch document type.
|
||||||
// Fields are serialized as-is to elasticsearch, where indices are built
|
// Fields are serialized as-is to elasticsearch, where indices are built
|
||||||
// to facilitate text search queries. Identifiers, Values, FilePath,
|
// to facilitate text search queries. Identifiers, Values, FilePath,
|
||||||
@@ -41,7 +42,6 @@ type KustomizationDocument struct {
|
|||||||
Kinds []string `json:"kinds,omitempty"`
|
Kinds []string `json:"kinds,omitempty"`
|
||||||
Identifiers []string `json:"identifiers,omitempty"`
|
Identifiers []string `json:"identifiers,omitempty"`
|
||||||
Values []string `json:"values,omitempty"`
|
Values []string `json:"values,omitempty"`
|
||||||
resFactory *resource.Factory
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type set map[string]struct{}
|
type set map[string]struct{}
|
||||||
@@ -52,7 +52,6 @@ func (doc *KustomizationDocument) Copy() *KustomizationDocument {
|
|||||||
Kinds: doc.Kinds,
|
Kinds: doc.Kinds,
|
||||||
Identifiers: doc.Identifiers,
|
Identifiers: doc.Identifiers,
|
||||||
Values: doc.Values,
|
Values: doc.Values,
|
||||||
resFactory: provider.NewDefaultDepProvider().GetResourceFactory(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -151,7 +150,7 @@ func (doc *KustomizationDocument) readBytes() ([]map[string]interface{}, error)
|
|||||||
}
|
}
|
||||||
|
|
||||||
configs := make([]map[string]interface{}, 0)
|
configs := make([]map[string]interface{}, 0)
|
||||||
ks, err := doc.resFactory.SliceFromBytes(data)
|
ks, err := fileReader.SliceFromBytes(data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("unable to parse resource: %v", err)
|
return nil, fmt.Errorf("unable to parse resource: %v", err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module sigs.k8s.io/kustomize/api/internal/crawl
|
module sigs.k8s.io/kustomize/api/internal/crawl
|
||||||
|
|
||||||
go 1.15
|
go 1.14
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/elastic/go-elasticsearch/v6 v6.8.5
|
github.com/elastic/go-elasticsearch/v6 v6.8.5
|
||||||
|
|||||||
@@ -145,7 +145,6 @@ github.com/go-toolsmith/pkgload v0.0.0-20181119091011-e9e65178eee8/go.mod h1:WoM
|
|||||||
github.com/go-toolsmith/pkgload v1.0.0/go.mod h1:5eFArkbO80v7Z0kdngIxsRXRMTaX4Ilcwuh3clNrQJc=
|
github.com/go-toolsmith/pkgload v1.0.0/go.mod h1:5eFArkbO80v7Z0kdngIxsRXRMTaX4Ilcwuh3clNrQJc=
|
||||||
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
||||||
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
||||||
github.com/gobuffalo/here v0.6.0/go.mod h1:wAG085dHOYqUpf+Ap+WOdrPTp5IYcDAs/x7PLa8Y5fM=
|
|
||||||
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
||||||
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU=
|
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU=
|
||||||
github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
|
github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
|
||||||
@@ -220,7 +219,6 @@ github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ
|
|||||||
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
||||||
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
||||||
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
github.com/imdario/mergo v0.3.5 h1:JboBksRwiiAJWvIYJVo46AfV+IAIKZpfrSzVKj42R4Q=
|
|
||||||
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
||||||
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
||||||
github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
|
github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
|
||||||
@@ -255,7 +253,6 @@ github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN
|
|||||||
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
||||||
github.com/mailru/easyjson v0.7.0 h1:aizVhC/NAAcKWb+5QsU1iNOZb4Yws5UO2I+aIprQITM=
|
github.com/mailru/easyjson v0.7.0 h1:aizVhC/NAAcKWb+5QsU1iNOZb4Yws5UO2I+aIprQITM=
|
||||||
github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
|
github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
|
||||||
github.com/markbates/pkger v0.17.1/go.mod h1:0JoVlrol20BSywW79rN3kdFFsE5xYM+rSCQDXbLhiuI=
|
|
||||||
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
||||||
github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
|
github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
|
||||||
github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
||||||
@@ -320,7 +317,6 @@ github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR
|
|||||||
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||||
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d/go.mod h1:w5+eXa0mYznDkHaMCXA4XYffjlH+cy1oyKbfzJXa2Do=
|
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d/go.mod h1:w5+eXa0mYznDkHaMCXA4XYffjlH+cy1oyKbfzJXa2Do=
|
||||||
github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo=
|
github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo=
|
||||||
github.com/sergi/go-diff v1.1.0 h1:we8PVUC3FE2uYfodKH/nBHMSetSfHDR6scGdBi+erh0=
|
|
||||||
github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
|
github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
|
||||||
github.com/shirou/gopsutil v0.0.0-20190901111213-e4ec7b275ada/go.mod h1:WWnYX4lzhCH5h/3YBfyVA3VbLYjlMZZAQcW9ojMexNc=
|
github.com/shirou/gopsutil v0.0.0-20190901111213-e4ec7b275ada/go.mod h1:WWnYX4lzhCH5h/3YBfyVA3VbLYjlMZZAQcW9ojMexNc=
|
||||||
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
|
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
|
||||||
@@ -359,8 +355,8 @@ github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e/go.mod h1:Qimiff
|
|||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
||||||
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
||||||
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
||||||
github.com/ulikunitz/xz v0.5.8 h1:ERv8V6GKqVi23rgu5cj9pVfVzJbOqAY2Ntl88O6c2nQ=
|
github.com/ulikunitz/xz v0.5.5 h1:pFrO0lVpTBXLpYw+pnLj6TbvHuyjXMfjGeCwSqCVwok=
|
||||||
github.com/ulikunitz/xz v0.5.8/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
|
github.com/ulikunitz/xz v0.5.5/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8=
|
||||||
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
||||||
github.com/ultraware/whitespace v0.0.4/go.mod h1:aVMh/gQve5Maj9hQ/hg+F75lr/X5A89uZnzAmWSineA=
|
github.com/ultraware/whitespace v0.0.4/go.mod h1:aVMh/gQve5Maj9hQ/hg+F75lr/X5A89uZnzAmWSineA=
|
||||||
github.com/uudashr/gocognit v0.0.0-20190926065955-1655d0de0517/go.mod h1:j44Ayx2KW4+oB6SWMv8KsmHzZrOInQav7D3cQMJ5JUM=
|
github.com/uudashr/gocognit v0.0.0-20190926065955-1655d0de0517/go.mod h1:j44Ayx2KW4+oB6SWMv8KsmHzZrOInQav7D3cQMJ5JUM=
|
||||||
@@ -373,8 +369,8 @@ github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q
|
|||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca h1:1CFlNzQhALwjS9mBAUkycX616GzgsuYUOCHA5+HSlXI=
|
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca h1:1CFlNzQhALwjS9mBAUkycX616GzgsuYUOCHA5+HSlXI=
|
||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
|
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
|
||||||
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf h1:gvEmqF83GB8R5XtrMseJb6A6R0OCtNAS8f4TmZg2dGc=
|
github.com/yujunz/go-getter v1.4.1-lite h1:FhvNc94AXMZkfqUwfMKhnQEC9phkphSGdPTL7tIdhOM=
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf/go.mod h1:bL0Pr07HEdsMZ1WBqZIxXj96r5LnFsY4LgPaPEGkw1k=
|
github.com/yujunz/go-getter v1.4.1-lite/go.mod h1:sbmqxXjyLunH1PkF3n7zSlnVeMvmYUuIl9ZVs/7NyCc=
|
||||||
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
||||||
go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
||||||
go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
||||||
@@ -507,13 +503,12 @@ gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.4 h1:/eiJrUcujPVeJ3xlSWaiNi3uSVmDGBK1pDHUHAnao1I=
|
gopkg.in/yaml.v2 v2.2.4 h1:/eiJrUcujPVeJ3xlSWaiNi3uSVmDGBK1pDHUHAnao1I=
|
||||||
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
|
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
|
||||||
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo=
|
gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71 h1:Xe2gvTZUJpsvOWUnvmL/tmhVBZUmHSvLbMjRj6NUUKo=
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
|
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
|
||||||
@@ -534,8 +529,8 @@ k8s.io/utils v0.0.0-20191114184206-e782cd3c129f/go.mod h1:sZAwmy6armz5eXlNoLmJcl
|
|||||||
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc=
|
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc=
|
||||||
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
||||||
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.7 h1:r0r8UEL0bL7X56HKUmhJZ+TP+nvRNGrDHHSLO7izlcQ=
|
sigs.k8s.io/kustomize/kyaml v0.9.3 h1:kZ5HnNmmnbndSXFivrAVM6u3Ik1dwu4kbpaV8KNwy8I=
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.7/go.mod h1:K9yg1k/HB/6xNOf5VH3LhTo1DK9/5ykSZO5uIv+Y/1k=
|
sigs.k8s.io/kustomize/kyaml v0.9.3/go.mod h1:UTm64bSWVdBUA8EQoYCxVOaBQxUdIOr5LKWxA4GNbkw=
|
||||||
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
||||||
sigs.k8s.io/yaml v1.1.0 h1:4A07+ZFc2wgJwo8YNlQpr1rVlgUDlxXHhPJciaPY5gs=
|
sigs.k8s.io/yaml v1.1.0 h1:4A07+ZFc2wgJwo8YNlQpr1rVlgUDlxXHhPJciaPY5gs=
|
||||||
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package generators
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// MakeConfigMap makes a configmap.
|
|
||||||
//
|
|
||||||
// ConfigMap: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#configmap-v1-core
|
|
||||||
//
|
|
||||||
// ConfigMaps and Secrets are similar.
|
|
||||||
//
|
|
||||||
// Both objects have a `data` field, which contains a map from keys to
|
|
||||||
// values that must be UTF-8 valid strings. Such data might be simple text,
|
|
||||||
// or whoever made the data may have done so by performing a base64 encoding
|
|
||||||
// on binary data. Regardless, k8s has no means to know this, so it treats
|
|
||||||
// the data field as a string.
|
|
||||||
//
|
|
||||||
// The ConfigMap has an additional field `binaryData`, also a map, but its
|
|
||||||
// values are _intended_ to be interpreted as a base64 encoding of []byte,
|
|
||||||
// by whatever makes use of the ConfigMap.
|
|
||||||
//
|
|
||||||
// In a ConfigMap, any key used in `data` cannot also be used in `binaryData`
|
|
||||||
// and vice-versa. A key must be unique across both maps.
|
|
||||||
func MakeConfigMap(
|
|
||||||
ldr ifc.KvLoader, args *types.ConfigMapArgs) (rn *yaml.RNode, err error) {
|
|
||||||
rn, err = makeBaseNode("ConfigMap", args.Name, args.Namespace)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
m, err := makeValidatedDataMap(ldr, args.Name, args.KvPairSources)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = rn.LoadMapIntoConfigMapData(m); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
copyLabelsAndAnnotations(rn, args.Options)
|
|
||||||
return rn, nil
|
|
||||||
}
|
|
||||||
@@ -1,223 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package generators_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/generators"
|
|
||||||
"sigs.k8s.io/kustomize/api/kv"
|
|
||||||
"sigs.k8s.io/kustomize/api/loader"
|
|
||||||
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var binaryHello = []byte{
|
|
||||||
0xff, // non-utf8
|
|
||||||
0x68, // h
|
|
||||||
0x65, // e
|
|
||||||
0x6c, // l
|
|
||||||
0x6c, // l
|
|
||||||
0x6f, // o
|
|
||||||
}
|
|
||||||
|
|
||||||
func manyHellos(count int) (result []byte) {
|
|
||||||
for i := 0; i < count; i++ {
|
|
||||||
result = append(result, binaryHello...)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMakeConfigMap(t *testing.T) {
|
|
||||||
type expected struct {
|
|
||||||
out string
|
|
||||||
errMsg string
|
|
||||||
}
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
args types.ConfigMapArgs
|
|
||||||
exp expected
|
|
||||||
}{
|
|
||||||
"construct config map from env": {
|
|
||||||
args: types.ConfigMapArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "envConfigMap",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
EnvSources: []string{
|
|
||||||
filepath.Join("configmap", "app.env"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: envConfigMap
|
|
||||||
data:
|
|
||||||
DB_PASSWORD: qwerty
|
|
||||||
DB_USERNAME: admin
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct config map from text file": {
|
|
||||||
args: types.ConfigMapArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "fileConfigMap1",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
FileSources: []string{
|
|
||||||
filepath.Join("configmap", "app-init.ini"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: fileConfigMap1
|
|
||||||
data:
|
|
||||||
app-init.ini: |
|
|
||||||
FOO=bar
|
|
||||||
BAR=baz
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct config map from text and binary file": {
|
|
||||||
args: types.ConfigMapArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "fileConfigMap2",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
FileSources: []string{
|
|
||||||
filepath.Join("configmap", "app-init.ini"),
|
|
||||||
filepath.Join("configmap", "app.bin"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: fileConfigMap2
|
|
||||||
data:
|
|
||||||
app-init.ini: |
|
|
||||||
FOO=bar
|
|
||||||
BAR=baz
|
|
||||||
binaryData:
|
|
||||||
app.bin: |
|
|
||||||
/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbG
|
|
||||||
xv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hl
|
|
||||||
bGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv/2
|
|
||||||
hlbGxv/2hlbGxv/2hlbGxv/2hlbGxv
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct config map from literal": {
|
|
||||||
args: types.ConfigMapArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "literalConfigMap1",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
LiteralSources: []string{"a=x", "b=y", "c=\"Hello World\"", "d='true'"},
|
|
||||||
},
|
|
||||||
Options: &types.GeneratorOptions{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"foo": "bar",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: literalConfigMap1
|
|
||||||
labels:
|
|
||||||
foo: 'bar'
|
|
||||||
data:
|
|
||||||
a: x
|
|
||||||
b: y
|
|
||||||
c: Hello World
|
|
||||||
d: "true"
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct config map from literal with GeneratorOptions in ConfigMapArgs": {
|
|
||||||
args: types.ConfigMapArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "literalConfigMap2",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
LiteralSources: []string{"a=x", "b=y", "c=\"Hello World\"", "d='true'"},
|
|
||||||
},
|
|
||||||
Options: &types.GeneratorOptions{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"veggie": "celery",
|
|
||||||
"dog": "beagle",
|
|
||||||
"cat": "annoying",
|
|
||||||
},
|
|
||||||
Annotations: map[string]string{
|
|
||||||
"river": "Missouri",
|
|
||||||
"city": "Iowa City",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: literalConfigMap2
|
|
||||||
labels:
|
|
||||||
cat: 'annoying'
|
|
||||||
dog: 'beagle'
|
|
||||||
veggie: 'celery'
|
|
||||||
annotations:
|
|
||||||
city: 'Iowa City'
|
|
||||||
river: 'Missouri'
|
|
||||||
data:
|
|
||||||
a: x
|
|
||||||
b: y
|
|
||||||
c: Hello World
|
|
||||||
d: "true"
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
fSys := filesys.MakeFsInMemory()
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("configmap", "app.env"),
|
|
||||||
[]byte("DB_USERNAME=admin\nDB_PASSWORD=qwerty\n"))
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("configmap", "app-init.ini"),
|
|
||||||
[]byte("FOO=bar\nBAR=baz\n"))
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("configmap", "app.bin"),
|
|
||||||
manyHellos(30))
|
|
||||||
kvLdr := kv.NewLoader(
|
|
||||||
loader.NewFileLoaderAtRoot(fSys),
|
|
||||||
valtest_test.MakeFakeValidator())
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
rn, err := MakeConfigMap(kvLdr, &tc.args)
|
|
||||||
if err != nil {
|
|
||||||
if !assert.EqualError(t, err, tc.exp.errMsg) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if tc.exp.errMsg != "" {
|
|
||||||
t.Fatalf("%s: should return error '%s'", n, tc.exp.errMsg)
|
|
||||||
}
|
|
||||||
output := rn.MustString()
|
|
||||||
if !assert.Equal(t, tc.exp.out, output) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package generators
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// MakeSecret makes a kubernetes Secret.
|
|
||||||
//
|
|
||||||
// Secret: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#secret-v1-core
|
|
||||||
//
|
|
||||||
// ConfigMaps and Secrets are similar.
|
|
||||||
//
|
|
||||||
// Like a ConfigMap, a Secret has a `data` field, but unlike a ConfigMap it has
|
|
||||||
// no `binaryData` field.
|
|
||||||
//
|
|
||||||
// All of a Secret's data is assumed to be opaque in nature, and assumed to be
|
|
||||||
// base64 encoded from its original representation, regardless of whether the
|
|
||||||
// original data was UTF-8 text or binary.
|
|
||||||
//
|
|
||||||
// This encoding provides no secrecy. It's just a neutral, common means to
|
|
||||||
// represent opaque text and binary data. Beneath the base64 encoding
|
|
||||||
// is presumably further encoding under control of the Secret's consumer.
|
|
||||||
//
|
|
||||||
// A Secret has string field `type` which holds an identifier, used by the
|
|
||||||
// client, to choose the algorithm to interpret the `data` field. Kubernetes
|
|
||||||
// cannot make use of this data; it's up to a controller or some pod's service
|
|
||||||
// to interpret the value, using `type` as a clue as to how to do this.
|
|
||||||
func MakeSecret(
|
|
||||||
ldr ifc.KvLoader, args *types.SecretArgs) (rn *yaml.RNode, err error) {
|
|
||||||
rn, err = makeBaseNode("Secret", args.Name, args.Namespace)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
t := "Opaque"
|
|
||||||
if args.Type != "" {
|
|
||||||
t = args.Type
|
|
||||||
}
|
|
||||||
if _, err := rn.Pipe(
|
|
||||||
yaml.FieldSetter{
|
|
||||||
Name: "type",
|
|
||||||
Value: yaml.NewStringRNode(t)}); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
m, err := makeValidatedDataMap(ldr, args.Name, args.KvPairSources)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = rn.LoadMapIntoSecretData(m); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
copyLabelsAndAnnotations(rn, args.Options)
|
|
||||||
return rn, nil
|
|
||||||
}
|
|
||||||
@@ -1,231 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package generators_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/generators"
|
|
||||||
"sigs.k8s.io/kustomize/api/kv"
|
|
||||||
"sigs.k8s.io/kustomize/api/loader"
|
|
||||||
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestMakeSecret(t *testing.T) {
|
|
||||||
type expected struct {
|
|
||||||
out string
|
|
||||||
errMsg string
|
|
||||||
}
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
args types.SecretArgs
|
|
||||||
exp expected
|
|
||||||
}{
|
|
||||||
"construct secret from env": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "envSecret",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
EnvSources: []string{
|
|
||||||
filepath.Join("secret", "app.env"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: envSecret
|
|
||||||
type: Opaque
|
|
||||||
data:
|
|
||||||
DB_PASSWORD: cXdlcnR5
|
|
||||||
DB_USERNAME: YWRtaW4=
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct secret from text file": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "fileSecret1",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
FileSources: []string{
|
|
||||||
filepath.Join("secret", "app-init.ini"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: fileSecret1
|
|
||||||
type: Opaque
|
|
||||||
data:
|
|
||||||
app-init.ini: Rk9PPWJhcgpCQVI9YmF6Cg==
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct secret from text and binary file": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "fileSecret2",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
FileSources: []string{
|
|
||||||
filepath.Join("secret", "app-init.ini"),
|
|
||||||
filepath.Join("secret", "app.bin"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: fileSecret2
|
|
||||||
type: Opaque
|
|
||||||
data:
|
|
||||||
app-init.ini: Rk9PPWJhcgpCQVI9YmF6Cg==
|
|
||||||
app.bin: //0=
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct secret from literal": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "literalSecret1",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
LiteralSources: []string{"a=x", "b=y", "c=\"Hello World\"", "d='true'"},
|
|
||||||
},
|
|
||||||
Options: &types.GeneratorOptions{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"foo": "bar",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: literalSecret1
|
|
||||||
labels:
|
|
||||||
foo: 'bar'
|
|
||||||
type: Opaque
|
|
||||||
data:
|
|
||||||
a: eA==
|
|
||||||
b: eQ==
|
|
||||||
c: SGVsbG8gV29ybGQ=
|
|
||||||
d: dHJ1ZQ==
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct secret with type": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "literalSecret1",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
LiteralSources: []string{"a=x"},
|
|
||||||
},
|
|
||||||
Options: &types.GeneratorOptions{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"foo": "bar",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
Type: "foobar",
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: literalSecret1
|
|
||||||
labels:
|
|
||||||
foo: 'bar'
|
|
||||||
type: foobar
|
|
||||||
data:
|
|
||||||
a: eA==
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"construct secret from literal with GeneratorOptions in SecretArgs": {
|
|
||||||
args: types.SecretArgs{
|
|
||||||
GeneratorArgs: types.GeneratorArgs{
|
|
||||||
Name: "literalSecret2",
|
|
||||||
KvPairSources: types.KvPairSources{
|
|
||||||
LiteralSources: []string{"a=x", "b=y", "c=\"Hello World\"", "d='true'"},
|
|
||||||
},
|
|
||||||
Options: &types.GeneratorOptions{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"veggie": "celery",
|
|
||||||
"dog": "beagle",
|
|
||||||
"cat": "annoying",
|
|
||||||
},
|
|
||||||
Annotations: map[string]string{
|
|
||||||
"river": "Missouri",
|
|
||||||
"city": "Iowa City",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: expected{
|
|
||||||
out: `apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: literalSecret2
|
|
||||||
labels:
|
|
||||||
cat: 'annoying'
|
|
||||||
dog: 'beagle'
|
|
||||||
veggie: 'celery'
|
|
||||||
annotations:
|
|
||||||
city: 'Iowa City'
|
|
||||||
river: 'Missouri'
|
|
||||||
type: Opaque
|
|
||||||
data:
|
|
||||||
a: eA==
|
|
||||||
b: eQ==
|
|
||||||
c: SGVsbG8gV29ybGQ=
|
|
||||||
d: dHJ1ZQ==
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
fSys := filesys.MakeFsInMemory()
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("secret", "app.env"),
|
|
||||||
[]byte("DB_USERNAME=admin\nDB_PASSWORD=qwerty\n"))
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("secret", "app-init.ini"),
|
|
||||||
[]byte("FOO=bar\nBAR=baz\n"))
|
|
||||||
fSys.WriteFile(
|
|
||||||
filesys.RootedPath("secret", "app.bin"),
|
|
||||||
[]byte{0xff, 0xfd})
|
|
||||||
kvLdr := kv.NewLoader(
|
|
||||||
loader.NewFileLoaderAtRoot(fSys),
|
|
||||||
valtest_test.MakeFakeValidator())
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
rn, err := MakeSecret(kvLdr, &tc.args)
|
|
||||||
if err != nil {
|
|
||||||
if !assert.EqualError(t, err, tc.exp.errMsg) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if tc.exp.errMsg != "" {
|
|
||||||
t.Fatalf("%s: should return error '%s'", n, tc.exp.errMsg)
|
|
||||||
}
|
|
||||||
output := rn.MustString()
|
|
||||||
if !assert.Equal(t, tc.exp.out, output) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,78 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package generators
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"github.com/go-errors/errors"
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func makeBaseNode(kind, name, namespace string) (*yaml.RNode, error) {
|
|
||||||
rn, err := yaml.Parse(fmt.Sprintf(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: %s
|
|
||||||
`, kind))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if name == "" {
|
|
||||||
return nil, errors.Errorf("a configmap must have a name")
|
|
||||||
}
|
|
||||||
if _, err := rn.Pipe(yaml.SetK8sName(name)); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if namespace != "" {
|
|
||||||
if _, err := rn.Pipe(yaml.SetK8sNamespace(namespace)); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return rn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func makeValidatedDataMap(
|
|
||||||
ldr ifc.KvLoader, name string, sources types.KvPairSources) (map[string]string, error) {
|
|
||||||
pairs, err := ldr.Load(sources)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.WrapPrefix(err, "loading KV pairs", 0)
|
|
||||||
}
|
|
||||||
knownKeys := make(map[string]string)
|
|
||||||
for _, p := range pairs {
|
|
||||||
// legal key: alphanumeric characters, '-', '_' or '.'
|
|
||||||
if err := ldr.Validator().ErrIfInvalidKey(p.Key); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if _, ok := knownKeys[p.Key]; ok {
|
|
||||||
return nil, errors.Errorf(
|
|
||||||
"configmap %s illegally repeats the key `%s`", name, p.Key)
|
|
||||||
}
|
|
||||||
knownKeys[p.Key] = p.Value
|
|
||||||
}
|
|
||||||
return knownKeys, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// copyLabelsAndAnnotations copies labels and annotations from
|
|
||||||
// GeneratorOptions into the given object.
|
|
||||||
func copyLabelsAndAnnotations(
|
|
||||||
rn *yaml.RNode, opts *types.GeneratorOptions) error {
|
|
||||||
if opts == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
for _, k := range yaml.SortedMapKeys(opts.Labels) {
|
|
||||||
v := opts.Labels[k]
|
|
||||||
if _, err := rn.Pipe(yaml.SetLabel(k, v)); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for _, k := range yaml.SortedMapKeys(opts.Annotations) {
|
|
||||||
v := opts.Annotations[k]
|
|
||||||
if _, err := rn.Pipe(yaml.SetAnnotation(k, v)); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -4,6 +4,10 @@
|
|||||||
package git
|
package git
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"log"
|
||||||
|
"os/exec"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -14,29 +18,70 @@ type Cloner func(repoSpec *RepoSpec) error
|
|||||||
// to say, some remote API, to obtain a local clone of
|
// to say, some remote API, to obtain a local clone of
|
||||||
// a remote repo.
|
// a remote repo.
|
||||||
func ClonerUsingGitExec(repoSpec *RepoSpec) error {
|
func ClonerUsingGitExec(repoSpec *RepoSpec) error {
|
||||||
r, err := newCmdRunner()
|
gitProgram, err := exec.LookPath("git")
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "no 'git' program on path")
|
||||||
|
}
|
||||||
|
repoSpec.Dir, err = filesys.NewTmpConfirmedDir()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
repoSpec.Dir = r.dir
|
|
||||||
if err = r.run("init"); err != nil {
|
cmd := exec.Command(
|
||||||
return err
|
gitProgram,
|
||||||
|
"clone",
|
||||||
|
"--depth=1",
|
||||||
|
repoSpec.CloneSpec(),
|
||||||
|
repoSpec.Dir.String())
|
||||||
|
out, err := cmd.CombinedOutput()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error cloning git repo: %s", out)
|
||||||
|
return errors.Wrapf(
|
||||||
|
err,
|
||||||
|
"trouble cloning git repo %v in %s",
|
||||||
|
repoSpec.CloneSpec(), repoSpec.Dir.String())
|
||||||
}
|
}
|
||||||
if err = r.run(
|
|
||||||
"remote", "add", "origin", repoSpec.CloneSpec()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
ref := "HEAD"
|
|
||||||
if repoSpec.Ref != "" {
|
if repoSpec.Ref != "" {
|
||||||
ref = repoSpec.Ref
|
cmd = exec.Command(
|
||||||
|
gitProgram,
|
||||||
|
"fetch",
|
||||||
|
"--depth=1",
|
||||||
|
"origin",
|
||||||
|
repoSpec.Ref)
|
||||||
|
cmd.Dir = repoSpec.Dir.String()
|
||||||
|
out, err = cmd.CombinedOutput()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error fetching ref: %s", out)
|
||||||
|
return errors.Wrapf(err, "trouble fetching %s", repoSpec.Ref)
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd = exec.Command(
|
||||||
|
gitProgram,
|
||||||
|
"checkout",
|
||||||
|
"FETCH_HEAD")
|
||||||
|
cmd.Dir = repoSpec.Dir.String()
|
||||||
|
out, err = cmd.CombinedOutput()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error checking out ref: %s", out)
|
||||||
|
return errors.Wrapf(err, "trouble checking out %s", repoSpec.Ref)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if err = r.run("fetch", "--depth=1", "origin", ref); err != nil {
|
|
||||||
return err
|
cmd = exec.Command(
|
||||||
|
gitProgram,
|
||||||
|
"submodule",
|
||||||
|
"update",
|
||||||
|
"--init",
|
||||||
|
"--recursive")
|
||||||
|
cmd.Dir = repoSpec.Dir.String()
|
||||||
|
out, err = cmd.CombinedOutput()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error fetching submodules: %s", out)
|
||||||
|
return errors.Wrapf(err, "trouble fetching submodules for %s", repoSpec.CloneSpec())
|
||||||
}
|
}
|
||||||
if err = r.run("checkout", "FETCH_HEAD"); err != nil {
|
|
||||||
return err
|
return nil
|
||||||
}
|
|
||||||
return r.run("submodule", "update", "--init", "--recursive")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoNothingCloner returns a cloner that only sets
|
// DoNothingCloner returns a cloner that only sets
|
||||||
|
|||||||
@@ -1,58 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package git
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os/exec"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/utils"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Arbitrary, but non-infinite, timeout for running commands.
|
|
||||||
const defaultDuration = 27 * time.Second
|
|
||||||
|
|
||||||
// gitRunner runs the external git binary.
|
|
||||||
type gitRunner struct {
|
|
||||||
gitProgram string
|
|
||||||
duration time.Duration
|
|
||||||
dir filesys.ConfirmedDir
|
|
||||||
}
|
|
||||||
|
|
||||||
// newCmdRunner returns a gitRunner if it can find the binary.
|
|
||||||
// It also creats a temp directory for cloning repos.
|
|
||||||
func newCmdRunner() (*gitRunner, error) {
|
|
||||||
gitProgram, err := exec.LookPath("git")
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrap(err, "no 'git' program on path")
|
|
||||||
}
|
|
||||||
dir, err := filesys.NewTmpConfirmedDir()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &gitRunner{
|
|
||||||
gitProgram: gitProgram,
|
|
||||||
duration: defaultDuration,
|
|
||||||
dir: dir,
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// run a command with a timeout.
|
|
||||||
func (r gitRunner) run(args ...string) error {
|
|
||||||
//nolint: gosec
|
|
||||||
cmd := exec.Command(r.gitProgram, args...)
|
|
||||||
cmd.Dir = r.dir.String()
|
|
||||||
return utils.TimedCall(
|
|
||||||
cmd.String(),
|
|
||||||
r.duration,
|
|
||||||
func() error {
|
|
||||||
_, err := cmd.CombinedOutput()
|
|
||||||
if err != nil {
|
|
||||||
return errors.Wrapf(err, "git cmd = '%s'", cmd.String())
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
})
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package conflict
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
|
||||||
"k8s.io/apimachinery/pkg/util/mergepatch"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
// conflictDetectorJson detects conflicts in a list of JSON patches.
|
|
||||||
type conflictDetectorJson struct {
|
|
||||||
resourceFactory *resource.Factory
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ resource.ConflictDetector = &conflictDetectorJson{}
|
|
||||||
|
|
||||||
func (cd *conflictDetectorJson) HasConflict(
|
|
||||||
p1, p2 *resource.Resource) (bool, error) {
|
|
||||||
return mergepatch.HasConflicts(p1.Map(), p2.Map())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cd *conflictDetectorJson) MergePatches(
|
|
||||||
patch1, patch2 *resource.Resource) (*resource.Resource, error) {
|
|
||||||
baseBytes, err := json.Marshal(patch1.Map())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
patchBytes, err := json.Marshal(patch2.Map())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
mergedBytes, err := jsonpatch.MergeMergePatches(baseBytes, patchBytes)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
mergedMap := make(map[string]interface{})
|
|
||||||
err = json.Unmarshal(mergedBytes, &mergedMap)
|
|
||||||
return cd.resourceFactory.FromMap(mergedMap), err
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package conflict
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"k8s.io/apimachinery/pkg/util/strategicpatch"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
// conflictDetectorSm detects conflicts in a list of strategic merge patches.
|
|
||||||
type conflictDetectorSm struct {
|
|
||||||
lookupPatchMeta strategicpatch.LookupPatchMeta
|
|
||||||
resourceFactory *resource.Factory
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ resource.ConflictDetector = &conflictDetectorSm{}
|
|
||||||
|
|
||||||
func (cd *conflictDetectorSm) HasConflict(
|
|
||||||
p1, p2 *resource.Resource) (bool, error) {
|
|
||||||
return strategicpatch.MergingMapsHaveConflicts(
|
|
||||||
p1.Map(), p2.Map(), cd.lookupPatchMeta)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cd *conflictDetectorSm) MergePatches(
|
|
||||||
patch1, patch2 *resource.Resource) (*resource.Resource, error) {
|
|
||||||
if cd.hasDeleteDirectiveMarker(patch2.Map()) {
|
|
||||||
if cd.hasDeleteDirectiveMarker(patch1.Map()) {
|
|
||||||
return nil, fmt.Errorf(
|
|
||||||
"cannot merge patches both containing '$patch: delete' directives")
|
|
||||||
}
|
|
||||||
patch1, patch2 = patch2, patch1
|
|
||||||
}
|
|
||||||
mergedMap, err := strategicpatch.MergeStrategicMergeMapPatchUsingLookupPatchMeta(
|
|
||||||
cd.lookupPatchMeta, patch1.Map(), patch2.Map())
|
|
||||||
return cd.resourceFactory.FromMap(mergedMap), err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cd *conflictDetectorSm) hasDeleteDirectiveMarker(
|
|
||||||
patch map[string]interface{}) bool {
|
|
||||||
if v, ok := patch["$patch"]; ok && v == "delete" {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
for _, v := range patch {
|
|
||||||
switch typedV := v.(type) {
|
|
||||||
case map[string]interface{}:
|
|
||||||
if cd.hasDeleteDirectiveMarker(typedV) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
case []interface{}:
|
|
||||||
for _, sv := range typedV {
|
|
||||||
typedE, ok := sv.(map[string]interface{})
|
|
||||||
if !ok {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if cd.hasDeleteDirectiveMarker(typedE) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package conflict
|
|
||||||
|
|
||||||
import (
|
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
|
||||||
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
||||||
sp "k8s.io/apimachinery/pkg/util/strategicpatch"
|
|
||||||
"k8s.io/client-go/kubernetes/scheme"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
type cdFactory struct {
|
|
||||||
rf *resource.Factory
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ resource.ConflictDetectorFactory = &cdFactory{}
|
|
||||||
|
|
||||||
// NewFactory returns a conflict detector factory.
|
|
||||||
// The detector uses a resource factory to convert resources to/from
|
|
||||||
// json/yaml/maps representations.
|
|
||||||
func NewFactory(rf *resource.Factory) resource.ConflictDetectorFactory {
|
|
||||||
return &cdFactory{rf: rf}
|
|
||||||
}
|
|
||||||
|
|
||||||
// New returns a conflict detector that's aware of the GVK type.
|
|
||||||
func (f *cdFactory) New(gvk resid.Gvk) (resource.ConflictDetector, error) {
|
|
||||||
// Convert to apimachinery representation of object
|
|
||||||
obj, err := scheme.Scheme.New(schema.GroupVersionKind{
|
|
||||||
Group: gvk.Group,
|
|
||||||
Version: gvk.Version,
|
|
||||||
Kind: gvk.Kind,
|
|
||||||
})
|
|
||||||
if err == nil {
|
|
||||||
meta, err := sp.NewPatchMetaFromStruct(obj)
|
|
||||||
return &conflictDetectorSm{
|
|
||||||
lookupPatchMeta: meta, resourceFactory: f.rf}, err
|
|
||||||
}
|
|
||||||
if runtime.IsNotRegisteredError(err) {
|
|
||||||
return &conflictDetectorJson{resourceFactory: f.rf}, nil
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
239
api/internal/k8sdeps/merge/merginator.go
Normal file
239
api/internal/k8sdeps/merge/merginator.go
Normal file
@@ -0,0 +1,239 @@
|
|||||||
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
package merge
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
jsonpatch "github.com/evanphx/json-patch"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
||||||
|
"k8s.io/apimachinery/pkg/util/mergepatch"
|
||||||
|
"k8s.io/apimachinery/pkg/util/strategicpatch"
|
||||||
|
"k8s.io/client-go/kubernetes/scheme"
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
|
)
|
||||||
|
|
||||||
|
type conflictDetector interface {
|
||||||
|
hasConflict(patch1, patch2 *resource.Resource) (bool, error)
|
||||||
|
findConflict(
|
||||||
|
conflictingPatchIdx int,
|
||||||
|
patches []*resource.Resource) (*resource.Resource, error)
|
||||||
|
mergePatches(patch1, patch2 *resource.Resource) (*resource.Resource, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type jsonMergePatch struct {
|
||||||
|
resourceFactory *resource.Factory
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ conflictDetector = &jsonMergePatch{}
|
||||||
|
|
||||||
|
func newJMPConflictDetector(rf *resource.Factory) conflictDetector {
|
||||||
|
return &jsonMergePatch{resourceFactory: rf}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (jmp *jsonMergePatch) hasConflict(
|
||||||
|
patch1, patch2 *resource.Resource) (bool, error) {
|
||||||
|
return mergepatch.HasConflicts(patch1.Map(), patch2.Map())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (jmp *jsonMergePatch) findConflict(
|
||||||
|
conflictingPatchIdx int,
|
||||||
|
patches []*resource.Resource) (*resource.Resource, error) {
|
||||||
|
for i, patch := range patches {
|
||||||
|
if i == conflictingPatchIdx {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !patches[conflictingPatchIdx].OrgId().Equals(patch.OrgId()) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
conflict, err := mergepatch.HasConflicts(
|
||||||
|
patch.Map(),
|
||||||
|
patches[conflictingPatchIdx].Map())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if conflict {
|
||||||
|
return patch, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (jmp *jsonMergePatch) mergePatches(
|
||||||
|
patch1, patch2 *resource.Resource) (*resource.Resource, error) {
|
||||||
|
baseBytes, err := json.Marshal(patch1.Map())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
patchBytes, err := json.Marshal(patch2.Map())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
mergedBytes, err := jsonpatch.MergeMergePatches(baseBytes, patchBytes)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
mergedMap := make(map[string]interface{})
|
||||||
|
err = json.Unmarshal(mergedBytes, &mergedMap)
|
||||||
|
return jmp.resourceFactory.FromMap(mergedMap), err
|
||||||
|
}
|
||||||
|
|
||||||
|
type strategicMergePatch struct {
|
||||||
|
lookupPatchMeta strategicpatch.LookupPatchMeta
|
||||||
|
rf *resource.Factory
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ conflictDetector = &strategicMergePatch{}
|
||||||
|
|
||||||
|
func newSMPConflictDetector(
|
||||||
|
versionedObj runtime.Object,
|
||||||
|
rf *resource.Factory) (conflictDetector, error) {
|
||||||
|
lookupPatchMeta, err := strategicpatch.NewPatchMetaFromStruct(versionedObj)
|
||||||
|
return &strategicMergePatch{lookupPatchMeta: lookupPatchMeta, rf: rf}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (smp *strategicMergePatch) hasConflict(
|
||||||
|
p1, p2 *resource.Resource) (bool, error) {
|
||||||
|
return strategicpatch.MergingMapsHaveConflicts(
|
||||||
|
p1.Map(), p2.Map(), smp.lookupPatchMeta)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (smp *strategicMergePatch) findConflict(
|
||||||
|
conflictingPatchIdx int,
|
||||||
|
patches []*resource.Resource) (*resource.Resource, error) {
|
||||||
|
for i, patch := range patches {
|
||||||
|
if i == conflictingPatchIdx {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !patches[conflictingPatchIdx].OrgId().Equals(patch.OrgId()) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
conflict, err := strategicpatch.MergingMapsHaveConflicts(
|
||||||
|
patch.Map(),
|
||||||
|
patches[conflictingPatchIdx].Map(),
|
||||||
|
smp.lookupPatchMeta)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if conflict {
|
||||||
|
return patch, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (smp *strategicMergePatch) mergePatches(
|
||||||
|
patch1, patch2 *resource.Resource) (*resource.Resource, error) {
|
||||||
|
if hasDeleteDirectiveMarker(patch2.Map()) {
|
||||||
|
if hasDeleteDirectiveMarker(patch1.Map()) {
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"cannot merge patches both containing '$patch: delete' directives")
|
||||||
|
}
|
||||||
|
patch1, patch2 = patch2, patch1
|
||||||
|
}
|
||||||
|
mergeJSONMap, err := strategicpatch.MergeStrategicMergeMapPatchUsingLookupPatchMeta(
|
||||||
|
smp.lookupPatchMeta, patch1.Map(), patch2.Map())
|
||||||
|
return smp.rf.FromMap(mergeJSONMap), err
|
||||||
|
}
|
||||||
|
|
||||||
|
type merginatorImpl struct {
|
||||||
|
rf *resource.Factory
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMerginator returns a new implementation of resmap.Merginator.
|
||||||
|
func NewMerginator(rf *resource.Factory) resmap.Merginator {
|
||||||
|
return &merginatorImpl{rf: rf}
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ resmap.Merginator = (*merginatorImpl)(nil)
|
||||||
|
|
||||||
|
// Merge merges the incoming resources into a new resmap.ResMap.
|
||||||
|
// Returns an error on conflict.
|
||||||
|
func (m *merginatorImpl) Merge(
|
||||||
|
patches []*resource.Resource) (resmap.ResMap, error) {
|
||||||
|
rc := resmap.New()
|
||||||
|
for ix, patch := range patches {
|
||||||
|
id := patch.OrgId()
|
||||||
|
existing := rc.GetMatchingResourcesByOriginalId(id.Equals)
|
||||||
|
if len(existing) == 0 {
|
||||||
|
rc.Append(patch)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if len(existing) > 1 {
|
||||||
|
return nil, fmt.Errorf("self conflict in patches")
|
||||||
|
}
|
||||||
|
|
||||||
|
versionedObj, err := scheme.Scheme.New(toSchemaGvk(id.Gvk))
|
||||||
|
if err != nil && !runtime.IsNotRegisteredError(err) {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var cd conflictDetector
|
||||||
|
if err != nil {
|
||||||
|
cd = newJMPConflictDetector(m.rf)
|
||||||
|
} else {
|
||||||
|
cd, err = newSMPConflictDetector(versionedObj, m.rf)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
conflict, err := cd.hasConflict(existing[0], patch)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if conflict {
|
||||||
|
conflictingPatch, err := cd.findConflict(ix, patches)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"conflict between %#v and %#v",
|
||||||
|
conflictingPatch.Map(), patch.Map())
|
||||||
|
}
|
||||||
|
merged, err := cd.mergePatches(existing[0], patch)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
rc.Replace(merged)
|
||||||
|
}
|
||||||
|
return rc, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// toSchemaGvk converts to a schema.GroupVersionKind.
|
||||||
|
func toSchemaGvk(x resid.Gvk) schema.GroupVersionKind {
|
||||||
|
return schema.GroupVersionKind{
|
||||||
|
Group: x.Group,
|
||||||
|
Version: x.Version,
|
||||||
|
Kind: x.Kind,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func hasDeleteDirectiveMarker(patch map[string]interface{}) bool {
|
||||||
|
if v, ok := patch["$patch"]; ok && v == "delete" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
for _, v := range patch {
|
||||||
|
switch typedV := v.(type) {
|
||||||
|
case map[string]interface{}:
|
||||||
|
if hasDeleteDirectiveMarker(typedV) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case []interface{}:
|
||||||
|
for _, sv := range typedV {
|
||||||
|
typedE, ok := sv.(map[string]interface{})
|
||||||
|
if !ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if hasDeleteDirectiveMarker(typedE) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
25
api/internal/merge/merginator.go
Normal file
25
api/internal/merge/merginator.go
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
// Copyright 2020 The Kubernetes Authors.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
package merge
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Merginator implements resmap.Merginator using kyaml libs.
|
||||||
|
type Merginator struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ resmap.Merginator = (*Merginator)(nil)
|
||||||
|
|
||||||
|
func NewMerginator(_ *resource.Factory) *Merginator {
|
||||||
|
return &Merginator{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Merge implements resmap.Merginator
|
||||||
|
func (m Merginator) Merge(
|
||||||
|
resources []*resource.Resource) (resmap.ResMap, error) {
|
||||||
|
panic("TODO(#Merginator): implement Merge")
|
||||||
|
}
|
||||||
4
api/internal/merge/merginator_test.go
Normal file
4
api/internal/merge/merginator_test.go
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
// Copyright 2020 The Kubernetes Authors.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
package merge_test
|
||||||
@@ -10,8 +10,8 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NameBackReferences is an association between a gvk.GVK (a ReferralTarget)
|
// NameBackReferences is an association between a gvk.GVK and a list
|
||||||
// and a list of Referrers that could refer to it.
|
// of FieldSpec instances that could refer to it.
|
||||||
//
|
//
|
||||||
// It is used to handle name changes, and can be thought of as a
|
// It is used to handle name changes, and can be thought of as a
|
||||||
// a contact list. If you change your own contact info (name,
|
// a contact list. If you change your own contact info (name,
|
||||||
@@ -19,20 +19,16 @@ import (
|
|||||||
// know about the change.
|
// know about the change.
|
||||||
//
|
//
|
||||||
// For example, ConfigMaps can be used by Pods and everything that
|
// For example, ConfigMaps can be used by Pods and everything that
|
||||||
// contains a Pod; Deployment, Job, StatefulSet, etc.
|
// contains a Pod; Deployment, Job, StatefulSet, etc. To change
|
||||||
// The ConfigMap is the ReferralTarget, the others are Referrers.
|
// the name of a ConfigMap instance from 'alice' to 'bob', one
|
||||||
//
|
// must visit all objects that could refer to the ConfigMap, see if
|
||||||
// If the the name of a ConfigMap instance changed from 'alice' to 'bob',
|
// they mention 'alice', and if so, change the reference to 'bob'.
|
||||||
// one must
|
|
||||||
// - visit all objects that could refer to the ConfigMap (the Referrers)
|
|
||||||
// - see if they mention 'alice',
|
|
||||||
// - if so, change the Referrer's name reference to 'bob'.
|
|
||||||
//
|
//
|
||||||
// The NameBackReferences instance to aid in this could look like
|
// The NameBackReferences instance to aid in this could look like
|
||||||
// {
|
// {
|
||||||
// kind: ConfigMap
|
// kind: ConfigMap
|
||||||
// version: v1
|
// version: v1
|
||||||
// fieldSpecs:
|
// FieldSpecs:
|
||||||
// - kind: Pod
|
// - kind: Pod
|
||||||
// version: v1
|
// version: v1
|
||||||
// path: spec/volumes/configMap/name
|
// path: spec/volumes/configMap/name
|
||||||
@@ -43,15 +39,13 @@ import (
|
|||||||
// (etc.)
|
// (etc.)
|
||||||
// }
|
// }
|
||||||
type NameBackReferences struct {
|
type NameBackReferences struct {
|
||||||
resid.Gvk `json:",inline,omitempty" yaml:",inline,omitempty"`
|
resid.Gvk `json:",inline,omitempty" yaml:",inline,omitempty"`
|
||||||
// TODO: rename json 'fieldSpecs' to 'referrers' for clarity.
|
FieldSpecs types.FsSlice `json:"FieldSpecs,omitempty" yaml:"FieldSpecs,omitempty"`
|
||||||
// This will, however, break anyone using a custom config.
|
|
||||||
Referrers types.FsSlice `json:"fieldSpecs,omitempty" yaml:"fieldSpecs,omitempty"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (n NameBackReferences) String() string {
|
func (n NameBackReferences) String() string {
|
||||||
var r []string
|
var r []string
|
||||||
for _, f := range n.Referrers {
|
for _, f := range n.FieldSpecs {
|
||||||
r = append(r, f.String())
|
r = append(r, f.String())
|
||||||
}
|
}
|
||||||
return n.Gvk.String() + ": (\n" +
|
return n.Gvk.String() + ": (\n" +
|
||||||
@@ -83,7 +77,7 @@ func (s nbrSlice) mergeOne(other NameBackReferences) (nbrSlice, error) {
|
|||||||
found := false
|
found := false
|
||||||
for _, c := range s {
|
for _, c := range s {
|
||||||
if c.Gvk.Equals(other.Gvk) {
|
if c.Gvk.Equals(other.Gvk) {
|
||||||
c.Referrers, err = c.Referrers.MergeAll(other.Referrers)
|
c.FieldSpecs, err = c.FieldSpecs.MergeAll(other.FieldSpecs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,13 +50,13 @@ func TestMergeAll(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "ConfigMap",
|
Kind: "ConfigMap",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice1,
|
FieldSpecs: fsSlice1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice2,
|
FieldSpecs: fsSlice2,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
nbrsSlice2 := nbrSlice{
|
nbrsSlice2 := nbrSlice{
|
||||||
@@ -64,13 +64,13 @@ func TestMergeAll(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "ConfigMap",
|
Kind: "ConfigMap",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice1,
|
FieldSpecs: fsSlice1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice2,
|
FieldSpecs: fsSlice2,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
expected := nbrSlice{
|
expected := nbrSlice{
|
||||||
@@ -78,13 +78,13 @@ func TestMergeAll(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "ConfigMap",
|
Kind: "ConfigMap",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice1,
|
FieldSpecs: fsSlice1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "Secret",
|
Kind: "Secret",
|
||||||
},
|
},
|
||||||
Referrers: fsSlice2,
|
FieldSpecs: fsSlice2,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
actual, err := nbrsSlice1.mergeAll(nbrsSlice2)
|
actual, err := nbrsSlice1.mergeAll(nbrsSlice2)
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ func TestAddNamereferenceFieldSpec(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindA",
|
Kind: "KindA",
|
||||||
},
|
},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindB",
|
Kind: "KindB",
|
||||||
@@ -89,7 +89,7 @@ func TestMerge(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindA",
|
Kind: "KindA",
|
||||||
},
|
},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindB",
|
Kind: "KindB",
|
||||||
@@ -103,7 +103,7 @@ func TestMerge(t *testing.T) {
|
|||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindA",
|
Kind: "KindA",
|
||||||
},
|
},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{
|
Gvk: resid.Gvk{
|
||||||
Kind: "KindC",
|
Kind: "KindC",
|
||||||
|
|||||||
@@ -13,10 +13,12 @@ import (
|
|||||||
. "sigs.k8s.io/kustomize/api/internal/plugins/execplugin"
|
. "sigs.k8s.io/kustomize/api/internal/plugins/execplugin"
|
||||||
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/utils"
|
"sigs.k8s.io/kustomize/api/internal/plugins/utils"
|
||||||
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
fLdr "sigs.k8s.io/kustomize/api/loader"
|
fLdr "sigs.k8s.io/kustomize/api/loader"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
|
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestExecPluginConfig(t *testing.T) {
|
func TestExecPluginConfig(t *testing.T) {
|
||||||
@@ -31,9 +33,10 @@ s/$BAR/bar baz/g
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
pvd := provider.NewDefaultDepProvider()
|
|
||||||
rf := resmap.NewFactory(
|
rf := resmap.NewFactory(
|
||||||
pvd.GetResourceFactory(), pvd.GetConflictDetectorFactory())
|
resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl()), nil)
|
||||||
|
v := valtest_test.MakeFakeValidator()
|
||||||
pluginConfig := rf.RF().FromMap(
|
pluginConfig := rf.RF().FromMap(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "someteam.example.com/v1",
|
"apiVersion": "someteam.example.com/v1",
|
||||||
@@ -45,7 +48,6 @@ s/$BAR/bar baz/g
|
|||||||
"argsFromFile": "sed-input.txt",
|
"argsFromFile": "sed-input.txt",
|
||||||
})
|
})
|
||||||
|
|
||||||
pluginConfig.RemoveBuildAnnotations()
|
|
||||||
p := NewExecPlugin(
|
p := NewExecPlugin(
|
||||||
pLdr.AbsolutePluginPath(
|
pLdr.AbsolutePluginPath(
|
||||||
konfig.DisabledPluginConfig(),
|
konfig.DisabledPluginConfig(),
|
||||||
@@ -60,7 +62,7 @@ s/$BAR/bar baz/g
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected err: %v", err)
|
t.Fatalf("unexpected err: %v", err)
|
||||||
}
|
}
|
||||||
p.Config(resmap.NewPluginHelpers(ldr, pvd.GetFieldValidator(), rf), yaml)
|
p.Config(resmap.NewPluginHelpers(ldr, v, rf), yaml)
|
||||||
|
|
||||||
expected := "someteam.example.com/v1/sedtransformer/SedTransformer"
|
expected := "someteam.example.com/v1/sedtransformer/SedTransformer"
|
||||||
if !strings.HasSuffix(p.Path(), expected) {
|
if !strings.HasSuffix(p.Path(), expected) {
|
||||||
|
|||||||
@@ -8,10 +8,11 @@ import (
|
|||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
. "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
. "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
||||||
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
"sigs.k8s.io/kustomize/api/loader"
|
"sigs.k8s.io/kustomize/api/loader"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
||||||
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
@@ -50,9 +51,8 @@ func TestLoader(t *testing.T) {
|
|||||||
BuildGoPlugin("builtin", "", "SecretGenerator").
|
BuildGoPlugin("builtin", "", "SecretGenerator").
|
||||||
BuildGoPlugin("someteam.example.com", "v1", "SomeServiceGenerator")
|
BuildGoPlugin("someteam.example.com", "v1", "SomeServiceGenerator")
|
||||||
defer th.Reset()
|
defer th.Reset()
|
||||||
p := provider.NewDefaultDepProvider()
|
rmF := resmap.NewFactory(resource.NewFactory(
|
||||||
rmF := resmap.NewFactory(
|
kunstruct.NewKunstructuredFactoryImpl()), nil)
|
||||||
p.GetResourceFactory(), p.GetConflictDetectorFactory())
|
|
||||||
fLdr, err := loader.NewLoader(
|
fLdr, err := loader.NewLoader(
|
||||||
loader.RestrictionRootOnly,
|
loader.RestrictionRootOnly,
|
||||||
filesys.Separator, filesys.MakeFsInMemory())
|
filesys.Separator, filesys.MakeFsInMemory())
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
package utils
|
package utils
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
@@ -14,8 +13,8 @@ import (
|
|||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
@@ -136,6 +135,9 @@ func GetResMapWithIDAnnotation(rm resmap.ResMap) (resmap.ResMap, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
annotations := r.GetAnnotations()
|
annotations := r.GetAnnotations()
|
||||||
|
if annotations == nil {
|
||||||
|
annotations = make(map[string]string)
|
||||||
|
}
|
||||||
annotations[idAnnotation] = string(idString)
|
annotations[idAnnotation] = string(idString)
|
||||||
r.SetAnnotations(annotations)
|
r.SetAnnotations(annotations)
|
||||||
}
|
}
|
||||||
@@ -145,65 +147,41 @@ func GetResMapWithIDAnnotation(rm resmap.ResMap) (resmap.ResMap, error) {
|
|||||||
// UpdateResMapValues updates the Resource value in the given ResMap
|
// UpdateResMapValues updates the Resource value in the given ResMap
|
||||||
// with the emitted Resource values in output.
|
// with the emitted Resource values in output.
|
||||||
func UpdateResMapValues(pluginName string, h *resmap.PluginHelpers, output []byte, rm resmap.ResMap) error {
|
func UpdateResMapValues(pluginName string, h *resmap.PluginHelpers, output []byte, rm resmap.ResMap) error {
|
||||||
mapFactory := h.ResmapFactory()
|
outputRM, err := h.ResmapFactory().NewResMapFromBytes(output)
|
||||||
resFactory := mapFactory.RF()
|
|
||||||
resources, err := resFactory.SliceFromBytes(output)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Don't use resources here, or error message will be unfriendly to plugin builders
|
for _, r := range outputRM.Resources() {
|
||||||
newMap, err := mapFactory.NewResMapFromBytes([]byte{})
|
// for each emitted Resource, find the matching Resource in the original ResMap
|
||||||
if err != nil {
|
// using its id
|
||||||
return err
|
annotations := r.GetAnnotations()
|
||||||
}
|
idString, ok := annotations[idAnnotation]
|
||||||
|
if !ok {
|
||||||
for _, r := range resources {
|
return fmt.Errorf("the transformer %s should not remove annotation %s",
|
||||||
removeIDAnnotation(r) // stale--not manipulated by plugin transformers
|
pluginName, idAnnotation)
|
||||||
|
|
||||||
// Add to the new map, checking for duplicates
|
|
||||||
if err := newMap.Append(r); err != nil {
|
|
||||||
prettyID, err := json.Marshal(r.CurId())
|
|
||||||
if err != nil {
|
|
||||||
prettyID = []byte(r.CurId().String())
|
|
||||||
}
|
|
||||||
return fmt.Errorf("plugin %s generated duplicate resource: %s", pluginName, prettyID)
|
|
||||||
}
|
}
|
||||||
|
id := resid.ResId{}
|
||||||
// Add to or update the old map
|
err := yaml.Unmarshal([]byte(idString), &id)
|
||||||
oldIdx, err := rm.GetIndexOfCurrentId(r.CurId())
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if oldIdx != -1 {
|
res, err := rm.GetByCurrentId(id)
|
||||||
rm.GetByIndex(oldIdx).ResetPrimaryData(r)
|
if err != nil {
|
||||||
} else {
|
return fmt.Errorf("unable to find unique match to %s", id.String())
|
||||||
if err := rm.Append(r); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
// remove the annotation set by Kustomize to track the resource
|
||||||
|
delete(annotations, idAnnotation)
|
||||||
// Remove items the transformer deleted from the old map
|
if len(annotations) == 0 {
|
||||||
for _, id := range rm.AllIds() {
|
annotations = nil
|
||||||
newIdx, _ := newMap.GetIndexOfCurrentId(id)
|
|
||||||
if newIdx == -1 {
|
|
||||||
rm.Remove(id)
|
|
||||||
}
|
}
|
||||||
}
|
r.SetAnnotations(annotations)
|
||||||
|
|
||||||
|
// update the resource value with the transformed object
|
||||||
|
res.ResetPrimaryData(r)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func removeIDAnnotation(r *resource.Resource) {
|
|
||||||
// remove the annotation set by Kustomize to track the resource
|
|
||||||
annotations := r.GetAnnotations()
|
|
||||||
delete(annotations, idAnnotation)
|
|
||||||
if len(annotations) == 0 {
|
|
||||||
annotations = nil
|
|
||||||
}
|
|
||||||
r.SetAnnotations(annotations)
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateResourceOptions updates the generator options for each resource in the
|
// UpdateResourceOptions updates the generator options for each resource in the
|
||||||
// given ResMap based on plugin provided annotations.
|
// given ResMap based on plugin provided annotations.
|
||||||
func UpdateResourceOptions(rm resmap.ResMap) (resmap.ResMap, error) {
|
func UpdateResourceOptions(rm resmap.ResMap) (resmap.ResMap, error) {
|
||||||
|
|||||||
@@ -9,10 +9,9 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
@@ -65,7 +64,7 @@ func strptr(s string) *string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestUpdateResourceOptions(t *testing.T) {
|
func TestUpdateResourceOptions(t *testing.T) {
|
||||||
rf := provider.NewDefaultDepProvider().GetResourceFactory()
|
rf := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
in := resmap.New()
|
in := resmap.New()
|
||||||
expected := resmap.New()
|
expected := resmap.New()
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
@@ -88,12 +87,28 @@ func TestUpdateResourceOptions(t *testing.T) {
|
|||||||
expected.Append(makeConfigMapOptions(rf, name, c.behavior, !c.needsHash))
|
expected.Append(makeConfigMapOptions(rf, name, c.behavior, !c.needsHash))
|
||||||
}
|
}
|
||||||
actual, err := UpdateResourceOptions(in)
|
actual, err := UpdateResourceOptions(in)
|
||||||
assert.NoError(t, err)
|
if err != nil {
|
||||||
assert.NoError(t, expected.ErrorIfNotEqualLists(actual))
|
t.Fatalf("unexpected error: %v", err.Error())
|
||||||
|
}
|
||||||
|
for i, a := range expected.Resources() {
|
||||||
|
b := actual.GetByIndex(i)
|
||||||
|
if b == nil {
|
||||||
|
t.Fatalf("resource %d missing from processed map", i)
|
||||||
|
}
|
||||||
|
if !a.Equals(b) {
|
||||||
|
t.Errorf("expected %v got %v", a, b)
|
||||||
|
}
|
||||||
|
if a.NeedHashSuffix() != b.NeedHashSuffix() {
|
||||||
|
t.Errorf("")
|
||||||
|
}
|
||||||
|
if a.Behavior() != b.Behavior() {
|
||||||
|
t.Errorf("expected %v got %v", a.Behavior(), b.Behavior())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestUpdateResourceOptionsWithInvalidHashAnnotationValues(t *testing.T) {
|
func TestUpdateResourceOptionsWithInvalidHashAnnotationValues(t *testing.T) {
|
||||||
rf := provider.NewDefaultDepProvider().GetResourceFactory()
|
rf := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
cases := []string{
|
cases := []string{
|
||||||
"",
|
"",
|
||||||
"FaLsE",
|
"FaLsE",
|
||||||
|
|||||||
@@ -19,7 +19,6 @@ import (
|
|||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/openapi"
|
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -236,18 +235,7 @@ func (kt *KustTarget) runGenerators(
|
|||||||
|
|
||||||
func (kt *KustTarget) configureExternalGenerators() ([]resmap.Generator, error) {
|
func (kt *KustTarget) configureExternalGenerators() ([]resmap.Generator, error) {
|
||||||
ra := accumulator.MakeEmptyAccumulator()
|
ra := accumulator.MakeEmptyAccumulator()
|
||||||
var generatorPaths []string
|
ra, err := kt.accumulateResources(ra, kt.kustomization.Generators)
|
||||||
for _, p := range kt.kustomization.Generators {
|
|
||||||
// handle inline generators
|
|
||||||
rm, err := kt.rFactory.NewResMapFromBytes([]byte(p))
|
|
||||||
if err != nil {
|
|
||||||
// not an inline config
|
|
||||||
generatorPaths = append(generatorPaths, p)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
ra.AppendAll(rm)
|
|
||||||
}
|
|
||||||
ra, err := kt.accumulateResources(ra, generatorPaths)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -272,18 +260,7 @@ func (kt *KustTarget) runTransformers(ra *accumulator.ResAccumulator) error {
|
|||||||
|
|
||||||
func (kt *KustTarget) configureExternalTransformers(transformers []string) ([]resmap.Transformer, error) {
|
func (kt *KustTarget) configureExternalTransformers(transformers []string) ([]resmap.Transformer, error) {
|
||||||
ra := accumulator.MakeEmptyAccumulator()
|
ra := accumulator.MakeEmptyAccumulator()
|
||||||
var transformerPaths []string
|
ra, err := kt.accumulateResources(ra, transformers)
|
||||||
for _, p := range transformers {
|
|
||||||
// handle inline transformers
|
|
||||||
rm, err := kt.rFactory.NewResMapFromBytes([]byte(p))
|
|
||||||
if err != nil {
|
|
||||||
// not an inline config
|
|
||||||
transformerPaths = append(transformerPaths, p)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
ra.AppendAll(rm)
|
|
||||||
}
|
|
||||||
ra, err := kt.accumulateResources(ra, transformerPaths)
|
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -383,11 +360,6 @@ func (kt *KustTarget) accumulateDirectory(
|
|||||||
return nil, errors.Wrapf(
|
return nil, errors.Wrapf(
|
||||||
err, "couldn't make target for path '%s'", ldr.Root())
|
err, "couldn't make target for path '%s'", ldr.Root())
|
||||||
}
|
}
|
||||||
err = openapi.SetSchemaVersion(subKt.Kustomization().OpenAPI, false)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrapf(
|
|
||||||
err, "couldn't set openapi version for path '%s'", ldr.Root())
|
|
||||||
}
|
|
||||||
if isComponent && subKt.kustomization.Kind != types.ComponentKind {
|
if isComponent && subKt.kustomization.Kind != types.ComponentKind {
|
||||||
return nil, fmt.Errorf(
|
return nil, fmt.Errorf(
|
||||||
"expected kind '%s' for path '%s' but got '%s'", types.ComponentKind, ldr.Root(), subKt.kustomization.Kind)
|
"expected kind '%s' for path '%s' but got '%s'", types.ComponentKind, ldr.Root(), subKt.kustomization.Kind)
|
||||||
|
|||||||
@@ -183,7 +183,8 @@ var transformerConfigurators = map[builtinhelpers.BuiltinPluginType]func(
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
var c struct {
|
var c struct {
|
||||||
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
||||||
|
Patches string `json:"patches,omitempty" yaml:"patches,omitempty"`
|
||||||
}
|
}
|
||||||
c.Paths = kt.kustomization.PatchesStrategicMerge
|
c.Paths = kt.kustomization.PatchesStrategicMerge
|
||||||
p := f()
|
p := f()
|
||||||
|
|||||||
@@ -8,10 +8,9 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
||||||
@@ -77,7 +76,8 @@ commonLabels:
|
|||||||
}
|
}
|
||||||
|
|
||||||
kt := makeKustTargetWithRf(
|
kt := makeKustTargetWithRf(
|
||||||
t, th.GetFSys(), "/", provider.NewDefaultDepProvider())
|
t, th.GetFSys(), "/",
|
||||||
|
resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl()))
|
||||||
for tn, tc := range testCases {
|
for tn, tc := range testCases {
|
||||||
t.Run(tn, func(t *testing.T) {
|
t.Run(tn, func(t *testing.T) {
|
||||||
th.WriteK("/", tc.content)
|
th.WriteK("/", tc.content)
|
||||||
@@ -148,8 +148,7 @@ metadata:
|
|||||||
{"op": "add", "path": "/spec/replica", "value": "3"}
|
{"op": "add", "path": "/spec/replica", "value": "3"}
|
||||||
]`)
|
]`)
|
||||||
|
|
||||||
pvd := provider.NewDefaultDepProvider()
|
resFactory := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl())
|
||||||
resFactory := pvd.GetResourceFactory()
|
|
||||||
|
|
||||||
resources := []*resource.Resource{
|
resources := []*resource.Resource{
|
||||||
resFactory.FromMapWithName("dply1", map[string]interface{}{
|
resFactory.FromMapWithName("dply1", map[string]interface{}{
|
||||||
@@ -244,16 +243,19 @@ metadata:
|
|||||||
t.Fatalf("unexpected error %v", err)
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expected.RemoveBuildAnnotations()
|
|
||||||
expYaml, err := expected.AsYaml()
|
|
||||||
assert.NoError(t, err)
|
|
||||||
|
|
||||||
kt := makeKustTargetWithRf(t, th.GetFSys(), "/whatever", pvd)
|
kt := makeKustTargetWithRf(
|
||||||
assert.NoError(t, kt.Load())
|
t, th.GetFSys(), "/whatever", resFactory)
|
||||||
|
err := kt.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected Resources error %v", err)
|
||||||
|
}
|
||||||
actual, err := kt.MakeCustomizedResMap()
|
actual, err := kt.MakeCustomizedResMap()
|
||||||
assert.NoError(t, err)
|
if err != nil {
|
||||||
actual.RemoveBuildAnnotations()
|
t.Fatalf("unexpected Resources error %v", err)
|
||||||
actYaml, err := actual.AsYaml()
|
}
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, expYaml, actYaml)
|
if err = expected.ErrorIfNotEqualLists(actual); err != nil {
|
||||||
|
t.Fatalf("unexpected inequality: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,12 +7,14 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
|
"sigs.k8s.io/kustomize/api/internal/k8sdeps/merge"
|
||||||
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
||||||
"sigs.k8s.io/kustomize/api/internal/target"
|
"sigs.k8s.io/kustomize/api/internal/target"
|
||||||
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
fLdr "sigs.k8s.io/kustomize/api/loader"
|
fLdr "sigs.k8s.io/kustomize/api/loader"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
valtest_test "sigs.k8s.io/kustomize/api/testutils/valtest"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -20,7 +22,9 @@ func makeAndLoadKustTarget(
|
|||||||
t *testing.T,
|
t *testing.T,
|
||||||
fSys filesys.FileSystem,
|
fSys filesys.FileSystem,
|
||||||
root string) *target.KustTarget {
|
root string) *target.KustTarget {
|
||||||
kt := makeKustTargetWithRf(t, fSys, root, provider.NewDefaultDepProvider())
|
kt := makeKustTargetWithRf(
|
||||||
|
t, fSys, root,
|
||||||
|
resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl()))
|
||||||
if err := kt.Load(); err != nil {
|
if err := kt.Load(); err != nil {
|
||||||
t.Fatalf("Unexpected load error %v", err)
|
t.Fatalf("Unexpected load error %v", err)
|
||||||
}
|
}
|
||||||
@@ -31,13 +35,13 @@ func makeKustTargetWithRf(
|
|||||||
t *testing.T,
|
t *testing.T,
|
||||||
fSys filesys.FileSystem,
|
fSys filesys.FileSystem,
|
||||||
root string,
|
root string,
|
||||||
pvd *provider.DepProvider) *target.KustTarget {
|
resourceFactory *resource.Factory) *target.KustTarget {
|
||||||
ldr, err := fLdr.NewLoader(fLdr.RestrictionRootOnly, root, fSys)
|
ldr, err := fLdr.NewLoader(fLdr.RestrictionRootOnly, root, fSys)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
rf := resmap.NewFactory(
|
rf := resmap.NewFactory(
|
||||||
pvd.GetResourceFactory(), pvd.GetConflictDetectorFactory())
|
resourceFactory, merge.NewMerginator(resourceFactory))
|
||||||
pc := konfig.DisabledPluginConfig()
|
pc := konfig.DisabledPluginConfig()
|
||||||
return target.NewKustTarget(
|
return target.NewKustTarget(
|
||||||
ldr,
|
ldr,
|
||||||
|
|||||||
@@ -1,36 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package utils
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
type errTimeOut struct {
|
|
||||||
duration time.Duration
|
|
||||||
cmd string
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewErrTimeOut(d time.Duration, c string) errTimeOut {
|
|
||||||
return errTimeOut{duration: d, cmd: c}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e errTimeOut) Error() string {
|
|
||||||
return fmt.Sprintf("hit %s timeout running '%s'", e.duration, e.cmd)
|
|
||||||
}
|
|
||||||
|
|
||||||
func IsErrTimeout(err error) bool {
|
|
||||||
if err == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
_, ok := err.(errTimeOut)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
_, ok = errors.Cause(err).(errTimeOut)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
// Copyright 2021 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package utils
|
|
||||||
|
|
||||||
import "strings"
|
|
||||||
|
|
||||||
// PathSplitter splits a slash delimited string, permitting escaped slashes.
|
|
||||||
func PathSplitter(path string) []string {
|
|
||||||
ps := strings.Split(path, "/")
|
|
||||||
var res []string
|
|
||||||
res = append(res, ps[0])
|
|
||||||
for i := 1; i < len(ps); i++ {
|
|
||||||
last := len(res) - 1
|
|
||||||
if strings.HasSuffix(res[last], `\`) {
|
|
||||||
res[last] = strings.TrimSuffix(res[last], `\`) + "/" + ps[i]
|
|
||||||
} else {
|
|
||||||
res = append(res, ps[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
// Copyright 2021 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package utils_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/utils"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestPathSplitter(t *testing.T) {
|
|
||||||
for _, tc := range []struct {
|
|
||||||
exp []string
|
|
||||||
path string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
path: "",
|
|
||||||
exp: []string{""},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "s",
|
|
||||||
exp: []string{"s"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "a/b/c",
|
|
||||||
exp: []string{"a", "b", "c"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: `a/b[]/c`,
|
|
||||||
exp: []string{"a", "b[]", "c"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: `a/b\/c/d\/e/f`,
|
|
||||||
exp: []string{"a", "b/c", "d/e", "f"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
// The actual reason for this.
|
|
||||||
path: `metadata/annotations/nginx.ingress.kubernetes.io\/auth-secret`,
|
|
||||||
exp: []string{
|
|
||||||
"metadata",
|
|
||||||
"annotations",
|
|
||||||
"nginx.ingress.kubernetes.io/auth-secret"},
|
|
||||||
},
|
|
||||||
} {
|
|
||||||
assert.Equal(t, tc.exp, PathSplitter(tc.path))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package utils
|
|
||||||
|
|
||||||
import (
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TimedCall runs fn, failing if it doesn't complete in the given duration.
|
|
||||||
// The description is used in the timeout error message.
|
|
||||||
func TimedCall(description string, d time.Duration, fn func() error) error {
|
|
||||||
done := make(chan error)
|
|
||||||
timer := time.NewTimer(d)
|
|
||||||
defer timer.Stop()
|
|
||||||
go func() { done <- fn() }()
|
|
||||||
select {
|
|
||||||
case err := <-done:
|
|
||||||
return err
|
|
||||||
case <-timer.C:
|
|
||||||
return NewErrTimeOut(d, description)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package utils_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"testing"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/utils"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
timeToWait = 10 * time.Millisecond
|
|
||||||
tooSlow = 2 * timeToWait
|
|
||||||
)
|
|
||||||
|
|
||||||
func errMsg(msg string) string {
|
|
||||||
return fmt.Sprintf("hit %s timeout running '%s'", timeToWait, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTimedCallFastNoError(t *testing.T) {
|
|
||||||
err := TimedCall(
|
|
||||||
"fast no error", timeToWait,
|
|
||||||
func() error { return nil })
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTimedCallFastWithError(t *testing.T) {
|
|
||||||
err := TimedCall(
|
|
||||||
"fast with error", timeToWait,
|
|
||||||
func() error { return assert.AnError })
|
|
||||||
if assert.Error(t, err) {
|
|
||||||
assert.EqualError(t, err, assert.AnError.Error())
|
|
||||||
} else {
|
|
||||||
t.Fail()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTimedCallSlowNoError(t *testing.T) {
|
|
||||||
err := TimedCall(
|
|
||||||
"slow no error", timeToWait,
|
|
||||||
func() error { time.Sleep(tooSlow); return nil })
|
|
||||||
if assert.Error(t, err) {
|
|
||||||
assert.EqualError(t, err, errMsg("slow no error"))
|
|
||||||
} else {
|
|
||||||
t.Fail()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTimedCallSlowWithError(t *testing.T) {
|
|
||||||
err := TimedCall(
|
|
||||||
"slow with error", timeToWait,
|
|
||||||
func() error { time.Sleep(tooSlow); return assert.AnError })
|
|
||||||
if assert.Error(t, err) {
|
|
||||||
assert.EqualError(t, err, errMsg("slow with error"))
|
|
||||||
} else {
|
|
||||||
t.Fail()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -10,10 +10,6 @@ import (
|
|||||||
// FieldValidator implements ifc.Validator to check
|
// FieldValidator implements ifc.Validator to check
|
||||||
// the values of various KRM string fields,
|
// the values of various KRM string fields,
|
||||||
// e.g. labels, annotations, names, namespaces.
|
// e.g. labels, annotations, names, namespaces.
|
||||||
//
|
|
||||||
// TODO: Have this use kyaml/yaml/internal/k8sgen/pkg/labels
|
|
||||||
// which has label and annotation validation code, but is internal
|
|
||||||
// so this impl would need to move to kyaml (a fine idea).
|
|
||||||
type FieldValidator struct {
|
type FieldValidator struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,105 +4,38 @@
|
|||||||
package wrappy
|
package wrappy
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/hasher"
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/internal/generators"
|
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// WNodeFactory makes instances of WNode.
|
// WNodeFactory makes instances of WNode.
|
||||||
//
|
|
||||||
// These instances in turn adapt
|
// These instances in turn adapt
|
||||||
// sigs.k8s.io/kustomize/kyaml/yaml.RNode
|
// sigs.k8s.io/kustomize/kyaml/yaml.RNode
|
||||||
// to implement ifc.Unstructured.
|
// to implement ifc.Unstructured.
|
||||||
// This factory is meant to implement ifc.KunstructuredFactory.
|
// This factory is meant to implement ifc.KunstructuredFactory.
|
||||||
//
|
|
||||||
// This implementation should be thin, as both WNode and WNodeFactory must be
|
|
||||||
// factored away (deleted) along with ifc.Kunstructured in favor of direct use
|
|
||||||
// of RNode methods upon completion of
|
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/2506.
|
|
||||||
//
|
|
||||||
// See also api/krusty/internal/provider/depprovider.go
|
|
||||||
type WNodeFactory struct {
|
type WNodeFactory struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ ifc.KunstructuredFactory = (*WNodeFactory)(nil)
|
var _ ifc.KunstructuredFactory = (*WNodeFactory)(nil)
|
||||||
|
|
||||||
func (k *WNodeFactory) SliceFromBytes(bs []byte) ([]ifc.Kunstructured, error) {
|
func (k *WNodeFactory) SliceFromBytes(bs []byte) ([]ifc.Kunstructured, error) {
|
||||||
yamlRNodes, err := kio.FromBytes(bs)
|
panic("TODO(#WNodeFactory): implement SliceFromBytes")
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var result []ifc.Kunstructured
|
|
||||||
for i := range yamlRNodes {
|
|
||||||
rn := yamlRNodes[i]
|
|
||||||
meta, err := rn.GetValidatedMetadata()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if !shouldDropObject(meta) {
|
|
||||||
if foundNil, path := rn.HasNilEntryInList(); foundNil {
|
|
||||||
return nil, fmt.Errorf("empty item at %v in object %v", path, rn)
|
|
||||||
}
|
|
||||||
result = append(result, FromRNode(rn))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// shouldDropObject returns true if the resource should not be accumulated.
|
|
||||||
func shouldDropObject(m yaml.ResourceMeta) bool {
|
|
||||||
_, y := m.ObjectMeta.Annotations[konfig.IgnoredByKustomizeAnnotation]
|
|
||||||
return y
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (k *WNodeFactory) FromMap(m map[string]interface{}) ifc.Kunstructured {
|
func (k *WNodeFactory) FromMap(m map[string]interface{}) ifc.Kunstructured {
|
||||||
rn, err := FromMap(m)
|
panic("TODO(#WNodeFactory): implement FromMap")
|
||||||
if err != nil {
|
|
||||||
// TODO(#WNodeFactory): handle or bubble error"
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return rn
|
|
||||||
}
|
|
||||||
|
|
||||||
// kustHash computes a hash of an unstructured object.
|
|
||||||
type kustHash struct{}
|
|
||||||
|
|
||||||
// Hash returns a hash of the given object
|
|
||||||
func (h *kustHash) Hash(m ifc.Kunstructured) (string, error) {
|
|
||||||
node, err := filtersutil.GetRNode(m)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return hasher.HashRNode(node)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (k *WNodeFactory) Hasher() ifc.KunstructuredHasher {
|
func (k *WNodeFactory) Hasher() ifc.KunstructuredHasher {
|
||||||
return &kustHash{}
|
panic("TODO(#WNodeFactory): implement Hasher")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MakeConfigMap makes a wrapped configmap.
|
|
||||||
func (k *WNodeFactory) MakeConfigMap(
|
func (k *WNodeFactory) MakeConfigMap(
|
||||||
ldr ifc.KvLoader, args *types.ConfigMapArgs) (ifc.Kunstructured, error) {
|
kvLdr ifc.KvLoader, args *types.ConfigMapArgs) (ifc.Kunstructured, error) {
|
||||||
rn, err := generators.MakeConfigMap(ldr, args)
|
panic("TODO(#WNodeFactory): implement MakeConfigMap")
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return FromRNode(rn), nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MakeSecret makes a wrapped secret.
|
|
||||||
func (k *WNodeFactory) MakeSecret(
|
func (k *WNodeFactory) MakeSecret(
|
||||||
ldr ifc.KvLoader, args *types.SecretArgs) (ifc.Kunstructured, error) {
|
kvLdr ifc.KvLoader, args *types.SecretArgs) (ifc.Kunstructured, error) {
|
||||||
rn, err := generators.MakeSecret(ldr, args)
|
panic("TODO(#WNodeFactory): implement MakeSecret")
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return FromRNode(rn), nil
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,321 +1,4 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
// Copyright 2020 The Kubernetes Authors.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
package wrappy_test
|
package wrappy
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/wrappy"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestHasher(t *testing.T) {
|
|
||||||
input := `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
data:
|
|
||||||
one: ""
|
|
||||||
binaryData:
|
|
||||||
two: ""
|
|
||||||
`
|
|
||||||
expect := "698h7c7t9m"
|
|
||||||
|
|
||||||
factory := &WNodeFactory{}
|
|
||||||
k, err := factory.SliceFromBytes([]byte(input))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
hasher := factory.Hasher()
|
|
||||||
result, err := hasher.Hash(k[0])
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if result != expect {
|
|
||||||
t.Fatalf("expect %s but got %s", expect, result)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSliceFromBytes(t *testing.T) {
|
|
||||||
factory := &WNodeFactory{}
|
|
||||||
testConfigMap :=
|
|
||||||
map[string]interface{}{
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"kind": "ConfigMap",
|
|
||||||
"metadata": map[string]interface{}{
|
|
||||||
"name": "winnie",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
testConfigMapList :=
|
|
||||||
map[string]interface{}{
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"kind": "ConfigMapList",
|
|
||||||
"items": []interface{}{
|
|
||||||
testConfigMap,
|
|
||||||
testConfigMap,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
testDeploymentSpec := map[string]interface{}{
|
|
||||||
"template": map[string]interface{}{
|
|
||||||
"spec": map[string]interface{}{
|
|
||||||
"hostAliases": []interface{}{
|
|
||||||
map[string]interface{}{
|
|
||||||
"hostnames": []interface{}{
|
|
||||||
"a.example.com",
|
|
||||||
},
|
|
||||||
"ip": "8.8.8.8",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
testDeploymentA := map[string]interface{}{
|
|
||||||
"apiVersion": "apps/v1",
|
|
||||||
"kind": "Deployment",
|
|
||||||
"metadata": map[string]interface{}{
|
|
||||||
"name": "deployment-a",
|
|
||||||
},
|
|
||||||
"spec": testDeploymentSpec,
|
|
||||||
}
|
|
||||||
testDeploymentB := map[string]interface{}{
|
|
||||||
"apiVersion": "apps/v1",
|
|
||||||
"kind": "Deployment",
|
|
||||||
"metadata": map[string]interface{}{
|
|
||||||
"name": "deployment-b",
|
|
||||||
},
|
|
||||||
"spec": testDeploymentSpec,
|
|
||||||
}
|
|
||||||
testDeploymentList :=
|
|
||||||
map[string]interface{}{
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"kind": "DeploymentList",
|
|
||||||
"items": []interface{}{
|
|
||||||
testDeploymentA,
|
|
||||||
testDeploymentB,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
type expected struct {
|
|
||||||
out []map[string]interface{}
|
|
||||||
isErr bool
|
|
||||||
}
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input []byte
|
|
||||||
exp expected
|
|
||||||
}{
|
|
||||||
"garbage": {
|
|
||||||
input: []byte("garbageIn: garbageOut"),
|
|
||||||
exp: expected{
|
|
||||||
isErr: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"noBytes": {
|
|
||||||
input: []byte{},
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"goodJson": {
|
|
||||||
input: []byte(`
|
|
||||||
{"apiVersion":"v1","kind":"ConfigMap","metadata":{"name":"winnie"}}
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testConfigMap},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"goodYaml1": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testConfigMap},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"goodYaml2": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testConfigMap, testConfigMap},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"localConfigYaml": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie-skip
|
|
||||||
annotations:
|
|
||||||
# this annotation causes the Resource to be ignored by kustomize
|
|
||||||
config.kubernetes.io/local-config: ""
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testConfigMap},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"garbageInOneOfTwoObjects": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
---
|
|
||||||
WOOOOOOOOOOOOOOOOOOOOOOOOT: woot
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
isErr: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"emptyObjects": {
|
|
||||||
input: []byte(`
|
|
||||||
---
|
|
||||||
#a comment
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"Missing .metadata.name in object": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Namespace
|
|
||||||
metadata:
|
|
||||||
annotations:
|
|
||||||
foo: bar
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
isErr: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"nil value in list": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: builtin
|
|
||||||
kind: ConfigMapGenerator
|
|
||||||
metadata:
|
|
||||||
name: kube100-site
|
|
||||||
labels:
|
|
||||||
app: web
|
|
||||||
testList:
|
|
||||||
- testA
|
|
||||||
-
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
isErr: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"List": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: List
|
|
||||||
items:
|
|
||||||
- apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
- apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{
|
|
||||||
testConfigMap,
|
|
||||||
testConfigMap},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"ConfigMapList": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMapList
|
|
||||||
items:
|
|
||||||
- apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
- apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: winnie
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testConfigMapList},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"listWithAnchors": {
|
|
||||||
input: []byte(`
|
|
||||||
apiVersion: v1
|
|
||||||
kind: DeploymentList
|
|
||||||
items:
|
|
||||||
- apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deployment-a
|
|
||||||
spec: &hostAliases
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
hostAliases:
|
|
||||||
- hostnames:
|
|
||||||
- a.example.com
|
|
||||||
ip: 8.8.8.8
|
|
||||||
- apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deployment-b
|
|
||||||
spec:
|
|
||||||
<<: *hostAliases
|
|
||||||
`),
|
|
||||||
exp: expected{
|
|
||||||
out: []map[string]interface{}{testDeploymentList},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
rs, err := factory.SliceFromBytes(tc.input)
|
|
||||||
if err != nil {
|
|
||||||
assert.True(t, tc.exp.isErr)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
assert.False(t, tc.exp.isErr)
|
|
||||||
assert.Equal(t, len(tc.exp.out), len(rs))
|
|
||||||
for i := range rs {
|
|
||||||
assert.Equal(
|
|
||||||
t, fmt.Sprintf("%v", tc.exp.out[i]), fmt.Sprintf("%v", rs[i].Map()))
|
|
||||||
if n != "listWithAnchors" {
|
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/3271
|
|
||||||
if !reflect.DeepEqual(tc.exp.out[i], rs[i].Map()) {
|
|
||||||
t.Fatalf("%s:\nexpected: %v\n actual: %v",
|
|
||||||
n, tc.exp.out[i], rs[i].Map())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -6,8 +6,6 @@ package wrappy
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"regexp"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
@@ -33,22 +31,10 @@ func NewWNode() *WNode {
|
|||||||
return FromRNode(yaml.NewRNode(nil))
|
return FromRNode(yaml.NewRNode(nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
func FromMap(m map[string]interface{}) (*WNode, error) {
|
|
||||||
n, err := yaml.FromMap(m)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return FromRNode(n), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func FromRNode(node *yaml.RNode) *WNode {
|
func FromRNode(node *yaml.RNode) *WNode {
|
||||||
return &WNode{node: node}
|
return &WNode{node: node}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (wn *WNode) AsRNode() *yaml.RNode {
|
|
||||||
return wn.node
|
|
||||||
}
|
|
||||||
|
|
||||||
func (wn *WNode) demandMetaData(label string) yaml.ResourceMeta {
|
func (wn *WNode) demandMetaData(label string) yaml.ResourceMeta {
|
||||||
meta, err := wn.node.GetMeta()
|
meta, err := wn.node.GetMeta()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -68,35 +54,9 @@ func (wn *WNode) GetAnnotations() map[string]string {
|
|||||||
return wn.demandMetaData("GetAnnotations").Annotations
|
return wn.demandMetaData("GetAnnotations").Annotations
|
||||||
}
|
}
|
||||||
|
|
||||||
// convertSliceIndex traverses the items in `fields` and find
|
|
||||||
// if there is a slice index in the item and change it to a
|
|
||||||
// valid Lookup field path. For example, 'ports[0]' will be
|
|
||||||
// converted to 'ports' and '0'.
|
|
||||||
func convertSliceIndex(fields []string) []string {
|
|
||||||
var res []string
|
|
||||||
for _, s := range fields {
|
|
||||||
if !strings.HasSuffix(s, "]") {
|
|
||||||
res = append(res, s)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
re := regexp.MustCompile(`^(.*)\[(\d+)\]$`)
|
|
||||||
groups := re.FindStringSubmatch(s)
|
|
||||||
if len(groups) == 0 {
|
|
||||||
// no match, add to result
|
|
||||||
res = append(res, s)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if groups[1] != "" {
|
|
||||||
res = append(res, groups[1])
|
|
||||||
}
|
|
||||||
res = append(res, groups[2])
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetFieldValue implements ifc.Kunstructured.
|
// GetFieldValue implements ifc.Kunstructured.
|
||||||
func (wn *WNode) GetFieldValue(path string) (interface{}, error) {
|
func (wn *WNode) GetFieldValue(path string) (interface{}, error) {
|
||||||
fields := convertSliceIndex(strings.Split(path, "."))
|
fields := strings.Split(path, ".")
|
||||||
rn, err := wn.node.Pipe(yaml.Lookup(fields...))
|
rn, err := wn.node.Pipe(yaml.Lookup(fields...))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -123,34 +83,14 @@ func (wn *WNode) GetFieldValue(path string) (interface{}, error) {
|
|||||||
// Return value as slice for SequenceNode kind
|
// Return value as slice for SequenceNode kind
|
||||||
if yn.Kind == yaml.SequenceNode {
|
if yn.Kind == yaml.SequenceNode {
|
||||||
var result []interface{}
|
var result []interface{}
|
||||||
if err := yn.Decode(&result); err != nil {
|
for _, node := range yn.Content {
|
||||||
return nil, err
|
result = append(result, node.Value)
|
||||||
}
|
}
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
if yn.Kind != yaml.ScalarNode {
|
|
||||||
return nil, fmt.Errorf("expected ScalarNode, got Kind=%d", yn.Kind)
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: When doing kustomize var replacement, which is likely a
|
// Return value value directly for all other (ScalarNode) kinds
|
||||||
// a primary use of this function and the reason it returns interface{}
|
return yn.Value, nil
|
||||||
// rather than string, we do conversion from Nodes to Go types and back
|
|
||||||
// to nodes. We should figure out how to do replacement using raw nodes,
|
|
||||||
// assuming we keep the var feature in kustomize.
|
|
||||||
// The other end of this is: refvar.go:updateNodeValue.
|
|
||||||
switch yn.Tag {
|
|
||||||
case yaml.NodeTagString:
|
|
||||||
return yn.Value, nil
|
|
||||||
case yaml.NodeTagInt:
|
|
||||||
return strconv.Atoi(yn.Value)
|
|
||||||
case yaml.NodeTagFloat:
|
|
||||||
return strconv.ParseFloat(yn.Value, 64)
|
|
||||||
case yaml.NodeTagBool:
|
|
||||||
return strconv.ParseBool(yn.Value)
|
|
||||||
default:
|
|
||||||
// Possibly this should be an error or log.
|
|
||||||
return yn.Value, nil
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetGvk implements ifc.Kunstructured.
|
// GetGvk implements ifc.Kunstructured.
|
||||||
@@ -160,16 +100,6 @@ func (wn *WNode) GetGvk() resid.Gvk {
|
|||||||
return resid.Gvk{Group: g, Version: v, Kind: meta.Kind}
|
return resid.Gvk{Group: g, Version: v, Kind: meta.Kind}
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetDataMap implements ifc.Kunstructured.
|
|
||||||
func (wn *WNode) GetDataMap() map[string]string {
|
|
||||||
return wn.node.GetDataMap()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetDataMap implements ifc.Kunstructured.
|
|
||||||
func (wn *WNode) SetDataMap(m map[string]string) {
|
|
||||||
wn.node.SetDataMap(m)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetKind implements ifc.Kunstructured.
|
// GetKind implements ifc.Kunstructured.
|
||||||
func (wn *WNode) GetKind() string {
|
func (wn *WNode) GetKind() string {
|
||||||
return wn.demandMetaData("GetKind").Kind
|
return wn.demandMetaData("GetKind").Kind
|
||||||
@@ -211,7 +141,12 @@ func (wn *WNode) GetString(path string) (string, error) {
|
|||||||
|
|
||||||
// Map implements ifc.Kunstructured.
|
// Map implements ifc.Kunstructured.
|
||||||
func (wn *WNode) Map() map[string]interface{} {
|
func (wn *WNode) Map() map[string]interface{} {
|
||||||
return wn.node.Map()
|
var result map[string]interface{}
|
||||||
|
if err := wn.node.YNode().Decode(&result); err != nil {
|
||||||
|
// Log and die since interface doesn't allow error.
|
||||||
|
log.Fatalf("failed to decode ynode: %v", err)
|
||||||
|
}
|
||||||
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
// MarshalJSON implements ifc.Kunstructured.
|
// MarshalJSON implements ifc.Kunstructured.
|
||||||
@@ -220,49 +155,47 @@ func (wn *WNode) MarshalJSON() ([]byte, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// MatchesAnnotationSelector implements ifc.Kunstructured.
|
// MatchesAnnotationSelector implements ifc.Kunstructured.
|
||||||
func (wn *WNode) MatchesAnnotationSelector(selector string) (bool, error) {
|
func (wn *WNode) MatchesAnnotationSelector(string) (bool, error) {
|
||||||
return wn.node.MatchesAnnotationSelector(selector)
|
panic("TODO(#WNode) MatchesAnnotationSelector; implement or drop from API")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MatchesLabelSelector implements ifc.Kunstructured.
|
// MatchesLabelSelector implements ifc.Kunstructured.
|
||||||
func (wn *WNode) MatchesLabelSelector(selector string) (bool, error) {
|
func (wn *WNode) MatchesLabelSelector(string) (bool, error) {
|
||||||
return wn.node.MatchesLabelSelector(selector)
|
panic("TODO(#WNode) MatchesLabelSelector; implement or drop from API")
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetAnnotations implements ifc.Kunstructured.
|
// SetAnnotations implements ifc.Kunstructured.
|
||||||
func (wn *WNode) SetAnnotations(annotations map[string]string) {
|
func (wn *WNode) SetAnnotations(annotations map[string]string) {
|
||||||
if err := wn.node.SetAnnotations(annotations); err != nil {
|
wn.setField(yaml.NewMapRNode(&annotations), yaml.MetadataField, yaml.AnnotationsField)
|
||||||
log.Fatal(err) // interface doesn't allow error.
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetGvk implements ifc.Kunstructured.
|
// SetGvk implements ifc.Kunstructured.
|
||||||
func (wn *WNode) SetGvk(gvk resid.Gvk) {
|
func (wn *WNode) SetGvk(gvk resid.Gvk) {
|
||||||
wn.setMapField(yaml.NewScalarRNode(gvk.Kind), yaml.KindField)
|
wn.setField(yaml.NewScalarRNode(gvk.Kind), yaml.KindField)
|
||||||
wn.setMapField(yaml.NewScalarRNode(gvk.ApiVersion()), yaml.APIVersionField)
|
wn.setField(yaml.NewScalarRNode(fmt.Sprintf("%s/%s", gvk.Group, gvk.Version)), yaml.APIVersionField)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetLabels implements ifc.Kunstructured.
|
// SetLabels implements ifc.Kunstructured.
|
||||||
func (wn *WNode) SetLabels(labels map[string]string) {
|
func (wn *WNode) SetLabels(labels map[string]string) {
|
||||||
if err := wn.node.SetLabels(labels); err != nil {
|
wn.setField(yaml.NewMapRNode(&labels), yaml.MetadataField, yaml.LabelsField)
|
||||||
log.Fatal(err) // interface doesn't allow error.
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetName implements ifc.Kunstructured.
|
// SetName implements ifc.Kunstructured.
|
||||||
func (wn *WNode) SetName(name string) {
|
func (wn *WNode) SetName(name string) {
|
||||||
wn.setMapField(yaml.NewScalarRNode(name), yaml.MetadataField, yaml.NameField)
|
wn.setField(yaml.NewScalarRNode(name), yaml.MetadataField, yaml.NameField)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetNamespace implements ifc.Kunstructured.
|
// SetNamespace implements ifc.Kunstructured.
|
||||||
func (wn *WNode) SetNamespace(ns string) {
|
func (wn *WNode) SetNamespace(ns string) {
|
||||||
if err := wn.node.SetNamespace(ns); err != nil {
|
wn.setField(yaml.NewScalarRNode(ns), yaml.MetadataField, yaml.NamespaceField)
|
||||||
log.Fatal(err) // interface doesn't allow error.
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (wn *WNode) setMapField(value *yaml.RNode, path ...string) {
|
func (wn *WNode) setField(value *yaml.RNode, path ...string) {
|
||||||
if err := wn.node.SetMapField(value, path...); err != nil {
|
err := wn.node.PipeE(
|
||||||
|
yaml.LookupCreate(yaml.MappingNode, path[0:len(path)-1]...),
|
||||||
|
yaml.SetField(path[len(path)-1], value),
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
// Log and die since interface doesn't allow error.
|
// Log and die since interface doesn't allow error.
|
||||||
log.Fatalf("failed to set field %v: %v", path, err)
|
log.Fatalf("failed to set field %v: %v", path, err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,9 +8,9 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/google/go-cmp/cmp"
|
"github.com/google/go-cmp/cmp"
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"gopkg.in/yaml.v3"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
|
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
kyaml "sigs.k8s.io/kustomize/kyaml/yaml"
|
kyaml "sigs.k8s.io/kustomize/kyaml/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -33,38 +33,6 @@ const (
|
|||||||
"area": "51",
|
"area": "51",
|
||||||
"greeting": "Take me to your leader."
|
"greeting": "Take me to your leader."
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"spec": {
|
|
||||||
"template": {
|
|
||||||
"spec": {
|
|
||||||
"containers": [
|
|
||||||
{
|
|
||||||
"env": [
|
|
||||||
{
|
|
||||||
"name": "CM_FOO",
|
|
||||||
"valueFrom": {
|
|
||||||
"configMapKeyRef": {
|
|
||||||
"key": "somekey",
|
|
||||||
"name": "myCm"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "SECRET_FOO",
|
|
||||||
"valueFrom": {
|
|
||||||
"secretKeyRef": {
|
|
||||||
"key": "someKey",
|
|
||||||
"name": "mySecret"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"image": "nginx:1.7.9",
|
|
||||||
"name": "nginx"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
@@ -390,51 +358,6 @@ func TestGetFieldValueReturnsMap(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetFieldValueReturnsStuff(t *testing.T) {
|
|
||||||
wn := NewWNode()
|
|
||||||
if err := wn.UnmarshalJSON([]byte(deploymentBiggerJson)); err != nil {
|
|
||||||
t.Fatalf("unexpected unmarshaljson err: %v", err)
|
|
||||||
}
|
|
||||||
expected := []interface{}{
|
|
||||||
map[string]interface{}{
|
|
||||||
"env": []interface{}{
|
|
||||||
map[string]interface{}{
|
|
||||||
"name": "CM_FOO",
|
|
||||||
"valueFrom": map[string]interface{}{
|
|
||||||
"configMapKeyRef": map[string]interface{}{
|
|
||||||
"key": "somekey",
|
|
||||||
"name": "myCm",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
map[string]interface{}{
|
|
||||||
"name": "SECRET_FOO",
|
|
||||||
"valueFrom": map[string]interface{}{
|
|
||||||
"secretKeyRef": map[string]interface{}{
|
|
||||||
"key": "someKey",
|
|
||||||
"name": "mySecret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"image": string("nginx:1.7.9"),
|
|
||||||
"name": string("nginx"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
actual, err := wn.GetFieldValue("spec.template.spec.containers")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("error getting field value: %v", err)
|
|
||||||
}
|
|
||||||
if diff := cmp.Diff(expected, actual); diff != "" {
|
|
||||||
t.Fatalf("actual map does not deep equal expected map:\n%v", diff)
|
|
||||||
}
|
|
||||||
// Cannot go deeper yet.
|
|
||||||
_, err = wn.GetFieldValue("spec.template.spec.containers.env")
|
|
||||||
if err == nil {
|
|
||||||
t.Fatalf("expected err %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetFieldValueReturnsSlice(t *testing.T) {
|
func TestGetFieldValueReturnsSlice(t *testing.T) {
|
||||||
bytes, err := yaml.Marshal(makeBigMap())
|
bytes, err := yaml.Marshal(makeBigMap())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -455,39 +378,6 @@ func TestGetFieldValueReturnsSlice(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetFieldValueReturnsSliceOfMappings(t *testing.T) {
|
|
||||||
bytes, err := yaml.Marshal(makeBigMap())
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("unexpected yaml.Marshal err: %v", err)
|
|
||||||
}
|
|
||||||
rNode, err := kyaml.Parse(string(bytes))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("unexpected yaml.Marshal err: %v", err)
|
|
||||||
}
|
|
||||||
wn := FromRNode(rNode)
|
|
||||||
expected := []interface{}{
|
|
||||||
map[string]interface{}{
|
|
||||||
"field1": "idx0foo",
|
|
||||||
"field2": "idx0bar",
|
|
||||||
},
|
|
||||||
map[string]interface{}{
|
|
||||||
"field1": "idx1foo",
|
|
||||||
"field2": "idx1bar",
|
|
||||||
},
|
|
||||||
map[string]interface{}{
|
|
||||||
"field1": "idx2foo",
|
|
||||||
"field2": "idx2bar",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
actual, err := wn.GetFieldValue("those")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("error getting slice: %v", err)
|
|
||||||
}
|
|
||||||
if diff := cmp.Diff(expected, actual); diff != "" {
|
|
||||||
t.Fatalf("actual slice does not deep equal expected slice:\n%v", diff)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetFieldValueReturnsString(t *testing.T) {
|
func TestGetFieldValueReturnsString(t *testing.T) {
|
||||||
wn := NewWNode()
|
wn := NewWNode()
|
||||||
if err := wn.UnmarshalJSON([]byte(deploymentBiggerJson)); err != nil {
|
if err := wn.UnmarshalJSON([]byte(deploymentBiggerJson)); err != nil {
|
||||||
@@ -558,10 +448,6 @@ func TestGetSlice(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestMapEmpty(t *testing.T) {
|
|
||||||
assert.Equal(t, 0, len(NewWNode().Map()))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMap(t *testing.T) {
|
func TestMap(t *testing.T) {
|
||||||
wn := NewWNode()
|
wn := NewWNode()
|
||||||
if err := wn.UnmarshalJSON([]byte(deploymentLittleJson)); err != nil {
|
if err := wn.UnmarshalJSON([]byte(deploymentLittleJson)); err != nil {
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ import (
|
|||||||
"k8s.io/apimachinery/pkg/util/yaml"
|
"k8s.io/apimachinery/pkg/util/yaml"
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/internal/k8sdeps/configmapandsecret"
|
"sigs.k8s.io/kustomize/api/internal/k8sdeps/configmapandsecret"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -117,6 +116,10 @@ func (kf *KunstructuredFactoryImpl) validate(u unstructured.Unstructured) error
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// nonKustomizableResourceAnnotation if set on a Resource will cause Kustomize to
|
||||||
|
// ignore the Resource rather than Kustomize it.
|
||||||
|
const ignoredByKustomizeResourceAnnotation = "config.kubernetes.io/local-config"
|
||||||
|
|
||||||
// skipResource returns true if the Resource should not be accumulated
|
// skipResource returns true if the Resource should not be accumulated
|
||||||
func (kf *KunstructuredFactoryImpl) skipResource(u unstructured.Unstructured) bool {
|
func (kf *KunstructuredFactoryImpl) skipResource(u unstructured.Unstructured) bool {
|
||||||
an := u.GetAnnotations()
|
an := u.GetAnnotations()
|
||||||
@@ -125,7 +128,7 @@ func (kf *KunstructuredFactoryImpl) skipResource(u unstructured.Unstructured) bo
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
// check if the Resource has opt-ed out of kustomize
|
// check if the Resource has opt-ed out of kustomize
|
||||||
_, found := an[konfig.IgnoredByKustomizeAnnotation]
|
_, found := an[ignoredByKustomizeResourceAnnotation]
|
||||||
return found
|
return found
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ package kunstruct
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
|
||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
jsonpatch "github.com/evanphx/json-patch"
|
||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
@@ -250,29 +249,6 @@ func (fs *UnstructAdapter) GetStringMap(path string) (map[string]string, error)
|
|||||||
return nil, NoFieldError{Field: path}
|
return nil, NoFieldError{Field: path}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (fs *UnstructAdapter) GetDataMap() map[string]string {
|
|
||||||
m, err := fs.GetStringMap("data")
|
|
||||||
if err != nil {
|
|
||||||
return map[string]string{}
|
|
||||||
}
|
|
||||||
return m
|
|
||||||
}
|
|
||||||
|
|
||||||
func (fs *UnstructAdapter) SetDataMap(m map[string]string) {
|
|
||||||
if m == nil {
|
|
||||||
unstructured.RemoveNestedField(fs.Object, "data")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s := make(map[string]interface{}, len(m))
|
|
||||||
for i, v := range m {
|
|
||||||
s[i] = v
|
|
||||||
}
|
|
||||||
err := unstructured.SetNestedMap(fs.Object, s, "data")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetMap returns value at the given fieldpath.
|
// GetMap returns value at the given fieldpath.
|
||||||
func (fs *UnstructAdapter) GetMap(path string) (map[string]interface{}, error) {
|
func (fs *UnstructAdapter) GetMap(path string) (map[string]interface{}, error) {
|
||||||
content, fields, found, err := fs.selectSubtree(path)
|
content, fields, found, err := fs.selectSubtree(path)
|
||||||
|
|||||||
@@ -6,8 +6,6 @@ package kunstruct
|
|||||||
import (
|
import (
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var kunstructured = NewKunstructuredFactoryImpl().FromMap(map[string]interface{}{
|
var kunstructured = NewKunstructuredFactoryImpl().FromMap(map[string]interface{}{
|
||||||
@@ -559,139 +557,3 @@ func compareValues(t *testing.T, name string, pathToField string, expectedValue
|
|||||||
t.Logf("%T value at `%s`", typedV, pathToField)
|
t.Logf("%T value at `%s`", typedV, pathToField)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestKunstGetDataMap(t *testing.T) {
|
|
||||||
emptyMap := map[string]string{}
|
|
||||||
testCases := map[string]struct {
|
|
||||||
theMap map[string]interface{}
|
|
||||||
expected map[string]string
|
|
||||||
}{
|
|
||||||
"actuallyNil": {
|
|
||||||
theMap: nil,
|
|
||||||
expected: emptyMap,
|
|
||||||
},
|
|
||||||
"empty": {
|
|
||||||
theMap: map[string]interface{}{},
|
|
||||||
expected: emptyMap,
|
|
||||||
},
|
|
||||||
"mostlyEmpty": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"hey": "there",
|
|
||||||
},
|
|
||||||
expected: emptyMap,
|
|
||||||
},
|
|
||||||
"noNameConfigMap": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"kind": "ConfigMap",
|
|
||||||
},
|
|
||||||
expected: emptyMap,
|
|
||||||
},
|
|
||||||
"configMap": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"apiVersion": "v1",
|
|
||||||
"kind": "ConfigMap",
|
|
||||||
"metadata": map[string]interface{}{
|
|
||||||
"name": "winnie",
|
|
||||||
},
|
|
||||||
"data": map[string]interface{}{
|
|
||||||
"wine": "cabernet",
|
|
||||||
"truck": "ford",
|
|
||||||
"rocket": "falcon9",
|
|
||||||
"planet": "mars",
|
|
||||||
"city": "brownsville",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
// order irrelevant, because assert.Equals is smart about maps.
|
|
||||||
expected: map[string]string{
|
|
||||||
"city": "brownsville",
|
|
||||||
"wine": "cabernet",
|
|
||||||
"planet": "mars",
|
|
||||||
"rocket": "falcon9",
|
|
||||||
"truck": "ford",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
kunstr := NewKunstructuredFactoryImpl().FromMap(tc.theMap)
|
|
||||||
m := kunstr.GetDataMap()
|
|
||||||
if !assert.Equal(t, tc.expected, m) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestKunstSetDataMap(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
theMap map[string]interface{}
|
|
||||||
input map[string]string
|
|
||||||
expected map[string]string
|
|
||||||
}{
|
|
||||||
"empty": {
|
|
||||||
theMap: map[string]interface{}{},
|
|
||||||
input: map[string]string{
|
|
||||||
"wine": "cabernet",
|
|
||||||
"truck": "ford",
|
|
||||||
},
|
|
||||||
expected: map[string]string{
|
|
||||||
"wine": "cabernet",
|
|
||||||
"truck": "ford",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"replace": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"foo": 3,
|
|
||||||
"data": map[string]string{
|
|
||||||
"rocket": "falcon9",
|
|
||||||
"planet": "mars",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
input: map[string]string{
|
|
||||||
"wine": "cabernet",
|
|
||||||
"truck": "ford",
|
|
||||||
},
|
|
||||||
expected: map[string]string{
|
|
||||||
"wine": "cabernet",
|
|
||||||
"truck": "ford",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"clear1": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"foo": 3,
|
|
||||||
"data": map[string]string{
|
|
||||||
"rocket": "falcon9",
|
|
||||||
"planet": "mars",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
input: map[string]string{},
|
|
||||||
expected: map[string]string{},
|
|
||||||
},
|
|
||||||
"clear2": {
|
|
||||||
theMap: map[string]interface{}{
|
|
||||||
"foo": 3,
|
|
||||||
"data": map[string]string{
|
|
||||||
"rocket": "falcon9",
|
|
||||||
"planet": "mars",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
input: nil,
|
|
||||||
expected: map[string]string{},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
kunstr := NewKunstructuredFactoryImpl().FromMap(tc.theMap)
|
|
||||||
kunstr.SetDataMap(tc.input)
|
|
||||||
m := kunstr.GetDataMap()
|
|
||||||
if !assert.Equal(t, tc.expected, m) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -51,11 +51,6 @@ commonLabels:
|
|||||||
group: apps
|
group: apps
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
|
|
||||||
- path: spec/template/spec/topologySpreadConstraints/labelSelector/matchLabels
|
|
||||||
create: false
|
|
||||||
group: apps
|
|
||||||
kind: Deployment
|
|
||||||
|
|
||||||
- path: spec/selector/matchLabels
|
- path: spec/selector/matchLabels
|
||||||
create: true
|
create: true
|
||||||
kind: ReplicaSet
|
kind: ReplicaSet
|
||||||
@@ -102,11 +97,6 @@ commonLabels:
|
|||||||
group: apps
|
group: apps
|
||||||
kind: StatefulSet
|
kind: StatefulSet
|
||||||
|
|
||||||
- path: spec/template/spec/topologySpreadConstraints/labelSelector/matchLabels
|
|
||||||
create: false
|
|
||||||
group: apps
|
|
||||||
kind: StatefulSet
|
|
||||||
|
|
||||||
- path: spec/volumeClaimTemplates[]/metadata/labels
|
- path: spec/volumeClaimTemplates[]/metadata/labels
|
||||||
create: true
|
create: true
|
||||||
group: apps
|
group: apps
|
||||||
|
|||||||
@@ -3,9 +3,6 @@
|
|||||||
|
|
||||||
package builtinpluginconsts
|
package builtinpluginconsts
|
||||||
|
|
||||||
// TODO: rename 'fieldSpecs' to 'referrers' for clarity.
|
|
||||||
// This will, however, break anyone using a custom config.
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
nameReferenceFieldSpecs = `
|
nameReferenceFieldSpecs = `
|
||||||
nameReference:
|
nameReference:
|
||||||
@@ -124,10 +121,6 @@ nameReference:
|
|||||||
kind: CronJob
|
kind: CronJob
|
||||||
- path: spec/configSource/configMap
|
- path: spec/configSource/configMap
|
||||||
kind: Node
|
kind: Node
|
||||||
- path: rules/resourceNames
|
|
||||||
kind: Role
|
|
||||||
- path: rules/resourceNames
|
|
||||||
kind: ClusterRole
|
|
||||||
|
|
||||||
- kind: Secret
|
- kind: Secret
|
||||||
version: v1
|
version: v1
|
||||||
@@ -265,8 +258,6 @@ nameReference:
|
|||||||
kind: Service
|
kind: Service
|
||||||
group: serving.knative.dev
|
group: serving.knative.dev
|
||||||
version: v1
|
version: v1
|
||||||
- path: spec/azureFile/secretName
|
|
||||||
kind: PersistentVolume
|
|
||||||
|
|
||||||
- kind: Service
|
- kind: Service
|
||||||
version: v1
|
version: v1
|
||||||
|
|||||||
@@ -19,32 +19,7 @@ func DefaultKustomizationFileName() string {
|
|||||||
return RecognizedKustomizationFileNames()[0]
|
return RecognizedKustomizationFileNames()[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
// IfApiMachineryElseKyaml returns true if executing the apimachinery code
|
|
||||||
// path, else we're executing the kyaml code paths.
|
|
||||||
func IfApiMachineryElseKyaml(s1, s2 string) string {
|
|
||||||
if !FlagEnableKyamlDefaultValue {
|
|
||||||
return s1
|
|
||||||
}
|
|
||||||
return s2
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// FlagEnableKyamlDefaultValue is the default value for the --enable_kyaml
|
|
||||||
// flag. This value is also used in unit tests. See provider.DepProvider.
|
|
||||||
//
|
|
||||||
// TODO(#3304): eliminate branching on this constant.
|
|
||||||
// Details: https://github.com/kubernetes-sigs/kustomize/issues/3304
|
|
||||||
//
|
|
||||||
// All tests should pass for either true or false values
|
|
||||||
// of this constant, without having to check its value.
|
|
||||||
// In the cases where there's a different outcome, either decide
|
|
||||||
// that the difference is acceptable, or make the difference go away.
|
|
||||||
//
|
|
||||||
// Historically, tests passed for enable_kyaml == false, i.e. using
|
|
||||||
// apimachinery libs. This doesn't mean the code was better, it just
|
|
||||||
// means regression tests preserved those outcomes.
|
|
||||||
FlagEnableKyamlDefaultValue = true
|
|
||||||
|
|
||||||
// An environment variable to consult for kustomization
|
// An environment variable to consult for kustomization
|
||||||
// configuration data. See:
|
// configuration data. See:
|
||||||
// https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
|
// https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
|
||||||
@@ -56,12 +31,6 @@ const (
|
|||||||
// A program name, for use in help, finding the XDG_CONFIG_DIR, etc.
|
// A program name, for use in help, finding the XDG_CONFIG_DIR, etc.
|
||||||
ProgramName = "kustomize"
|
ProgramName = "kustomize"
|
||||||
|
|
||||||
// ConfigAnnoDomain is configuration-related annotation namespace.
|
|
||||||
ConfigAnnoDomain = "config.kubernetes.io"
|
|
||||||
|
|
||||||
// If a resource has this annotation, kustomize will drop it.
|
|
||||||
IgnoredByKustomizeAnnotation = ConfigAnnoDomain + "/local-config"
|
|
||||||
|
|
||||||
// Label key that indicates the resources are built from Kustomize
|
// Label key that indicates the resources are built from Kustomize
|
||||||
ManagedbyLabelKey = "app.kubernetes.io/managed-by"
|
ManagedbyLabelKey = "app.kubernetes.io/managed-by"
|
||||||
|
|
||||||
|
|||||||
@@ -93,70 +93,3 @@ resources:
|
|||||||
t.Fatalf("unexpected error: %q", err)
|
t.Fatalf("unexpected error: %q", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestResourceHasAnchor(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("/app", `
|
|
||||||
resources:
|
|
||||||
- ingress.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/ingress.yaml", `
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: blog
|
|
||||||
spec:
|
|
||||||
tls:
|
|
||||||
- hosts:
|
|
||||||
- xyz.me
|
|
||||||
- www.xyz.me
|
|
||||||
secretName: cert-tls
|
|
||||||
rules:
|
|
||||||
- host: xyz.me
|
|
||||||
http: &xxx_rules
|
|
||||||
paths:
|
|
||||||
- path: /
|
|
||||||
pathType: Prefix
|
|
||||||
backend:
|
|
||||||
service:
|
|
||||||
name: service
|
|
||||||
port:
|
|
||||||
number: 80
|
|
||||||
- host: www.xyz.me
|
|
||||||
http: *xxx_rules
|
|
||||||
`)
|
|
||||||
m := th.Run("/app", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: blog
|
|
||||||
spec:
|
|
||||||
rules:
|
|
||||||
- host: xyz.me
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- backend:
|
|
||||||
service:
|
|
||||||
name: service
|
|
||||||
port:
|
|
||||||
number: 80
|
|
||||||
path: /
|
|
||||||
pathType: Prefix
|
|
||||||
- host: www.xyz.me
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- backend:
|
|
||||||
service:
|
|
||||||
name: service
|
|
||||||
port:
|
|
||||||
number: 80
|
|
||||||
path: /
|
|
||||||
pathType: Prefix
|
|
||||||
tls:
|
|
||||||
- hosts:
|
|
||||||
- xyz.me
|
|
||||||
- www.xyz.me
|
|
||||||
secretName: cert-tls
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,102 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package krusty_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestBasicIO_1(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
opts := th.MakeDefaultOptions()
|
|
||||||
if !opts.UseKyaml {
|
|
||||||
// This test won't pass under apimachinery, because in the bowels of
|
|
||||||
// that code (see GetAnnotations in v0.17.0 of
|
|
||||||
// k8s.io/apimachinery/pkg/apis/meta/v1/unstructured/unstructured.go)
|
|
||||||
// an error returned from NestedStringMap is discarded, and an
|
|
||||||
// empty annotation map is silently returned, making this test fail
|
|
||||||
// The swallowed error arises from code like:
|
|
||||||
// var v interface{}
|
|
||||||
// v = true
|
|
||||||
// if str, ok := v.(string); ok {
|
|
||||||
// save the value in a map (doesn't happen)
|
|
||||||
// } else {
|
|
||||||
// return an error (that is then ignored by GetAnnotations)
|
|
||||||
// }
|
|
||||||
// The error happens when any annotation value can be interpreted as
|
|
||||||
// a boolean or number. Such annotations cannot be successfully applied
|
|
||||||
// to an object in a cluster unless they are quoted.
|
|
||||||
t.SkipNow()
|
|
||||||
}
|
|
||||||
th.WriteK(".", `
|
|
||||||
resources:
|
|
||||||
- service.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("service.yaml", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
annotations:
|
|
||||||
port: 8080
|
|
||||||
happy: true
|
|
||||||
color: green
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
`)
|
|
||||||
m := th.Run(".", opts)
|
|
||||||
// The annotations are sorted by key, hence the order change.
|
|
||||||
// Quotes are added intentionally.
|
|
||||||
th.AssertActualEqualsExpected(
|
|
||||||
m, `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
annotations:
|
|
||||||
color: green
|
|
||||||
happy: "true"
|
|
||||||
port: "8080"
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestBasicIO_2(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
opts := th.MakeDefaultOptions()
|
|
||||||
th.WriteK(".", `
|
|
||||||
resources:
|
|
||||||
- service.yaml
|
|
||||||
`)
|
|
||||||
// All the annotation values are quoted in the input.
|
|
||||||
th.WriteF("service.yaml", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
annotations:
|
|
||||||
port: "8080"
|
|
||||||
happy: "true"
|
|
||||||
color: green
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
`)
|
|
||||||
m := th.Run(".", opts)
|
|
||||||
// The annotations are sorted by key, hence the order change.
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
annotations:
|
|
||||||
color: green
|
|
||||||
happy: "true"
|
|
||||||
port: "8080"
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
@@ -39,8 +39,8 @@ resources:
|
|||||||
configMapGenerator:
|
configMapGenerator:
|
||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
literals:
|
literals:
|
||||||
- testValue=purple
|
- testValue=1
|
||||||
- otherValue=green
|
- otherValue=10
|
||||||
`)
|
`)
|
||||||
th.WriteF("/app/base/deploy.yaml", `
|
th.WriteF("/app/base/deploy.yaml", `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@@ -64,8 +64,8 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- testValue=blue
|
- testValue=2
|
||||||
- compValue=red
|
- compValue=5
|
||||||
`)
|
`)
|
||||||
th.WriteF("/app/comp/stub.yaml", `
|
th.WriteF("/app/comp/stub.yaml", `
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@@ -125,12 +125,14 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
compValue: red
|
compValue: "5"
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: blue
|
testValue: "2"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: comp-my-configmap-97647ckcmg
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: comp-my-configmap-kc6k2kmkh9
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -154,7 +156,7 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- otherValue=orange
|
- otherValue=9
|
||||||
`),
|
`),
|
||||||
writeK("/app/prod", `
|
writeK("/app/prod", `
|
||||||
resources:
|
resources:
|
||||||
@@ -177,12 +179,14 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
compValue: red
|
compValue: "5"
|
||||||
otherValue: orange
|
otherValue: "9"
|
||||||
testValue: blue
|
testValue: "2"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: comp-my-configmap-g486mb229k
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: comp-my-configmap-55249mf5kb
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -208,7 +212,7 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- otherValue=orange
|
- otherValue=9
|
||||||
`),
|
`),
|
||||||
writeK("/app/prod", `
|
writeK("/app/prod", `
|
||||||
resources:
|
resources:
|
||||||
@@ -230,12 +234,14 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
compValue: red
|
compValue: "5"
|
||||||
otherValue: orange
|
otherValue: "9"
|
||||||
testValue: blue
|
testValue: "2"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: comp-my-configmap-g486mb229k
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: comp-my-configmap-55249mf5kb
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -273,11 +279,11 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: purple
|
testValue: "1"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: my-configmap-9cd648hm8f
|
name: my-configmap-2g9c94mhb8
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -288,12 +294,14 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
compValue: red
|
compValue: "5"
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: blue
|
testValue: "2"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: comp-my-configmap-97647ckcmg
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: comp-my-configmap-kc6k2kmkh9
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -319,8 +327,8 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- compValue=red
|
- compValue=5
|
||||||
- testValue=blue
|
- testValue=2
|
||||||
`),
|
`),
|
||||||
},
|
},
|
||||||
runPath: "/app/direct-component",
|
runPath: "/app/direct-component",
|
||||||
@@ -334,12 +342,14 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
compValue: red
|
compValue: "5"
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: blue
|
testValue: "2"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: my-configmap-97647ckcmg
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: my-configmap-kc6k2kmkh9
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
"missing-optional-component-api-version": {
|
"missing-optional-component-api-version": {
|
||||||
@@ -350,7 +360,7 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- otherValue=orange
|
- otherValue=9
|
||||||
`),
|
`),
|
||||||
},
|
},
|
||||||
runPath: "/app/prod",
|
runPath: "/app/prod",
|
||||||
@@ -364,11 +374,13 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
otherValue: orange
|
otherValue: "9"
|
||||||
testValue: purple
|
testValue: "1"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: my-configmap-6hhdg8gkdg
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
|
name: my-configmap-5g7gh5mgt5
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -411,11 +423,11 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: purple
|
testValue: "1"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: my-configmap-a-b-9cd648hm8f
|
name: my-configmap-a-b-2g9c94mhb8
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
kind: Deployment
|
kind: Deployment
|
||||||
@@ -426,11 +438,11 @@ spec:
|
|||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
otherValue: green
|
otherValue: "10"
|
||||||
testValue: purple
|
testValue: "1"
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
name: my-configmap-b-9cd648hm8f
|
name: my-configmap-b-2g9c94mhb8
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -562,7 +574,7 @@ configMapGenerator:
|
|||||||
- name: my-configmap
|
- name: my-configmap
|
||||||
behavior: merge
|
behavior: merge
|
||||||
literals:
|
literals:
|
||||||
- otherValue=orange
|
- otherValue=9
|
||||||
`),
|
`),
|
||||||
},
|
},
|
||||||
runPath: "/app/prod",
|
runPath: "/app/prod",
|
||||||
|
|||||||
@@ -4,121 +4,11 @@
|
|||||||
package krusty_test
|
package krusty_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Numbers and booleans are quoted
|
|
||||||
func TestGeneratorIntVsStringNoMerge(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK(".", `
|
|
||||||
resources:
|
|
||||||
- service.yaml
|
|
||||||
configMapGenerator:
|
|
||||||
- name: bob
|
|
||||||
literals:
|
|
||||||
- fruit=Indian Gooseberry
|
|
||||||
- year=2020
|
|
||||||
- crisis=true
|
|
||||||
`)
|
|
||||||
th.WriteF("service.yaml", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
`)
|
|
||||||
m := th.Run(".", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(
|
|
||||||
m, `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: demo
|
|
||||||
spec:
|
|
||||||
clusterIP: None
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
crisis: "true"
|
|
||||||
fruit: Indian Gooseberry
|
|
||||||
year: "2020"
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: bob-79t79mt227
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGeneratorIntVsStringWithMerge(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("base", `
|
|
||||||
configMapGenerator:
|
|
||||||
- name: bob
|
|
||||||
literals:
|
|
||||||
- fruit=Indian Gooseberry
|
|
||||||
- year=2020
|
|
||||||
- crisis=true
|
|
||||||
`)
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
configMapGenerator:
|
|
||||||
- name: bob
|
|
||||||
behavior: merge
|
|
||||||
literals:
|
|
||||||
- month=12
|
|
||||||
`)
|
|
||||||
m := th.Run("overlay", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `apiVersion: v1
|
|
||||||
data:
|
|
||||||
crisis: "true"
|
|
||||||
fruit: Indian Gooseberry
|
|
||||||
month: "12"
|
|
||||||
year: "2020"
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: bob-bk46gm59c6
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGeneratorFromProperties(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("base", `
|
|
||||||
configMapGenerator:
|
|
||||||
- name: test-configmap
|
|
||||||
behavior: create
|
|
||||||
envs:
|
|
||||||
- properties
|
|
||||||
`)
|
|
||||||
th.WriteF("base/properties", `
|
|
||||||
VAR1=100
|
|
||||||
`)
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
configMapGenerator:
|
|
||||||
- name: test-configmap
|
|
||||||
behavior: "merge"
|
|
||||||
envs:
|
|
||||||
- properties
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/properties", `
|
|
||||||
VAR2=200
|
|
||||||
`)
|
|
||||||
m := th.Run("overlay", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `apiVersion: v1
|
|
||||||
data:
|
|
||||||
VAR1: "100"
|
|
||||||
VAR2: "200"
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: test-configmap-hdghb5ddkg
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate a Secret and a ConfigMap from the same data
|
// Generate a Secret and a ConfigMap from the same data
|
||||||
// to compare the result.
|
// to compare the result.
|
||||||
func TestGeneratorBasics(t *testing.T) {
|
func TestGeneratorBasics(t *testing.T) {
|
||||||
@@ -169,9 +59,10 @@ electromagnetic
|
|||||||
strong nuclear
|
strong nuclear
|
||||||
weak nuclear
|
weak nuclear
|
||||||
`)
|
`)
|
||||||
opts := th.MakeDefaultOptions()
|
|
||||||
m := th.Run("/app", opts)
|
m := th.Run("/app", th.MakeDefaultOptions())
|
||||||
expFmt := `apiVersion: v1
|
th.AssertActualEqualsExpected(m, `
|
||||||
|
apiVersion: v1
|
||||||
data:
|
data:
|
||||||
MOUNTAIN: everest
|
MOUNTAIN: everest
|
||||||
OCEAN: pacific
|
OCEAN: pacific
|
||||||
@@ -204,30 +95,15 @@ apiVersion: v1
|
|||||||
data:
|
data:
|
||||||
MOUNTAIN: ZXZlcmVzdA==
|
MOUNTAIN: ZXZlcmVzdA==
|
||||||
OCEAN: cGFjaWZpYw==
|
OCEAN: cGFjaWZpYw==
|
||||||
forces.txt: %s
|
forces.txt: CmdyYXZpdGF0aW9uYWwKZWxlY3Ryb21hZ25ldGljCnN0cm9uZyBudWNsZWFyCndlYWsgbnVjbGVhcgo=
|
||||||
fruit: YXBwbGU=
|
fruit: YXBwbGU=
|
||||||
passphrase: %s
|
passphrase: CkxpZmUgaXMgc2hvcnQuCkJ1dCB0aGUgeWVhcnMgYXJlIGxvbmcuCk5vdCB3aGlsZSB0aGUgZXZpbCBkYXlzIGNvbWUgbm90Lgo=
|
||||||
vegetable: YnJvY2NvbGk=
|
vegetable: YnJvY2NvbGk=
|
||||||
kind: Secret
|
kind: Secret
|
||||||
metadata:
|
metadata:
|
||||||
name: blah-bob-%s
|
name: blah-bob-ftht6hfgmb
|
||||||
type: Opaque
|
type: Opaque
|
||||||
`
|
`)
|
||||||
th.AssertActualEqualsExpected(
|
|
||||||
m,
|
|
||||||
// TODO(#3304): DECISION - kyaml better; not a bug.
|
|
||||||
opts.IfApiMachineryElseKyaml(
|
|
||||||
fmt.Sprintf(
|
|
||||||
expFmt,
|
|
||||||
`CmdyYXZpdGF0aW9uYWwKZWxlY3Ryb21hZ25ldGljCnN0cm9uZyBudWNsZWFyCndlYWsgbnVjbGVhcgo=`,
|
|
||||||
`CkxpZmUgaXMgc2hvcnQuCkJ1dCB0aGUgeWVhcnMgYXJlIGxvbmcuCk5vdCB3aGlsZSB0aGUgZXZpbCBkYXlzIGNvbWUgbm90Lgo=`,
|
|
||||||
`ftht6hfgmb`),
|
|
||||||
fmt.Sprintf(
|
|
||||||
expFmt, `|
|
|
||||||
CmdyYXZpdGF0aW9uYWwKZWxlY3Ryb21hZ25ldGljCnN0cm9uZyBudWNsZWFyCndlYWsgbn
|
|
||||||
VjbGVhcgo=`, `|
|
|
||||||
CkxpZmUgaXMgc2hvcnQuCkJ1dCB0aGUgeWVhcnMgYXJlIGxvbmcuCk5vdCB3aGlsZSB0aG
|
|
||||||
UgZXZpbCBkYXlzIGNvbWUgbm90Lgo=`, `9t25t44gg4`)))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: These should be errors instead.
|
// TODO: These should be errors instead.
|
||||||
@@ -283,68 +159,6 @@ metadata:
|
|||||||
`)
|
`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIssue3393(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK(".", `
|
|
||||||
resources:
|
|
||||||
- cm.yaml
|
|
||||||
configMapGenerator:
|
|
||||||
- name: project
|
|
||||||
behavior: merge
|
|
||||||
literals:
|
|
||||||
- ANOTHER_ENV_VARIABLE="bar"
|
|
||||||
`)
|
|
||||||
th.WriteF("cm.yaml", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: project
|
|
||||||
data:
|
|
||||||
A_FIRST_ENV_VARIABLE: "foo"
|
|
||||||
`)
|
|
||||||
m := th.Run(".", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
A_FIRST_ENV_VARIABLE: foo
|
|
||||||
ANOTHER_ENV_VARIABLE: bar
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: project
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGeneratorSimpleOverlay(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("base", `
|
|
||||||
namePrefix: p-
|
|
||||||
configMapGenerator:
|
|
||||||
- name: cm
|
|
||||||
behavior: create
|
|
||||||
literals:
|
|
||||||
- fruit=apple
|
|
||||||
`)
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
configMapGenerator:
|
|
||||||
- name: cm
|
|
||||||
behavior: merge
|
|
||||||
literals:
|
|
||||||
- veggie=broccoli
|
|
||||||
`)
|
|
||||||
m := th.Run("overlay", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
fruit: apple
|
|
||||||
veggie: broccoli
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: p-cm-877mt5hc89
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGeneratorOverlays(t *testing.T) {
|
func TestGeneratorOverlays(t *testing.T) {
|
||||||
th := kusttest_test.MakeHarness(t)
|
th := kusttest_test.MakeHarness(t)
|
||||||
th.WriteK("/app/base1", `
|
th.WriteK("/app/base1", `
|
||||||
@@ -401,6 +215,8 @@ data:
|
|||||||
from: overlay
|
from: overlay
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
name: p1-com1-8tc62428t2
|
name: p1-com1-8tc62428t2
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@@ -408,6 +224,8 @@ data:
|
|||||||
from: overlay
|
from: overlay
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
name: p2-com2-87mcggf7d7
|
name: p2-com2-87mcggf7d7
|
||||||
`)
|
`)
|
||||||
}
|
}
|
||||||
@@ -454,6 +272,8 @@ data:
|
|||||||
foo: bar
|
foo: bar
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
name: o1-cm-ft9mmdc8c6
|
name: o1-cm-ft9mmdc8c6
|
||||||
---
|
---
|
||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
@@ -462,43 +282,8 @@ data:
|
|||||||
foo: bar
|
foo: bar
|
||||||
kind: ConfigMap
|
kind: ConfigMap
|
||||||
metadata:
|
metadata:
|
||||||
|
annotations: {}
|
||||||
|
labels: {}
|
||||||
name: cm-o2-5k95kd76ft
|
name: cm-o2-5k95kd76ft
|
||||||
`)
|
`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestConfigMapGeneratorLiteralNewline(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("/app", `
|
|
||||||
generators:
|
|
||||||
- configmaps.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/configmaps.yaml", `
|
|
||||||
apiVersion: builtin
|
|
||||||
kind: ConfigMapGenerator
|
|
||||||
metadata:
|
|
||||||
name: testing
|
|
||||||
literals:
|
|
||||||
- |
|
|
||||||
initial.txt=greetings
|
|
||||||
everyone
|
|
||||||
- |
|
|
||||||
final.txt=different
|
|
||||||
behavior
|
|
||||||
---
|
|
||||||
`)
|
|
||||||
m := th.Run("/app", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(
|
|
||||||
m, `
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
final.txt: |
|
|
||||||
different
|
|
||||||
behavior
|
|
||||||
initial.txt: |
|
|
||||||
greetings
|
|
||||||
everyone
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: testing-tt4769fb52
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ spec:
|
|||||||
action: fly
|
action: fly
|
||||||
`)
|
`)
|
||||||
th.WriteF("/app/base/mykind.yaml", `
|
th.WriteF("/app/base/mykind.yaml", `
|
||||||
apiVersion: jingfang.example.com/v1
|
apiVersion: jingfang.example.com/v1beta1
|
||||||
kind: MyKind
|
kind: MyKind
|
||||||
metadata:
|
metadata:
|
||||||
name: mykind
|
name: mykind
|
||||||
@@ -236,7 +236,7 @@ kind: Secret
|
|||||||
metadata:
|
metadata:
|
||||||
name: x-crdsecret
|
name: x-crdsecret
|
||||||
---
|
---
|
||||||
apiVersion: jingfang.example.com/v1
|
apiVersion: jingfang.example.com/v1beta1
|
||||||
kind: MyKind
|
kind: MyKind
|
||||||
metadata:
|
metadata:
|
||||||
name: x-mykind
|
name: x-mykind
|
||||||
@@ -285,7 +285,7 @@ kind: Secret
|
|||||||
metadata:
|
metadata:
|
||||||
name: prod-x-crdsecret
|
name: prod-x-crdsecret
|
||||||
---
|
---
|
||||||
apiVersion: jingfang.example.com/v1
|
apiVersion: jingfang.example.com/v1beta1
|
||||||
kind: MyKind
|
kind: MyKind
|
||||||
metadata:
|
metadata:
|
||||||
name: prod-x-mykind
|
name: prod-x-mykind
|
||||||
|
|||||||
@@ -1,227 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package krusty_test
|
|
||||||
|
|
||||||
// In the following structure `top` patches `left-deploy` and `right-deploy` to reference the
|
|
||||||
// configMap in `bottom`. This test verifies `configMapRef.name` in `left-deploy` and
|
|
||||||
// `right-deploy` is modified correctly.
|
|
||||||
//
|
|
||||||
// top
|
|
||||||
// / \
|
|
||||||
// left right
|
|
||||||
// / \ / \
|
|
||||||
// left-service bottom bottom right-service
|
|
||||||
// | \ / |
|
|
||||||
// left-deploy bottom right-deploy
|
|
||||||
// |
|
|
||||||
// configMap
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestConfMapNameResolutionInDiamondWithPatches(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteK("bottom", `
|
|
||||||
configMapGenerator:
|
|
||||||
- name: bottom
|
|
||||||
literals:
|
|
||||||
- KEY=value
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("left-service", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("left-service/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: left-deploy
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: left-deploy
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/name: left-pod
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: left-pod
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: left-image:v1.0
|
|
||||||
name: service
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("right-service", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("right-service/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: right-deploy
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: right-deploy
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/name: right-pod
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: right-pod
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: right-image:v1.0
|
|
||||||
name: service
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("top", `
|
|
||||||
resources:
|
|
||||||
- ./left
|
|
||||||
- ./right
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("top/left", `
|
|
||||||
resources:
|
|
||||||
- ../../left-service
|
|
||||||
- ./bottom
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- target:
|
|
||||||
kind: Deployment
|
|
||||||
name: left-deploy
|
|
||||||
patch: |-
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: ignored-by-kustomize
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: service
|
|
||||||
env:
|
|
||||||
- name: MAPPED_CONFIG_ITEM
|
|
||||||
valueFrom:
|
|
||||||
configMapKeyRef:
|
|
||||||
name: left-bottom
|
|
||||||
key: KEY
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("top/left/bottom", `
|
|
||||||
namePrefix: left-
|
|
||||||
|
|
||||||
resources:
|
|
||||||
- ../../../bottom
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("top/right", `
|
|
||||||
resources:
|
|
||||||
- ../../right-service
|
|
||||||
- ./bottom
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- target:
|
|
||||||
kind: Deployment
|
|
||||||
name: right-deploy
|
|
||||||
patch: |-
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: ignored-by-kustomize
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: service
|
|
||||||
env:
|
|
||||||
- name: MAPPED_CONFIG_ITEM
|
|
||||||
valueFrom:
|
|
||||||
configMapKeyRef:
|
|
||||||
name: right-bottom
|
|
||||||
key: KEY
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("top/right/bottom", `
|
|
||||||
namePrefix: right-
|
|
||||||
|
|
||||||
resources:
|
|
||||||
- ../../../bottom
|
|
||||||
`)
|
|
||||||
|
|
||||||
m := th.Run("top", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: left-deploy
|
|
||||||
name: left-deploy
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/name: left-pod
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: left-pod
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- env:
|
|
||||||
- name: MAPPED_CONFIG_ITEM
|
|
||||||
valueFrom:
|
|
||||||
configMapKeyRef:
|
|
||||||
key: KEY
|
|
||||||
name: left-bottom-9f2t6f5h6d
|
|
||||||
image: left-image:v1.0
|
|
||||||
name: service
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
KEY: value
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: left-bottom-9f2t6f5h6d
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: right-deploy
|
|
||||||
name: right-deploy
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/name: right-pod
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/name: right-pod
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- env:
|
|
||||||
- name: MAPPED_CONFIG_ITEM
|
|
||||||
valueFrom:
|
|
||||||
configMapKeyRef:
|
|
||||||
key: KEY
|
|
||||||
name: right-bottom-9f2t6f5h6d
|
|
||||||
image: right-image:v1.0
|
|
||||||
name: service
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
KEY: value
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: right-bottom-9f2t6f5h6d
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
@@ -1,622 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package krusty_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Regression test for https://github.com/kubernetes-sigs/kustomize/issues/3280
|
|
||||||
// GVKN shouldn't change with default options
|
|
||||||
func TestKeepOriginalGVKN(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("apps/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("apps/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("apps", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
m := th.Run("apps", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/3280
|
|
||||||
// These tests document behavior that will change
|
|
||||||
func TestChangeName(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("apps/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("apps/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("apps", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// name should become `new-name`
|
|
||||||
m := th.Run("apps", options)
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestChangeKind(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("apps/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("apps/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("apps", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// kind should become `StatefulSet`
|
|
||||||
m := th.Run("apps", options)
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestChangeNameAndKind(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("apps/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("apps/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("apps", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// kind should become `StatefulSet`
|
|
||||||
// name should become `new-name`
|
|
||||||
m := th.Run("apps", options)
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/3280
|
|
||||||
// Should be able to refer to a resource with either its
|
|
||||||
// original GVKN or its current one
|
|
||||||
func TestPatchOriginalName(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
th.WriteF("base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
th.WriteK("base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// name should become `new-name`
|
|
||||||
m := th.Run("overlay", options)
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPatchNewName(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
th.WriteF("base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
th.WriteK("base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// depPatch cannot find target with the name `new-name`
|
|
||||||
// because base/patch.yaml can't yet edit the name
|
|
||||||
assert.Error(t, th.RunWithErr("overlay", options))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPatchOriginalNameAndKind(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
th.WriteF("base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
th.WriteK("base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// kind should become `StatefulSet`
|
|
||||||
// name should become `new-name`
|
|
||||||
m := th.Run("overlay", options)
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPatchNewNameAndKind(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
th.WriteF("base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
th.WriteK("base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// depPatch cannot find target with kind `StatefulSet` and name `new-name`
|
|
||||||
// because base/patch.yaml can't yet edit the kind or name
|
|
||||||
assert.Error(t, th.RunWithErr("overlay", options))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use original name, but new kind
|
|
||||||
// Should fail, even after #3280 is done, because this ID is invalid
|
|
||||||
func TestPatchOriginalNameAndNewKind(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteF("base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: old-name
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
th.WriteF("base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
`)
|
|
||||||
th.WriteK("base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("overlay/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: new-name
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// depPatch cannot find target with kind `Deployment` and name `new-name`
|
|
||||||
// because the resource never had this GVKN
|
|
||||||
assert.Error(t, th.RunWithErr("overlay", options))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Here is a structure of a kustomization of two components, component1
|
|
||||||
// and component2, that both use a shared deployment definition, which
|
|
||||||
// component2 adjusts by changing the kind via a patch. This test case
|
|
||||||
// checks that it does not have a conflicting definition.
|
|
||||||
// Currently documents broken behavior.
|
|
||||||
//
|
|
||||||
// root
|
|
||||||
// / \
|
|
||||||
// component1/overlay component2/overlay
|
|
||||||
// | |
|
|
||||||
// component1/base component2/base
|
|
||||||
// \ /
|
|
||||||
// base
|
|
||||||
//
|
|
||||||
// This is the directory layout:
|
|
||||||
//
|
|
||||||
// ├── component1
|
|
||||||
// │ ├── base
|
|
||||||
// │ │ └── kustomization.yaml
|
|
||||||
// │ └── overlay
|
|
||||||
// │ └── kustomization.yaml
|
|
||||||
// ├── component2
|
|
||||||
// │ ├── base
|
|
||||||
// │ │ └── kustomization.yaml
|
|
||||||
// │ └── overlay
|
|
||||||
// │ └── kustomization.yaml
|
|
||||||
// ├── shared
|
|
||||||
// │ ├── kustomization.yaml
|
|
||||||
// │ └── deployment.yaml
|
|
||||||
// ├── kustomization.yaml
|
|
||||||
|
|
||||||
func TestBaseReuseNameAndKindConflict(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteK("/app/shared", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/shared/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: my-deploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
image: nginx
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app/component1/base", `
|
|
||||||
resources:
|
|
||||||
- ../../shared
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/component1/overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
|
|
||||||
namePrefix: overlay-
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app/component2/base", `
|
|
||||||
resources:
|
|
||||||
- ../../shared
|
|
||||||
|
|
||||||
patches:
|
|
||||||
- path: patch.yaml
|
|
||||||
target:
|
|
||||||
kind: Deployment
|
|
||||||
name: my-deploy
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/component2/base/patch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: StatefulSet
|
|
||||||
metadata:
|
|
||||||
name: my-stateful-set
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/component2/overlay", `
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
|
|
||||||
namePrefix: overlay-
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app", `
|
|
||||||
resources:
|
|
||||||
- component1/overlay
|
|
||||||
- component2/overlay
|
|
||||||
`)
|
|
||||||
|
|
||||||
options := th.MakeDefaultOptions()
|
|
||||||
options.AllowResourceIdChanges = true
|
|
||||||
|
|
||||||
// Error occurs when app/component2/base tries to load the shared resources
|
|
||||||
// because the kind is not (yet) allowed to change yet
|
|
||||||
// so it loads a second Deployment with the name my-deploy
|
|
||||||
// instead of a StatefulSet as specified by the patch.
|
|
||||||
// Will be fixed by #3280.
|
|
||||||
assert.Error(t, th.RunWithErr("overlay", options))
|
|
||||||
}
|
|
||||||
@@ -1,79 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package krusty_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestInlineTransformer(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeEnhancedHarness(t)
|
|
||||||
defer th.Reset()
|
|
||||||
|
|
||||||
th.WriteF("/app/resource.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: whatever
|
|
||||||
data: {}
|
|
||||||
`)
|
|
||||||
th.WriteK("/app", `
|
|
||||||
resources:
|
|
||||||
- resource.yaml
|
|
||||||
transformers:
|
|
||||||
- |-
|
|
||||||
apiVersion: builtin
|
|
||||||
kind: NamespaceTransformer
|
|
||||||
metadata:
|
|
||||||
name: not-important-to-example
|
|
||||||
namespace: test
|
|
||||||
fieldSpecs:
|
|
||||||
- path: metadata/namespace
|
|
||||||
create: true
|
|
||||||
`)
|
|
||||||
|
|
||||||
expected := `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
data: {}
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: whatever
|
|
||||||
namespace: test
|
|
||||||
`
|
|
||||||
|
|
||||||
m := th.Run("/app", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, expected)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestInlineGenerator(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeEnhancedHarness(t)
|
|
||||||
defer th.Reset()
|
|
||||||
|
|
||||||
th.WriteK("/app", `
|
|
||||||
generators:
|
|
||||||
- |-
|
|
||||||
apiVersion: builtin
|
|
||||||
kind: ConfigMapGenerator
|
|
||||||
metadata:
|
|
||||||
name: mymap
|
|
||||||
literals:
|
|
||||||
- FRUIT=apple
|
|
||||||
- VEGETABLE=carrot
|
|
||||||
`)
|
|
||||||
|
|
||||||
expected := `
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
FRUIT: apple
|
|
||||||
VEGETABLE: carrot
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: mymap-kfd8tf729k
|
|
||||||
`
|
|
||||||
|
|
||||||
m := th.Run("/app", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, expected)
|
|
||||||
}
|
|
||||||
@@ -1,387 +0,0 @@
|
|||||||
package krusty_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Checks that a patch at the top of the stack can refer to resources
|
|
||||||
// by an intermediate name after it has gone through multiple name
|
|
||||||
// transformations.
|
|
||||||
// Ref: Issue #3455
|
|
||||||
func TestIntermediateName(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("/app/gcp", `
|
|
||||||
namePrefix: gcp-
|
|
||||||
resources:
|
|
||||||
- ../emea
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/gcp/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: prod-foo
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/emea", `
|
|
||||||
namePrefix: emea-
|
|
||||||
resources:
|
|
||||||
- ../prod
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/prod", `
|
|
||||||
namePrefix: prod-
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
m := th.Run("/app/gcp", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: gcp-emea-prod-foo
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tests that if resources in different layers (containing name
|
|
||||||
// transformations) have the same name, there is no conflict
|
|
||||||
func TestIntermediateNameSameNameDifferentLayer(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("/app/gcp", `
|
|
||||||
namePrefix: gcp-
|
|
||||||
resources:
|
|
||||||
- ../emea
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/gcp/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: prod-foo
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/emea", `
|
|
||||||
namePrefix: emea-
|
|
||||||
resources:
|
|
||||||
- ../prod
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/emea/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/prod", `
|
|
||||||
namePrefix: prod-
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
m := th.Run("/app/gcp", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: gcp-emea-prod-foo
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: gcp-emea-foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Same as above test but tries to refer to the name foo
|
|
||||||
// instead of prod-foo
|
|
||||||
func TestIntermediateNameAmbiguous(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
th.WriteK("/app/gcp", `
|
|
||||||
namePrefix: gcp-
|
|
||||||
resources:
|
|
||||||
- ../emea
|
|
||||||
patchesStrategicMerge:
|
|
||||||
- depPatch.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/gcp/depPatch.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
replicas: 999
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/emea", `
|
|
||||||
namePrefix: emea-
|
|
||||||
resources:
|
|
||||||
- ../prod
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/emea/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/prod", `
|
|
||||||
namePrefix: prod-
|
|
||||||
resources:
|
|
||||||
- ../base
|
|
||||||
`)
|
|
||||||
th.WriteK("/app/base", `
|
|
||||||
resources:
|
|
||||||
- deployment.yaml
|
|
||||||
`)
|
|
||||||
th.WriteF("/app/base/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: foo
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`)
|
|
||||||
err := th.RunWithErr("/app/gcp", th.MakeDefaultOptions())
|
|
||||||
assert.Error(t, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test for issue #3228
|
|
||||||
// References to resources by their intermediate names after multiple
|
|
||||||
// name transformations should be supported
|
|
||||||
func TestIntermediateNameSecretKeyRefDiamond(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
|
|
||||||
th.WriteK(".", `
|
|
||||||
namePrefix: project-
|
|
||||||
resources:
|
|
||||||
- app`)
|
|
||||||
|
|
||||||
th.WriteK("/app", `
|
|
||||||
resources:
|
|
||||||
- resources/deployment.yaml
|
|
||||||
- resources/xql
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app/resources/xql", `
|
|
||||||
resources:
|
|
||||||
- xql-zero
|
|
||||||
- xql-one
|
|
||||||
configurations:
|
|
||||||
- ./kustomizeconfig.yaml
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("/app/resources/xql/kustomizeconfig.yaml", `
|
|
||||||
varReference:
|
|
||||||
- path: spec/template/spec/containers/env/valueFrom/secretKeyRef/name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app/resources/xql/xql-one", `
|
|
||||||
namePrefix: xql-one-
|
|
||||||
resources:
|
|
||||||
- ../../../../bases/xql
|
|
||||||
secretGenerator:
|
|
||||||
- name: xql-secret
|
|
||||||
behavior: merge
|
|
||||||
envs:
|
|
||||||
- config/xql-one-secret.env
|
|
||||||
vars:
|
|
||||||
- name: PROJECT_XQL_ONE_SECRET_NAME
|
|
||||||
objref:
|
|
||||||
kind: Secret
|
|
||||||
name: xql-secret
|
|
||||||
apiVersion: v1
|
|
||||||
fieldref:
|
|
||||||
fieldpath: metadata.name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("/app/resources/xql/xql-one/config/xql-one-secret.env", `
|
|
||||||
arg=1
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/app/resources/xql/xql-zero", `
|
|
||||||
namePrefix: xql-zero-
|
|
||||||
resources:
|
|
||||||
- ../../../../bases/xql
|
|
||||||
secretGenerator:
|
|
||||||
- name: xql-secret
|
|
||||||
behavior: merge
|
|
||||||
envs:
|
|
||||||
- config/xql-zero-secret.env
|
|
||||||
vars:
|
|
||||||
- name: PROJECT_XQL_ZERO_SECRET_NAME
|
|
||||||
objref:
|
|
||||||
kind: Secret
|
|
||||||
name: xql-secret
|
|
||||||
apiVersion: v1
|
|
||||||
fieldref:
|
|
||||||
fieldpath: metadata.name
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("/app/resources/xql/xql-zero/config/xql-zero-secret.env", `
|
|
||||||
arg=0
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("app/resources/deployment.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: app
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: app
|
|
||||||
image: example.com/app:latest
|
|
||||||
imagePullPolicy: Always
|
|
||||||
env:
|
|
||||||
- name: XQL_ZERO_ARG
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
name: $(PROJECT_XQL_ZERO_SECRET_NAME)
|
|
||||||
key: arg
|
|
||||||
- name: XQL_ZERO_PASSWORD
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
name: xql-zero-xql-secret
|
|
||||||
key: password
|
|
||||||
- name: XQL_ONE_ARG
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
name: $(PROJECT_XQL_ONE_SECRET_NAME)
|
|
||||||
key: arg
|
|
||||||
- name: XQL_ONE_PASSWORD
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
name: xql-one-xql-secret
|
|
||||||
key: password
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteK("/bases/xql", `
|
|
||||||
secretGenerator:
|
|
||||||
- name: xql-secret
|
|
||||||
envs:
|
|
||||||
- config/xql-secret.env
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("bases/xql/config/xql-secret.env", `
|
|
||||||
password=SUPER_SECRET_PASSWORD
|
|
||||||
`)
|
|
||||||
|
|
||||||
m := th.Run(".", th.MakeDefaultOptions())
|
|
||||||
th.AssertActualEqualsExpected(m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: project-app
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- env:
|
|
||||||
- name: XQL_ZERO_ARG
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
key: arg
|
|
||||||
name: project-xql-zero-xql-secret-6khmtc56hm
|
|
||||||
- name: XQL_ZERO_PASSWORD
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
key: password
|
|
||||||
name: project-xql-zero-xql-secret-6khmtc56hm
|
|
||||||
- name: XQL_ONE_ARG
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
key: arg
|
|
||||||
name: project-xql-one-xql-secret-79mhmf5dgt
|
|
||||||
- name: XQL_ONE_PASSWORD
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
key: password
|
|
||||||
name: project-xql-one-xql-secret-79mhmf5dgt
|
|
||||||
image: example.com/app:latest
|
|
||||||
imagePullPolicy: Always
|
|
||||||
name: app
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
arg: MA==
|
|
||||||
password: U1VQRVJfU0VDUkVUX1BBU1NXT1JE
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: project-xql-zero-xql-secret-6khmtc56hm
|
|
||||||
type: Opaque
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
data:
|
|
||||||
arg: MQ==
|
|
||||||
password: U1VQRVJfU0VDUkVUX1BBU1NXT1JE
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: project-xql-one-xql-secret-79mhmf5dgt
|
|
||||||
type: Opaque
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
@@ -5,13 +5,13 @@ package provider
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/internal/conflict"
|
"sigs.k8s.io/kustomize/api/internal/k8sdeps/merge"
|
||||||
k8sconflict "sigs.k8s.io/kustomize/api/internal/k8sdeps/conflict"
|
kmerge "sigs.k8s.io/kustomize/api/internal/merge"
|
||||||
"sigs.k8s.io/kustomize/api/internal/validate"
|
"sigs.k8s.io/kustomize/api/internal/validate"
|
||||||
"sigs.k8s.io/kustomize/api/internal/wrappy"
|
"sigs.k8s.io/kustomize/api/internal/wrappy"
|
||||||
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/k8sdeps/validator"
|
"sigs.k8s.io/kustomize/api/k8sdeps/validator"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -95,27 +95,17 @@ import (
|
|||||||
// would really reduce the work
|
// would really reduce the work
|
||||||
// (e.g. drop Vars, drop ReplacementTranformer).
|
// (e.g. drop Vars, drop ReplacementTranformer).
|
||||||
//
|
//
|
||||||
// - resource.ConflictDetector
|
// - resmap.Merginator
|
||||||
//
|
//
|
||||||
// 1) api/internal/k8sdeps/conflict.conflictDetectorJson
|
// 1) api/internal/k8sdeps/merge.Merginator
|
||||||
// api/internal/k8sdeps/conflict.conflictDetectorSm
|
|
||||||
//
|
//
|
||||||
// Uses k8s.io/apimachinery/pkg/util/strategicpatch,
|
// Uses k8s.io/apimachinery/pkg/util/strategicpatch,
|
||||||
// apimachinery/pkg/util/mergepatch, etc. to merge
|
// apimachinery/pkg/util/mergepatch, etc. to merge
|
||||||
// resource.Resource instances.
|
// resource.Resource instances.
|
||||||
//
|
//
|
||||||
// 2) api/internal/conflict.smPatchMergeOnlyDetector
|
// 2) api/internal/merge.Merginator
|
||||||
//
|
//
|
||||||
// At time of writing, this doesn't report conflicts,
|
// At time of writing, this is unimplemented.
|
||||||
// but it does know how to merge patches. Conflict
|
|
||||||
// reporting isn't vital to kustomize function. It's
|
|
||||||
// rare that a person would configure one transformer
|
|
||||||
// with many patches, much less so many that it became
|
|
||||||
// hard to spot conflicts. In the case of an undetected
|
|
||||||
// conflict, the last patch applied wins, likely what
|
|
||||||
// the user wants anyway. Regardless, the effect of this
|
|
||||||
// is plainly visible and usable in the output, even if
|
|
||||||
// a conflict happened but wasn't reported as an error.
|
|
||||||
//
|
//
|
||||||
// - ifc.Validator
|
// - ifc.Validator
|
||||||
//
|
//
|
||||||
@@ -126,7 +116,6 @@ import (
|
|||||||
//
|
//
|
||||||
// 2) api/internal/validate.FieldValidator
|
// 2) api/internal/validate.FieldValidator
|
||||||
//
|
//
|
||||||
// See TODO inside the validator for status.
|
|
||||||
// At time of writing, this is a do-nothing
|
// At time of writing, this is a do-nothing
|
||||||
// validator as it's not critical to kustomize function.
|
// validator as it's not critical to kustomize function.
|
||||||
//
|
//
|
||||||
@@ -149,20 +138,18 @@ import (
|
|||||||
// If you're reading this, plan not done.
|
// If you're reading this, plan not done.
|
||||||
//
|
//
|
||||||
type DepProvider struct {
|
type DepProvider struct {
|
||||||
kFactory ifc.KunstructuredFactory
|
resourceFactory *resource.Factory
|
||||||
resourceFactory *resource.Factory
|
merginator resmap.Merginator
|
||||||
conflictDectectorFactory resource.ConflictDetectorFactory
|
fieldValidator ifc.Validator
|
||||||
fieldValidator ifc.Validator
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeK8sdepBasedInstances() *DepProvider {
|
func makeK8sdepBasedInstances() *DepProvider {
|
||||||
kf := kunstruct.NewKunstructuredFactoryImpl()
|
kf := kunstruct.NewKunstructuredFactoryImpl()
|
||||||
rf := resource.NewFactory(kf)
|
rf := resource.NewFactory(kf)
|
||||||
return &DepProvider{
|
return &DepProvider{
|
||||||
kFactory: kf,
|
resourceFactory: rf,
|
||||||
resourceFactory: rf,
|
merginator: merge.NewMerginator(rf),
|
||||||
conflictDectectorFactory: k8sconflict.NewFactory(rf),
|
fieldValidator: validator.NewKustValidator(),
|
||||||
fieldValidator: validator.NewKustValidator(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -170,10 +157,9 @@ func makeKyamlBasedInstances() *DepProvider {
|
|||||||
kf := &wrappy.WNodeFactory{}
|
kf := &wrappy.WNodeFactory{}
|
||||||
rf := resource.NewFactory(kf)
|
rf := resource.NewFactory(kf)
|
||||||
return &DepProvider{
|
return &DepProvider{
|
||||||
kFactory: kf,
|
resourceFactory: rf,
|
||||||
resourceFactory: rf,
|
merginator: kmerge.NewMerginator(rf),
|
||||||
conflictDectectorFactory: conflict.NewFactory(),
|
fieldValidator: validate.NewFieldValidator(),
|
||||||
fieldValidator: validate.NewFieldValidator(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -184,20 +170,12 @@ func NewDepProvider(useKyaml bool) *DepProvider {
|
|||||||
return makeK8sdepBasedInstances()
|
return makeK8sdepBasedInstances()
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewDefaultDepProvider() *DepProvider {
|
|
||||||
return NewDepProvider(konfig.FlagEnableKyamlDefaultValue)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dp *DepProvider) GetKunstructuredFactory() ifc.KunstructuredFactory {
|
|
||||||
return dp.kFactory
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dp *DepProvider) GetResourceFactory() *resource.Factory {
|
func (dp *DepProvider) GetResourceFactory() *resource.Factory {
|
||||||
return dp.resourceFactory
|
return dp.resourceFactory
|
||||||
}
|
}
|
||||||
|
|
||||||
func (dp *DepProvider) GetConflictDetectorFactory() resource.ConflictDetectorFactory {
|
func (dp *DepProvider) GetMerginator() resmap.Merginator {
|
||||||
return dp.conflictDectectorFactory
|
return dp.merginator
|
||||||
}
|
}
|
||||||
|
|
||||||
func (dp *DepProvider) GetFieldValidator() ifc.Validator {
|
func (dp *DepProvider) GetFieldValidator() ifc.Validator {
|
||||||
@@ -1,283 +0,0 @@
|
|||||||
// Copyright 2021 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package krusty_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
kusttest_test "sigs.k8s.io/kustomize/api/testutils/kusttest"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestIssue3377(t *testing.T) {
|
|
||||||
th := kusttest_test.MakeHarness(t)
|
|
||||||
opts := th.MakeDefaultOptions()
|
|
||||||
th.WriteK(".", `
|
|
||||||
resources:
|
|
||||||
- service-a.yaml
|
|
||||||
- service-b.yaml
|
|
||||||
|
|
||||||
patchesJson6902:
|
|
||||||
- path: service-a-patch.yaml
|
|
||||||
target:
|
|
||||||
version: v1
|
|
||||||
group: networking.k8s.io
|
|
||||||
kind: Ingress
|
|
||||||
name: service-a
|
|
||||||
- path: service-b-patch.yaml
|
|
||||||
target:
|
|
||||||
version: v1
|
|
||||||
group: networking.k8s.io
|
|
||||||
kind: Ingress
|
|
||||||
name: service-b
|
|
||||||
vars:
|
|
||||||
- name: SERVICE_A_DNS_NAME
|
|
||||||
objref:
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
name: service-a
|
|
||||||
fieldref:
|
|
||||||
fieldpath: spec.rules[0].host
|
|
||||||
- name: SERVICE_B_DNS_NAME
|
|
||||||
objref:
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
name: service-b
|
|
||||||
fieldref:
|
|
||||||
fieldpath: spec.rules[0].host
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("service-a.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: service-a
|
|
||||||
spec:
|
|
||||||
replicas: 1
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: repository/service-a:v1
|
|
||||||
imagePullPolicy: Always
|
|
||||||
name: service-b
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
env:
|
|
||||||
- name: REDIRECT_URI
|
|
||||||
value: "http://$(SERVICE_B_DNS_NAME):80/oauth_redir"
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: service-a
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
spec:
|
|
||||||
type: LoadBalancer
|
|
||||||
ports:
|
|
||||||
- port: 8080
|
|
||||||
selector:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
---
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: service-a
|
|
||||||
spec:
|
|
||||||
rules:
|
|
||||||
- host: service-a.k8s.com
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- path: /
|
|
||||||
pathType: Prefix
|
|
||||||
backend:
|
|
||||||
service:
|
|
||||||
name: service-a
|
|
||||||
port:
|
|
||||||
number: 8080
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("service-a-patch.yaml", `
|
|
||||||
- op: replace
|
|
||||||
path: /spec/rules/0/host
|
|
||||||
value: new-service-a.k8s.com
|
|
||||||
`)
|
|
||||||
|
|
||||||
th.WriteF("service-b.yaml", `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: service-b
|
|
||||||
spec:
|
|
||||||
replicas: 1
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: repository/service-b:v1
|
|
||||||
imagePullPolicy: Always
|
|
||||||
name: service-b
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
env:
|
|
||||||
- name: REDIRECT_URI
|
|
||||||
value: "http://$(SERVICE_A_DNS_NAME):80/oauth_redir"
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: service-b
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
spec:
|
|
||||||
type: LoadBalancer
|
|
||||||
ports:
|
|
||||||
- port: 8080
|
|
||||||
selector:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
---
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: service-b
|
|
||||||
spec:
|
|
||||||
rules:
|
|
||||||
- host: service-b.k8s.com
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- path: /
|
|
||||||
pathType: Prefix
|
|
||||||
backend:
|
|
||||||
service:
|
|
||||||
name: service-b
|
|
||||||
port:
|
|
||||||
number: 8080
|
|
||||||
`)
|
|
||||||
th.WriteF("service-b-patch.yaml", `
|
|
||||||
- op: replace
|
|
||||||
path: /spec/rules/0/host
|
|
||||||
value: new-service-b.k8s.com
|
|
||||||
`)
|
|
||||||
m := th.Run(".", opts)
|
|
||||||
th.AssertActualEqualsExpected(
|
|
||||||
m, `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: service-a
|
|
||||||
spec:
|
|
||||||
replicas: 1
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- env:
|
|
||||||
- name: REDIRECT_URI
|
|
||||||
value: http://new-service-b.k8s.com:80/oauth_redir
|
|
||||||
image: repository/service-a:v1
|
|
||||||
imagePullPolicy: Always
|
|
||||||
name: service-b
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
name: service-a
|
|
||||||
spec:
|
|
||||||
ports:
|
|
||||||
- port: 8080
|
|
||||||
selector:
|
|
||||||
app.kubernetes.io/component: service-a
|
|
||||||
type: LoadBalancer
|
|
||||||
---
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: service-a
|
|
||||||
spec:
|
|
||||||
rules:
|
|
||||||
- host: new-service-a.k8s.com
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- backend:
|
|
||||||
service:
|
|
||||||
name: service-a
|
|
||||||
port:
|
|
||||||
number: 8080
|
|
||||||
path: /
|
|
||||||
pathType: Prefix
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: service-b
|
|
||||||
spec:
|
|
||||||
replicas: 1
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- env:
|
|
||||||
- name: REDIRECT_URI
|
|
||||||
value: http://new-service-a.k8s.com:80/oauth_redir
|
|
||||||
image: repository/service-b:v1
|
|
||||||
imagePullPolicy: Always
|
|
||||||
name: service-b
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
name: service-b
|
|
||||||
spec:
|
|
||||||
ports:
|
|
||||||
- port: 8080
|
|
||||||
selector:
|
|
||||||
app.kubernetes.io/component: service-b
|
|
||||||
type: LoadBalancer
|
|
||||||
---
|
|
||||||
apiVersion: networking.k8s.io/v1
|
|
||||||
kind: Ingress
|
|
||||||
metadata:
|
|
||||||
name: service-b
|
|
||||||
spec:
|
|
||||||
rules:
|
|
||||||
- host: new-service-b.k8s.com
|
|
||||||
http:
|
|
||||||
paths:
|
|
||||||
- backend:
|
|
||||||
service:
|
|
||||||
name: service-b
|
|
||||||
port:
|
|
||||||
number: 8080
|
|
||||||
path: /
|
|
||||||
pathType: Prefix
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
@@ -11,12 +11,11 @@ import (
|
|||||||
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
||||||
"sigs.k8s.io/kustomize/api/internal/target"
|
"sigs.k8s.io/kustomize/api/internal/target"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
|
"sigs.k8s.io/kustomize/api/krusty/internal/provider"
|
||||||
fLdr "sigs.k8s.io/kustomize/api/loader"
|
fLdr "sigs.k8s.io/kustomize/api/loader"
|
||||||
"sigs.k8s.io/kustomize/api/provenance"
|
"sigs.k8s.io/kustomize/api/provenance"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kyaml/openapi"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Kustomizer performs kustomizations. It's meant to behave
|
// Kustomizer performs kustomizations. It's meant to behave
|
||||||
@@ -54,7 +53,7 @@ func MakeKustomizer(fSys filesys.FileSystem, o *Options) *Kustomizer {
|
|||||||
func (b *Kustomizer) Run(path string) (resmap.ResMap, error) {
|
func (b *Kustomizer) Run(path string) (resmap.ResMap, error) {
|
||||||
resmapFactory := resmap.NewFactory(
|
resmapFactory := resmap.NewFactory(
|
||||||
b.depProvider.GetResourceFactory(),
|
b.depProvider.GetResourceFactory(),
|
||||||
b.depProvider.GetConflictDetectorFactory())
|
b.depProvider.GetMerginator())
|
||||||
lr := fLdr.RestrictionNone
|
lr := fLdr.RestrictionNone
|
||||||
if b.options.LoadRestrictions == types.LoadRestrictionsRootOnly {
|
if b.options.LoadRestrictions == types.LoadRestrictionsRootOnly {
|
||||||
lr = fLdr.RestrictionRootOnly
|
lr = fLdr.RestrictionRootOnly
|
||||||
@@ -74,10 +73,6 @@ func (b *Kustomizer) Run(path string) (resmap.ResMap, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
err = openapi.SetSchemaVersion(kt.Kustomization().OpenAPI, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var m resmap.ResMap
|
var m resmap.ResMap
|
||||||
m, err = kt.MakeCustomizedResMap()
|
m, err = kt.MakeCustomizedResMap()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -90,7 +85,7 @@ func (b *Kustomizer) Run(path string) (resmap.ResMap, error) {
|
|||||||
t := builtins.LabelTransformerPlugin{
|
t := builtins.LabelTransformerPlugin{
|
||||||
Labels: map[string]string{
|
Labels: map[string]string{
|
||||||
konfig.ManagedbyLabelKey: fmt.Sprintf(
|
konfig.ManagedbyLabelKey: fmt.Sprintf(
|
||||||
"kustomize-%s", provenance.GetProvenance().Semver())},
|
"kustomize-%s", provenance.GetProvenance().Version)},
|
||||||
FieldSpecs: []types.FieldSpec{{
|
FieldSpecs: []types.FieldSpec{{
|
||||||
Path: "metadata/labels",
|
Path: "metadata/labels",
|
||||||
CreateIfNotPresent: true,
|
CreateIfNotPresent: true,
|
||||||
@@ -98,6 +93,5 @@ func (b *Kustomizer) Run(path string) (resmap.ResMap, error) {
|
|||||||
}
|
}
|
||||||
t.Transform(m)
|
t.Transform(m)
|
||||||
}
|
}
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user