mirror of https://github.com/fluxcd/flux2.git
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
269 lines
6.8 KiB
Go
269 lines
6.8 KiB
Go
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
"text/template"
|
|
"time"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/mattn/go-shellwords"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
k8syaml "k8s.io/apimachinery/pkg/util/yaml"
|
|
"k8s.io/client-go/tools/clientcmd"
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
"sigs.k8s.io/controller-runtime/pkg/envtest"
|
|
)
|
|
|
|
func readYamlObjects(objectFile string) ([]unstructured.Unstructured, error) {
|
|
obj, err := os.ReadFile(objectFile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
objects := []unstructured.Unstructured{}
|
|
reader := k8syaml.NewYAMLReader(bufio.NewReader(bytes.NewReader(obj)))
|
|
for {
|
|
doc, err := reader.Read()
|
|
if err != nil {
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
}
|
|
unstructuredObj := &unstructured.Unstructured{}
|
|
decoder := k8syaml.NewYAMLOrJSONDecoder(bytes.NewBuffer(doc), len(doc))
|
|
err = decoder.Decode(unstructuredObj)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
objects = append(objects, *unstructuredObj)
|
|
}
|
|
return objects, nil
|
|
}
|
|
|
|
// A KubeManager that can create objects that are subject to a test.
|
|
type testEnvKubeManager struct {
|
|
client client.WithWatch
|
|
testEnv *envtest.Environment
|
|
}
|
|
|
|
func (m *testEnvKubeManager) NewClient(kubeconfig string, kubecontext string) (client.WithWatch, error) {
|
|
return m.client, nil
|
|
}
|
|
|
|
func (m *testEnvKubeManager) CreateObjects(clientObjects []unstructured.Unstructured) error {
|
|
for _, obj := range clientObjects {
|
|
// First create the object then set its status if present in the
|
|
// yaml file. Make a copy first since creating an object may overwrite
|
|
// the status.
|
|
createObj := obj.DeepCopy()
|
|
err := m.client.Create(context.Background(), createObj)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
obj.SetResourceVersion(createObj.GetResourceVersion())
|
|
err = m.client.Status().Update(context.Background(), &obj)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *testEnvKubeManager) Stop() error {
|
|
if m.testEnv == nil {
|
|
return fmt.Errorf("do nothing because testEnv is nil")
|
|
}
|
|
return m.testEnv.Stop()
|
|
}
|
|
|
|
func NewTestEnvKubeManager(testClusterMode TestClusterMode) (*testEnvKubeManager, error) {
|
|
switch testClusterMode {
|
|
case TestEnvClusterMode:
|
|
useExistingCluster := false
|
|
testEnv := &envtest.Environment{
|
|
UseExistingCluster: &useExistingCluster,
|
|
CRDDirectoryPaths: []string{"manifests"},
|
|
}
|
|
cfg, err := testEnv.Start()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
user, err := testEnv.ControlPlane.AddUser(envtest.User{
|
|
Name: "envtest-admin",
|
|
Groups: []string{"system:masters"},
|
|
}, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
kubeConfig, err := user.KubeConfig()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmpFilename := filepath.Join("/tmp", "kubeconfig-"+time.Nanosecond.String())
|
|
ioutil.WriteFile(tmpFilename, kubeConfig, 0644)
|
|
rootArgs.kubeconfig = tmpFilename
|
|
k8sClient, err := client.NewWithWatch(cfg, client.Options{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &testEnvKubeManager{
|
|
testEnv: testEnv,
|
|
client: k8sClient,
|
|
}, nil
|
|
case ExistingClusterMode:
|
|
// TEST_KUBECONFIG is mandatory to prevent destroying a current cluster accidentally.
|
|
testKubeConfig := os.Getenv("TEST_KUBECONFIG")
|
|
if testKubeConfig == "" {
|
|
return nil, fmt.Errorf("environment variable TEST_KUBECONFIG is required to run tests against an existing cluster")
|
|
}
|
|
rootArgs.kubeconfig = testKubeConfig
|
|
|
|
useExistingCluster := true
|
|
config, err := clientcmd.BuildConfigFromFlags("", testKubeConfig)
|
|
testEnv := &envtest.Environment{
|
|
UseExistingCluster: &useExistingCluster,
|
|
Config: config,
|
|
}
|
|
cfg, err := testEnv.Start()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
k8sClient, err := client.NewWithWatch(cfg, client.Options{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &testEnvKubeManager{
|
|
testEnv: testEnv,
|
|
client: k8sClient,
|
|
}, nil
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
type TestClusterMode int
|
|
|
|
const (
|
|
TestEnvClusterMode = TestClusterMode(iota + 1)
|
|
ExistingClusterMode
|
|
)
|
|
|
|
// Structure used for each test to load objects into kubernetes, run
|
|
// commands and assert on the expected output.
|
|
type cmdTestCase struct {
|
|
// The command line arguments to test.
|
|
args string
|
|
// When true, the test expects the command to fail.
|
|
wantError bool
|
|
// String literal that contains the expected test output.
|
|
goldenValue string
|
|
// Filename that contains the expected test output.
|
|
goldenFile string
|
|
// Filename that contains yaml objects to load into Kubernetes
|
|
objectFile string
|
|
// TestClusterMode to bootstrap and testing, default to Fake
|
|
testClusterMode TestClusterMode
|
|
// TemplateValues enable template preprocessing for the golden file, or golden value
|
|
templateValues map[string]string
|
|
}
|
|
|
|
func (cmd *cmdTestCase) runTestCmd(t *testing.T) {
|
|
km, err := NewTestEnvKubeManager(cmd.testClusterMode)
|
|
if err != nil {
|
|
t.Fatalf("Error creating kube manager: '%v'", err)
|
|
}
|
|
|
|
if km != nil {
|
|
defer km.Stop()
|
|
}
|
|
|
|
if cmd.objectFile != "" {
|
|
clientObjects, err := readYamlObjects(cmd.objectFile)
|
|
if err != nil {
|
|
t.Fatalf("Error loading yaml: '%v'", err)
|
|
}
|
|
err = km.CreateObjects(clientObjects)
|
|
if err != nil {
|
|
t.Fatalf("Error creating test objects: '%v'", err)
|
|
}
|
|
}
|
|
|
|
actual, err := executeCommand(cmd.args)
|
|
if (err != nil) != cmd.wantError {
|
|
t.Fatalf("Expected error='%v', Got: %v", cmd.wantError, err)
|
|
}
|
|
if err != nil {
|
|
actual = err.Error()
|
|
}
|
|
|
|
var expected string
|
|
if cmd.goldenValue != "" {
|
|
if cmd.templateValues != nil {
|
|
expected, err = executeGoldenTemplate(cmd.goldenValue, cmd.templateValues)
|
|
if err != nil {
|
|
t.Fatalf("Error executing golden template: %s", err)
|
|
}
|
|
} else {
|
|
expected = cmd.goldenValue
|
|
}
|
|
}
|
|
if cmd.goldenFile != "" {
|
|
goldenFileContent, err := os.ReadFile(cmd.goldenFile)
|
|
if err != nil {
|
|
t.Fatalf("Error reading golden file: '%s'", err)
|
|
}
|
|
if cmd.templateValues != nil {
|
|
expected, err = executeGoldenTemplate(string(goldenFileContent), cmd.templateValues)
|
|
if err != nil {
|
|
t.Fatalf("Error executing golden template file '%s': %s", cmd.goldenFile, err)
|
|
}
|
|
} else {
|
|
expected = string(goldenFileContent)
|
|
}
|
|
}
|
|
|
|
diff := cmp.Diff(expected, actual)
|
|
if diff != "" {
|
|
t.Errorf("Mismatch from '%s' (-want +got):\n%s", cmd.goldenFile, diff)
|
|
}
|
|
}
|
|
|
|
func executeGoldenTemplate(goldenValue string, templateValues map[string]string) (string, error) {
|
|
tmpl := template.Must(template.New("golden").Parse(goldenValue))
|
|
var out bytes.Buffer
|
|
if err := tmpl.Execute(&out, templateValues); err != nil {
|
|
return "", err
|
|
}
|
|
return out.String(), nil
|
|
}
|
|
|
|
// Run the command and return the captured output.
|
|
func executeCommand(cmd string) (string, error) {
|
|
args, err := shellwords.Parse(cmd)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
rootCmd.SetOut(buf)
|
|
rootCmd.SetErr(buf)
|
|
rootCmd.SetArgs(args)
|
|
|
|
logger.stderr = rootCmd.ErrOrStderr()
|
|
|
|
_, err = rootCmd.ExecuteC()
|
|
result := buf.String()
|
|
|
|
return result, err
|
|
}
|