mirror of
https://github.com/argoproj/argo-cd
synced 2026-04-21 17:07:16 +00:00
314 lines
7.9 KiB
Go
314 lines
7.9 KiB
Go
package controller
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
)
|
|
|
|
func TestIsHookOfType(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
hookType HookType
|
|
annot map[string]string
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "ArgoCD PreDelete hook",
|
|
hookType: PreDeleteHookType,
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Helm PreDelete hook",
|
|
hookType: PreDeleteHookType,
|
|
annot: map[string]string{"helm.sh/hook": "pre-delete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PostDelete hook",
|
|
hookType: PostDeleteHookType,
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Helm PostDelete hook",
|
|
hookType: PostDeleteHookType,
|
|
annot: map[string]string{"helm.sh/hook": "post-delete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Not a hook",
|
|
hookType: PreDeleteHookType,
|
|
annot: map[string]string{"some-other": "annotation"},
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "Wrong hook type",
|
|
hookType: PreDeleteHookType,
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete"},
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "Nil annotations",
|
|
hookType: PreDeleteHookType,
|
|
annot: nil,
|
|
expected: false,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
obj := &unstructured.Unstructured{}
|
|
obj.SetAnnotations(tt.annot)
|
|
result := isHookOfType(obj, tt.hookType)
|
|
assert.Equal(t, tt.expected, result)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsHook(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
annot map[string]string
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "ArgoCD PreDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PostDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PreSync hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreSync"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Not a hook",
|
|
annot: map[string]string{"some-other": "annotation"},
|
|
expected: false,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
obj := &unstructured.Unstructured{}
|
|
obj.SetAnnotations(tt.annot)
|
|
result := isHook(obj)
|
|
assert.Equal(t, tt.expected, result)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsPreDeleteHook(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
annot map[string]string
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "ArgoCD PreDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Helm PreDelete hook",
|
|
annot: map[string]string{"helm.sh/hook": "pre-delete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PostDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete"},
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "Helm PreDelete & PreDelete hook",
|
|
annot: map[string]string{"helm.sh/hook": "pre-delete,post-delete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PostDelete & PreDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete,PreDelete"},
|
|
expected: true,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
obj := &unstructured.Unstructured{}
|
|
obj.SetAnnotations(tt.annot)
|
|
result := isPreDeleteHook(obj)
|
|
assert.Equal(t, tt.expected, result)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsPostDeleteHook(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
annot map[string]string
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "ArgoCD PostDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Helm PostDelete hook",
|
|
annot: map[string]string{"helm.sh/hook": "post-delete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "ArgoCD PreDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreDelete"},
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "ArgoCD PostDelete & PreDelete hook",
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PostDelete,PreDelete"},
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "Helm PostDelete & PreDelete hook",
|
|
annot: map[string]string{"helm.sh/hook": "post-delete,pre-delete"},
|
|
expected: true,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
obj := &unstructured.Unstructured{}
|
|
obj.SetAnnotations(tt.annot)
|
|
result := isPostDeleteHook(obj)
|
|
assert.Equal(t, tt.expected, result)
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestPartitionTargetObjsForSync covers partitionTargetObjsForSync in state.go.
|
|
func TestPartitionTargetObjsForSync(t *testing.T) {
|
|
newObj := func(name string, annot map[string]string) *unstructured.Unstructured {
|
|
u := &unstructured.Unstructured{}
|
|
u.SetName(name)
|
|
u.SetAnnotations(annot)
|
|
return u
|
|
}
|
|
|
|
tests := []struct {
|
|
name string
|
|
in []*unstructured.Unstructured
|
|
wantNames []string
|
|
wantPreDelete bool
|
|
wantPostDelete bool
|
|
}{
|
|
{
|
|
name: "PostSync with PreDelete and PostDelete in same annotation stays in sync set",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("combined", map[string]string{"argocd.argoproj.io/hook": "PostSync,PreDelete,PostDelete"}),
|
|
},
|
|
wantNames: []string{"combined"},
|
|
wantPreDelete: true,
|
|
wantPostDelete: true,
|
|
},
|
|
{
|
|
name: "PreDelete-only manifest excluded from sync",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("pre-del", map[string]string{"argocd.argoproj.io/hook": "PreDelete"}),
|
|
},
|
|
wantNames: nil,
|
|
wantPreDelete: true,
|
|
wantPostDelete: false,
|
|
},
|
|
{
|
|
name: "PostDelete-only manifest excluded from sync",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("post-del", map[string]string{"argocd.argoproj.io/hook": "PostDelete"}),
|
|
},
|
|
wantNames: nil,
|
|
wantPreDelete: false,
|
|
wantPostDelete: true,
|
|
},
|
|
{
|
|
name: "Helm pre-delete only excluded from sync",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("helm-pre-del", map[string]string{"helm.sh/hook": "pre-delete"}),
|
|
},
|
|
wantNames: nil,
|
|
wantPreDelete: true,
|
|
wantPostDelete: false,
|
|
},
|
|
{
|
|
name: "Helm pre-install with pre-delete stays in sync (sync-phase hook wins)",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("helm-mixed", map[string]string{"helm.sh/hook": "pre-install,pre-delete"}),
|
|
},
|
|
wantNames: []string{"helm-mixed"},
|
|
wantPreDelete: true,
|
|
wantPostDelete: false,
|
|
},
|
|
{
|
|
name: "Non-hook resource unchanged",
|
|
in: []*unstructured.Unstructured{
|
|
newObj("pod", map[string]string{"app": "x"}),
|
|
},
|
|
wantNames: []string{"pod"},
|
|
wantPreDelete: false,
|
|
wantPostDelete: false,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got, hasPre, hasPost := partitionTargetObjsForSync(tt.in)
|
|
var names []string
|
|
for _, o := range got {
|
|
names = append(names, o.GetName())
|
|
}
|
|
assert.Equal(t, tt.wantNames, names)
|
|
assert.Equal(t, tt.wantPreDelete, hasPre, "hasPreDeleteHooks")
|
|
assert.Equal(t, tt.wantPostDelete, hasPost, "hasPostDeleteHooks")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestMultiHookOfType(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
hookType []HookType
|
|
annot map[string]string
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "helm PreDelete & PostDelete hook",
|
|
hookType: []HookType{PreDeleteHookType, PostDeleteHookType},
|
|
annot: map[string]string{"helm.sh/hook": "pre-delete,post-delete"},
|
|
expected: true,
|
|
},
|
|
|
|
{
|
|
name: "ArgoCD PreDelete & PostDelete hook",
|
|
hookType: []HookType{PreDeleteHookType, PostDeleteHookType},
|
|
annot: map[string]string{"argocd.argoproj.io/hook": "PreDelete,PostDelete"},
|
|
expected: true,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
obj := &unstructured.Unstructured{}
|
|
obj.SetAnnotations(tt.annot)
|
|
|
|
for _, hookType := range tt.hookType {
|
|
result := isHookOfType(obj, hookType)
|
|
assert.Equal(t, tt.expected, result)
|
|
}
|
|
})
|
|
}
|
|
}
|