Files
argo-cd/test/e2e/fixture/applicationsets/expectation.go
2026-01-12 16:35:49 +02:00

259 lines
8.9 KiB
Go

package applicationsets
import (
"fmt"
"reflect"
"strings"
"testing"
"github.com/argoproj/gitops-engine/pkg/diff"
"github.com/argoproj/gitops-engine/pkg/health"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"github.com/argoproj/argo-cd/v3/pkg/apis/application/v1alpha1"
"github.com/argoproj/argo-cd/v3/test/e2e/fixture/applicationsets/utils"
)
type state = string
const (
failed = "failed"
pending = "pending"
succeeded = "succeeded"
)
// Expectation returns succeeded on succes condition, or pending/failed on failure, along with
// a message to describe the success/failure condition.
type Expectation func(c *Consequences) (state state, message string)
// Success asserts that the last command was successful
func Success(message string) Expectation {
return func(c *Consequences) (state, string) {
if c.actions.lastError != nil {
return failed, fmt.Sprintf("error: %v", c.actions.lastError)
}
if !strings.Contains(c.actions.lastOutput, message) {
return failed, fmt.Sprintf("output did not contain '%s'", message)
}
return succeeded, fmt.Sprintf("no error and output contained '%s'", message)
}
}
// OutputContains asserts that the last command output contains the expected substring
func OutputContains(expected string) Expectation {
return func(c *Consequences) (state, string) {
if c.actions.lastError != nil {
return failed, fmt.Sprintf("error: %v", c.actions.lastError)
}
if !strings.Contains(c.actions.lastOutput, expected) {
return failed, fmt.Sprintf("output did not contain '%s', got: %s", expected, c.actions.lastOutput)
}
return succeeded, fmt.Sprintf("output contained '%s'", expected)
}
}
// Error asserts that the last command was an error with substring match
func Error(message, err string) Expectation {
return func(c *Consequences) (state, string) {
if c.actions.lastError == nil {
return failed, "no error"
}
if !strings.Contains(c.actions.lastOutput, message) {
return failed, fmt.Sprintf("output does not contain '%s'", message)
}
if !strings.Contains(c.actions.lastError.Error(), err) {
return failed, fmt.Sprintf("error does not contain '%s'", message)
}
return succeeded, fmt.Sprintf("error '%s'", message)
}
}
// ApplicationsExist checks whether each of the 'expectedApps' exist in the namespace, and are
// equivalent to provided values.
func ApplicationsExist(expectedApps []v1alpha1.Application) Expectation {
return func(c *Consequences) (state, string) {
for _, expectedApp := range expectedApps {
foundApp := c.app(expectedApp.Name)
if foundApp == nil {
return pending, fmt.Sprintf("missing app '%s'", expectedApp.QualifiedName())
}
if !appsAreEqual(expectedApp, *foundApp) {
diff, err := getDiff(filterFields(expectedApp), filterFields(*foundApp))
if err != nil {
return failed, err.Error()
}
return pending, fmt.Sprintf("apps are not equal: '%s', diff: %s\n", expectedApp.QualifiedName(), diff)
}
}
return succeeded, "all apps successfully found"
}
}
// ApplicationSetHasHealthStatus checks whether the ApplicationSet has the expected health status.
func ApplicationSetHasHealthStatus(applicationSetName string, expectedHealthStatus health.HealthStatusCode) Expectation {
return func(c *Consequences) (state, string) {
// retrieve the application set
foundApplicationSet := c.applicationSet(applicationSetName)
if foundApplicationSet == nil {
return pending, fmt.Sprintf("application set '%s' not found", applicationSetName)
}
if foundApplicationSet.Status.Health.Status != expectedHealthStatus {
return pending, fmt.Sprintf("application set health status is '%s', expected '%s'",
foundApplicationSet.Status.Health.Status, expectedHealthStatus)
}
return succeeded, fmt.Sprintf("application set has expected health status '%s'", expectedHealthStatus)
}
}
// ApplicationSetHasConditions checks whether each of the 'expectedConditions' exist in the ApplicationSet status, and are
// equivalent to provided values.
func ApplicationSetHasConditions(expectedConditions []v1alpha1.ApplicationSetCondition) Expectation {
return func(c *Consequences) (state, string) {
// retrieve the application set
foundApplicationSet := c.applicationSet(c.context.GetName())
if foundApplicationSet == nil {
return pending, fmt.Sprintf("application set '%s' not found", c.context.GetName())
}
if !conditionsAreEqual(&expectedConditions, &foundApplicationSet.Status.Conditions) {
diff, err := getConditionDiff(expectedConditions, foundApplicationSet.Status.Conditions)
if err != nil {
return failed, err.Error()
}
return pending, fmt.Sprintf("application set conditions are not equal: '%s', diff: %s\n", expectedConditions, diff)
}
return succeeded, "application set successfully found"
}
}
// ApplicationsDoNotExist checks that each of the 'expectedApps' no longer exist in the namespace
func ApplicationsDoNotExist(expectedApps []v1alpha1.Application) Expectation {
return func(c *Consequences) (state, string) {
for _, expectedApp := range expectedApps {
foundApp := c.app(expectedApp.Name)
if foundApp != nil {
return pending, fmt.Sprintf("app '%s' should no longer exist", expectedApp.QualifiedName())
}
}
return succeeded, "all apps do not exist"
}
}
// Pod checks whether a specified condition is true for any of the pods in the namespace
func Pod(t *testing.T, predicate func(p corev1.Pod) bool) Expectation {
t.Helper()
return func(_ *Consequences) (state, string) {
pods, err := pods(t, utils.ApplicationsResourcesNamespace)
if err != nil {
return failed, err.Error()
}
for _, pod := range pods.Items {
if predicate(pod) {
return succeeded, fmt.Sprintf("pod predicate matched pod named '%s'", pod.GetName())
}
}
return pending, "pod predicate does not match pods"
}
}
func pods(t *testing.T, namespace string) (*corev1.PodList, error) {
t.Helper()
fixtureClient := utils.GetE2EFixtureK8sClient(t)
pods, err := fixtureClient.KubeClientset.CoreV1().Pods(namespace).List(t.Context(), metav1.ListOptions{})
return pods, err
}
// getDiff returns a string containing a comparison result of two applications (for test output/debug purposes)
func getDiff(orig, newApplication v1alpha1.Application) (string, error) {
bytes, _, err := diff.CreateTwoWayMergePatch(orig, newApplication, orig)
if err != nil {
return "", err
}
return string(bytes), nil
}
// getConditionDiff returns a string containing a comparison result of two ApplicationSetCondition (for test output/debug purposes)
func getConditionDiff(orig, newApplicationSetCondition []v1alpha1.ApplicationSetCondition) (string, error) {
if len(orig) != len(newApplicationSetCondition) {
return fmt.Sprintf("mismatch between condition sizes: %v %v", len(orig), len(newApplicationSetCondition)), nil
}
var bytes []byte
for index := range orig {
b, _, err := diff.CreateTwoWayMergePatch(orig[index], newApplicationSetCondition[index], orig[index])
if err != nil {
return "", err
}
bytes = append(bytes, b...)
}
return string(bytes), nil
}
// filterFields returns a copy of Application, but with unnecessary (for testing) fields removed
func filterFields(input v1alpha1.Application) v1alpha1.Application {
spec := input.Spec
metaCopy := input.ObjectMeta.DeepCopy()
output := v1alpha1.Application{
ObjectMeta: metav1.ObjectMeta{
Labels: metaCopy.Labels,
Annotations: metaCopy.Annotations,
Name: metaCopy.Name,
Namespace: metaCopy.Namespace,
Finalizers: metaCopy.Finalizers,
},
Spec: v1alpha1.ApplicationSpec{
Source: &v1alpha1.ApplicationSource{
Path: spec.GetSource().Path,
RepoURL: spec.GetSource().RepoURL,
TargetRevision: spec.GetSource().TargetRevision,
},
Destination: v1alpha1.ApplicationDestination{
Server: spec.Destination.Server,
Name: spec.Destination.Name,
Namespace: spec.Destination.Namespace,
},
Project: spec.Project,
},
}
return output
}
// filterConditionFields returns a copy of ApplicationSetCondition, but with unnecessary (for testing) fields removed
func filterConditionFields(input *[]v1alpha1.ApplicationSetCondition) *[]v1alpha1.ApplicationSetCondition {
var filteredConditions []v1alpha1.ApplicationSetCondition
for _, condition := range *input {
newCondition := &v1alpha1.ApplicationSetCondition{
Type: condition.Type,
Status: condition.Status,
Message: condition.Message,
Reason: condition.Reason,
}
filteredConditions = append(filteredConditions, *newCondition)
}
return &filteredConditions
}
// appsAreEqual returns true if the apps are equal, comparing only fields of interest
func appsAreEqual(one v1alpha1.Application, two v1alpha1.Application) bool {
return reflect.DeepEqual(filterFields(one), filterFields(two))
}
// conditionsAreEqual returns true if the appset status conditions are equal, comparing only fields of interest
func conditionsAreEqual(one, two *[]v1alpha1.ApplicationSetCondition) bool {
return reflect.DeepEqual(filterConditionFields(one), filterConditionFields(two))
}