mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-06-14 10:30:59 +00:00
[doc]: https://github.com/golang/go/wiki/Modules#releasing-modules-v2-or-higher Per this Go modules [doc] a repo or branch that's already tagged v2 or higher should increment the major version (e.g. go to v3) when releasing their first Go module-based packages. At the moment, the kustomize repo has these top level packages in the sigs.k8s.io/kustomize module: - `cmd` - holds main program for kustomize Conceivably someone can depend on this package for integration tests. - `internal` - intentionally unreleased subpackages - `k8sdeps` - an adapter wrapping k8s dependencies This exists only for use in pre-Go-modules kustomize-into-kubectl integration and won't live much longer (as everything involved is switching to Go modules). - `pkg` - kustomize packages for export This should shrink in later versions, since the surface area is too large, containing sub-packages that should be in 'internal'. - `plugin` - holds main programs for plugins This PR changes the top level go.mod file from ``` module sigs.k8s.io/kustomize ``` to ``` module sigs.k8s.io/kustomize/v3 ``` and adjusts all import statements to reflect the change.
337 lines
7.8 KiB
Go
337 lines
7.8 KiB
Go
// Copyright 2019 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package accumulator_test
|
|
|
|
import (
|
|
"bytes"
|
|
"log"
|
|
"os"
|
|
"strings"
|
|
"testing"
|
|
|
|
"sigs.k8s.io/kustomize/v3/k8sdeps/kunstruct"
|
|
. "sigs.k8s.io/kustomize/v3/pkg/accumulator"
|
|
"sigs.k8s.io/kustomize/v3/pkg/gvk"
|
|
"sigs.k8s.io/kustomize/v3/pkg/resmap"
|
|
"sigs.k8s.io/kustomize/v3/pkg/resmaptest"
|
|
"sigs.k8s.io/kustomize/v3/pkg/resource"
|
|
"sigs.k8s.io/kustomize/v3/pkg/transformers/config"
|
|
"sigs.k8s.io/kustomize/v3/pkg/types"
|
|
)
|
|
|
|
func makeResAccumulator(t *testing.T) (*ResAccumulator, *resource.Factory) {
|
|
ra := MakeEmptyAccumulator()
|
|
err := ra.MergeConfig(config.MakeDefaultConfig())
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
rf := resource.NewFactory(
|
|
kunstruct.NewKunstructuredFactoryImpl())
|
|
err = ra.AppendAll(
|
|
resmaptest_test.NewRmBuilder(t, rf).
|
|
Add(map[string]interface{}{
|
|
"apiVersion": "apps/v1",
|
|
"kind": "Deployment",
|
|
"metadata": map[string]interface{}{
|
|
"name": "deploy1",
|
|
},
|
|
"spec": map[string]interface{}{
|
|
"template": map[string]interface{}{
|
|
"spec": map[string]interface{}{
|
|
"containers": []interface{}{
|
|
map[string]interface{}{
|
|
"command": []interface{}{
|
|
"myserver",
|
|
"--somebackendService $(SERVICE_ONE)",
|
|
"--yetAnother $(SERVICE_TWO)",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}}).
|
|
Add(map[string]interface{}{
|
|
"apiVersion": "v1",
|
|
"kind": "Service",
|
|
"metadata": map[string]interface{}{
|
|
"name": "backendOne",
|
|
}}).
|
|
Add(map[string]interface{}{
|
|
"apiVersion": "v1",
|
|
"kind": "Service",
|
|
"metadata": map[string]interface{}{
|
|
"name": "backendTwo",
|
|
}}).ResMap())
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
return ra, rf
|
|
}
|
|
|
|
func TestResolveVarsHappy(t *testing.T) {
|
|
ra, _ := makeResAccumulator(t)
|
|
err := ra.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne"},
|
|
},
|
|
{
|
|
Name: "SERVICE_TWO",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendTwo"},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
err = ra.ResolveVars()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
c := getCommand(find("deploy1", ra.ResMap()))
|
|
if c != "myserver --somebackendService backendOne --yetAnother backendTwo" {
|
|
t.Fatalf("unexpected command: %s", c)
|
|
}
|
|
}
|
|
|
|
func TestResolveVarsOneUnused(t *testing.T) {
|
|
ra, _ := makeResAccumulator(t)
|
|
err := ra.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne"},
|
|
},
|
|
{
|
|
Name: "SERVICE_UNUSED",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendTwo"},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
var buf bytes.Buffer
|
|
log.SetOutput(&buf)
|
|
defer func() {
|
|
log.SetOutput(os.Stderr)
|
|
}()
|
|
err = ra.ResolveVars()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
expectLog(t, buf, "well-defined vars that were never replaced: SERVICE_UNUSED")
|
|
c := getCommand(find("deploy1", ra.ResMap()))
|
|
if c != "myserver --somebackendService backendOne --yetAnother $(SERVICE_TWO)" {
|
|
t.Fatalf("unexpected command: %s", c)
|
|
}
|
|
}
|
|
|
|
func expectLog(t *testing.T, log bytes.Buffer, expect string) {
|
|
if !strings.Contains(log.String(), expect) {
|
|
t.Fatalf("expected log containing '%s', got '%s'", expect, log.String())
|
|
}
|
|
}
|
|
|
|
func TestResolveVarsVarNeedsDisambiguation(t *testing.T) {
|
|
ra, rf := makeResAccumulator(t)
|
|
|
|
rm0 := resmap.New()
|
|
err := rm0.Append(
|
|
rf.FromMap(
|
|
map[string]interface{}{
|
|
"apiVersion": "v1",
|
|
"kind": "Service",
|
|
"metadata": map[string]interface{}{
|
|
"name": "backendOne",
|
|
"namespace": "fooNamespace",
|
|
},
|
|
}))
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
err = ra.AppendAll(rm0)
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
|
|
err = ra.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne",
|
|
},
|
|
},
|
|
})
|
|
if err == nil {
|
|
t.Fatalf("expected error")
|
|
}
|
|
if !strings.Contains(
|
|
err.Error(), "unable to disambiguate") {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveVarsGoodResIdBadField(t *testing.T) {
|
|
ra, _ := makeResAccumulator(t)
|
|
err := ra.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne"},
|
|
FieldRef: types.FieldSelector{FieldPath: "nope_nope_nope"},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
err = ra.ResolveVars()
|
|
if err == nil {
|
|
t.Fatalf("expected error")
|
|
}
|
|
if !strings.Contains(
|
|
err.Error(),
|
|
"not found in corresponding resource") {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveVarsUnmappableVar(t *testing.T) {
|
|
ra, _ := makeResAccumulator(t)
|
|
err := ra.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_THREE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "doesNotExist"},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
err = ra.ResolveVars()
|
|
if err == nil {
|
|
t.Fatalf("expected error")
|
|
}
|
|
if !strings.Contains(
|
|
err.Error(),
|
|
"cannot be mapped to a field in the set of known resources") {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveVarsWithNoambiguation(t *testing.T) {
|
|
ra1, rf := makeResAccumulator(t)
|
|
err := ra1.MergeVars([]types.Var{
|
|
{
|
|
Name: "SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne",
|
|
},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
|
|
// Create another accumulator having a resource with different prefix
|
|
ra2 := MakeEmptyAccumulator()
|
|
|
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
|
Add(map[string]interface{}{
|
|
"apiVersion": "apps/v1",
|
|
"kind": "Deployment",
|
|
"metadata": map[string]interface{}{
|
|
"name": "deploy2",
|
|
},
|
|
"spec": map[string]interface{}{
|
|
"template": map[string]interface{}{
|
|
"spec": map[string]interface{}{
|
|
"containers": []interface{}{
|
|
map[string]interface{}{
|
|
"command": []interface{}{
|
|
"myserver",
|
|
"--somebackendService $(SUB_SERVICE_ONE)",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}}).
|
|
Add(map[string]interface{}{
|
|
"apiVersion": "v1",
|
|
"kind": "Service",
|
|
"metadata": map[string]interface{}{
|
|
"name": "backendOne",
|
|
}}).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)
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
|
|
err = ra2.MergeVars([]types.Var{
|
|
{
|
|
Name: "SUB_SERVICE_ONE",
|
|
ObjRef: types.Target{
|
|
Gvk: gvk.Gvk{Version: "v1", Kind: "Service"},
|
|
Name: "backendOne",
|
|
},
|
|
},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
err = ra1.MergeAccumulator(ra2)
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
|
|
err = ra1.ResolveVars()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err: %v", err)
|
|
}
|
|
}
|
|
|
|
func find(name string, resMap resmap.ResMap) *resource.Resource {
|
|
for _, r := range resMap.Resources() {
|
|
if r.GetName() == name {
|
|
return r
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Assumes arg is a deployment, returns the command of first container.
|
|
func getCommand(r *resource.Resource) string {
|
|
var m map[string]interface{}
|
|
var c []interface{}
|
|
m, _ = r.Map()["spec"].(map[string]interface{})
|
|
m, _ = m["template"].(map[string]interface{})
|
|
m, _ = m["spec"].(map[string]interface{})
|
|
c, _ = m["containers"].([]interface{})
|
|
m, _ = c[0].(map[string]interface{})
|
|
|
|
cmd, _ := m["command"].([]interface{})
|
|
n := make([]string, len(cmd))
|
|
for i, v := range cmd {
|
|
n[i] = v.(string)
|
|
}
|
|
return strings.Join(n, " ")
|
|
}
|