add support for bulk deletion to ark restore delete
refactor util methods into a common package move utility methods into cli package instead of common rename util.go to common.go Signed-off-by: Shubheksha Jalan <jshubheksha@gmail.com>pull/865/head
parent
a28327b47e
commit
e3232b7eb6
|
@ -17,10 +17,7 @@ limitations under the License.
|
||||||
package backup
|
package backup
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
@ -34,6 +31,7 @@ import (
|
||||||
"github.com/heptio/ark/pkg/backup"
|
"github.com/heptio/ark/pkg/backup"
|
||||||
"github.com/heptio/ark/pkg/client"
|
"github.com/heptio/ark/pkg/client"
|
||||||
"github.com/heptio/ark/pkg/cmd"
|
"github.com/heptio/ark/pkg/cmd"
|
||||||
|
"github.com/heptio/ark/pkg/cmd/cli"
|
||||||
"github.com/heptio/ark/pkg/cmd/util/flag"
|
"github.com/heptio/ark/pkg/cmd/util/flag"
|
||||||
clientset "github.com/heptio/ark/pkg/generated/clientset/versioned"
|
clientset "github.com/heptio/ark/pkg/generated/clientset/versioned"
|
||||||
)
|
)
|
||||||
|
@ -117,31 +115,16 @@ func (o *DeleteOptions) Validate(c *cobra.Command, args []string, f client.Facto
|
||||||
hasSelector = o.Selector.LabelSelector != nil
|
hasSelector = o.Selector.LabelSelector != nil
|
||||||
)
|
)
|
||||||
|
|
||||||
if !xor(hasNames, hasAll, hasSelector) {
|
if !cli.Xor(hasNames, hasAll, hasSelector) {
|
||||||
return errors.New("you must specify exactly one of: specific backup name(s), the --all flag, or the --selector flag")
|
return errors.New("you must specify exactly one of: specific backup name(s), the --all flag, or the --selector flag")
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// xor returns true if exactly one of the provided values is true,
|
|
||||||
// or false otherwise.
|
|
||||||
func xor(val bool, vals ...bool) bool {
|
|
||||||
res := val
|
|
||||||
|
|
||||||
for _, v := range vals {
|
|
||||||
if res && v {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
res = res || v
|
|
||||||
}
|
|
||||||
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run performs the delete backup operation.
|
// Run performs the delete backup operation.
|
||||||
func (o *DeleteOptions) Run() error {
|
func (o *DeleteOptions) Run() error {
|
||||||
if !o.Confirm && !getConfirmation() {
|
if !o.Confirm && !cli.GetConfirmation() {
|
||||||
// Don't do anything unless we get confirmation
|
// Don't do anything unless we get confirmation
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -197,28 +180,3 @@ func (o *DeleteOptions) Run() error {
|
||||||
|
|
||||||
return kubeerrs.NewAggregate(errs)
|
return kubeerrs.NewAggregate(errs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getConfirmation() bool {
|
|
||||||
reader := bufio.NewReader(os.Stdin)
|
|
||||||
|
|
||||||
for {
|
|
||||||
fmt.Printf("Are you sure you want to continue (Y/N)? ")
|
|
||||||
|
|
||||||
confirmation, err := reader.ReadString('\n')
|
|
||||||
if err != nil {
|
|
||||||
fmt.Fprintf(os.Stderr, "error reading user input: %v\n", err)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
confirmation = strings.TrimSpace(confirmation)
|
|
||||||
if len(confirmation) != 1 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
switch strings.ToLower(confirmation) {
|
|
||||||
case "y":
|
|
||||||
return true
|
|
||||||
case "n":
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
/*
|
||||||
|
Copyright 2018 the Heptio Ark 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 cli
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetConfirmation ensures that the user confirms the action before proceeding.
|
||||||
|
func GetConfirmation() bool {
|
||||||
|
reader := bufio.NewReader(os.Stdin)
|
||||||
|
|
||||||
|
for {
|
||||||
|
fmt.Printf("Are you sure you want to continue (Y/N)? ")
|
||||||
|
|
||||||
|
confirmation, err := reader.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "error reading user input: %v\n", err)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
confirmation = strings.TrimSpace(confirmation)
|
||||||
|
if len(confirmation) != 1 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch strings.ToLower(confirmation) {
|
||||||
|
case "y":
|
||||||
|
return true
|
||||||
|
case "n":
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Xor returns true if exactly one of the provided values is true,
|
||||||
|
// or false otherwise.
|
||||||
|
func Xor(val bool, vals ...bool) bool {
|
||||||
|
res := val
|
||||||
|
|
||||||
|
for _, v := range vals {
|
||||||
|
if res && v {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
res = res || v
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
|
@ -19,29 +19,147 @@ package restore
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
"github.com/spf13/pflag"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/labels"
|
||||||
|
kubeerrs "k8s.io/apimachinery/pkg/util/errors"
|
||||||
|
|
||||||
|
arkv1api "github.com/heptio/ark/pkg/apis/ark/v1"
|
||||||
"github.com/heptio/ark/pkg/client"
|
"github.com/heptio/ark/pkg/client"
|
||||||
"github.com/heptio/ark/pkg/cmd"
|
"github.com/heptio/ark/pkg/cmd"
|
||||||
|
"github.com/heptio/ark/pkg/cmd/cli"
|
||||||
|
"github.com/heptio/ark/pkg/cmd/util/flag"
|
||||||
|
clientset "github.com/heptio/ark/pkg/generated/clientset/versioned"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// NewDeleteCommand creates and returns a new cobra command for deleting restores.
|
||||||
func NewDeleteCommand(f client.Factory, use string) *cobra.Command {
|
func NewDeleteCommand(f client.Factory, use string) *cobra.Command {
|
||||||
|
o := &DeleteOptions{}
|
||||||
c := &cobra.Command{
|
c := &cobra.Command{
|
||||||
Use: fmt.Sprintf("%s NAME", use),
|
Use: fmt.Sprintf("%s [NAMES]", use),
|
||||||
Short: "Delete a restore",
|
Short: "Delete restores",
|
||||||
Args: cobra.ExactArgs(1),
|
Example: ` # delete a restore named "restore-1"
|
||||||
|
ark restore delete restore-1
|
||||||
|
|
||||||
|
# delete a restore named "restore-1" without prompting for confirmation
|
||||||
|
ark restore delete restore-1 --confirm
|
||||||
|
|
||||||
|
# delete restores named "restore-1" and "restore-2"
|
||||||
|
ark restore delete restore-1 restore-2
|
||||||
|
|
||||||
|
# delete all restores labelled with foo=bar"
|
||||||
|
ark restore delete --selector foo=bar
|
||||||
|
|
||||||
|
# delete all restores
|
||||||
|
ark restore delete --all`,
|
||||||
|
|
||||||
Run: func(c *cobra.Command, args []string) {
|
Run: func(c *cobra.Command, args []string) {
|
||||||
arkClient, err := f.Client()
|
cmd.CheckError(o.Complete(f, args))
|
||||||
cmd.CheckError(err)
|
cmd.CheckError(o.Validate(c, f, args))
|
||||||
|
cmd.CheckError(o.Run())
|
||||||
|
|
||||||
name := args[0]
|
|
||||||
|
|
||||||
err = arkClient.ArkV1().Restores(f.Namespace()).Delete(name, nil)
|
|
||||||
cmd.CheckError(err)
|
|
||||||
|
|
||||||
fmt.Printf("Restore %q deleted\n", name)
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
o.BindFlags(c.Flags())
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DeleteOptions contains parameters used for deleting a restore.
|
||||||
|
type DeleteOptions struct {
|
||||||
|
Names []string
|
||||||
|
All bool
|
||||||
|
Selector flag.LabelSelector
|
||||||
|
Confirm bool
|
||||||
|
client clientset.Interface
|
||||||
|
namespace string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Complete fills in the correct values for all the options.
|
||||||
|
func (o *DeleteOptions) Complete(f client.Factory, args []string) error {
|
||||||
|
o.namespace = f.Namespace()
|
||||||
|
client, err := f.Client()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
o.client = client
|
||||||
|
o.Names = args
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate validates the fields of the DeleteOptions struct.
|
||||||
|
func (o *DeleteOptions) Validate(c *cobra.Command, f client.Factory, args []string) error {
|
||||||
|
if o.client == nil {
|
||||||
|
return errors.New("Ark client is not set; unable to proceed")
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
hasNames = len(o.Names) > 0
|
||||||
|
hasAll = o.All
|
||||||
|
hasSelector = o.Selector.LabelSelector != nil
|
||||||
|
)
|
||||||
|
if !cli.Xor(hasNames, hasAll, hasSelector) {
|
||||||
|
return errors.New("you must specify exactly one of: specific restore name(s), the --all flag, or the --selector flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run performs the deletion of restore(s).
|
||||||
|
func (o *DeleteOptions) Run() error {
|
||||||
|
if !o.Confirm && !cli.GetConfirmation() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
restores []*arkv1api.Restore
|
||||||
|
errs []error
|
||||||
|
)
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case len(o.Names) > 0:
|
||||||
|
for _, name := range o.Names {
|
||||||
|
restore, err := o.client.ArkV1().Restores(o.namespace).Get(name, metav1.GetOptions{})
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, errors.WithStack(err))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
restores = append(restores, restore)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
selector := labels.Everything().String()
|
||||||
|
if o.Selector.LabelSelector != nil {
|
||||||
|
selector = o.Selector.String()
|
||||||
|
}
|
||||||
|
res, err := o.client.ArkV1().Restores(o.namespace).List(metav1.ListOptions{
|
||||||
|
LabelSelector: selector,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, errors.WithStack(err))
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range res.Items {
|
||||||
|
restores = append(restores, &res.Items[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(restores) == 0 {
|
||||||
|
fmt.Println("No restores found")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for _, r := range restores {
|
||||||
|
err := o.client.ArkV1().Restores(r.Namespace).Delete(r.Name, nil)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, errors.WithStack(err))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
fmt.Printf("Restore %q deleted\n", r.Name)
|
||||||
|
}
|
||||||
|
return kubeerrs.NewAggregate(errs)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BindFlags binds the options for this command to the flags.
|
||||||
|
func (o *DeleteOptions) BindFlags(flags *pflag.FlagSet) {
|
||||||
|
flags.BoolVar(&o.Confirm, "confirm", o.Confirm, "Confirm deletion")
|
||||||
|
flags.BoolVar(&o.All, "all", o.All, "Delete all restores")
|
||||||
|
flags.VarP(&o.Selector, "selector", "l", "Delete all restores matching this label selector")
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue