From d147cba76af6ba1173583a6fc90d8b7cc4b283a7 Mon Sep 17 00:00:00 2001 From: sangitaray2021 Date: Thu, 19 Sep 2024 02:26:14 +0530 Subject: [PATCH 1/4] Added tracking for deleted namespace status check in restore flow Signed-off-by: sangitaray2021 fixed unittest Signed-off-by: sangitaray2021 refactored tracker execution and caller Signed-off-by: sangitaray2021 added change log Signed-off-by: sangitaray2021 Author: sangitaray2021 Author: sangitaray2021 Date: Thu Sep 19 02:26:14 2024 +0530 Signed-off-by: sangitaray2021 --- changelogs/unreleased/8233-sangitaray2021 | 1 + pkg/controller/restore_controller.go | 23 +++--- pkg/restore/request.go | 26 +++---- pkg/restore/restore.go | 38 +++++----- pkg/restore/restore_test.go | 34 +++++---- .../kube/namespace_deletionstatus_tracker.go | 71 +++++++++++++++++++ pkg/util/kube/utils.go | 17 ++++- pkg/util/kube/utils_test.go | 33 ++++++--- 8 files changed, 178 insertions(+), 65 deletions(-) create mode 100644 changelogs/unreleased/8233-sangitaray2021 create mode 100644 pkg/util/kube/namespace_deletionstatus_tracker.go diff --git a/changelogs/unreleased/8233-sangitaray2021 b/changelogs/unreleased/8233-sangitaray2021 new file mode 100644 index 0000000000..82236c3001 --- /dev/null +++ b/changelogs/unreleased/8233-sangitaray2021 @@ -0,0 +1 @@ +Added tracking for deleted namespace status check in restore flow. \ No newline at end of file diff --git a/pkg/controller/restore_controller.go b/pkg/controller/restore_controller.go index aa55715c2c..4cd5e39ffb 100644 --- a/pkg/controller/restore_controller.go +++ b/pkg/controller/restore_controller.go @@ -559,17 +559,18 @@ func (r *restoreReconciler) runValidatedRestore(restore *api.Restore, info backu } restoreReq := &pkgrestore.Request{ - Log: restoreLog, - Restore: restore, - Backup: info.backup, - PodVolumeBackups: podVolumeBackups, - VolumeSnapshots: volumeSnapshots, - BackupReader: backupFile, - ResourceModifiers: resourceModifiers, - DisableInformerCache: r.disableInformerCache, - CSIVolumeSnapshots: csiVolumeSnapshots, - BackupVolumeInfoMap: backupVolumeInfoMap, - RestoreVolumeInfoTracker: volume.NewRestoreVolInfoTracker(restore, restoreLog, r.globalCrClient), + Log: restoreLog, + Restore: restore, + Backup: info.backup, + PodVolumeBackups: podVolumeBackups, + VolumeSnapshots: volumeSnapshots, + BackupReader: backupFile, + ResourceModifiers: resourceModifiers, + DisableInformerCache: r.disableInformerCache, + CSIVolumeSnapshots: csiVolumeSnapshots, + BackupVolumeInfoMap: backupVolumeInfoMap, + RestoreVolumeInfoTracker: volume.NewRestoreVolInfoTracker(restore, restoreLog, r.globalCrClient), + NamespaceDeletionStatusTracker: kubeutil.NewNamespaceDeletionStatusTracker(), } restoreWarnings, restoreErrors := r.restorer.RestoreWithResolvers(restoreReq, actionsResolver, pluginManager) diff --git a/pkg/restore/request.go b/pkg/restore/request.go index d6e341c4ff..d2f652f793 100644 --- a/pkg/restore/request.go +++ b/pkg/restore/request.go @@ -29,6 +29,7 @@ import ( "github.com/vmware-tanzu/velero/internal/volume" velerov1api "github.com/vmware-tanzu/velero/pkg/apis/velero/v1" "github.com/vmware-tanzu/velero/pkg/itemoperation" + "github.com/vmware-tanzu/velero/pkg/util/kube" ) const ( @@ -52,18 +53,19 @@ func resourceKey(obj runtime.Object) string { type Request struct { *velerov1api.Restore - Log logrus.FieldLogger - Backup *velerov1api.Backup - PodVolumeBackups []*velerov1api.PodVolumeBackup - VolumeSnapshots []*volume.Snapshot - BackupReader io.Reader - RestoredItems map[itemKey]restoredItemStatus - itemOperationsList *[]*itemoperation.RestoreOperation - ResourceModifiers *resourcemodifiers.ResourceModifiers - DisableInformerCache bool - CSIVolumeSnapshots []*snapshotv1api.VolumeSnapshot - BackupVolumeInfoMap map[string]volume.BackupVolumeInfo - RestoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker + Log logrus.FieldLogger + Backup *velerov1api.Backup + PodVolumeBackups []*velerov1api.PodVolumeBackup + VolumeSnapshots []*volume.Snapshot + BackupReader io.Reader + RestoredItems map[itemKey]restoredItemStatus + itemOperationsList *[]*itemoperation.RestoreOperation + ResourceModifiers *resourcemodifiers.ResourceModifiers + DisableInformerCache bool + CSIVolumeSnapshots []*snapshotv1api.VolumeSnapshot + BackupVolumeInfoMap map[string]volume.BackupVolumeInfo + RestoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker + NamespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker } type restoredItemStatus struct { diff --git a/pkg/restore/restore.go b/pkg/restore/restore.go index 9ae0b0f663..2717f420d5 100644 --- a/pkg/restore/restore.go +++ b/pkg/restore/restore.go @@ -102,22 +102,23 @@ type Restorer interface { // kubernetesRestorer implements Restorer for restoring into a Kubernetes cluster. type kubernetesRestorer struct { - discoveryHelper discovery.Helper - dynamicFactory client.DynamicFactory - namespaceClient corev1.NamespaceInterface - podVolumeRestorerFactory podvolume.RestorerFactory - podVolumeTimeout time.Duration - resourceTerminatingTimeout time.Duration - resourceTimeout time.Duration - resourcePriorities types.Priorities - fileSystem filesystem.Interface - pvRenamer func(string) (string, error) - logger logrus.FieldLogger - podCommandExecutor podexec.PodCommandExecutor - podGetter cache.Getter - credentialFileStore credentials.FileStore - kbClient crclient.Client - multiHookTracker *hook.MultiHookTracker + discoveryHelper discovery.Helper + dynamicFactory client.DynamicFactory + namespaceClient corev1.NamespaceInterface + podVolumeRestorerFactory podvolume.RestorerFactory + podVolumeTimeout time.Duration + resourceTerminatingTimeout time.Duration + resourceTimeout time.Duration + resourcePriorities types.Priorities + fileSystem filesystem.Interface + pvRenamer func(string) (string, error) + logger logrus.FieldLogger + podCommandExecutor podexec.PodCommandExecutor + podGetter cache.Getter + credentialFileStore credentials.FileStore + kbClient crclient.Client + multiHookTracker *hook.MultiHookTracker + namespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker } // NewKubernetesRestorer creates a new kubernetesRestorer. @@ -323,6 +324,7 @@ func (kr *kubernetesRestorer) RestoreWithResolvers( backupVolumeInfoMap: req.BackupVolumeInfoMap, restoreVolumeInfoTracker: req.RestoreVolumeInfoTracker, hooksWaitExecutor: hooksWaitExecutor, + namespaceDeletionStatusTracker: req.NamespaceDeletionStatusTracker, } return restoreCtx.execute() @@ -371,6 +373,7 @@ type restoreContext struct { backupVolumeInfoMap map[string]volume.BackupVolumeInfo restoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker hooksWaitExecutor *hooksWaitExecutor + namespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker } type resourceClientKey struct { @@ -718,6 +721,7 @@ func (ctx *restoreContext) processSelectedResource( ns, ctx.namespaceClient, ctx.resourceTerminatingTimeout, + ctx.namespaceDeletionStatusTracker, ) if err != nil { errs.AddVeleroError(err) @@ -1119,7 +1123,7 @@ func (ctx *restoreContext) restoreItem(obj *unstructured.Unstructured, groupReso // namespace into which the resource is being restored into exists. // This is the *remapped* namespace that we are ensuring exists. nsToEnsure := getNamespace(ctx.log, archive.GetItemFilePath(ctx.restoreDir, "namespaces", "", obj.GetNamespace()), namespace) - _, nsCreated, err := kube.EnsureNamespaceExistsAndIsReady(nsToEnsure, ctx.namespaceClient, ctx.resourceTerminatingTimeout) + _, nsCreated, err := kube.EnsureNamespaceExistsAndIsReady(nsToEnsure, ctx.namespaceClient, ctx.resourceTerminatingTimeout, ctx.namespaceDeletionStatusTracker) if err != nil { errs.AddVeleroError(err) return warnings, errs, itemExists diff --git a/pkg/restore/restore_test.go b/pkg/restore/restore_test.go index 8e239e9a70..ef08253a27 100644 --- a/pkg/restore/restore_test.go +++ b/pkg/restore/restore_test.go @@ -58,6 +58,7 @@ import ( uploadermocks "github.com/vmware-tanzu/velero/pkg/podvolume/mocks" "github.com/vmware-tanzu/velero/pkg/test" "github.com/vmware-tanzu/velero/pkg/types" + "github.com/vmware-tanzu/velero/pkg/util/kube" kubeutil "github.com/vmware-tanzu/velero/pkg/util/kube" . "github.com/vmware-tanzu/velero/pkg/util/results" ) @@ -2292,10 +2293,11 @@ func TestShouldRestore(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - dynamicFactory: client.NewDynamicFactory(h.DynamicClient), - namespaceClient: h.KubeClient.CoreV1().Namespaces(), - resourceTerminatingTimeout: time.Millisecond, + log: h.log, + dynamicFactory: client.NewDynamicFactory(h.DynamicClient), + namespaceClient: h.KubeClient.CoreV1().Namespaces(), + resourceTerminatingTimeout: time.Millisecond, + namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), } for _, resource := range tc.apiResources { @@ -3710,9 +3712,10 @@ func newHarness(t *testing.T) *harness { fileSystem: test.NewFakeFileSystem(), // unsupported - podVolumeRestorerFactory: nil, - podVolumeTimeout: 0, - kbClient: kbClient, + podVolumeRestorerFactory: nil, + podVolumeTimeout: 0, + kbClient: kbClient, + namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), }, log: log, } @@ -3899,9 +3902,10 @@ func TestIsAlreadyExistsError(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - dynamicFactory: client.NewDynamicFactory(h.DynamicClient), - namespaceClient: h.KubeClient.CoreV1().Namespaces(), + log: h.log, + dynamicFactory: client.NewDynamicFactory(h.DynamicClient), + namespaceClient: h.KubeClient.CoreV1().Namespaces(), + namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), } if test.apiResource != nil { @@ -4018,7 +4022,8 @@ func TestHasCSIVolumeSnapshot(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, + log: h.log, + namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), } if tc.vs != nil { @@ -4118,9 +4123,10 @@ func TestHasSnapshotDataUpload(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - kbClient: h.restorer.kbClient, - restore: tc.restore, + log: h.log, + kbClient: h.restorer.kbClient, + restore: tc.restore, + namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), } if tc.duResult != nil { diff --git a/pkg/util/kube/namespace_deletionstatus_tracker.go b/pkg/util/kube/namespace_deletionstatus_tracker.go new file mode 100644 index 0000000000..9f91245421 --- /dev/null +++ b/pkg/util/kube/namespace_deletionstatus_tracker.go @@ -0,0 +1,71 @@ +/* +Copyright 2018 the Velero contributors. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package kube + +import ( + "fmt" + "sync" + + "k8s.io/apimachinery/pkg/util/sets" +) + +// NamespaceDeletionStatusTracker keeps track of in-progress backups. +type NamespaceDeletionStatusTracker interface { + // Add informs the tracker that a polling is in progress to check namespace deletion status. + Add(ns, name string) + // Delete informs the tracker that a namespace deletion is completed. + Delete(ns, name string) + // Contains returns true if the tracker is tracking the namespace deletion progress. + Contains(ns, name string) bool +} + +type namespaceDeletionStatusTracker struct { + lock sync.RWMutex + isNameSpacePresentInPollingSet sets.Set[string] +} + +// NewNamespaceDeletionStatusTracker returns a new NamespaceDeletionStatusTracker. +func NewNamespaceDeletionStatusTracker() NamespaceDeletionStatusTracker { + return &namespaceDeletionStatusTracker{ + isNameSpacePresentInPollingSet: sets.New[string](), + } +} + +func (bt *namespaceDeletionStatusTracker) Add(ns, name string) { + bt.lock.Lock() + defer bt.lock.Unlock() + + bt.isNameSpacePresentInPollingSet.Insert(namespaceDeletionStatusTrackerKey(ns, name)) +} + +func (bt *namespaceDeletionStatusTracker) Delete(ns, name string) { + bt.lock.Lock() + defer bt.lock.Unlock() + + bt.isNameSpacePresentInPollingSet.Delete(namespaceDeletionStatusTrackerKey(ns, name)) +} + +func (bt *namespaceDeletionStatusTracker) Contains(ns, name string) bool { + bt.lock.RLock() + defer bt.lock.RUnlock() + + return bt.isNameSpacePresentInPollingSet.Has(namespaceDeletionStatusTrackerKey(ns, name)) +} + +func namespaceDeletionStatusTrackerKey(ns, name string) string { + return fmt.Sprintf("%s/%s", ns, name) +} diff --git a/pkg/util/kube/utils.go b/pkg/util/kube/utils.go index d50e687287..c733555f2a 100644 --- a/pkg/util/kube/utils.go +++ b/pkg/util/kube/utils.go @@ -74,11 +74,14 @@ func NamespaceAndName(objMeta metav1.Object) string { // // namespace already exists and is not ready, this function will return (false, false, nil). // If the namespace exists and is marked for deletion, this function will wait up to the timeout for it to fully delete. -func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client corev1client.NamespaceInterface, timeout time.Duration) (ready bool, nsCreated bool, err error) { +func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client corev1client.NamespaceInterface, timeout time.Duration, namespaceDeletionStatusTracker NamespaceDeletionStatusTracker) (ready bool, nsCreated bool, err error) { // nsCreated tells whether the namespace was created by this method // required for keeping track of number of restored items // if namespace is marked for deletion, and we timed out, report an error var terminatingNamespace bool + + var namespaceAlreadyInDeletionTracker bool + err = wait.PollUntilContextTimeout(context.Background(), time.Second, timeout, true, func(ctx context.Context) (bool, error) { clusterNS, err := client.Get(ctx, namespace.Name, metav1.GetOptions{}) // if namespace is marked for deletion, and we timed out, report an error @@ -92,8 +95,12 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core // Return the err and exit the loop. return true, err } - if clusterNS != nil && (clusterNS.GetDeletionTimestamp() != nil || clusterNS.Status.Phase == corev1api.NamespaceTerminating) { + if namespaceDeletionStatusTracker.Contains(clusterNS.Name, clusterNS.Name) { + namespaceAlreadyInDeletionTracker = true + return true, errors.Errorf("namespace %s is already present in the polling set, skipping execution", namespace.Name) + } + // Marked for deletion, keep waiting terminatingNamespace = true return false, nil @@ -107,7 +114,13 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core // err will be set if we timed out or encountered issues retrieving the namespace, if err != nil { if terminatingNamespace { + // If the namespace is marked for deletion, and we timed out, adding it in tracker + namespaceDeletionStatusTracker.Add(namespace.Name, namespace.Name) return false, nsCreated, errors.Wrapf(err, "timed out waiting for terminating namespace %s to disappear before restoring", namespace.Name) + + } else if namespaceAlreadyInDeletionTracker { + // If the namespace is already in the tracker, return an error. + return false, nsCreated, err } return false, nsCreated, errors.Wrapf(err, "error getting namespace %s", namespace.Name) } diff --git a/pkg/util/kube/utils_test.go b/pkg/util/kube/utils_test.go index 6feb9bd946..9a4bd5e135 100644 --- a/pkg/util/kube/utils_test.go +++ b/pkg/util/kube/utils_test.go @@ -47,14 +47,16 @@ func TestNamespaceAndName(t *testing.T) { func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { tests := []struct { - name string - expectNSFound bool - nsPhase corev1.NamespacePhase - nsDeleting bool - expectCreate bool - alreadyExists bool - expectedResult bool - expectedCreatedResult bool + name string + expectNSFound bool + nsPhase corev1.NamespacePhase + nsDeleting bool + expectCreate bool + alreadyExists bool + expectedResult bool + expectedCreatedResult bool + nsAlreadyInTerminationTracker bool + namespaceDeletionStatusTracker NamespaceDeletionStatusTracker }{ { name: "namespace found, not deleting", @@ -95,8 +97,17 @@ func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { expectedResult: false, expectedCreatedResult: false, }, + { + name: "same namespace found earlier, terminating phase already tracked", + expectNSFound: true, + nsPhase: corev1.NamespaceTerminating, + expectedResult: false, + expectedCreatedResult: false, + nsAlreadyInTerminationTracker: true, + }, } + namespaceDeletionStatusTracker := NewNamespaceDeletionStatusTracker() for _, test := range tests { t.Run(test.name, func(t *testing.T) { namespace := &corev1.Namespace{ @@ -132,7 +143,11 @@ func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { nsClient.On("Create", namespace).Return(namespace, nil) } - result, nsCreated, _ := EnsureNamespaceExistsAndIsReady(namespace, nsClient, timeout) + if test.nsAlreadyInTerminationTracker { + namespaceDeletionStatusTracker.Add("test", "test") + } + + result, nsCreated, _ := EnsureNamespaceExistsAndIsReady(namespace, nsClient, timeout, namespaceDeletionStatusTracker) assert.Equal(t, test.expectedResult, result) assert.Equal(t, test.expectedCreatedResult, nsCreated) From 963e878596e923e95d26ab377881bf04e0e2797b Mon Sep 17 00:00:00 2001 From: sangitaray2021 Date: Thu, 26 Sep 2024 14:23:01 +0530 Subject: [PATCH 2/4] fixed linter issuer Signed-off-by: sangitaray2021 --- pkg/util/kube/utils.go | 1 - 1 file changed, 1 deletion(-) diff --git a/pkg/util/kube/utils.go b/pkg/util/kube/utils.go index c733555f2a..a6d8c5d608 100644 --- a/pkg/util/kube/utils.go +++ b/pkg/util/kube/utils.go @@ -117,7 +117,6 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core // If the namespace is marked for deletion, and we timed out, adding it in tracker namespaceDeletionStatusTracker.Add(namespace.Name, namespace.Name) return false, nsCreated, errors.Wrapf(err, "timed out waiting for terminating namespace %s to disappear before restoring", namespace.Name) - } else if namespaceAlreadyInDeletionTracker { // If the namespace is already in the tracker, return an error. return false, nsCreated, err From e0690310d12823f101ff60640583382260517fd4 Mon Sep 17 00:00:00 2001 From: sangitaray2021 Date: Tue, 29 Oct 2024 01:12:14 +0530 Subject: [PATCH 3/4] incorporated PR comments Signed-off-by: sangitaray2021 --- pkg/controller/restore_controller.go | 24 +++++------ pkg/restore/request.go | 26 ++++++------ pkg/restore/restore.go | 42 +++++++++---------- pkg/restore/restore_test.go | 38 ++++++++--------- ....go => resource_deletionstatus_tracker.go} | 34 +++++++-------- pkg/util/kube/utils.go | 6 +-- pkg/util/kube/utils_test.go | 26 ++++++------ 7 files changed, 98 insertions(+), 98 deletions(-) rename pkg/util/kube/{namespace_deletionstatus_tracker.go => resource_deletionstatus_tracker.go} (52%) diff --git a/pkg/controller/restore_controller.go b/pkg/controller/restore_controller.go index 4cd5e39ffb..336f2e1944 100644 --- a/pkg/controller/restore_controller.go +++ b/pkg/controller/restore_controller.go @@ -559,18 +559,18 @@ func (r *restoreReconciler) runValidatedRestore(restore *api.Restore, info backu } restoreReq := &pkgrestore.Request{ - Log: restoreLog, - Restore: restore, - Backup: info.backup, - PodVolumeBackups: podVolumeBackups, - VolumeSnapshots: volumeSnapshots, - BackupReader: backupFile, - ResourceModifiers: resourceModifiers, - DisableInformerCache: r.disableInformerCache, - CSIVolumeSnapshots: csiVolumeSnapshots, - BackupVolumeInfoMap: backupVolumeInfoMap, - RestoreVolumeInfoTracker: volume.NewRestoreVolInfoTracker(restore, restoreLog, r.globalCrClient), - NamespaceDeletionStatusTracker: kubeutil.NewNamespaceDeletionStatusTracker(), + Log: restoreLog, + Restore: restore, + Backup: info.backup, + PodVolumeBackups: podVolumeBackups, + VolumeSnapshots: volumeSnapshots, + BackupReader: backupFile, + ResourceModifiers: resourceModifiers, + DisableInformerCache: r.disableInformerCache, + CSIVolumeSnapshots: csiVolumeSnapshots, + BackupVolumeInfoMap: backupVolumeInfoMap, + RestoreVolumeInfoTracker: volume.NewRestoreVolInfoTracker(restore, restoreLog, r.globalCrClient), + ResourceDeletionStatusTracker: kubeutil.NewResourceDeletionStatusTracker(), } restoreWarnings, restoreErrors := r.restorer.RestoreWithResolvers(restoreReq, actionsResolver, pluginManager) diff --git a/pkg/restore/request.go b/pkg/restore/request.go index d2f652f793..0e41b10ca4 100644 --- a/pkg/restore/request.go +++ b/pkg/restore/request.go @@ -53,19 +53,19 @@ func resourceKey(obj runtime.Object) string { type Request struct { *velerov1api.Restore - Log logrus.FieldLogger - Backup *velerov1api.Backup - PodVolumeBackups []*velerov1api.PodVolumeBackup - VolumeSnapshots []*volume.Snapshot - BackupReader io.Reader - RestoredItems map[itemKey]restoredItemStatus - itemOperationsList *[]*itemoperation.RestoreOperation - ResourceModifiers *resourcemodifiers.ResourceModifiers - DisableInformerCache bool - CSIVolumeSnapshots []*snapshotv1api.VolumeSnapshot - BackupVolumeInfoMap map[string]volume.BackupVolumeInfo - RestoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker - NamespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker + Log logrus.FieldLogger + Backup *velerov1api.Backup + PodVolumeBackups []*velerov1api.PodVolumeBackup + VolumeSnapshots []*volume.Snapshot + BackupReader io.Reader + RestoredItems map[itemKey]restoredItemStatus + itemOperationsList *[]*itemoperation.RestoreOperation + ResourceModifiers *resourcemodifiers.ResourceModifiers + DisableInformerCache bool + CSIVolumeSnapshots []*snapshotv1api.VolumeSnapshot + BackupVolumeInfoMap map[string]volume.BackupVolumeInfo + RestoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker + ResourceDeletionStatusTracker kube.ResourceDeletionStatusTracker } type restoredItemStatus struct { diff --git a/pkg/restore/restore.go b/pkg/restore/restore.go index 2717f420d5..619db5b87e 100644 --- a/pkg/restore/restore.go +++ b/pkg/restore/restore.go @@ -102,23 +102,23 @@ type Restorer interface { // kubernetesRestorer implements Restorer for restoring into a Kubernetes cluster. type kubernetesRestorer struct { - discoveryHelper discovery.Helper - dynamicFactory client.DynamicFactory - namespaceClient corev1.NamespaceInterface - podVolumeRestorerFactory podvolume.RestorerFactory - podVolumeTimeout time.Duration - resourceTerminatingTimeout time.Duration - resourceTimeout time.Duration - resourcePriorities types.Priorities - fileSystem filesystem.Interface - pvRenamer func(string) (string, error) - logger logrus.FieldLogger - podCommandExecutor podexec.PodCommandExecutor - podGetter cache.Getter - credentialFileStore credentials.FileStore - kbClient crclient.Client - multiHookTracker *hook.MultiHookTracker - namespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker + discoveryHelper discovery.Helper + dynamicFactory client.DynamicFactory + namespaceClient corev1.NamespaceInterface + podVolumeRestorerFactory podvolume.RestorerFactory + podVolumeTimeout time.Duration + resourceTerminatingTimeout time.Duration + resourceTimeout time.Duration + resourcePriorities types.Priorities + fileSystem filesystem.Interface + pvRenamer func(string) (string, error) + logger logrus.FieldLogger + podCommandExecutor podexec.PodCommandExecutor + podGetter cache.Getter + credentialFileStore credentials.FileStore + kbClient crclient.Client + multiHookTracker *hook.MultiHookTracker + resourceDeletionStatusTracker kube.ResourceDeletionStatusTracker } // NewKubernetesRestorer creates a new kubernetesRestorer. @@ -324,7 +324,7 @@ func (kr *kubernetesRestorer) RestoreWithResolvers( backupVolumeInfoMap: req.BackupVolumeInfoMap, restoreVolumeInfoTracker: req.RestoreVolumeInfoTracker, hooksWaitExecutor: hooksWaitExecutor, - namespaceDeletionStatusTracker: req.NamespaceDeletionStatusTracker, + resourceDeletionStatusTracker: req.ResourceDeletionStatusTracker, } return restoreCtx.execute() @@ -373,7 +373,7 @@ type restoreContext struct { backupVolumeInfoMap map[string]volume.BackupVolumeInfo restoreVolumeInfoTracker *volume.RestoreVolumeInfoTracker hooksWaitExecutor *hooksWaitExecutor - namespaceDeletionStatusTracker kube.NamespaceDeletionStatusTracker + resourceDeletionStatusTracker kube.ResourceDeletionStatusTracker } type resourceClientKey struct { @@ -721,7 +721,7 @@ func (ctx *restoreContext) processSelectedResource( ns, ctx.namespaceClient, ctx.resourceTerminatingTimeout, - ctx.namespaceDeletionStatusTracker, + ctx.resourceDeletionStatusTracker, ) if err != nil { errs.AddVeleroError(err) @@ -1123,7 +1123,7 @@ func (ctx *restoreContext) restoreItem(obj *unstructured.Unstructured, groupReso // namespace into which the resource is being restored into exists. // This is the *remapped* namespace that we are ensuring exists. nsToEnsure := getNamespace(ctx.log, archive.GetItemFilePath(ctx.restoreDir, "namespaces", "", obj.GetNamespace()), namespace) - _, nsCreated, err := kube.EnsureNamespaceExistsAndIsReady(nsToEnsure, ctx.namespaceClient, ctx.resourceTerminatingTimeout, ctx.namespaceDeletionStatusTracker) + _, nsCreated, err := kube.EnsureNamespaceExistsAndIsReady(nsToEnsure, ctx.namespaceClient, ctx.resourceTerminatingTimeout, ctx.resourceDeletionStatusTracker) if err != nil { errs.AddVeleroError(err) return warnings, errs, itemExists diff --git a/pkg/restore/restore_test.go b/pkg/restore/restore_test.go index ef08253a27..9617af146b 100644 --- a/pkg/restore/restore_test.go +++ b/pkg/restore/restore_test.go @@ -2293,11 +2293,11 @@ func TestShouldRestore(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - dynamicFactory: client.NewDynamicFactory(h.DynamicClient), - namespaceClient: h.KubeClient.CoreV1().Namespaces(), - resourceTerminatingTimeout: time.Millisecond, - namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), + log: h.log, + dynamicFactory: client.NewDynamicFactory(h.DynamicClient), + namespaceClient: h.KubeClient.CoreV1().Namespaces(), + resourceTerminatingTimeout: time.Millisecond, + resourceDeletionStatusTracker: kube.NewResourceDeletionStatusTracker(), } for _, resource := range tc.apiResources { @@ -3712,10 +3712,10 @@ func newHarness(t *testing.T) *harness { fileSystem: test.NewFakeFileSystem(), // unsupported - podVolumeRestorerFactory: nil, - podVolumeTimeout: 0, - kbClient: kbClient, - namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), + podVolumeRestorerFactory: nil, + podVolumeTimeout: 0, + kbClient: kbClient, + resourceDeletionStatusTracker: kube.NewResourceDeletionStatusTracker(), }, log: log, } @@ -3902,10 +3902,10 @@ func TestIsAlreadyExistsError(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - dynamicFactory: client.NewDynamicFactory(h.DynamicClient), - namespaceClient: h.KubeClient.CoreV1().Namespaces(), - namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), + log: h.log, + dynamicFactory: client.NewDynamicFactory(h.DynamicClient), + namespaceClient: h.KubeClient.CoreV1().Namespaces(), + resourceDeletionStatusTracker: kube.NewResourceDeletionStatusTracker(), } if test.apiResource != nil { @@ -4022,8 +4022,8 @@ func TestHasCSIVolumeSnapshot(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), + log: h.log, + resourceDeletionStatusTracker: kube.NewResourceDeletionStatusTracker(), } if tc.vs != nil { @@ -4123,10 +4123,10 @@ func TestHasSnapshotDataUpload(t *testing.T) { h := newHarness(t) ctx := &restoreContext{ - log: h.log, - kbClient: h.restorer.kbClient, - restore: tc.restore, - namespaceDeletionStatusTracker: kube.NewNamespaceDeletionStatusTracker(), + log: h.log, + kbClient: h.restorer.kbClient, + restore: tc.restore, + resourceDeletionStatusTracker: kube.NewResourceDeletionStatusTracker(), } if tc.duResult != nil { diff --git a/pkg/util/kube/namespace_deletionstatus_tracker.go b/pkg/util/kube/resource_deletionstatus_tracker.go similarity index 52% rename from pkg/util/kube/namespace_deletionstatus_tracker.go rename to pkg/util/kube/resource_deletionstatus_tracker.go index 9f91245421..7e7bb20a9f 100644 --- a/pkg/util/kube/namespace_deletionstatus_tracker.go +++ b/pkg/util/kube/resource_deletionstatus_tracker.go @@ -23,49 +23,49 @@ import ( "k8s.io/apimachinery/pkg/util/sets" ) -// NamespaceDeletionStatusTracker keeps track of in-progress backups. -type NamespaceDeletionStatusTracker interface { +// resourceDeletionStatusTracker keeps track of in-progress backups. +type ResourceDeletionStatusTracker interface { // Add informs the tracker that a polling is in progress to check namespace deletion status. - Add(ns, name string) + Add(kind, ns, name string) // Delete informs the tracker that a namespace deletion is completed. - Delete(ns, name string) + Delete(kind, ns, name string) // Contains returns true if the tracker is tracking the namespace deletion progress. - Contains(ns, name string) bool + Contains(kind, ns, name string) bool } -type namespaceDeletionStatusTracker struct { +type resourceDeletionStatusTracker struct { lock sync.RWMutex isNameSpacePresentInPollingSet sets.Set[string] } -// NewNamespaceDeletionStatusTracker returns a new NamespaceDeletionStatusTracker. -func NewNamespaceDeletionStatusTracker() NamespaceDeletionStatusTracker { - return &namespaceDeletionStatusTracker{ +// NewResourceDeletionStatusTracker returns a new ResourceDeletionStatusTracker. +func NewResourceDeletionStatusTracker() ResourceDeletionStatusTracker { + return &resourceDeletionStatusTracker{ isNameSpacePresentInPollingSet: sets.New[string](), } } -func (bt *namespaceDeletionStatusTracker) Add(ns, name string) { +func (bt *resourceDeletionStatusTracker) Add(kind, ns, name string) { bt.lock.Lock() defer bt.lock.Unlock() - bt.isNameSpacePresentInPollingSet.Insert(namespaceDeletionStatusTrackerKey(ns, name)) + bt.isNameSpacePresentInPollingSet.Insert(resourceDeletionStatusTrackerKey(kind, ns, name)) } -func (bt *namespaceDeletionStatusTracker) Delete(ns, name string) { +func (bt *resourceDeletionStatusTracker) Delete(kind, ns, name string) { bt.lock.Lock() defer bt.lock.Unlock() - bt.isNameSpacePresentInPollingSet.Delete(namespaceDeletionStatusTrackerKey(ns, name)) + bt.isNameSpacePresentInPollingSet.Delete(resourceDeletionStatusTrackerKey(kind, ns, name)) } -func (bt *namespaceDeletionStatusTracker) Contains(ns, name string) bool { +func (bt *resourceDeletionStatusTracker) Contains(kind, ns, name string) bool { bt.lock.RLock() defer bt.lock.RUnlock() - return bt.isNameSpacePresentInPollingSet.Has(namespaceDeletionStatusTrackerKey(ns, name)) + return bt.isNameSpacePresentInPollingSet.Has(resourceDeletionStatusTrackerKey(kind, ns, name)) } -func namespaceDeletionStatusTrackerKey(ns, name string) string { - return fmt.Sprintf("%s/%s", ns, name) +func resourceDeletionStatusTrackerKey(kind, ns, name string) string { + return fmt.Sprintf("%s/%s/%s", kind, ns, name) } diff --git a/pkg/util/kube/utils.go b/pkg/util/kube/utils.go index a6d8c5d608..ee798aeaf4 100644 --- a/pkg/util/kube/utils.go +++ b/pkg/util/kube/utils.go @@ -74,7 +74,7 @@ func NamespaceAndName(objMeta metav1.Object) string { // // namespace already exists and is not ready, this function will return (false, false, nil). // If the namespace exists and is marked for deletion, this function will wait up to the timeout for it to fully delete. -func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client corev1client.NamespaceInterface, timeout time.Duration, namespaceDeletionStatusTracker NamespaceDeletionStatusTracker) (ready bool, nsCreated bool, err error) { +func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client corev1client.NamespaceInterface, timeout time.Duration, resourceDeletionStatusTracker ResourceDeletionStatusTracker) (ready bool, nsCreated bool, err error) { // nsCreated tells whether the namespace was created by this method // required for keeping track of number of restored items // if namespace is marked for deletion, and we timed out, report an error @@ -96,7 +96,7 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core return true, err } if clusterNS != nil && (clusterNS.GetDeletionTimestamp() != nil || clusterNS.Status.Phase == corev1api.NamespaceTerminating) { - if namespaceDeletionStatusTracker.Contains(clusterNS.Name, clusterNS.Name) { + if resourceDeletionStatusTracker.Contains(clusterNS.Kind, clusterNS.Name, clusterNS.Name) { namespaceAlreadyInDeletionTracker = true return true, errors.Errorf("namespace %s is already present in the polling set, skipping execution", namespace.Name) } @@ -115,7 +115,7 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core if err != nil { if terminatingNamespace { // If the namespace is marked for deletion, and we timed out, adding it in tracker - namespaceDeletionStatusTracker.Add(namespace.Name, namespace.Name) + resourceDeletionStatusTracker.Add(namespace.Kind, namespace.Name, namespace.Name) return false, nsCreated, errors.Wrapf(err, "timed out waiting for terminating namespace %s to disappear before restoring", namespace.Name) } else if namespaceAlreadyInDeletionTracker { // If the namespace is already in the tracker, return an error. diff --git a/pkg/util/kube/utils_test.go b/pkg/util/kube/utils_test.go index 9a4bd5e135..35792c5272 100644 --- a/pkg/util/kube/utils_test.go +++ b/pkg/util/kube/utils_test.go @@ -47,16 +47,16 @@ func TestNamespaceAndName(t *testing.T) { func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { tests := []struct { - name string - expectNSFound bool - nsPhase corev1.NamespacePhase - nsDeleting bool - expectCreate bool - alreadyExists bool - expectedResult bool - expectedCreatedResult bool - nsAlreadyInTerminationTracker bool - namespaceDeletionStatusTracker NamespaceDeletionStatusTracker + name string + expectNSFound bool + nsPhase corev1.NamespacePhase + nsDeleting bool + expectCreate bool + alreadyExists bool + expectedResult bool + expectedCreatedResult bool + nsAlreadyInTerminationTracker bool + ResourceDeletionStatusTracker ResourceDeletionStatusTracker }{ { name: "namespace found, not deleting", @@ -107,7 +107,7 @@ func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { }, } - namespaceDeletionStatusTracker := NewNamespaceDeletionStatusTracker() + resourceDeletionStatusTracker := NewResourceDeletionStatusTracker() for _, test := range tests { t.Run(test.name, func(t *testing.T) { namespace := &corev1.Namespace{ @@ -144,10 +144,10 @@ func TestEnsureNamespaceExistsAndIsReady(t *testing.T) { } if test.nsAlreadyInTerminationTracker { - namespaceDeletionStatusTracker.Add("test", "test") + resourceDeletionStatusTracker.Add(namespace.Kind, "test", "test") } - result, nsCreated, _ := EnsureNamespaceExistsAndIsReady(namespace, nsClient, timeout, namespaceDeletionStatusTracker) + result, nsCreated, _ := EnsureNamespaceExistsAndIsReady(namespace, nsClient, timeout, resourceDeletionStatusTracker) assert.Equal(t, test.expectedResult, result) assert.Equal(t, test.expectedCreatedResult, nsCreated) From 6a5e8c279e8a5874fd83a049434769a2d4c79abe Mon Sep 17 00:00:00 2001 From: sangitaray2021 Date: Wed, 6 Nov 2024 23:01:21 +0530 Subject: [PATCH 4/4] resolved comments Signed-off-by: sangitaray2021 --- pkg/util/kube/resource_deletionstatus_tracker.go | 2 +- pkg/util/kube/utils.go | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/pkg/util/kube/resource_deletionstatus_tracker.go b/pkg/util/kube/resource_deletionstatus_tracker.go index 7e7bb20a9f..9f2b1bd142 100644 --- a/pkg/util/kube/resource_deletionstatus_tracker.go +++ b/pkg/util/kube/resource_deletionstatus_tracker.go @@ -23,7 +23,7 @@ import ( "k8s.io/apimachinery/pkg/util/sets" ) -// resourceDeletionStatusTracker keeps track of in-progress backups. +// resourceDeletionStatusTracker keeps track of items pending deletion. type ResourceDeletionStatusTracker interface { // Add informs the tracker that a polling is in progress to check namespace deletion status. Add(kind, ns, name string) diff --git a/pkg/util/kube/utils.go b/pkg/util/kube/utils.go index ee798aeaf4..5d64f117ba 100644 --- a/pkg/util/kube/utils.go +++ b/pkg/util/kube/utils.go @@ -119,7 +119,7 @@ func EnsureNamespaceExistsAndIsReady(namespace *corev1api.Namespace, client core return false, nsCreated, errors.Wrapf(err, "timed out waiting for terminating namespace %s to disappear before restoring", namespace.Name) } else if namespaceAlreadyInDeletionTracker { // If the namespace is already in the tracker, return an error. - return false, nsCreated, err + return false, nsCreated, errors.Wrapf(err, "skipping polling for terminating namespace %s", namespace.Name) } return false, nsCreated, errors.Wrapf(err, "error getting namespace %s", namespace.Name) }