argo-cd/test/e2e/hook_test.go
Matthieu MOREL f770a2a73a
chore: enable require-error rule from errorlint linter on test folder (#18684)
Signed-off-by: Matthieu MOREL <matthieu.morel35@gmail.com>
2024-06-15 18:54:09 +00:00

428 lines
14 KiB
Go

package e2e
import (
"context"
"fmt"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"github.com/argoproj/gitops-engine/pkg/health"
. "github.com/argoproj/gitops-engine/pkg/sync/common"
. "github.com/argoproj/argo-cd/v2/pkg/apis/application/v1alpha1"
. "github.com/argoproj/argo-cd/v2/test/e2e/fixture"
. "github.com/argoproj/argo-cd/v2/test/e2e/fixture/app"
. "github.com/argoproj/argo-cd/v2/util/errors"
)
func TestPreSyncHookSuccessful(t *testing.T) {
// special-case that the pod remains in the running state, but we don't really care, because this is only used for
// determining overall operation status is a sync with >1 wave/phase
testHookSuccessful(t, HookTypePreSync)
}
func TestSyncHookSuccessful(t *testing.T) {
testHookSuccessful(t, HookTypeSync)
}
func TestPostSyncHookSuccessful(t *testing.T) {
testHookSuccessful(t, HookTypePostSync)
}
// make sure we can run a standard sync hook
func testHookSuccessful(t *testing.T, hookType HookType) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", fmt.Sprintf(`[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "%s"}}]`, hookType)).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(ResourceSyncStatusIs("Pod", "pod", SyncStatusCodeSynced)).
Expect(ResourceHealthIs("Pod", "pod", health.HealthStatusHealthy)).
Expect(ResourceResultNumbering(2)).
Expect(ResourceResultIs(ResourceResult{Version: "v1", Kind: "Pod", Namespace: DeploymentNamespace(), Name: "hook", Message: "pod/hook created", HookType: hookType, HookPhase: OperationSucceeded, SyncPhase: SyncPhase(hookType)}))
}
func TestPostDeleteHook(t *testing.T) {
Given(t).
Path("post-delete-hook").
When().
CreateApp().
Refresh(RefreshTypeNormal).
Delete(true).
Then().
Expect(DoesNotExist()).
AndAction(func() {
hooks, err := KubeClientset.CoreV1().Pods(DeploymentNamespace()).List(context.Background(), metav1.ListOptions{})
CheckError(err)
assert.Len(t, hooks.Items, 1)
assert.Equal(t, "hook", hooks.Items[0].Name)
})
}
// make sure that that hooks do not appear in "argocd app diff"
func TestHookDiff(t *testing.T) {
Given(t).
Path("hook").
When().
CreateApp().
Then().
And(func(_ *Application) {
output, err := RunCli("app", "diff", Name())
require.Error(t, err)
assert.Contains(t, output, "name: pod")
assert.NotContains(t, output, "name: hook")
})
}
// make sure that if pre-sync fails, we fail the app and we do not create the pod
func TestPreSyncHookFailure(t *testing.T) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PreSync"}}]`).
// make hook fail
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command", "value": ["false"]}]`).
CreateApp().
IgnoreErrors().
Sync().
Then().
Expect(Error("hook Failed PreSync", "")).
// make sure resource are also printed
Expect(Error("pod OutOfSync Missing", "")).
Expect(OperationPhaseIs(OperationFailed)).
// if a pre-sync hook fails, we should not start the main sync
Expect(SyncStatusIs(SyncStatusCodeOutOfSync)).
Expect(ResourceResultNumbering(1)).
Expect(ResourceSyncStatusIs("Pod", "pod", SyncStatusCodeOutOfSync))
}
// make sure that if sync fails, we fail the app and we did create the pod
func TestSyncHookFailure(t *testing.T) {
Given(t).
Path("hook").
When().
// make hook fail
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
IgnoreErrors().
Sync().
Then().
Expect(OperationPhaseIs(OperationFailed)).
// even thought the hook failed, we expect the pod to be in sync
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(ResourceResultNumbering(2)).
Expect(ResourceSyncStatusIs("Pod", "pod", SyncStatusCodeSynced))
}
// make sure that if the deployments fails, we still get success and synced
func TestSyncHookResourceFailure(t *testing.T) {
Given(t).
Path("hook-and-deployment").
When().
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(HealthIs(health.HealthStatusProgressing))
}
// make sure that if post-sync fails, we fail the app and we did not create the pod
func TestPostSyncHookFailure(t *testing.T) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PostSync"}}]`).
// make hook fail
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
IgnoreErrors().
Sync().
Then().
Expect(OperationPhaseIs(OperationFailed)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(ResourceResultNumbering(2)).
Expect(ResourceSyncStatusIs("Pod", "pod", SyncStatusCodeSynced))
}
// make sure that if the pod fails, we do not run the post-sync hook
func TestPostSyncHookPodFailure(t *testing.T) {
Given(t).
Path("hook").
When().
IgnoreErrors().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PostSync"}}]`).
// make pod fail
PatchFile("pod.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
Sync().
Then().
// TODO - I feel like this should be a failure, not success
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(ResourceSyncStatusIs("Pod", "pod", SyncStatusCodeSynced)).
Expect(ResourceHealthIs("Pod", "pod", health.HealthStatusDegraded)).
Expect(ResourceResultNumbering(1)).
Expect(NotPod(func(p v1.Pod) bool { return p.Name == "hook" }))
}
func TestSyncFailHookPodFailure(t *testing.T) {
// Tests that a SyncFail hook will successfully run upon a pod failure (which leads to a sync failure)
Given(t).
Path("hook").
When().
IgnoreErrors().
AddFile("sync-fail-hook.yaml", `
apiVersion: v1
kind: Pod
metadata:
annotations:
argocd.argoproj.io/hook: SyncFail
name: sync-fail-hook
spec:
containers:
- command:
- "true"
image: "quay.io/argoprojlabs/argocd-e2e-container:0.1"
imagePullPolicy: IfNotPresent
name: main
restartPolicy: Never
`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PostSync"}}]`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
Sync().
Then().
Expect(ResourceResultIs(ResourceResult{Version: "v1", Kind: "Pod", Namespace: DeploymentNamespace(), Name: "sync-fail-hook", Message: "pod/sync-fail-hook created", HookType: HookTypeSyncFail, HookPhase: OperationSucceeded, SyncPhase: SyncPhaseSyncFail})).
Expect(OperationPhaseIs(OperationFailed))
}
func TestSyncFailHookPodFailureSyncFailFailure(t *testing.T) {
// Tests that a failing SyncFail hook will successfully be marked as failed
Given(t).
Path("hook").
When().
IgnoreErrors().
AddFile("successful-sync-fail-hook.yaml", `
apiVersion: v1
kind: Pod
metadata:
annotations:
argocd.argoproj.io/hook: SyncFail
name: successful-sync-fail-hook
spec:
containers:
- command:
- "true"
image: "quay.io/argoprojlabs/argocd-e2e-container:0.1"
imagePullPolicy: IfNotPresent
name: main
restartPolicy: Never
`).
AddFile("failed-sync-fail-hook.yaml", `
apiVersion: v1
kind: Pod
metadata:
annotations:
argocd.argoproj.io/hook: SyncFail
name: failed-sync-fail-hook
spec:
containers:
- command:
- "false"
image: "quay.io/argoprojlabs/argocd-e2e-container:0.1"
imagePullPolicy: IfNotPresent
name: main
restartPolicy: Never
`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PostSync"}}]`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
Sync().
Then().
Expect(ResourceResultIs(ResourceResult{Version: "v1", Kind: "Pod", Namespace: DeploymentNamespace(), Name: "successful-sync-fail-hook", Message: "pod/successful-sync-fail-hook created", HookType: HookTypeSyncFail, HookPhase: OperationSucceeded, SyncPhase: SyncPhaseSyncFail})).
Expect(ResourceResultIs(ResourceResult{Version: "v1", Kind: "Pod", Namespace: DeploymentNamespace(), Name: "failed-sync-fail-hook", Message: `container "main" failed with exit code 1`, HookType: HookTypeSyncFail, HookPhase: OperationFailed, SyncPhase: SyncPhaseSyncFail})).
Expect(OperationPhaseIs(OperationFailed))
}
// make sure that we delete the hook on success
func TestHookDeletePolicyHookSucceededHookExit0(t *testing.T) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "HookSucceeded"}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(NotPod(func(p v1.Pod) bool { return p.Name == "hook" }))
}
// make sure that we delete the hook on failure, if policy is set
func TestHookDeletePolicyHookSucceededHookExit1(t *testing.T) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "HookSucceeded"}]`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
IgnoreErrors().
Sync().
Then().
Expect(OperationPhaseIs(OperationFailed)).
Expect(ResourceResultNumbering(2)).
Expect(Pod(func(p v1.Pod) bool { return p.Name == "hook" }))
}
// make sure that we do NOT delete the hook on success if failure policy is set
func TestHookDeletePolicyHookFailedHookExit0(t *testing.T) {
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "HookFailed"}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(ResourceResultNumbering(2)).
Expect(Pod(func(p v1.Pod) bool { return p.Name == "hook" }))
}
// make sure that we do delete the hook on failure if failure policy is set
func TestHookDeletePolicyHookFailedHookExit1(t *testing.T) {
Given(t).
Path("hook").
When().
IgnoreErrors().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "HookFailed"}]`).
PatchFile("hook.yaml", `[{"op": "replace", "path": "/spec/containers/0/command/0", "value": "false"}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationFailed)).
Expect(ResourceResultNumbering(2)).
Expect(NotPod(func(p v1.Pod) bool { return p.Name == "hook" }))
}
// make sure that we can run the hook twice
func TestHookBeforeHookCreation(t *testing.T) {
var creationTimestamp1 string
Given(t).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "BeforeHookCreation"}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(HealthIs(health.HealthStatusHealthy)).
Expect(ResourceResultNumbering(2)).
// the app will be in health+n-sync before this hook has run
Expect(Pod(func(p v1.Pod) bool { return p.Name == "hook" })).
And(func(_ *Application) {
var err error
creationTimestamp1, err = getCreationTimestamp()
CheckError(err)
assert.NotEmpty(t, creationTimestamp1)
// pause to ensure that timestamp will change
time.Sleep(2 * time.Second)
}).
When().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(HealthIs(health.HealthStatusHealthy)).
Expect(ResourceResultNumbering(2)).
Expect(Pod(func(p v1.Pod) bool { return p.Name == "hook" })).
And(func(_ *Application) {
creationTimestamp2, err := getCreationTimestamp()
CheckError(err)
assert.NotEmpty(t, creationTimestamp2)
assert.NotEqual(t, creationTimestamp1, creationTimestamp2)
})
}
// edge-case where we are unable to delete the hook because it is still running
func TestHookBeforeHookCreationFailure(t *testing.T) {
Given(t).
Timeout(1).
Path("hook").
When().
PatchFile("hook.yaml", `[
{"op": "add", "path": "/metadata/annotations/argocd.argoproj.io~1hook-delete-policy", "value": "BeforeHookCreation"},
{"op": "replace", "path": "/spec/containers/0/command", "value": ["sleep", "3"]}
]`).
CreateApp().
IgnoreErrors().
Sync().
DoNotIgnoreErrors().
TerminateOp().
Then().
Expect(OperationPhaseIs(OperationFailed)).
Expect(ResourceResultNumbering(2))
}
func getCreationTimestamp() (string, error) {
return Run(".", "kubectl", "-n", DeploymentNamespace(), "get", "pod", "hook", "-o", "jsonpath={.metadata.creationTimestamp}")
}
// make sure that we never create something annotated with Skip
func TestHookSkip(t *testing.T) {
Given(t).
Path("hook").
When().
// should not create this pod
PatchFile("pod.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "Skip"}}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(ResourceResultNumbering(1)).
Expect(NotPod(func(p v1.Pod) bool { return p.Name == "pod" }))
}
// make sure that we do NOT name non-hook resources in they are unnamed
func TestNamingNonHookResource(t *testing.T) {
Given(t).
Async(true).
Path("hook").
When().
PatchFile("pod.yaml", `[{"op": "remove", "path": "/metadata/name"}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationFailed))
}
// make sure that we name hook resources in they are unnamed
func TestAutomaticallyNamingUnnamedHook(t *testing.T) {
Given(t).
Async(true).
Path("hook").
When().
PatchFile("hook.yaml", `[{"op": "remove", "path": "/metadata/name"}]`).
// make this part of two sync tasks
PatchFile("hook.yaml", `[{"op": "replace", "path": "/metadata/annotations", "value": {"argocd.argoproj.io/hook": "PreSync,PostSync"}}]`).
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
And(func(app *Application) {
resources := app.Status.OperationState.SyncResult.Resources
assert.Len(t, resources, 3)
// make sure we don't use the same name
assert.Contains(t, resources[0].Name, "presync")
assert.Contains(t, resources[2].Name, "postsync")
})
}