/* Copyright 2019 The Kubernetes Authors. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package plugins import ( "fmt" "strings" "testing" "sigs.k8s.io/kustomize/v3/internal/loadertest" "sigs.k8s.io/kustomize/v3/k8sdeps/kunstruct" "sigs.k8s.io/kustomize/v3/pkg/resmap" "sigs.k8s.io/kustomize/v3/pkg/resource" "sigs.k8s.io/kustomize/v3/pkg/types" ) func TestExecPluginConfig(t *testing.T) { path := "/app" rf := resmap.NewFactory( resource.NewFactory( kunstruct.NewKunstructuredFactoryImpl()), nil) ldr := loadertest.NewFakeLoader(path) pluginConfig := rf.RF().FromMap( map[string]interface{}{ "apiVersion": "someteam.example.com/v1", "kind": "SedTransformer", "metadata": map[string]interface{}{ "name": "some-random-name", }, "argsOneLiner": "one two", "argsFromFile": "sed-input.txt", }) ldr.AddFile("/app/sed-input.txt", []byte(` s/$FOO/foo/g s/$BAR/bar/g \ \ \ `)) p := NewExecPlugin( AbsolutePluginPath( DefaultPluginConfig(), pluginConfig.OrgId())) yaml, err := pluginConfig.AsYAML() if err != nil { t.Fatalf("unexpected err: %v", err) } p.Config(ldr, rf, yaml) expected := "/kustomize/plugin/someteam.example.com/v1/sedtransformer/SedTransformer" if !strings.HasSuffix(p.path, expected) { t.Fatalf("expected suffix '%s', got '%s'", expected, p.path) } expected = `apiVersion: someteam.example.com/v1 argsFromFile: sed-input.txt argsOneLiner: one two kind: SedTransformer metadata: name: some-random-name ` if expected != string(p.cfg) { t.Fatalf("expected cfg '%s', got '%s'", expected, string(p.cfg)) } if len(p.args) != 5 { t.Fatalf("unexpected arg len %d, %v", len(p.args), p.args) } if p.args[0] != "one" || p.args[1] != "two" || p.args[2] != "s/$FOO/foo/g" || p.args[3] != "s/$BAR/bar/g" || p.args[4] != "\\ \\ \\ " { t.Fatalf("unexpected arg array: %v", p.args) } } func makeConfigMap(rf *resource.Factory, name, behavior string, hashValue *string) *resource.Resource { r := rf.FromMap(map[string]interface{}{ "apiVersion": "v1", "kind": "ConfigMap", "metadata": map[string]interface{}{"name": name}, }) annotations := map[string]string{} if behavior != "" { annotations[behaviorAnnotation] = behavior } if hashValue != nil { annotations[hashAnnotation] = *hashValue } if len(annotations) > 0 { r.SetAnnotations(annotations) } return r } func makeConfigMapOptions(rf *resource.Factory, name, behavior string, disableHash bool) *resource.Resource { return rf.FromMapAndOption(map[string]interface{}{ "apiVersion": "v1", "kind": "ConfigMap", "metadata": map[string]interface{}{"name": name}, }, &types.GeneratorArgs{Behavior: behavior}, &types.GeneratorOptions{DisableNameSuffixHash: disableHash}) } func strptr(s string) *string { return &s } func TestUpdateResourceOptions(t *testing.T) { p := NewExecPlugin("") rf := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl()) in := resmap.New() expected := resmap.New() cases := []struct { behavior string needsHash bool hashValue *string }{ {hashValue: strptr("false")}, {hashValue: strptr("true"), needsHash: true}, {behavior: "replace"}, {behavior: "merge"}, {behavior: "create"}, {behavior: "nonsense"}, {behavior: "merge", hashValue: strptr("false")}, {behavior: "merge", hashValue: strptr("true"), needsHash: true}, } for i, c := range cases { name := fmt.Sprintf("test%d", i) in.Append(makeConfigMap(rf, name, c.behavior, c.hashValue)) expected.Append(makeConfigMapOptions(rf, name, c.behavior, !c.needsHash)) } actual, err := p.updateResourceOptions(in) if err != nil { 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) { p := NewExecPlugin("") rf := resource.NewFactory(kunstruct.NewKunstructuredFactoryImpl()) cases := []string{ "", "FaLsE", "TrUe", "potato", } for i, c := range cases { name := fmt.Sprintf("test%d", i) in := resmap.New() in.Append(makeConfigMap(rf, name, "", &c)) _, err := p.updateResourceOptions(in) if err == nil { t.Errorf("expected error from value %q", c) } } }