mirror of
https://github.com/open-cluster-management-io/ocm.git
synced 2026-05-21 00:24:08 +00:00
refactor gc controller (#229)
Signed-off-by: Zhiwei Yin <zyin@redhat.com>
This commit is contained in:
@@ -191,6 +191,10 @@ var _ = ginkgo.BeforeSuite(func() {
|
||||
err = features.HubMutableFeatureGate.Set("ManagedClusterAutoApproval=true")
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
// enable resourceCleanup feature gate
|
||||
err = features.HubMutableFeatureGate.Set("ResourceCleanup=true")
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
// start hub controller
|
||||
var ctx context.Context
|
||||
startHub = func() {
|
||||
|
||||
@@ -6,16 +6,31 @@ import (
|
||||
|
||||
"github.com/onsi/ginkgo/v2"
|
||||
"github.com/onsi/gomega"
|
||||
"github.com/openshift/library-go/pkg/operator/v1helpers"
|
||||
"k8s.io/apimachinery/pkg/api/errors"
|
||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||
"k8s.io/apimachinery/pkg/util/rand"
|
||||
|
||||
clusterv1 "open-cluster-management.io/api/cluster/v1"
|
||||
workapiv1 "open-cluster-management.io/api/work/v1"
|
||||
|
||||
testinghelpers "open-cluster-management.io/ocm/pkg/registration/helpers/testing"
|
||||
)
|
||||
|
||||
var (
|
||||
mclClusterRoleName = func(clusterName string) string {
|
||||
return fmt.Sprintf("open-cluster-management:managedcluster:%s", clusterName)
|
||||
}
|
||||
mclClusterRoleBindingName = func(clusterName string) string {
|
||||
return fmt.Sprintf("open-cluster-management:managedcluster:%s", clusterName)
|
||||
}
|
||||
registrationRoleBindingName = func(clusterName string) string {
|
||||
return fmt.Sprintf("open-cluster-management:managedcluster:%s:registration", clusterName)
|
||||
}
|
||||
workRoleBindingName = func(clusterName string) string {
|
||||
return fmt.Sprintf("open-cluster-management:managedcluster:%s:work", clusterName)
|
||||
}
|
||||
)
|
||||
|
||||
var _ = ginkgo.Describe("Cluster deleting", func() {
|
||||
var managedCluster *clusterv1.ManagedCluster
|
||||
|
||||
@@ -31,7 +46,7 @@ var _ = ginkgo.Describe("Cluster deleting", func() {
|
||||
}
|
||||
})
|
||||
|
||||
ginkgo.It("deleting cluster should wait until all manifestworks are deleted", func() {
|
||||
ginkgo.It("Cluster is deleting, all addons and non-addon works should be deleted", func() {
|
||||
_, err := clusterClient.ClusterV1().ManagedClusters().Create(context.Background(), managedCluster, metav1.CreateOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
@@ -40,44 +55,258 @@ var _ = ginkgo.Describe("Cluster deleting", func() {
|
||||
return err
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.Succeed())
|
||||
|
||||
manifestWork := testinghelpers.NewManifestWork(managedCluster.Name, "work1", []string{}, nil)
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Create(context.Background(), manifestWork, metav1.CreateOptions{})
|
||||
addon1 := testinghelpers.NewManagedClusterAddons("addon1", managedCluster.Name, []string{"hold"}, nil)
|
||||
_, err = addOnClient.AddonV1alpha1().ManagedClusterAddOns(managedCluster.Name).Create(context.Background(),
|
||||
addon1, metav1.CreateOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
manifestWork1 := testinghelpers.NewManifestWork(managedCluster.Name, "work1", []string{"hold"}, nil, nil, nil)
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Create(context.Background(), manifestWork1, metav1.CreateOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
manifestWork2 := testinghelpers.NewManifestWork(managedCluster.Name, "work2", []string{"hold"}, nil,
|
||||
map[string]string{clusterv1.CleanupPriorityAnnotationKey: "100"}, nil)
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Create(context.Background(), manifestWork2, metav1.CreateOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
manifestWork3 := testinghelpers.NewManifestWork(managedCluster.Name, "work3", []string{"hold"}, nil,
|
||||
map[string]string{clusterv1.CleanupPriorityAnnotationKey: "20"}, nil)
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Create(context.Background(), manifestWork3, metav1.CreateOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
roleBindingName := fmt.Sprintf("open-cluster-management:managedcluster:%s:work", managedCluster.Name)
|
||||
// check rbac are created
|
||||
gomega.Eventually(func() bool {
|
||||
if _, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(), roleBindingName, metav1.GetOptions{}); err != nil {
|
||||
if _, err := kubeClient.RbacV1().ClusterRoles().Get(context.Background(),
|
||||
mclClusterRoleName(managedCluster.Name), metav1.GetOptions{}); err != nil {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
if _, err := kubeClient.RbacV1().ClusterRoleBindings().Get(context.Background(),
|
||||
mclClusterRoleBindingName(managedCluster.Name), metav1.GetOptions{}); err != nil {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
if _, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(),
|
||||
registrationRoleBindingName(managedCluster.Name), metav1.GetOptions{}); err != nil {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
if _, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(),
|
||||
workRoleBindingName(managedCluster.Name), metav1.GetOptions{}); err != nil {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// delete cluster
|
||||
err = clusterClient.ClusterV1().ManagedClusters().Delete(context.Background(), managedCluster.Name, metav1.DeleteOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
err = kubeClient.RbacV1().RoleBindings(managedCluster.Name).Delete(context.Background(), roleBindingName, metav1.DeleteOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
roleBinding, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(), roleBindingName, metav1.GetOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
gomega.Expect(len(roleBinding.Finalizers)).Should(gomega.Equal(1))
|
||||
gomega.Expect(roleBinding.Finalizers[0]).Should(gomega.Equal(workapiv1.ManifestWorkFinalizer))
|
||||
gomega.Expect(roleBinding.DeletionTimestamp.IsZero()).Should(gomega.BeFalse())
|
||||
|
||||
// Delete work
|
||||
err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Delete(context.Background(), "work1", metav1.DeleteOptions{})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
// addons should be deleting and manifestworks should not be deleting
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).Get(context.Background(), "work1", metav1.GetOptions{})
|
||||
addon, err := addOnClient.AddonV1alpha1().ManagedClusterAddOns(managedCluster.Name).Get(context.Background(),
|
||||
"addon1", metav1.GetOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
if addon.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
|
||||
works, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).List(context.Background(), metav1.ListOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
for _, work := range works.Items {
|
||||
if !work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// remove finalizer on addon1
|
||||
gomega.Eventually(func() bool {
|
||||
addon, err := addOnClient.AddonV1alpha1().ManagedClusterAddOns(managedCluster.Name).Get(context.Background(),
|
||||
"addon1", metav1.GetOptions{})
|
||||
if errors.IsNotFound(err) {
|
||||
return true
|
||||
}
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
addon.Finalizers = []string{}
|
||||
_, err = addOnClient.AddonV1alpha1().ManagedClusterAddOns(managedCluster.Name).Update(context.Background(),
|
||||
addon, metav1.UpdateOptions{})
|
||||
return err == nil
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// addon should be deleted. work1 should be deleting ,the other work should not be deleting
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := addOnClient.AddonV1alpha1().ManagedClusterAddOns(managedCluster.Name).Get(context.Background(),
|
||||
"addon1", metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(), roleBindingName, metav1.GetOptions{})
|
||||
works, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).List(context.Background(), metav1.ListOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
if len(works.Items) != 3 {
|
||||
return false
|
||||
}
|
||||
for _, work := range works.Items {
|
||||
if work.Name == "work1" && work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
if work.Name == "work2" && !work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
if work.Name == "work3" && !work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// remove finalizer on work1
|
||||
gomega.Eventually(func() bool {
|
||||
work, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).Get(context.Background(),
|
||||
"work1", metav1.GetOptions{})
|
||||
if errors.IsNotFound(err) {
|
||||
return true
|
||||
}
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
work.Finalizers = []string{}
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Update(context.Background(),
|
||||
work, metav1.UpdateOptions{})
|
||||
return err == nil
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// work1 should be deleted, work3 should be deleting, work2 should not be deleting
|
||||
gomega.Eventually(func() bool {
|
||||
works, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).List(context.Background(), metav1.ListOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
if len(works.Items) != 2 {
|
||||
return false
|
||||
}
|
||||
for _, work := range works.Items {
|
||||
if work.Name == "work3" && work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
if work.Name == "work2" && !work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// remove finalizer on work3
|
||||
gomega.Eventually(func() bool {
|
||||
work, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).Get(context.Background(),
|
||||
"work3", metav1.GetOptions{})
|
||||
if errors.IsNotFound(err) {
|
||||
return true
|
||||
}
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
work.Finalizers = []string{}
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Update(context.Background(),
|
||||
work, metav1.UpdateOptions{})
|
||||
return err == nil
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// work3 should be deleted, work2 should be deleting
|
||||
gomega.Eventually(func() bool {
|
||||
works, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).List(context.Background(), metav1.ListOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
if len(works.Items) != 1 {
|
||||
return false
|
||||
}
|
||||
for _, work := range works.Items {
|
||||
if work.Name == "work2" && !work.DeletionTimestamp.IsZero() {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// managedCluster should have deleting condition
|
||||
gomega.Eventually(func() bool {
|
||||
cluster, err := clusterClient.ClusterV1().ManagedClusters().Get(context.Background(), managedCluster.Name, metav1.GetOptions{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
condition := v1helpers.FindCondition(cluster.Status.Conditions, clusterv1.ManagedClusterConditionDeleting)
|
||||
if condition == nil {
|
||||
return false
|
||||
}
|
||||
if condition.Reason != clusterv1.ConditionDeletingReasonResourceRemaining {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// remove finalizer on work2
|
||||
gomega.Eventually(func() bool {
|
||||
work, err := workClient.WorkV1().ManifestWorks(managedCluster.Name).Get(context.Background(),
|
||||
"work2", metav1.GetOptions{})
|
||||
if errors.IsNotFound(err) {
|
||||
return true
|
||||
}
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
work.Finalizers = []string{}
|
||||
_, err = workClient.WorkV1().ManifestWorks(managedCluster.Name).Update(context.Background(),
|
||||
work, metav1.UpdateOptions{})
|
||||
return err == nil
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// all rbac should be deleted
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := kubeClient.RbacV1().ClusterRoles().Get(context.Background(),
|
||||
mclClusterRoleName(managedCluster.Name), metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := kubeClient.RbacV1().ClusterRoleBindings().Get(context.Background(),
|
||||
mclClusterRoleBindingName(managedCluster.Name), metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(),
|
||||
registrationRoleBindingName(managedCluster.Name), metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := kubeClient.RbacV1().RoleBindings(managedCluster.Name).Get(context.Background(),
|
||||
workRoleBindingName(managedCluster.Name), metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
}, eventuallyTimeout, eventuallyInterval).Should(gomega.BeTrue())
|
||||
|
||||
// managedCluster should be deleted
|
||||
gomega.Eventually(func() bool {
|
||||
_, err := clusterClient.ClusterV1().ManagedClusters().Get(context.Background(), managedCluster.Name, metav1.GetOptions{})
|
||||
return errors.IsNotFound(err)
|
||||
|
||||
Reference in New Issue
Block a user