mirror of
https://github.com/vmware-tanzu/velero.git
synced 2026-01-04 20:24:02 +00:00
* log a warning instead of erroring if additional item can't be found Signed-off-by: Steve Kriss <krisss@vmware.com> * always show backup warning/error count in get/describe Signed-off-by: Steve Kriss <krisss@vmware.com> * changelog Signed-off-by: Steve Kriss <krisss@vmware.com>
130 lines
3.7 KiB
Go
130 lines
3.7 KiB
Go
/*
|
|
Copyright 2017, 2019, 2020 the Velero contributors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package output
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"sort"
|
|
"time"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/util/duration"
|
|
|
|
velerov1api "github.com/vmware-tanzu/velero/pkg/apis/velero/v1"
|
|
)
|
|
|
|
var (
|
|
backupColumns = []metav1.TableColumnDefinition{
|
|
// name needs Type and Format defined for the decorator to identify it:
|
|
// https://github.com/kubernetes/kubernetes/blob/v1.15.3/pkg/printers/tableprinter.go#L204
|
|
{Name: "Name", Type: "string", Format: "name"},
|
|
{Name: "Status"},
|
|
{Name: "Errors"},
|
|
{Name: "Warnings"},
|
|
{Name: "Created"},
|
|
{Name: "Expires"},
|
|
{Name: "Storage Location"},
|
|
{Name: "Selector"},
|
|
}
|
|
)
|
|
|
|
func printBackupList(list *velerov1api.BackupList) []metav1.TableRow {
|
|
sortBackupsByPrefixAndTimestamp(list)
|
|
rows := make([]metav1.TableRow, 0, len(list.Items))
|
|
|
|
for i := range list.Items {
|
|
rows = append(rows, printBackup(&list.Items[i])...)
|
|
}
|
|
return rows
|
|
}
|
|
|
|
// sort by default alphabetically, but if backups stem from a common schedule
|
|
// (detected by the presence of a 14-digit timestamp suffix), then within that
|
|
// group, sort by newest to oldest (i.e. prefix ASC, suffix DESC)
|
|
var timestampSuffix = regexp.MustCompile("-[0-9]{14}$")
|
|
|
|
func sortBackupsByPrefixAndTimestamp(list *velerov1api.BackupList) {
|
|
sort.Slice(list.Items, func(i, j int) bool {
|
|
iSuffixIndex := timestampSuffix.FindStringIndex(list.Items[i].Name)
|
|
jSuffixIndex := timestampSuffix.FindStringIndex(list.Items[j].Name)
|
|
|
|
// one/both don't have a timestamp suffix, so sort alphabetically
|
|
if iSuffixIndex == nil || jSuffixIndex == nil {
|
|
return list.Items[i].Name < list.Items[j].Name
|
|
}
|
|
|
|
// different prefixes, so sort alphabetically
|
|
if list.Items[i].Name[0:iSuffixIndex[0]] != list.Items[j].Name[0:jSuffixIndex[0]] {
|
|
return list.Items[i].Name < list.Items[j].Name
|
|
}
|
|
|
|
// same prefixes, so sort based on suffix (desc)
|
|
return list.Items[i].Name[iSuffixIndex[0]:] >= list.Items[j].Name[jSuffixIndex[0]:]
|
|
})
|
|
}
|
|
|
|
func printBackup(backup *velerov1api.Backup) []metav1.TableRow {
|
|
row := metav1.TableRow{
|
|
Object: runtime.RawExtension{Object: backup},
|
|
}
|
|
|
|
var expiration time.Time
|
|
if backup.Status.Expiration != nil {
|
|
expiration = backup.Status.Expiration.Time
|
|
}
|
|
if expiration.IsZero() && backup.Spec.TTL.Duration > 0 {
|
|
expiration = backup.CreationTimestamp.Add(backup.Spec.TTL.Duration)
|
|
}
|
|
|
|
status := string(backup.Status.Phase)
|
|
if status == "" {
|
|
status = string(velerov1api.BackupPhaseNew)
|
|
}
|
|
if backup.DeletionTimestamp != nil && !backup.DeletionTimestamp.Time.IsZero() {
|
|
status = "Deleting"
|
|
}
|
|
|
|
row.Cells = append(row.Cells,
|
|
backup.Name,
|
|
status,
|
|
backup.Status.Errors,
|
|
backup.Status.Warnings,
|
|
backup.Status.StartTimestamp,
|
|
humanReadableTimeFromNow(expiration),
|
|
backup.Spec.StorageLocation,
|
|
metav1.FormatLabelSelector(backup.Spec.LabelSelector),
|
|
)
|
|
|
|
return []metav1.TableRow{row}
|
|
}
|
|
|
|
func humanReadableTimeFromNow(when time.Time) string {
|
|
if when.IsZero() {
|
|
return "n/a"
|
|
}
|
|
|
|
now := time.Now()
|
|
switch {
|
|
case when == now || when.After(now):
|
|
return duration.ShortHumanDuration(when.Sub(now))
|
|
default:
|
|
return fmt.Sprintf("%s ago", duration.ShortHumanDuration(now.Sub(when)))
|
|
}
|
|
}
|