diff --git a/pkg/handlers/backup.go b/pkg/handlers/backup.go index 8a39ae2976..9a56fafee4 100644 --- a/pkg/handlers/backup.go +++ b/pkg/handlers/backup.go @@ -110,8 +110,8 @@ func (h *Handler) ListBackups(w http.ResponseWriter, r *http.Request) { } type ListInstanceBackupsResponse struct { - Error string `json:"error,omitempty"` - Backups []*snapshottypes.ReplicatedBackup `json:"backups"` + Error string `json:"error,omitempty"` + Backups []*snapshottypes.Backup `json:"backups"` } func (h *Handler) ListInstanceBackups(w http.ResponseWriter, r *http.Request) { diff --git a/pkg/kotsadmsnapshot/backup.go b/pkg/kotsadmsnapshot/backup.go index ed6cef89cb..502e43a811 100644 --- a/pkg/kotsadmsnapshot/backup.go +++ b/pkg/kotsadmsnapshot/backup.go @@ -747,7 +747,7 @@ func ListBackupsForApp(ctx context.Context, kotsadmNamespace string, appID strin backup := types.Backup{ Name: veleroBackup.Name, - Status: string(veleroBackup.Status.Phase), + Status: types.GetStatusFromBackupPhase(veleroBackup.Status.Phase), AppID: appID, } @@ -770,7 +770,7 @@ func ListBackupsForApp(ctx context.Context, kotsadmNamespace string, appID strin backup.Sequence = s } if backup.Status == "" { - backup.Status = "New" + backup.Status = types.BackupStatusInProgress } trigger, ok := veleroBackup.Annotations[types.BackupTriggerAnnotation] @@ -783,7 +783,7 @@ func ListBackupsForApp(ctx context.Context, kotsadmNamespace string, appID strin backup.SupportBundleID = supportBundleID } - if backup.Status != "New" && backup.Status != "InProgress" { + if backup.Status != types.BackupStatusInProgress { volumeSummary, err := getSnapshotVolumeSummary(ctx, &veleroBackup) if err != nil { return nil, errors.Wrap(err, "failed to get volume summary") @@ -801,7 +801,7 @@ func ListBackupsForApp(ctx context.Context, kotsadmNamespace string, appID strin return backups, nil } -func ListInstanceBackups(ctx context.Context, kotsadmNamespace string) ([]*types.ReplicatedBackup, error) { +func ListInstanceBackups(ctx context.Context, kotsadmNamespace string) ([]*types.Backup, error) { cfg, err := k8sutil.GetClusterConfig() if err != nil { return nil, errors.Wrap(err, "failed to get cluster config") @@ -831,94 +831,113 @@ func ListInstanceBackups(ctx context.Context, kotsadmNamespace string) ([]*types return nil, errors.Wrap(err, "failed to list velero backups") } - replicatedBackupsMap := map[string]*types.ReplicatedBackup{} + return getBackupsFromVeleroBackups(ctx, veleroBackups.Items) +} - for _, veleroBackup := range veleroBackups.Items { - // TODO: Enforce version? +// getBackupsFromVeleroBackups returns an array of `Backup` structs, consisting of Replicated's representation of a backup +// from an array of Velero backups `VolumeSummary`'s and . +func getBackupsFromVeleroBackups(ctx context.Context, veleroBackups []velerov1.Backup) ([]*types.Backup, error) { + result := make(map[string]*types.Backup, 0) + + for _, veleroBackup := range veleroBackups { + // filter out non instance backups if !types.IsInstanceBackup(veleroBackup) { continue } - - backup := types.Backup{ - Name: veleroBackup.Name, - Status: string(veleroBackup.Status.Phase), - IncludedApps: make([]types.App, 0), - } - - if veleroBackup.Status.StartTimestamp != nil { - backup.StartedAt = &veleroBackup.Status.StartTimestamp.Time - } - if veleroBackup.Status.CompletionTimestamp != nil { - backup.FinishedAt = &veleroBackup.Status.CompletionTimestamp.Time - } - if veleroBackup.Status.Expiration != nil { - backup.ExpiresAt = &veleroBackup.Status.Expiration.Time - } - if backup.Status == "" { - backup.Status = "New" + veleroStatus := veleroBackup.Status + backupName := types.GetBackupName(veleroBackup) + if _, ok := result[backupName]; !ok { + result[backupName] = &types.Backup{ + Name: backupName, + Status: types.GetStatusFromBackupPhase(veleroStatus.Phase), + Trigger: types.GetBackupTrigger(veleroBackup), + ExpectedBackupCount: types.GetInstanceBackupCount(veleroBackup), + IncludedApps: []types.App{}, + } } - - trigger, ok := veleroBackup.Annotations[types.BackupTriggerAnnotation] - if ok { - backup.Trigger = trigger + backup := result[backupName] + backup.BackupCount++ + // backup uses the oldest velero backup start time as its start time + if veleroStatus.StartTimestamp != nil { + if backup.StartedAt == nil || veleroStatus.StartTimestamp.Time.Before(*backup.StartedAt) { + backup.StartedAt = &veleroStatus.StartTimestamp.Time + } } - appAnnotationStr, _ := veleroBackup.Annotations[types.BackupAppsSequencesAnnotation] - if len(appAnnotationStr) > 0 { - var apps map[string]int64 - if err := json.Unmarshal([]byte(appAnnotationStr), &apps); err != nil { - return nil, errors.Wrap(err, "failed to unmarshal apps sequences") + // backup uses the first expiration date as its expiration timestamp + if veleroStatus.Expiration != nil { + if backup.ExpiresAt == nil || veleroStatus.Expiration.Time.Before(*backup.ExpiresAt) { + backup.ExpiresAt = &veleroStatus.Expiration.Time } - for slug, sequence := range apps { - a, err := store.GetStore().GetAppFromSlug(slug) - if err != nil { - if store.GetStore().IsNotFound(err) { - // app might not exist in current installation - continue - } - return nil, errors.Wrap(err, "failed to get app from slug") - } + } - backup.IncludedApps = append(backup.IncludedApps, types.App{ - Slug: slug, - Sequence: sequence, - Name: a.Name, - AppIconURI: a.IconURI, - }) + // backup uses the most recent completion date as its completion timestamp + if veleroStatus.CompletionTimestamp != nil { + if backup.FinishedAt == nil || veleroStatus.CompletionTimestamp.Time.After(*backup.FinishedAt) { + backup.FinishedAt = &veleroStatus.CompletionTimestamp.Time } } - // get volume information - if backup.Status != "New" && backup.Status != "InProgress" { - volumeSummary, err := getSnapshotVolumeSummary(ctx, &veleroBackup) - if err != nil { - return nil, errors.Wrap(err, "failed to get volume summary") - } + backup.Status = types.RollupStatus([]types.BackupStatus{backup.Status, types.GetStatusFromBackupPhase(veleroStatus.Phase)}) - backup.VolumeCount = volumeSummary.VolumeCount - backup.VolumeSuccessCount = volumeSummary.VolumeSuccessCount - backup.VolumeBytes = volumeSummary.VolumeBytes - backup.VolumeSizeHuman = volumeSummary.VolumeSizeHuman + // get volume information + volumeSummary, err := getSnapshotVolumeSummary(ctx, &veleroBackup) + if err != nil { + return nil, fmt.Errorf("failed to get volume summary for backup %s: %w", backupName, err) } - // group the velero backups by the name we present to the user - backupName := types.GetBackupName(veleroBackup) - if _, ok := replicatedBackupsMap[backupName]; !ok { - replicatedBackupsMap[backupName] = &types.ReplicatedBackup{ - Name: backupName, - Backups: []types.Backup{}, - ExpectedBackupCount: types.GetInstanceBackupCount(veleroBackup), - } + backup.VolumeCount += volumeSummary.VolumeCount + backup.VolumeSuccessCount += volumeSummary.VolumeSuccessCount + backup.VolumeBytes += volumeSummary.VolumeBytes + backup.VolumeSizeHuman = units.HumanSize(float64(backup.VolumeBytes)) + + apps, err := getAppsFromAppSequences(veleroBackup) + if err != nil { + return nil, fmt.Errorf("failed to get apps from app sequences for backup %s: %w", backupName, err) } - replicatedBackupsMap[backupName].Backups = append(replicatedBackupsMap[backupName].Backups, backup) + backup.IncludedApps = append(backup.IncludedApps, apps...) } - replicatedBackups := []*types.ReplicatedBackup{} - for _, rb := range replicatedBackupsMap { - replicatedBackups = append(replicatedBackups, rb) + backups := []*types.Backup{} + for _, backup := range result { + // we consider a backup to have failed if the number of backups that actually exist is less than the expected number + if backup.ExpectedBackupCount != backup.BackupCount { + backup.Status = types.BackupStatusFailed + } + backups = append(backups, backup) } - return replicatedBackups, nil + return backups, nil +} + +// getAppsFromAppSequences returns a list of `App` structs from the backup sequence annotation. +func getAppsFromAppSequences(veleroBackup velerov1.Backup) ([]types.App, error) { + apps := []types.App{} + appAnnotationStr, _ := veleroBackup.Annotations[types.BackupAppsSequencesAnnotation] + if len(appAnnotationStr) > 0 { + var appsSequences map[string]int64 + if err := json.Unmarshal([]byte(appAnnotationStr), &appsSequences); err != nil { + return nil, fmt.Errorf("failed to unmarshal apps sequences: %w", err) + } + for slug, sequence := range appsSequences { + a, err := store.GetStore().GetAppFromSlug(slug) + if err != nil { + if store.GetStore().IsNotFound(err) { + // app might not exist in current installation + continue + } + return nil, fmt.Errorf("failed to get app from slug: %w", err) + } + + apps = append(apps, types.App{ + Slug: slug, + Sequence: sequence, + Name: a.Name, + AppIconURI: a.IconURI, + }) + } + } + return apps, nil } func getSnapshotVolumeSummary(ctx context.Context, veleroBackup *velerov1.Backup) (*types.VolumeSummary, error) { @@ -1077,7 +1096,7 @@ func HasUnfinishedApplicationBackup(ctx context.Context, kotsadmNamespace string } for _, backup := range backups { - if backup.Status == "New" || backup.Status == "InProgress" { + if backup.Status == types.BackupStatusInProgress { return true, nil } } @@ -1086,16 +1105,14 @@ func HasUnfinishedApplicationBackup(ctx context.Context, kotsadmNamespace string } func HasUnfinishedInstanceBackup(ctx context.Context, kotsadmNamespace string) (bool, error) { - replicatedBackups, err := ListInstanceBackups(ctx, kotsadmNamespace) + backups, err := ListInstanceBackups(ctx, kotsadmNamespace) if err != nil { return false, errors.Wrap(err, "failed to list backups") } - for _, replicatedBackup := range replicatedBackups { - for _, backup := range replicatedBackup.Backups { - if backup.Status == "New" || backup.Status == "InProgress" { - return true, nil - } + for _, backup := range backups { + if backup.Status == types.BackupStatusInProgress { + return true, nil } } diff --git a/pkg/kotsadmsnapshot/backup_test.go b/pkg/kotsadmsnapshot/backup_test.go index d81fd78cc5..8c4889f9e5 100644 --- a/pkg/kotsadmsnapshot/backup_test.go +++ b/pkg/kotsadmsnapshot/backup_test.go @@ -3163,10 +3163,12 @@ func TestListBackupsForApp(t *testing.T) { }, expectedBackups: []*types.Backup{ { - AppID: "app-1", - Name: "app-backup-app-1", - Status: "Completed", - VolumeSizeHuman: "0B", + AppID: "app-1", + Name: "app-backup-app-1", + Status: "Completed", + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, }, }, }, @@ -3201,13 +3203,15 @@ func TestListBackupsForApp(t *testing.T) { }, expectedBackups: []*types.Backup{ { - AppID: "app-1", - Name: "app-backup-app-1", - Status: "Completed", - StartedAt: &startTs, - FinishedAt: &completionTs, - ExpiresAt: &expirationTs, - VolumeSizeHuman: "0B", + AppID: "app-1", + Name: "app-backup-app-1", + Status: "Completed", + StartedAt: &startTs, + FinishedAt: &completionTs, + ExpiresAt: &expirationTs, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, }, }, }, @@ -3255,14 +3259,16 @@ func TestListBackupsForApp(t *testing.T) { }, expectedBackups: []*types.Backup{ { - AppID: "app-1", - Name: "app-backup-app-1", - Status: "Completed", - Trigger: "schedule", - VolumeSizeHuman: "2kB", - VolumeBytes: 2000, - VolumeSuccessCount: 1, - VolumeCount: 1, + AppID: "app-1", + Name: "app-backup-app-1", + Status: "Completed", + Trigger: "schedule", + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "2kB", + VolumeBytes: 2000, + VolumeSuccessCount: 1, + VolumeCount: 1, + }, }, }, }, @@ -3331,7 +3337,7 @@ func TestListInstanceBackups(t *testing.T) { setup func(mockStore *mock_store.MockStore) veleroClientBuilder veleroclient.VeleroClientBuilder k8sClientBuilder k8sclient.K8sClientsetBuilder - expectedBackups []*types.ReplicatedBackup + expectedBackups []*types.Backup wantErr string }{ { @@ -3379,7 +3385,7 @@ func TestListInstanceBackups(t *testing.T) { testBsl, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{}, + expectedBackups: []*types.Backup{}, }, { name: "non instance backups are excluded", @@ -3415,17 +3421,15 @@ func TestListInstanceBackups(t *testing.T) { }, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{ + expectedBackups: []*types.Backup{ { Name: "instance-backup", ExpectedBackupCount: 1, - Backups: []types.Backup{ - { - Name: "instance-backup", - Status: "Completed", - IncludedApps: []types.App{}, - VolumeSizeHuman: "0B", - }, + BackupCount: 1, + Status: "Completed", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", }, }, }, @@ -3479,29 +3483,21 @@ func TestListInstanceBackups(t *testing.T) { }, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{ + expectedBackups: []*types.Backup{ { Name: "aggregated-repl-backup", ExpectedBackupCount: 2, - Backups: []types.Backup{ - { - Name: "app-backup", - Status: "Completed", - IncludedApps: []types.App{}, - VolumeSizeHuman: "0B", - }, - { - Name: "infra-backup", - Status: "Completed", - IncludedApps: []types.App{}, - VolumeSizeHuman: "0B", - }, + BackupCount: 2, + Status: "Completed", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", }, }, }, }, { - name: "included apps are populated ", + name: "included apps are populated", setup: func(mockStore *mock_store.MockStore) { mockStore.EXPECT().GetAppFromSlug("app-1").Times(1).Return(&apptypes.App{ ID: "1", @@ -3534,24 +3530,113 @@ func TestListInstanceBackups(t *testing.T) { }, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{ + expectedBackups: []*types.Backup{ { Name: "some-backup", + Status: "Completed", ExpectedBackupCount: 1, - Backups: []types.Backup{ + BackupCount: 1, + IncludedApps: []types.App{ { - Name: "some-backup", - Status: "Completed", - IncludedApps: []types.App{ - { - Slug: "app-1", - Sequence: 1, - Name: "App 1", - AppIconURI: "https://some-url.com/icon.png", - }, + Slug: "app-1", + Sequence: 1, + Name: "App 1", + AppIconURI: "https://some-url.com/icon.png", + }, + }, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, + }, + }, + }, + { + name: "if improved dr, included apps are populated from the included backups", + setup: func(mockStore *mock_store.MockStore) { + mockStore.EXPECT().GetAppFromSlug("app-1").Times(1).Return(&apptypes.App{ + ID: "1", + Name: "App 1", + Slug: "app-1", + IconURI: "https://some-url.com/icon.png", + }, nil) + mockStore.EXPECT().GetAppFromSlug("app-2").Times(1).Return(&apptypes.App{ + ID: "2", + Name: "App 2", + Slug: "app-2", + IconURI: "https://some-url.com/icon.png", + }, nil) + }, + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.BackupAppsSequencesAnnotation: "{\"app-1\":1}", + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", }, - VolumeSizeHuman: "0B", }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.BackupAppsSequencesAnnotation: "{\"app-2\":1}", + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + Status: "Completed", + ExpectedBackupCount: 2, + BackupCount: 2, + IncludedApps: []types.App{ + { + Slug: "app-2", + Sequence: 1, + Name: "App 2", + AppIconURI: "https://some-url.com/icon.png", + }, + { + Slug: "app-1", + Sequence: 1, + Name: "App 1", + AppIconURI: "https://some-url.com/icon.png", + }, + }, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", }, }, }, @@ -3584,20 +3669,89 @@ func TestListInstanceBackups(t *testing.T) { }, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{ + expectedBackups: []*types.Backup{ { Name: "some-backup", ExpectedBackupCount: 1, - Backups: []types.Backup{ - { - Name: "some-backup", - Status: "Completed", - StartedAt: &startTs, - FinishedAt: &completionTs, - ExpiresAt: &expirationTs, - IncludedApps: []types.App{}, - VolumeSizeHuman: "0B", + BackupCount: 1, + Status: "Completed", + StartedAt: &startTs, + FinishedAt: &completionTs, + ExpiresAt: &expirationTs, + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, + }, + }, + }, + { + name: "if improved dr, timestamps are populated based on the timestamps of the included backups", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + StartTimestamp: &metav1.Time{Time: startTs}, + CompletionTimestamp: &metav1.Time{Time: completionTs.Add(-1 * time.Minute)}, + Expiration: &metav1.Time{Time: expirationTs.Add(1 * time.Minute)}, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + StartTimestamp: &metav1.Time{Time: startTs.Add(1 * time.Minute)}, + CompletionTimestamp: &metav1.Time{Time: completionTs}, + Expiration: &metav1.Time{Time: expirationTs}, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 2, + Status: "Completed", + StartedAt: &startTs, + FinishedAt: &completionTs, + ExpiresAt: &expirationTs, + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", }, }, }, @@ -3643,22 +3797,345 @@ func TestListInstanceBackups(t *testing.T) { }, ).VeleroV1(), }, - expectedBackups: []*types.ReplicatedBackup{ + expectedBackups: []*types.Backup{ { Name: "some-backup-with-volumes", ExpectedBackupCount: 1, - Backups: []types.Backup{ - { - Name: "some-backup-with-volumes", - Status: "Completed", - Trigger: "manual", - VolumeSizeHuman: "2kB", - VolumeBytes: 2000, - VolumeSuccessCount: 1, - VolumeCount: 1, - IncludedApps: []types.App{}, + BackupCount: 1, + Status: "Completed", + Trigger: "manual", + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "2kB", + VolumeBytes: 2000, + VolumeSuccessCount: 1, + VolumeCount: 1, + }, + IncludedApps: []types.App{}, + }, + }, + }, + { + name: "if improved dr, volume info is populated from pod volume backups from both backups", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, }, }, + &velerov1.PodVolumeBackup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup-pod-volume-backup", + Namespace: "velero", + Labels: map[string]string{ + "velero.io/backup-name": "app-backup", + }, + }, + Status: velerov1.PodVolumeBackupStatus{ + Phase: velerov1.PodVolumeBackupPhaseCompleted, + Progress: velerov1.PodVolumeOperationProgress{ + BytesDone: 2000, + }, + }, + }, + &velerov1.PodVolumeBackup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup-pod-volume-backup", + Namespace: "velero", + Labels: map[string]string{ + "velero.io/backup-name": "infra-backup", + }, + }, + Status: velerov1.PodVolumeBackupStatus{ + Phase: velerov1.PodVolumeBackupPhaseCompleted, + Progress: velerov1.PodVolumeOperationProgress{ + BytesDone: 3000, + }, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 2, + Status: "Completed", + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "5kB", + VolumeBytes: 5000, + VolumeSuccessCount: 2, + VolumeCount: 2, + }, + IncludedApps: []types.App{}, + }, + }, + }, + { + name: "if expected backup count is not equal to actual backup count, it is marked as failed", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 1, + Status: "Failed", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, + }, + }, + }, + { + name: "status is in progress if any of the backups are in progress", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseInProgress, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 2, + Status: "InProgress", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, + }, + }, + }, + { + name: "status is deleting if any of the backups are deleting and none is in progress", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseDeleting, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 2, + Status: "Deleting", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, + }, + }, + }, + { + name: "status is failed if one backup is failed and the other completed", + k8sClientBuilder: &k8sclient.MockBuilder{ + Client: fake.NewSimpleClientset( + veleroNamespaceConfigmap, + veleroDeployment, + ), + }, + veleroClientBuilder: &veleroclient.MockBuilder{ + Client: velerofake.NewSimpleClientset( + testBsl, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "app-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeApp, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseCompleted, + }, + }, + &velerov1.Backup{ + ObjectMeta: metav1.ObjectMeta{ + Name: "infra-backup", + Namespace: "velero", + Labels: map[string]string{ + types.InstanceBackupNameLabel: "aggregated-repl-backup", + }, + Annotations: map[string]string{ + types.InstanceBackupVersionAnnotation: types.InstanceBackupVersionCurrent, + types.InstanceBackupAnnotation: "true", + types.InstanceBackupTypeAnnotation: types.InstanceBackupTypeInfra, + types.InstanceBackupCountAnnotation: "2", + }, + }, + Status: velerov1.BackupStatus{ + Phase: velerov1.BackupPhaseFailed, + }, + }, + ).VeleroV1(), + }, + expectedBackups: []*types.Backup{ + { + Name: "aggregated-repl-backup", + ExpectedBackupCount: 2, + BackupCount: 2, + Status: "Failed", + IncludedApps: []types.App{}, + VolumeSummary: types.VolumeSummary{ + VolumeSizeHuman: "0B", + }, }, }, }, diff --git a/pkg/kotsadmsnapshot/types/types.go b/pkg/kotsadmsnapshot/types/types.go index d07e888c58..be97cda5e3 100644 --- a/pkg/kotsadmsnapshot/types/types.go +++ b/pkg/kotsadmsnapshot/types/types.go @@ -2,6 +2,7 @@ package types import ( "strconv" + "strings" "time" velerov1 "github.com/vmware-tanzu/velero/pkg/apis/velero/v1" @@ -59,29 +60,38 @@ type App struct { AppIconURI string `json:"iconUri"` } -// ReplicatedBackup holds both the infrastructure and app backups for an EC cluster -type ReplicatedBackup struct { - Name string `json:"name"` - // number of backups expected to exist for the ReplicatedBackup to be considered complete - ExpectedBackupCount int `json:"expectedBackupCount"` - Backups []Backup `json:"backups"` -} +// BackupStatus represents the status of a backup +type BackupStatus string +const ( + // BackupStatusInProgress indicates that the backup is currently in progress + BackupStatusInProgress BackupStatus = "InProgress" + // BackupStatusCompleted indicates that the backup has been completed successfully + BackupStatusCompleted BackupStatus = "Completed" + // BackupStatusFailed indicates that the backup has failed + BackupStatusFailed BackupStatus = "Failed" + // BackupStatusDeleting indicates that the backup is being deleted + BackupStatusDeleting BackupStatus = "Deleting" +) + +// Backup represnts a replicated backup working as an abstraction layer between Replicated and +// Velero backups. These can be either infrastructure/instance, app backups or both. type Backup struct { - Name string `json:"name"` - Status string `json:"status"` - Trigger string `json:"trigger"` - AppID string `json:"appID"` // TODO: remove with app backups - Sequence int64 `json:"sequence"` // TODO: remove with app backups - StartedAt *time.Time `json:"startedAt,omitempty"` - FinishedAt *time.Time `json:"finishedAt,omitempty"` - ExpiresAt *time.Time `json:"expiresAt,omitempty"` - VolumeCount int `json:"volumeCount"` - VolumeSuccessCount int `json:"volumeSuccessCount"` - VolumeBytes int64 `json:"volumeBytes"` - VolumeSizeHuman string `json:"volumeSizeHuman"` - SupportBundleID string `json:"supportBundleId,omitempty"` - IncludedApps []App `json:"includedApps,omitempty"` + Name string `json:"name"` + Status BackupStatus `json:"status"` + Trigger string `json:"trigger"` + AppID string `json:"appID"` // TODO: remove with app backups + Sequence int64 `json:"sequence"` // TODO: remove with app backups + StartedAt *time.Time `json:"startedAt,omitempty"` + FinishedAt *time.Time `json:"finishedAt,omitempty"` + ExpiresAt *time.Time `json:"expiresAt,omitempty"` + SupportBundleID string `json:"supportBundleId,omitempty"` + IncludedApps []App `json:"includedApps,omitempty"` + // number of velero backups expected to exist for the Backup to be considered done + ExpectedBackupCount int `json:"expectedBackupCount"` + // number of velero backups that actually exist + BackupCount int `json:"backupCount"` + VolumeSummary } type BackupDetail struct { @@ -238,3 +248,45 @@ func GetInstanceBackupCount(veleroBackup velerov1.Backup) int { } return 1 } + +// GetBackupTrigger returns the trigger of the backup from the velero backup object annotation. +func GetBackupTrigger(veleroBackup velerov1.Backup) string { + if val, ok := veleroBackup.GetAnnotations()[BackupTriggerAnnotation]; ok { + return val + } + return "" +} + +// GetStatusFromBackupPhase returns our backup status from the velero backup phase. +func GetStatusFromBackupPhase(phase velerov1.BackupPhase) BackupStatus { + switch { + case phase == velerov1.BackupPhaseNew || phase == velerov1.BackupPhaseInProgress: + return BackupStatusInProgress + case phase == velerov1.BackupPhaseCompleted: + return BackupStatusCompleted + case strings.Contains(strings.ToLower(string(phase)), "fail"): + return BackupStatusFailed + case phase == velerov1.BackupPhaseDeleting: + return BackupStatusDeleting + default: + return BackupStatusInProgress + } +} + +// RollupStatus returns the overall status of a list of backup statuses. This is particularly useful when we have multiple +// velero backups for a single Replicated backup. +func RollupStatus(backupStatuses []BackupStatus) BackupStatus { + result := BackupStatusCompleted + + for _, backupStatus := range backupStatuses { + switch { + case backupStatus == BackupStatusInProgress: + return BackupStatusInProgress + case backupStatus == BackupStatusDeleting: + result = BackupStatusDeleting + case backupStatus == BackupStatusFailed && result != BackupStatusDeleting: + result = BackupStatusFailed + } + } + return result +} diff --git a/pkg/kotsadmsnapshot/types/types_test.go b/pkg/kotsadmsnapshot/types/types_test.go new file mode 100644 index 0000000000..6c14bfb1b8 --- /dev/null +++ b/pkg/kotsadmsnapshot/types/types_test.go @@ -0,0 +1,144 @@ +package types + +import ( + "slices" + "strings" + "testing" + + "github.com/stretchr/testify/assert" + velerov1 "github.com/vmware-tanzu/velero/pkg/apis/velero/v1" +) + +func TestRollupStatus(t *testing.T) { + tests := []struct { + backupStatuses []BackupStatus + expected BackupStatus + }{ + { + backupStatuses: []BackupStatus{ + BackupStatusInProgress, + BackupStatusInProgress, + }, + expected: BackupStatusInProgress, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusInProgress, + BackupStatusFailed, + }, + expected: BackupStatusInProgress, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusInProgress, + BackupStatusCompleted, + }, + expected: BackupStatusInProgress, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusInProgress, + BackupStatusDeleting, + }, + expected: BackupStatusInProgress, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusDeleting, + BackupStatusDeleting, + }, + expected: BackupStatusDeleting, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusDeleting, + BackupStatusFailed, + }, + expected: BackupStatusDeleting, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusDeleting, + BackupStatusCompleted, + }, + expected: BackupStatusDeleting, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusFailed, + BackupStatusFailed, + }, + expected: BackupStatusFailed, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusFailed, + BackupStatusCompleted, + }, + expected: BackupStatusFailed, + }, + { + backupStatuses: []BackupStatus{ + BackupStatusCompleted, + BackupStatusCompleted, + }, + expected: BackupStatusCompleted, + }, + } + + for _, test := range tests { + name := "" + for _, status := range test.backupStatuses { + name += string(status) + "-" + } + name = strings.TrimSuffix(name, "-") + t.Run(name, func(t *testing.T) { + assert.Equal(t, test.expected, RollupStatus(test.backupStatuses)) + // Reverse the order of the statuses and check if the result is the same + slices.Reverse(test.backupStatuses) + assert.Equal(t, test.expected, RollupStatus(test.backupStatuses)) + }) + } +} + +func TestGetStatusFromBackupPhase(t *testing.T) { + tests := []struct { + phase string + expected BackupStatus + }{ + { + phase: string(velerov1.BackupPhaseNew), + expected: BackupStatusInProgress, + }, + { + phase: string(velerov1.BackupPhaseInProgress), + expected: BackupStatusInProgress, + }, + { + phase: string(velerov1.BackupPhaseCompleted), + expected: BackupStatusCompleted, + }, + { + phase: string(velerov1.BackupPhaseFailed), + expected: BackupStatusFailed, + }, + { + phase: "SomeNewFailState", + expected: BackupStatusFailed, + }, + { + phase: string(velerov1.BackupPhaseDeleting), + expected: BackupStatusDeleting, + }, + { + phase: "SomeUnknownNewState", + expected: BackupStatusInProgress, + }, + } + + for _, test := range tests { + t.Run(test.phase, func(t *testing.T) { + assert.Equal(t, test.expected, GetStatusFromBackupPhase(velerov1.BackupPhase(test.phase))) + }) + } +} diff --git a/web/src/components/snapshots/Snapshots.jsx b/web/src/components/snapshots/Snapshots.jsx index d21f064fee..a2607c68f3 100644 --- a/web/src/components/snapshots/Snapshots.jsx +++ b/web/src/components/snapshots/Snapshots.jsx @@ -108,12 +108,8 @@ class Snapshots extends Component { } const response = await res.json(); - const snapshots = response.backups?.flatMap( - (replBackup) => replBackup?.backups - ); - this.setState({ - snapshots: snapshots?.sort((a, b) => + snapshots: response.backups?.sort((a, b) => b.startedAt ? new Date(b.startedAt) - new Date(a.startedAt) : -99999999