diff --git a/apis/dataprotection/v1alpha1/backup_types.go b/apis/dataprotection/v1alpha1/backup_types.go index 25f93afe762..3ae54ec6c1d 100644 --- a/apis/dataprotection/v1alpha1/backup_types.go +++ b/apis/dataprotection/v1alpha1/backup_types.go @@ -194,6 +194,18 @@ type BackupStatus struct { // +optional VolumeSnapshots []VolumeSnapshotStatus `json:"volumeSnapshots,omitempty"` + // Records the parent backup name for incremental or differential backup. + // When the parent backup is deleted, the backup will also be deleted. + // + // +optional + ParentBackupName string `json:"parentBackupName,omitempty"` + + // Records the base full backup name for incremental backup or differential backup. + // When the base backup is deleted, the backup will also be deleted. + // + // +optional + BaseBackupName string `json:"baseBackupName,omitempty"` + // Records any additional information for the backup. // // +optional diff --git a/apis/dataprotection/v1alpha1/backuppolicy_types.go b/apis/dataprotection/v1alpha1/backuppolicy_types.go index a7610645715..71879a2cb0f 100644 --- a/apis/dataprotection/v1alpha1/backuppolicy_types.go +++ b/apis/dataprotection/v1alpha1/backuppolicy_types.go @@ -221,6 +221,12 @@ type BackupMethod struct { // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$` Name string `json:"name"` + // The name of the compatible full backup method, used by incremental backups. + // + // +optional + // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$` + CompatibleMethod string `json:"compatibleMethod,omitempty"` + // Specifies whether to take snapshots of persistent volumes. If true, // the ActionSetName is not required, the controller will use the CSI volume // snapshotter to create the snapshot. diff --git a/apis/dataprotection/v1alpha1/backuppolicytemplate_types.go b/apis/dataprotection/v1alpha1/backuppolicytemplate_types.go index 2a50aa6235f..360df823025 100644 --- a/apis/dataprotection/v1alpha1/backuppolicytemplate_types.go +++ b/apis/dataprotection/v1alpha1/backuppolicytemplate_types.go @@ -82,6 +82,12 @@ type BackupMethodTPL struct { // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$` Name string `json:"name"` + // The name of the compatible full backup method, used by incremental backups. + // + // +optional + // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$` + CompatibleMethod string `json:"compatibleMethod,omitempty"` + // Specifies whether to take snapshots of persistent volumes. If true, // the ActionSetName is not required, the controller will use the CSI volume // snapshotter to create the snapshot. diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml index e50acf17dcd..672973f1844 100644 --- a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml +++ b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml @@ -73,6 +73,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, + used by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicytemplates.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicytemplates.yaml index ba430dc6ac7..60d55ba99c2 100644 --- a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicytemplates.yaml +++ b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicytemplates.yaml @@ -76,6 +76,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, + used by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml index 23060c13a49..363975c4c09 100644 --- a/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml +++ b/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml @@ -298,6 +298,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, used + by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. @@ -1013,6 +1018,11 @@ spec: backupRepoName: description: The name of the backup repository. type: string + baseBackupName: + description: |- + Records the base full backup name for incremental backup or differential backup. + When the base backup is deleted, the backup will also be deleted. + type: string completionTimestamp: description: |- Records the time when the backup operation was completed. @@ -1092,6 +1102,11 @@ spec: kopiaRepoPath: description: Records the path of the Kopia repository. type: string + parentBackupName: + description: |- + Records the parent backup name for incremental or differential backup. + When the parent backup is deleted, the backup will also be deleted. + type: string path: description: |- The directory within the backup repository where the backup data is stored. diff --git a/controllers/apps/transformer_cluster_backup_policy.go b/controllers/apps/transformer_cluster_backup_policy.go index f6867accd71..249ab5a75d1 100644 --- a/controllers/apps/transformer_cluster_backup_policy.go +++ b/controllers/apps/transformer_cluster_backup_policy.go @@ -414,11 +414,12 @@ func (r *backupPolicyBuilder) syncBackupMethods(backupPolicy *dpv1alpha1.BackupP } for _, backupMethodTPL := range r.backupPolicyTPL.Spec.BackupMethods { backupMethod := dpv1alpha1.BackupMethod{ - Name: backupMethodTPL.Name, - ActionSetName: backupMethodTPL.ActionSetName, - SnapshotVolumes: backupMethodTPL.SnapshotVolumes, - TargetVolumes: backupMethodTPL.TargetVolumes, - RuntimeSettings: backupMethodTPL.RuntimeSettings, + Name: backupMethodTPL.Name, + CompatibleMethod: backupMethodTPL.CompatibleMethod, + ActionSetName: backupMethodTPL.ActionSetName, + SnapshotVolumes: backupMethodTPL.SnapshotVolumes, + TargetVolumes: backupMethodTPL.TargetVolumes, + RuntimeSettings: backupMethodTPL.RuntimeSettings, } if m, ok := oldBackupMethodMap[backupMethodTPL.Name]; ok { backupMethod = m diff --git a/controllers/dataprotection/actionset_controller_test.go b/controllers/dataprotection/actionset_controller_test.go index e3495edaf52..1697cd4e21d 100644 --- a/controllers/dataprotection/actionset_controller_test.go +++ b/controllers/dataprotection/actionset_controller_test.go @@ -55,14 +55,14 @@ var _ = Describe("ActionSet Controller test", func() { Context("create a actionSet", func() { It("should be available", func() { - as := testdp.NewFakeActionSet(&testCtx) + as := testdp.NewFakeActionSet(&testCtx, nil) Expect(as).ShouldNot(BeNil()) }) }) Context("validate a actionSet", func() { It("validate withParameters", func() { - as := testdp.NewFakeActionSet(&testCtx) + as := testdp.NewFakeActionSet(&testCtx, nil) Expect(as).ShouldNot(BeNil()) By("set invalid withParameters and schema") Expect(testapps.ChangeObj(&testCtx, as, func(action *dpv1alpha1.ActionSet) { diff --git a/controllers/dataprotection/backup_controller.go b/controllers/dataprotection/backup_controller.go index ee086f96dca..504048e39df 100644 --- a/controllers/dataprotection/backup_controller.go +++ b/controllers/dataprotection/backup_controller.go @@ -245,6 +245,11 @@ func (r *BackupReconciler) deleteBackupFiles(reqCtx intctrlutil.RequestCtx, back // handleDeletingPhase handles the deletion of backup. It will delete the backup CR // and the backup workload(job). func (r *BackupReconciler) handleDeletingPhase(reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) (ctrl.Result, error) { + // delete related backups + if err := r.deleteRelatedBackups(reqCtx, backup); err != nil { + return intctrlutil.RequeueWithError(err, reqCtx.Log, "") + } + // if backup phase is Deleting, delete the backup reference workloads, // backup data stored in backup repository and volume snapshots. // TODO(ldm): if backup is being used by restore, do not delete it. @@ -395,28 +400,11 @@ func (r *BackupReconciler) prepareBackupRequest( if err != nil { return nil, err } - request.ActionSet = actionSet - - // check continuous backups should have backupschedule label - if request.ActionSet.Spec.BackupType == dpv1alpha1.BackupTypeContinuous { - if _, ok := request.Labels[dptypes.BackupScheduleLabelKey]; !ok { - return nil, fmt.Errorf("continuous backup is only allowed to be created by backupSchedule") - } - backupSchedule := &dpv1alpha1.BackupSchedule{} - if err := request.Client.Get(reqCtx.Ctx, client.ObjectKey{Name: backup.Labels[dptypes.BackupScheduleLabelKey], - Namespace: backup.Namespace}, backupSchedule); err != nil { - return nil, err - } - if backupSchedule.Status.Phase != dpv1alpha1.BackupSchedulePhaseAvailable { - return nil, fmt.Errorf("create continuous backup by failed backupschedule %s/%s", - backupSchedule.Namespace, backupSchedule.Name) - } - } - // validate parameters if err := dputils.ValidateParameters(actionSet, backup.Spec.Parameters, true); err != nil { return nil, fmt.Errorf("fails to validate parameters with actionset %s: %v", actionSet.Name, err) } + request.ActionSet = actionSet } // check encryption config @@ -432,13 +420,25 @@ func (r *BackupReconciler) prepareBackupRequest( } request.BackupPolicy = backupPolicy + request.BackupMethod = backupMethod + + switch dpv1alpha1.BackupType(request.GetBackupType()) { + case dpv1alpha1.BackupTypeIncremental: + request, err = prepare4Incremental(request) + case dpv1alpha1.BackupTypeContinuous: + err = validateContinuousBackup(backup, reqCtx, request.Client) + } + if err != nil { + return nil, err + } + if !snapshotVolumes { // if use volume snapshot, ignore backup repo if err = HandleBackupRepo(request); err != nil { return nil, err } } - request.BackupMethod = backupMethod + return request, nil } @@ -527,6 +527,14 @@ func (r *BackupReconciler) patchBackupStatus( request.Status.Phase = dpv1alpha1.BackupPhaseRunning request.Status.StartTimestamp = &metav1.Time{Time: r.clock.Now().UTC()} + // set status parent backup and base backup name + if request.ParentBackup != nil { + request.Status.ParentBackupName = request.ParentBackup.Name + } + if request.BaseBackup != nil { + request.Status.BaseBackupName = request.BaseBackup.Name + } + if err = dpbackup.SetExpirationByCreationTime(request.Backup); err != nil { return err } @@ -751,6 +759,33 @@ func (r *BackupReconciler) deleteExternalResources( return deleteRelatedObjectList(reqCtx, r.Client, &appsv1.StatefulSetList{}, namespaces, labels) } +// deleteRelatedBackups deletes the related backups. +func (r *BackupReconciler) deleteRelatedBackups( + reqCtx intctrlutil.RequestCtx, + backup *dpv1alpha1.Backup) error { + backupList := &dpv1alpha1.BackupList{} + labels := map[string]string{ + dptypes.BackupPolicyLabelKey: backup.Spec.BackupPolicyName, + } + if err := r.Client.List(reqCtx.Ctx, backupList, + client.InNamespace(backup.Namespace), client.MatchingLabels(labels)); client.IgnoreNotFound(err) != nil { + return err + } + for i := range backupList.Items { + bp := &backupList.Items[i] + // delete backups related to the current backup + // files in the related backup's status.path will be deleted by its own associated deleter + if bp.Status.ParentBackupName != backup.Name && bp.Status.BaseBackupName != backup.Name { + continue + } + if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, bp); err != nil { + return err + } + reqCtx.Log.Info("delete the related backup", "backup", fmt.Sprintf("%s/%s", bp.Namespace, bp.Name)) + } + return nil +} + // PatchBackupObjectMeta patches backup object metaObject include cluster snapshot. func PatchBackupObjectMeta( original *dpv1alpha1.Backup, @@ -922,3 +957,56 @@ func setClusterSnapshotAnnotation(request *dpbackup.Request, cluster *kbappsv1.C request.Backup.Annotations[constant.ClusterSnapshotAnnotationKey] = *clusterString return nil } + +// validateContinuousBackup validates the continuous backup. +func validateContinuousBackup(backup *dpv1alpha1.Backup, reqCtx intctrlutil.RequestCtx, cli client.Client) error { + // validate if the continuous backup is created by a backupSchedule. + if _, ok := backup.Labels[dptypes.BackupScheduleLabelKey]; !ok { + return fmt.Errorf("continuous backup is only allowed to be created by backupSchedule") + } + backupSchedule := &dpv1alpha1.BackupSchedule{} + if err := cli.Get(reqCtx.Ctx, client.ObjectKey{Name: backup.Labels[dptypes.BackupScheduleLabelKey], + Namespace: backup.Namespace}, backupSchedule); err != nil { + return err + } + if backupSchedule.Status.Phase != dpv1alpha1.BackupSchedulePhaseAvailable { + return fmt.Errorf("create continuous backup by failed backupschedule %s/%s", + backupSchedule.Namespace, backupSchedule.Name) + } + return nil +} + +// prepare4Incremental prepares for incremental backup +func prepare4Incremental(request *dpbackup.Request) (*dpbackup.Request, error) { + // get and validate parent backup + parentBackup, err := GetParentBackup(request.Ctx, request.Client, request.Backup, request.BackupMethod) + if err != nil { + return nil, err + } + parentBackupType, err := dputils.GetBackupTypeByMethodName(request.RequestCtx, + request.Client, parentBackup.Spec.BackupMethod, request.BackupPolicy) + if err != nil { + return nil, err + } + request.ParentBackup = parentBackup + // get and validate base backup + switch parentBackupType { + case dpv1alpha1.BackupTypeFull: + request.BaseBackup = request.ParentBackup + case dpv1alpha1.BackupTypeIncremental: + baseBackup := &dpv1alpha1.Backup{} + baseBackupName := request.ParentBackup.Status.BaseBackupName + if len(baseBackupName) == 0 { + return nil, fmt.Errorf("backup %s/%s base backup name is empty", + request.ParentBackup.Namespace, request.ParentBackup.Name) + } + if err := request.Client.Get(request.Ctx, client.ObjectKey{Name: baseBackupName, + Namespace: request.ParentBackup.Namespace}, baseBackup); err != nil { + return nil, fmt.Errorf("failed to get base backup %s/%s: %w", request.ParentBackup.Namespace, baseBackupName, err) + } + request.BaseBackup = baseBackup + default: + return nil, fmt.Errorf("parent backup type is %s, but only full and incremental backup are supported", parentBackupType) + } + return request, nil +} diff --git a/controllers/dataprotection/backup_controller_test.go b/controllers/dataprotection/backup_controller_test.go index f9ceb194bb9..702b69e8a57 100644 --- a/controllers/dataprotection/backup_controller_test.go +++ b/controllers/dataprotection/backup_controller_test.go @@ -100,6 +100,7 @@ var _ = Describe("Backup Controller test", func() { When("with default settings", func() { var ( actionSet *dpv1alpha1.ActionSet + incActionSet *dpv1alpha1.ActionSet backupPolicy *dpv1alpha1.BackupPolicy repoPVCName string cluster *kbappsv1.Cluster @@ -108,8 +109,9 @@ var _ = Describe("Backup Controller test", func() { ) BeforeEach(func() { - By("creating an actionSet") - actionSet = testdp.NewFakeActionSet(&testCtx) + By("creating actionSets") + actionSet = testdp.NewFakeActionSet(&testCtx, nil) + incActionSet = testdp.NewFakeIncActionSet(&testCtx) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) @@ -117,7 +119,7 @@ var _ = Describe("Backup Controller test", func() { By("creating backup repo") _, repoPVCName = testdp.NewFakeBackupRepo(&testCtx, nil) - By("creating a backupPolicy from actionSet: " + actionSet.Name) + By("creating a backupPolicy from actionSets: " + actionSet.Name + ", " + incActionSet.Name) backupPolicy = testdp.NewFakeBackupPolicy(&testCtx, nil) cluster = clusterInfo.Cluster @@ -365,7 +367,7 @@ var _ = Describe("Backup Controller test", func() { })).Should(Succeed()) }) - It("create an backup using fallbackLabelSelector", func() { + It("create a backup using fallbackLabelSelector", func() { podFactory := func(name string) *testapps.MockPodFactory { return testapps.NewPodFactory(testCtx.DefaultNamespace, name). AddAppInstanceLabel(testdp.ClusterName). @@ -779,6 +781,161 @@ var _ = Describe("Backup Controller test", func() { })).Should(Succeed()) }) }) + + Context("create incremental backup", func() { + const ( + incBackupName = "inc-backup-" + scheduleName = "schedule" + ) + var ( + fullBackup *dpv1alpha1.Backup + fullBackupKey types.NamespacedName + now = metav1.Now() + ) + + getJobKey := func(bp *dpv1alpha1.Backup) client.ObjectKey { + return client.ObjectKey{ + Name: dpbackup.GenerateBackupJobName(bp, dpbackup.BackupDataJobNamePrefix+"-0"), + Namespace: bp.Namespace, + } + } + + newFakeIncBackup := func(name, parentName string, scheduled bool) *dpv1alpha1.Backup { + return testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) { + backup.Name = name + backup.Spec.BackupMethod = testdp.IncBackupMethodName + backup.Spec.ParentBackupName = parentName + if scheduled { + backup.Labels[dptypes.BackupScheduleLabelKey] = scheduleName + } + }) + } + + step := func() *metav1.Time { + bak := now + now = metav1.Time{Time: now.Add(time.Hour)} + return &bak + } + + mockBackupStatus := func(backup *dpv1alpha1.Backup, parentBackup, baseBackup string) { + backupStatus := dpv1alpha1.BackupStatus{ + Phase: dpv1alpha1.BackupPhaseCompleted, + ParentBackupName: parentBackup, + BaseBackupName: baseBackup, + TimeRange: &dpv1alpha1.BackupTimeRange{ + Start: step(), + End: step(), + }, + } + testdp.PatchBackupStatus(&testCtx, client.ObjectKeyFromObject(backup), backupStatus) + } + + checkBackupParentAndBase := func(backup *dpv1alpha1.Backup, parentBackup, baseBackup string) { + Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, fetched *dpv1alpha1.Backup) { + g.Expect(fetched.Status.ParentBackupName).NotTo(HaveLen(0)) + g.Expect(fetched.Status.ParentBackupName).To(Equal(parentBackup)) + g.Expect(fetched.Status.BaseBackupName).NotTo(HaveLen(0)) + g.Expect(fetched.Status.BaseBackupName).To(Equal(baseBackup)) + g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseRunning)) + })).Should(Succeed()) + } + + checkBackupCompleted := func(backup *dpv1alpha1.Backup) { + testdp.PatchK8sJobStatus(&testCtx, getJobKey(backup), batchv1.JobComplete) + Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, fetched *dpv1alpha1.Backup) { + g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseCompleted)) + })).Should(Succeed()) + } + + checkBackupDeleting := func(backup types.NamespacedName) { + Eventually(testapps.CheckObj(&testCtx, backup, func(g Gomega, fetched *dpv1alpha1.Backup) { + g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseDeleting)) + })).Should(Succeed()) + } + + mockIncBackupAndComplete := func(scheduled bool, backupName, parentName, expectedParent, expectedBase string) types.NamespacedName { + incBackup := newFakeIncBackup(backupName, parentName, scheduled) + By("check backup parent and base") + checkBackupParentAndBase(incBackup, expectedParent, expectedBase) + By("check backup completed") + checkBackupCompleted(incBackup) + mockBackupStatus(incBackup, expectedParent, expectedBase) + return client.ObjectKeyFromObject(incBackup) + } + + BeforeEach(func() { + By("creating a full backup from backupPolicy " + testdp.BackupPolicyName) //nolint:goconst + fullBackup = testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) { + backup.Labels[dptypes.BackupScheduleLabelKey] = scheduleName + }) + fullBackupKey = client.ObjectKeyFromObject(fullBackup) + }) + + It("creates an incremental backup based on a specific backup", func() { + By("waiting for the full backup " + fullBackupKey.String() + " to complete") + checkBackupCompleted(fullBackup) + mockBackupStatus(fullBackup, "", "") + By("creating an incremental backup from the specific full backup " + fullBackupKey.String()) + incBackup1 := mockIncBackupAndComplete(false, incBackupName+"1", fullBackup.Name, fullBackup.Name, fullBackup.Name) + By("creating an incremental backup from the specific incremental backup " + incBackup1.String()) + _ = mockIncBackupAndComplete(false, incBackupName+"2", incBackup1.Name, incBackup1.Name, fullBackup.Name) + }) + + It("creates an incremental backup without specific backup", func() { + By("waiting for the full backup " + fullBackupKey.String() + " to complete") + checkBackupCompleted(fullBackup) + mockBackupStatus(fullBackup, "", "") + By("creating an incremental backup" + incBackupName + "1 without specific backup") + incBackup1 := mockIncBackupAndComplete(true, incBackupName+"1", "", fullBackup.Name, fullBackup.Name) + By("creating an incremental backup" + incBackupName + "2 without specific backup") + _ = mockIncBackupAndComplete(false, incBackupName+"2", "", incBackup1.Name, fullBackup.Name) + By("creating an incremental backup" + incBackupName + "3 with the schedule label, it prefers the latest schedule backup as parent") + incBackup3 := mockIncBackupAndComplete(true, incBackupName+"3", "", incBackup1.Name, fullBackup.Name) + By("creating an incremental backup" + incBackupName + "4 without the schedule label, it prefers the latest backup as parent") + _ = mockIncBackupAndComplete(false, incBackupName+"4", "", incBackup3.Name, fullBackup.Name) + By("creating a new full backup from backupPolicy " + testdp.BackupPolicyName) + fullBackup1 := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) { + backup.Name = "full-bakcup-1" + }) + fullBackupKey1 := client.ObjectKeyFromObject(fullBackup1) + By("waiting for the full backup " + fullBackupKey1.String() + " to complete") + checkBackupCompleted(fullBackup1) + mockBackupStatus(fullBackup1, "", "") + By("creating an incremental backup " + incBackupName + "5, it prefers the latest full backup as parent") + _ = mockIncBackupAndComplete(false, incBackupName+"5", "", fullBackup1.Name, fullBackup1.Name) + + }) + + It("creates an incremental backup without valid parent backups", func() { + By("creating an incremental backup without specific parent backup") + incBackup1 := newFakeIncBackup(incBackupName+"1", "", false) + incBackupKey1 := client.ObjectKeyFromObject(incBackup1) + By("check backup failed") + Eventually(testapps.CheckObj(&testCtx, incBackupKey1, func(g Gomega, fetched *dpv1alpha1.Backup) { + g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed)) + })).Should(Succeed()) + }) + + It("deletes incremental backups", func() { + By("waiting for the full backup to complete, the full backup: " + fullBackupKey.String()) + checkBackupCompleted(fullBackup) + mockBackupStatus(fullBackup, "", "") + By("creating an incremental backup " + incBackupName + "1") + incBackup1 := mockIncBackupAndComplete(false, incBackupName+"1", fullBackup.Name, fullBackup.Name, fullBackup.Name) + By("creating an incremental backup " + incBackupName + "2") + incBackup2 := mockIncBackupAndComplete(false, incBackupName+"2", incBackup1.Name, incBackup1.Name, fullBackup.Name) + By("creating an incremental backup " + incBackupName + "3") + incBackup3 := mockIncBackupAndComplete(false, incBackupName+"3", "", incBackup2.Name, fullBackup.Name) + By("deleting an incremental backup" + incBackupName + "2 will delete its child backup") + testapps.DeleteObject(&testCtx, incBackup2, &dpv1alpha1.Backup{}) + checkBackupDeleting(incBackup2) + checkBackupDeleting(incBackup3) + By("deleting a base backup" + fullBackupKey.String() + " will delete all related incremental backups") + testapps.DeleteObject(&testCtx, fullBackupKey, &dpv1alpha1.Backup{}) + checkBackupDeleting(fullBackupKey) + checkBackupDeleting(incBackup1) + }) + }) }) When("with exceptional settings", func() { @@ -855,7 +1012,7 @@ var _ = Describe("Backup Controller test", func() { }) It("should fail because actionSet's backup type isn't Full", func() { - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) actionSetKey := client.ObjectKeyFromObject(actionSet) Eventually(testapps.GetAndChangeObj(&testCtx, actionSetKey, func(fetched *dpv1alpha1.ActionSet) { fetched.Spec.BackupType = dpv1alpha1.BackupTypeIncremental @@ -874,7 +1031,7 @@ var _ = Describe("Backup Controller test", func() { Context("create continuous backup", func() { It("should fail when continuous backup don't have backupschedule label", func() { By("create actionset with continuous backuptype") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) actionSetKey := client.ObjectKeyFromObject(actionSet) Eventually(testapps.GetAndChangeObj(&testCtx, actionSetKey, func(fetched *dpv1alpha1.ActionSet) { fetched.Spec.BackupType = dpv1alpha1.BackupTypeContinuous @@ -894,7 +1051,7 @@ var _ = Describe("Backup Controller test", func() { It("continue reconcile when continuous backup is Failed after fixing the issue", func() { By("create actionset and backupRepo for continuous backup") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(actionSet), func(fetched *dpv1alpha1.ActionSet) { fetched.Spec.BackupType = dpv1alpha1.BackupTypeContinuous })).Should(Succeed()) @@ -977,7 +1134,7 @@ var _ = Describe("Backup Controller test", func() { repo, repoPVCName = testdp.NewFakeBackupRepo(&testCtx, nil) By("creating actionSet") - _ = testdp.NewFakeActionSet(&testCtx) + _ = testdp.NewFakeActionSet(&testCtx, nil) }) Context("explicitly specify backup repo", func() { @@ -1128,7 +1285,7 @@ var _ = Describe("Backup Controller test", func() { BeforeEach(func() { By("creating an actionSet") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) diff --git a/controllers/dataprotection/backuppolicy_controller_test.go b/controllers/dataprotection/backuppolicy_controller_test.go index d2625fa1c65..c9df090a4a6 100644 --- a/controllers/dataprotection/backuppolicy_controller_test.go +++ b/controllers/dataprotection/backuppolicy_controller_test.go @@ -75,7 +75,7 @@ var _ = Describe("BackupPolicy Controller test", func() { It("backup policy should be available for target", func() { By("creating actionSet used by backup policy") - as := testdp.NewFakeActionSet(&testCtx) + as := testdp.NewFakeActionSet(&testCtx, nil) Expect(as).ShouldNot(BeNil()) By("creating backupPolicy and its status should be available") @@ -85,7 +85,7 @@ var _ = Describe("BackupPolicy Controller test", func() { It("test backup policy with targets", func() { By("creating actionSet used by backup policy") - as := testdp.NewFakeActionSet(&testCtx) + as := testdp.NewFakeActionSet(&testCtx, nil) Expect(as).ShouldNot(BeNil()) By("creating backupPolicy") diff --git a/controllers/dataprotection/backuppolicytemplate_controller_test.go b/controllers/dataprotection/backuppolicytemplate_controller_test.go index 8a1f419127c..9f4c35d2f23 100644 --- a/controllers/dataprotection/backuppolicytemplate_controller_test.go +++ b/controllers/dataprotection/backuppolicytemplate_controller_test.go @@ -96,7 +96,7 @@ var _ = Describe("", func() { })).Should(Succeed()) By("should be available") - testdp.NewFakeActionSet(&testCtx) + testdp.NewFakeActionSet(&testCtx, nil) Eventually(testapps.CheckObj(&testCtx, key, func(g Gomega, pobj *dpv1alpha1.BackupPolicyTemplate) { g.Expect(pobj.Status.ObservedGeneration).To(Equal(bpt.Generation)) g.Expect(pobj.Status.Phase).To(Equal(dpv1alpha1.AvailablePhase)) @@ -109,7 +109,7 @@ var _ = Describe("", func() { scheduleName2 = "test2" ) By("set backup parameters and schema in acitionSet") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) testdp.MockActionSetWithSchema(&testCtx, actionSet) bpt := testdp.NewBackupPolicyTemplateFactory(BackupPolicyTemplateName). AddBackupMethod(BackupMethod, false, testdp.ActionSetName). diff --git a/controllers/dataprotection/backupschedule_controller_test.go b/controllers/dataprotection/backupschedule_controller_test.go index 25c4a82ab04..bc7687c6983 100644 --- a/controllers/dataprotection/backupschedule_controller_test.go +++ b/controllers/dataprotection/backupschedule_controller_test.go @@ -97,7 +97,7 @@ var _ = Describe("Backup Schedule Controller", func() { BeforeEach(func() { By("creating an actionSet") - actionSet = testdp.NewFakeActionSet(&testCtx) + actionSet = testdp.NewFakeActionSet(&testCtx, nil) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) diff --git a/controllers/dataprotection/gc_controller_test.go b/controllers/dataprotection/gc_controller_test.go index 150a16aa350..6d8e07639e3 100644 --- a/controllers/dataprotection/gc_controller_test.go +++ b/controllers/dataprotection/gc_controller_test.go @@ -93,7 +93,7 @@ var _ = Describe("Data Protection Garbage Collection Controller", func() { BeforeEach(func() { By("creating an actionSet") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) diff --git a/controllers/dataprotection/log_collection_controller_test.go b/controllers/dataprotection/log_collection_controller_test.go index b4936e04dba..5c93ae02de9 100644 --- a/controllers/dataprotection/log_collection_controller_test.go +++ b/controllers/dataprotection/log_collection_controller_test.go @@ -80,7 +80,7 @@ var _ = Describe("Log Collection Controller", func() { BeforeEach(func() { By("creating an actionSet") - actionSet := testdp.NewFakeActionSet(&testCtx) + actionSet := testdp.NewFakeActionSet(&testCtx, nil) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) diff --git a/controllers/dataprotection/restore_controller_test.go b/controllers/dataprotection/restore_controller_test.go index e5cd5c9e944..e8c03eebd15 100644 --- a/controllers/dataprotection/restore_controller_test.go +++ b/controllers/dataprotection/restore_controller_test.go @@ -23,12 +23,14 @@ import ( "fmt" "strconv" "strings" + "time" . "github.com/onsi/ginkgo/v2" . "github.com/onsi/gomega" batchv1 "k8s.io/api/batch/v1" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/meta" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/types" "sigs.k8s.io/controller-runtime/pkg/client" @@ -120,7 +122,7 @@ var _ = Describe("Restore Controller test", func() { BeforeEach(func() { By("creating an actionSet") - actionSet = testdp.NewFakeActionSet(&testCtx) + actionSet = testdp.NewFakeActionSet(&testCtx, nil) By("creating storage provider") _ = testdp.NewFakeStorageProvider(&testCtx, nil) @@ -133,11 +135,15 @@ var _ = Describe("Restore Controller test", func() { mockBackupCompleted, useVolumeSnapshot, isSerialPolicy bool, + backupType dpv1alpha1.BackupType, expectRestorePhase dpv1alpha1.RestorePhase, change func(f *testdp.MockRestoreFactory), - changeBackupStatus func(b *dpv1alpha1.Backup)) *dpv1alpha1.Restore { + changeBackupStatus func(b *dpv1alpha1.Backup), + backupNames ...string, + ) *dpv1alpha1.Restore { By("create a completed backup") - backup := mockBackupForRestore(actionSet.Name, repo.Name, repoPVCName, mockBackupCompleted, useVolumeSnapshot) + backup := mockBackupForRestore(actionSet.Name, repo.Name, repoPVCName, mockBackupCompleted, + useVolumeSnapshot, backupType, backupNames...) if changeBackupStatus != nil { Expect(testapps.ChangeObjStatus(&testCtx, backup, func() { changeBackupStatus(backup) @@ -218,7 +224,7 @@ var _ = Describe("Restore Controller test", func() { } testRestoreWithVolumeClaimsTemplate := func(replicas, startingIndex int) { - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(replicas), int32(startingIndex), nil) @@ -259,7 +265,7 @@ var _ = Describe("Restore Controller test", func() { Context("with restore fails", func() { It("test restore is Failed when backup is not completed", func() { By("expect for restore is Failed ") - initResourcesAndWaitRestore(false, false, true, dpv1alpha1.RestorePhaseFailed, + initResourcesAndWaitRestore(false, false, true, "", dpv1alpha1.RestorePhaseFailed, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(3), int32(0), nil) @@ -268,7 +274,7 @@ var _ = Describe("Restore Controller test", func() { It("test restore is failed when check failed in new action", func() { By("expect for restore is Failed") - restore := initResourcesAndWaitRestore(true, false, true, dpv1alpha1.RestorePhaseFailed, + restore := initResourcesAndWaitRestore(true, false, true, "", dpv1alpha1.RestorePhaseFailed, func(f *testdp.MockRestoreFactory) { f.Get().Spec.Backup.Name = "wrongBackup" }, nil) @@ -281,7 +287,7 @@ var _ = Describe("Restore Controller test", func() { It("test restore is failed when validate failed in new action", func() { By("expect for restore is Failed") - restore := initResourcesAndWaitRestore(false, false, true, dpv1alpha1.RestorePhaseFailed, func(f *testdp.MockRestoreFactory) { + restore := initResourcesAndWaitRestore(false, false, true, "", dpv1alpha1.RestorePhaseFailed, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(3), int32(0), nil) }, nil) @@ -294,7 +300,7 @@ var _ = Describe("Restore Controller test", func() { It("test restore is Failed when restore job is not Failed", func() { By("expect for restore is Failed ") - restore := initResourcesAndWaitRestore(true, false, true, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, true, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(3), int32(0), nil) @@ -332,7 +338,7 @@ var _ = Describe("Restore Controller test", func() { testdp.MockActionSetWithSchema(&testCtx, actionSet) replicas := 3 startingIndex := 0 - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(replicas), int32(startingIndex), nil) @@ -349,7 +355,7 @@ var _ = Describe("Restore Controller test", func() { It("test volumeClaimsTemplate when volumeClaimRestorePolicy is Serial", func() { replicas := 2 startingIndex := 1 - restore := initResourcesAndWaitRestore(true, false, true, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, true, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(replicas), int32(startingIndex), nil) @@ -391,7 +397,7 @@ var _ = Describe("Restore Controller test", func() { }) It("test dataSourceRef", func() { - initResourcesAndWaitRestore(true, true, false, dpv1alpha1.RestorePhaseAsDataSource, + initResourcesAndWaitRestore(true, true, false, "", dpv1alpha1.RestorePhaseAsDataSource, func(f *testdp.MockRestoreFactory) { f.SetDataSourceRef(testdp.DataVolumeName, testdp.DataVolumeMountPath) }, nil) @@ -400,7 +406,7 @@ var _ = Describe("Restore Controller test", func() { It("test when dataRestorePolicy is OneToOne", func() { startingIndex := 0 restoredReplicas := 2 - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(restoredReplicas), int32(startingIndex), nil) @@ -438,7 +444,7 @@ var _ = Describe("Restore Controller test", func() { startingIndex := 0 restoredReplicas := 2 sourcePodName := "pod-0" - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, testdp.DataVolumeMountPath, "", int32(restoredReplicas), int32(startingIndex), nil) @@ -469,6 +475,7 @@ var _ = Describe("Restore Controller test", func() { By("mock jobs are completed and wait for restore is completed") mockAndCheckRestoreCompleted(restore) }) + }) Context("test postReady stage", func() { @@ -487,7 +494,7 @@ var _ = Describe("Restore Controller test", func() { matchLabels := map[string]string{ constant.AppInstanceLabelKey: testdp.ClusterName, } - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetConnectCredential(testdp.ClusterName).SetJobActionConfig(matchLabels).SetExecActionConfig(matchLabels) }, nil) @@ -525,7 +532,7 @@ var _ = Describe("Restore Controller test", func() { constant.AppInstanceLabelKey: testdp.ClusterName, } - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetJobActionConfig(matchLabels).SetExecActionConfig(matchLabels) }, func(b *dpv1alpha1.Backup) { @@ -581,7 +588,7 @@ var _ = Describe("Restore Controller test", func() { constant.AppInstanceLabelKey: testdp.ClusterName, } - restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.RestorePhaseRunning, + restore := initResourcesAndWaitRestore(true, false, false, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetJobActionConfig(matchLabels).SetExecActionConfig(matchLabels) f.SetParameters(testdp.TestParameters) @@ -596,17 +603,102 @@ var _ = Describe("Restore Controller test", func() { Context("test cross namespace", func() { It("should wait for preparation of backup repo", func() { By("creating a restore in a different namespace from backup") - initResourcesAndWaitRestore(true, false, true, dpv1alpha1.RestorePhaseRunning, + initResourcesAndWaitRestore(true, false, true, "", dpv1alpha1.RestorePhaseRunning, func(f *testdp.MockRestoreFactory) { f.SetNamespace(namespace2) }, nil) }) }) + + Context("test restore from incremental backup", func() { + var ( + baseBackup *dpv1alpha1.Backup + parentBackupName string + ancestorBackups = []*dpv1alpha1.Backup{} + cnt = 0 + ) + + genIncBackupName := func() string { + cnt++ + return fmt.Sprintf("inc-backup-%d", cnt) + } + + BeforeEach(func() { + By("mock completed full backup and parent incremental backup") + baseBackup = mockBackupForRestore(actionSet.Name, repo.Name, repoPVCName, true, false, dpv1alpha1.BackupTypeFull) + actionSet = testdp.NewFakeIncActionSet(&testCtx) + parentBackupName = baseBackup.Name + for i := 0; i < 3; i++ { + backup := mockBackupForRestore(actionSet.Name, repo.Name, repoPVCName, true, false, dpv1alpha1.BackupTypeIncremental, + genIncBackupName(), parentBackupName, baseBackup.Name) + ancestorBackups = append(ancestorBackups, backup) + parentBackupName = backup.Name + } + }) + + AfterEach(func() { + ancestorBackups = []*dpv1alpha1.Backup{} + cnt = 0 + }) + + It("test restore from incremental backup", func() { + replicas, startingIndex := 3, 0 + restore := initResourcesAndWaitRestore(true, false, false, dpv1alpha1.BackupTypeIncremental, dpv1alpha1.RestorePhaseRunning, + func(f *testdp.MockRestoreFactory) { + f.SetVolumeClaimsTemplate(testdp.MysqlTemplateName, testdp.DataVolumeName, + testdp.DataVolumeMountPath, "", int32(replicas), int32(startingIndex), nil) + f.SetPrepareDataRequiredPolicy(dpv1alpha1.OneToOneRestorePolicy, "") + }, nil, genIncBackupName(), parentBackupName, baseBackup.Name) + + By("wait for creating jobs and pvcs") + checkJobAndPVCSCount(restore, replicas, replicas, 0) + By("check job env") + ancestorIncrementalBackupNames := []string{} + for _, backup := range ancestorBackups { + ancestorIncrementalBackupNames = append(ancestorIncrementalBackupNames, backup.Name) + } + expectedEnv := map[string]string{ + dptypes.DPAncestorIncrementalBackupNames: strings.Join(ancestorIncrementalBackupNames, ","), + dptypes.DPBaseBackupName: baseBackup.Name, + } + jobList := &batchv1.JobList{} + Expect(k8sClient.List(ctx, jobList, + client.MatchingLabels{dprestore.DataProtectionRestoreLabelKey: restore.Name}, + client.InNamespace(testCtx.DefaultNamespace))).Should(Succeed()) + for _, job := range jobList.Items { + cnt := 0 + for _, env := range job.Spec.Template.Spec.Containers[0].Env { + if value, ok := expectedEnv[env.Name]; ok { + Expect(env.Value).Should(Equal(value)) + cnt++ + } + } + Expect(cnt).To(Equal(len(expectedEnv))) + } + By("mock jobs are completed and wait for restore is completed") + mockAndCheckRestoreCompleted(restore) + }) + }) }) }) -func mockBackupForRestore(actionSetName, repoName, backupPVCName string, mockBackupCompleted, useVolumeSnapshotBackup bool) *dpv1alpha1.Backup { - backup := testdp.NewFakeBackup(&testCtx, nil) +func mockBackupForRestore( + actionSetName, repoName, backupPVCName string, + mockBackupCompleted, useVolumeSnapshotBackup bool, + backupType dpv1alpha1.BackupType, + backupNames ...string, +) *dpv1alpha1.Backup { + backup := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) { + if len(backupNames) > 0 { + backup.Name = backupNames[0] + } + if backupType == dpv1alpha1.BackupTypeIncremental { + if len(backupNames) > 1 { + backup.Spec.ParentBackupName = backupNames[1] + } + backup.Spec.BackupMethod = testdp.IncBackupMethodName + } + }) // wait for backup is failed by backup controller. // it will be failed if the backupPolicy is not created. Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, tmpBackup *dpv1alpha1.Backup) { @@ -621,7 +713,14 @@ func mockBackupForRestore(actionSetName, repoName, backupPVCName string, mockBac backupMethodName = testdp.VSBackupMethodName testdp.MockBackupVSStatusActions(backup) } - backup.Status.Path = "/backup-data" + if backupType == dpv1alpha1.BackupTypeIncremental { + backupMethodName = testdp.IncBackupMethodName + if len(backupNames) > 2 { + backup.Status.ParentBackupName = backupNames[1] + backup.Status.BaseBackupName = backupNames[2] + } + } + backup.Status.Path = "/backup-data" + "/" + backup.Name backup.Status.Phase = dpv1alpha1.BackupPhaseCompleted backup.Status.BackupRepoName = repoName backup.Status.PersistentVolumeClaimName = backupPVCName @@ -631,6 +730,14 @@ func mockBackupForRestore(actionSetName, repoName, backupPVCName string, mockBac PortName: testdp.PortName, } testdp.MockBackupStatusMethod(backup, backupMethodName, testdp.DataVolumeName, actionSetName) + backup.Status.TimeRange = &dpv1alpha1.BackupTimeRange{ + Start: &metav1.Time{}, + End: &metav1.Time{}, + } + fakeClock.Step(time.Hour) + backup.Status.TimeRange.Start.Time = fakeClock.Now() + fakeClock.Step(time.Hour) + backup.Status.TimeRange.End.Time = fakeClock.Now() })).Should(Succeed()) } return backup diff --git a/controllers/dataprotection/utils.go b/controllers/dataprotection/utils.go index 4e80e905d37..eddd556312c 100644 --- a/controllers/dataprotection/utils.go +++ b/controllers/dataprotection/utils.go @@ -24,6 +24,7 @@ import ( "encoding/json" "fmt" "reflect" + "sort" "strings" "sync" @@ -581,6 +582,179 @@ func fromFlattenName(flatten string) (name string, namespace string) { return } +// GetParentBackup returns the parent backup of the backup. +// If parentBackupName is specified, the backup should be a on-demand backup, +// then validate and return the parent backup. +// If parentBackupName is not specified, find the latest valid parent backup. +func GetParentBackup(ctx context.Context, cli client.Client, backup *dpv1alpha1.Backup, + backupMethod *dpv1alpha1.BackupMethod) (*dpv1alpha1.Backup, error) { + if backup == nil || backupMethod == nil { + return nil, fmt.Errorf("backup or backupMethod is nil") + } + var scheduleName string + if schedule, ok := backup.Labels[dptypes.BackupScheduleLabelKey]; ok && len(schedule) > 0 { + scheduleName = schedule + } + parentBackup := &dpv1alpha1.Backup{} + if len(backup.Spec.ParentBackupName) != 0 { + // only on-demand backup can specify parent backup + if len(scheduleName) != 0 { + return nil, fmt.Errorf("schedule backup cannot specify parent backup") + } + if err := cli.Get(ctx, client.ObjectKey{ + Namespace: backup.Namespace, + Name: backup.Spec.ParentBackupName, + }, parentBackup); err != nil { + return nil, err + } + if err := ValidateParentBackup(backup, parentBackup, backupMethod); err != nil { + return nil, fmt.Errorf("failed to validate specified parent backup %s: %w", backup.Spec.ParentBackupName, err) + } + return parentBackup, nil + } + parentBackup, err := FindParentBackupIfNotSet(ctx, cli, backup, backupMethod, scheduleName) + if err != nil { + return nil, fmt.Errorf("failed to find parent backup: %w", err) + } + if parentBackup == nil { + return nil, fmt.Errorf("failed to find a valid parent backup for backup %s/%s", backup.Namespace, backup.Name) + } + return parentBackup, nil +} + +// FindParentBackupIfNotSet finds the latest valid parent backup for the incremental backup. +// a. return the latest full backup when it is newer than the base backup of the latest incremental backup, +// or when the base backup of the latest incremental backup is not found. +// b. return the latest incremental backup. +// c. return the latest full backup if incremental backups are not found. +// For scheduled backups, find the parent within scheduled backups, which have the schedule label, +// if not found, find the full backup as the parent within all backups. +// For on-demand backups, find the parent within all backups. +func FindParentBackupIfNotSet(ctx context.Context, cli client.Client, backup *dpv1alpha1.Backup, + backupMethod *dpv1alpha1.BackupMethod, scheduleName string) (*dpv1alpha1.Backup, error) { + getLatestBackup := func(backupList []*dpv1alpha1.Backup) *dpv1alpha1.Backup { + if len(backupList) == 0 { + return nil + } + // sort by stop time in descending order + sort.Slice(backupList, func(i, j int) bool { + i, j = j, i + return dputils.CompareWithBackupStopTime(*backupList[i], *backupList[j]) + }) + return backupList[0] + } + getLatestParentBackup := func(labels map[string]string, incremental bool) (*dpv1alpha1.Backup, error) { + backupList := &dpv1alpha1.BackupList{} + if err := cli.List(ctx, backupList, client.InNamespace(backup.Namespace), + client.MatchingLabels(labels)); err != nil && !apierrors.IsNotFound(err) { + return nil, err + } + filteredbackupList := FilterParentBackups(backupList, backup, backupMethod, incremental) + return getLatestBackup(filteredbackupList), nil + } + + labelMap := map[string]string{} + // with backup policy label + labelMap[dptypes.BackupPolicyLabelKey] = backup.Spec.BackupPolicyName + // with the schedule label if specified schedule + if len(scheduleName) != 0 { + labelMap[dptypes.BackupScheduleLabelKey] = scheduleName + } + // 1. get the latest incremental backups + labelMap[dptypes.BackupTypeLabelKey] = string(dpv1alpha1.BackupTypeIncremental) + latestIncrementalBackup, err := getLatestParentBackup(labelMap, true) + if err != nil { + return nil, err + } + // 2. get the latest full backups + labelMap[dptypes.BackupTypeLabelKey] = string(dpv1alpha1.BackupTypeFull) + latestFullBackup, err := getLatestParentBackup(labelMap, false) + if err != nil { + return nil, err + } + // 3. prefer the latest backup; if it is an incremental backup, it should be based on the latest full backup. + if latestIncrementalBackup != nil && latestFullBackup != nil { + if !dputils.CompareWithBackupStopTime(*latestIncrementalBackup, *latestFullBackup) && + latestIncrementalBackup.Status.BaseBackupName == latestFullBackup.Name { + return latestIncrementalBackup, nil + } + // the base backup of the latest incremental backup is not found, + // or the latest full backup is newer than the base backup of the latest incremental backup + return latestFullBackup, nil + } + // 4. get the latest unscheduled full backup if scheduled backups not found + if len(scheduleName) != 0 && latestFullBackup == nil { + delete(labelMap, dptypes.BackupScheduleLabelKey) + latestFullBackup, err = getLatestParentBackup(labelMap, false) + if err != nil { + return nil, err + } + } + // 5. only full backup found + if latestFullBackup != nil { + return latestFullBackup, nil + } + // illegal case: no full backup found but incremental backup found + if latestIncrementalBackup != nil { + return nil, fmt.Errorf("illegal incremental backup %s/%s", latestIncrementalBackup.Namespace, + latestIncrementalBackup.Name) + } + // 6. no backup found + return nil, nil +} + +// FilterParentBackups filters the parent backups by backup phase, backup method and end time. +func FilterParentBackups(backupList *dpv1alpha1.BackupList, targetBackup *dpv1alpha1.Backup, + backupMethod *dpv1alpha1.BackupMethod, incremental bool) []*dpv1alpha1.Backup { + var res []*dpv1alpha1.Backup + if backupList == nil || len(backupList.Items) == 0 { + return res + } + for i, backup := range backupList.Items { + if err := ValidateParentBackup(targetBackup, &backup, backupMethod); err != nil { + continue + } + // backups are listed by backup type label, validate if the backup method matches + // the backup type specified by label value. + if incremental { + if backup.Spec.BackupMethod != targetBackup.Spec.BackupMethod { + continue + } + } else { + if backup.Spec.BackupMethod != backupMethod.CompatibleMethod { + continue + } + } + res = append(res, &backupList.Items[i]) + } + return res +} + +// ValidateParentBackup validates the parent backup. +func ValidateParentBackup(backup *dpv1alpha1.Backup, parentBackup *dpv1alpha1.Backup, + backupMethod *dpv1alpha1.BackupMethod) error { + // validate parent backup is completed + if parentBackup.Status.Phase != dpv1alpha1.BackupPhaseCompleted { + return fmt.Errorf("parent backup %s/%s is not completed", parentBackup.Namespace, parentBackup.Name) + } + // validate if parent backup policy is consistent with the backup policy + if parentBackup.Spec.BackupPolicyName != backup.Spec.BackupPolicyName { + return fmt.Errorf("parent backup %s/%s policy %s is not consistent with the backup", + parentBackup.Namespace, parentBackup.Name, parentBackup.Spec.BackupPolicyName) + } + // validate if parent backup method is compatible with the backup method + if backup.Spec.BackupMethod != parentBackup.Spec.BackupMethod && + backupMethod.CompatibleMethod != parentBackup.Spec.BackupMethod { + return fmt.Errorf("parent backup %s/%s method %s is invalid for incremental backup", + parentBackup.Namespace, parentBackup.Name, parentBackup.Spec.BackupMethod) + } + // valiate parent end time + if parentBackup.GetEndTime().IsZero() { + return fmt.Errorf("parent backup %s/%s end time is zero", parentBackup.Namespace, parentBackup.Name) + } + return nil +} + // restore functions func getPopulatePVCName(pvcUID types.UID) string { diff --git a/controllers/dataprotection/volumepopulator_controller_test.go b/controllers/dataprotection/volumepopulator_controller_test.go index e7502f9cd17..571ddd0038c 100644 --- a/controllers/dataprotection/volumepopulator_controller_test.go +++ b/controllers/dataprotection/volumepopulator_controller_test.go @@ -101,7 +101,7 @@ var _ = Describe("Volume Populator Controller test", func() { BeforeEach(func() { By("create actionSet") - actionSet = testdp.NewFakeActionSet(&testCtx) + actionSet = testdp.NewFakeActionSet(&testCtx, nil) }) initResources := func(volumeBinding storagev1.VolumeBindingMode, useVolumeSnapshotBackup, mockBackupCompleted bool) *corev1.PersistentVolumeClaim { @@ -109,7 +109,7 @@ var _ = Describe("Volume Populator Controller test", func() { createStorageClass(volumeBinding) By("create backup") - backup := mockBackupForRestore(actionSet.Name, "", "", mockBackupCompleted, useVolumeSnapshotBackup) + backup := mockBackupForRestore(actionSet.Name, "", "", mockBackupCompleted, useVolumeSnapshotBackup, "") By("create restore ") restore := testdp.NewRestoreFactory(testCtx.DefaultNamespace, testdp.RestoreName). diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml index e50acf17dcd..672973f1844 100644 --- a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml +++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml @@ -73,6 +73,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, + used by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicytemplates.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicytemplates.yaml index ba430dc6ac7..60d55ba99c2 100644 --- a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicytemplates.yaml +++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicytemplates.yaml @@ -76,6 +76,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, + used by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml index 23060c13a49..363975c4c09 100644 --- a/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml +++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml @@ -298,6 +298,11 @@ spec: For volume snapshot backup, the actionSet is not required, the controller will use the CSI volume snapshotter to create the snapshot. type: string + compatibleMethod: + description: The name of the compatible full backup method, used + by incremental backups. + pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$ + type: string env: description: Specifies the environment variables for the backup workload. @@ -1013,6 +1018,11 @@ spec: backupRepoName: description: The name of the backup repository. type: string + baseBackupName: + description: |- + Records the base full backup name for incremental backup or differential backup. + When the base backup is deleted, the backup will also be deleted. + type: string completionTimestamp: description: |- Records the time when the backup operation was completed. @@ -1092,6 +1102,11 @@ spec: kopiaRepoPath: description: Records the path of the Kopia repository. type: string + parentBackupName: + description: |- + Records the parent backup name for incremental or differential backup. + When the parent backup is deleted, the backup will also be deleted. + type: string path: description: |- The directory within the backup repository where the backup data is stored. diff --git a/docs/developer_docs/api-reference/backup.md b/docs/developer_docs/api-reference/backup.md index ec66150f503..38f80be99fc 100644 --- a/docs/developer_docs/api-reference/backup.md +++ b/docs/developer_docs/api-reference/backup.md @@ -1915,6 +1915,18 @@ string
compatibleMethod
The name of the compatible full backup method, used by incremental backups.
+snapshotVolumes
compatibleMethod
The name of the compatible full backup method, used by incremental backups.
+snapshotVolumes
parentBackupName
Records the parent backup name for incremental or differential backup. +When the parent backup is deleted, the backup will also be deleted.
+baseBackupName
Records the base full backup name for incremental backup or differential backup. +When the base backup is deleted, the backup will also be deleted.
+extras