|
|
@ -19,6 +19,8 @@ package main
|
|
|
|
import (
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
@ -59,8 +61,10 @@ type resumable interface {
|
|
|
|
|
|
|
|
|
|
|
|
type resumeCommand struct {
|
|
|
|
type resumeCommand struct {
|
|
|
|
apiType
|
|
|
|
apiType
|
|
|
|
object resumable
|
|
|
|
client client.WithWatch
|
|
|
|
list listResumable
|
|
|
|
list listResumable
|
|
|
|
|
|
|
|
namespace string
|
|
|
|
|
|
|
|
shouldReconcile bool
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type listResumable interface {
|
|
|
|
type listResumable interface {
|
|
|
@ -68,6 +72,11 @@ type listResumable interface {
|
|
|
|
resumeItem(i int) resumable
|
|
|
|
resumeItem(i int) resumable
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
type reconcileResponse struct {
|
|
|
|
|
|
|
|
resumable
|
|
|
|
|
|
|
|
err error
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (resume resumeCommand) run(cmd *cobra.Command, args []string) error {
|
|
|
|
func (resume resumeCommand) run(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 1 && !resumeArgs.all {
|
|
|
|
if len(args) < 1 && !resumeArgs.all {
|
|
|
|
return fmt.Errorf("%s name is required", resume.humanKind)
|
|
|
|
return fmt.Errorf("%s name is required", resume.humanKind)
|
|
|
@ -80,52 +89,162 @@ func (resume resumeCommand) run(cmd *cobra.Command, args []string) error {
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
resume.client = kubeClient
|
|
|
|
|
|
|
|
resume.namespace = *kubeconfigArgs.Namespace
|
|
|
|
|
|
|
|
|
|
|
|
var listOpts []client.ListOption
|
|
|
|
// require waiting for the object(s) if the user has not provided the --wait flag and gave exactly
|
|
|
|
listOpts = append(listOpts, client.InNamespace(*kubeconfigArgs.Namespace))
|
|
|
|
// one object to resume. This is necessary to maintain backwards compatibility with prior versions
|
|
|
|
if len(args) > 0 {
|
|
|
|
// of this command. Otherwise just follow the value of the --wait flag (including its default).
|
|
|
|
listOpts = append(listOpts, client.MatchingFields{
|
|
|
|
resume.shouldReconcile = !resumeCmd.PersistentFlags().Changed("wait") && len(args) == 1 || resumeArgs.wait
|
|
|
|
"metadata.name": args[0],
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
err = kubeClient.List(ctx, resume.list.asClientList(), listOpts...)
|
|
|
|
resumables, err := resume.getPatchedResumables(ctx, args)
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
wg.Add(len(resumables))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
resultChan := make(chan reconcileResponse, len(resumables))
|
|
|
|
|
|
|
|
for _, r := range resumables {
|
|
|
|
|
|
|
|
go func(res resumable) {
|
|
|
|
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
resultChan <- resume.reconcile(ctx, res)
|
|
|
|
|
|
|
|
}(r)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
|
|
|
|
defer close(resultChan)
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
reconcileResps := make([]reconcileResponse, 0, len(resumables))
|
|
|
|
|
|
|
|
for c := range resultChan {
|
|
|
|
|
|
|
|
reconcileResps = append(reconcileResps, c)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
resume.printMessage(reconcileResps)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// getPatchedResumables returns a list of the given resumable objects that have been patched to be resumed.
|
|
|
|
|
|
|
|
// If the args slice is empty, it patches all resumable objects in the given namespace.
|
|
|
|
|
|
|
|
func (resume *resumeCommand) getPatchedResumables(ctx context.Context, args []string) ([]resumable, error) {
|
|
|
|
|
|
|
|
if len(args) < 1 {
|
|
|
|
|
|
|
|
objs, err := resume.patch(ctx, []client.ListOption{
|
|
|
|
|
|
|
|
client.InNamespace(resume.namespace),
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("failed patching objects: %w", err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return objs, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var resumables []resumable
|
|
|
|
|
|
|
|
processed := make(map[string]struct{}, len(args))
|
|
|
|
|
|
|
|
for _, arg := range args {
|
|
|
|
|
|
|
|
if _, has := processed[arg]; has {
|
|
|
|
|
|
|
|
continue // skip object that user might have provided more than once
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
processed[arg] = struct{}{}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
objs, err := resume.patch(ctx, []client.ListOption{
|
|
|
|
|
|
|
|
client.InNamespace(resume.namespace),
|
|
|
|
|
|
|
|
client.MatchingFields{
|
|
|
|
|
|
|
|
"metadata.name": arg,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
resumables = append(resumables, objs...)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return resumables, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Patches resumable objects by setting their status to unsuspended.
|
|
|
|
|
|
|
|
// Returns a slice of resumables that have been patched and any error encountered during patching.
|
|
|
|
|
|
|
|
func (resume resumeCommand) patch(ctx context.Context, listOpts []client.ListOption) ([]resumable, error) {
|
|
|
|
|
|
|
|
if err := resume.client.List(ctx, resume.list.asClientList(), listOpts...); err != nil {
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if resume.list.len() == 0 {
|
|
|
|
if resume.list.len() == 0 {
|
|
|
|
logger.Failuref("no %s objects found in %s namespace", resume.kind, *kubeconfigArgs.Namespace)
|
|
|
|
logger.Failuref("no %s objects found in %s namespace", resume.kind, resume.namespace)
|
|
|
|
return nil
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var resumables []resumable
|
|
|
|
|
|
|
|
|
|
|
|
for i := 0; i < resume.list.len(); i++ {
|
|
|
|
for i := 0; i < resume.list.len(); i++ {
|
|
|
|
logger.Actionf("resuming %s %s in %s namespace", resume.humanKind, resume.list.resumeItem(i).asClientObject().GetName(), *kubeconfigArgs.Namespace)
|
|
|
|
|
|
|
|
obj := resume.list.resumeItem(i)
|
|
|
|
obj := resume.list.resumeItem(i)
|
|
|
|
|
|
|
|
logger.Actionf("resuming %s %s in %s namespace", resume.humanKind, obj.asClientObject().GetName(), resume.namespace)
|
|
|
|
|
|
|
|
|
|
|
|
patch := client.MergeFrom(obj.deepCopyClientObject())
|
|
|
|
patch := client.MergeFrom(obj.deepCopyClientObject())
|
|
|
|
obj.setUnsuspended()
|
|
|
|
obj.setUnsuspended()
|
|
|
|
if err := kubeClient.Patch(ctx, obj.asClientObject(), patch); err != nil {
|
|
|
|
if err := resume.client.Patch(ctx, obj.asClientObject(), patch); err != nil {
|
|
|
|
return err
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
resumables = append(resumables, obj)
|
|
|
|
|
|
|
|
|
|
|
|
logger.Successf("%s resumed", resume.humanKind)
|
|
|
|
logger.Successf("%s resumed", resume.humanKind)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return resumables, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Waits for resumable object to be reconciled and returns the object and any error encountered while waiting.
|
|
|
|
|
|
|
|
// Returns an empty reconcileResponse, if shouldReconcile is false.
|
|
|
|
|
|
|
|
func (resume resumeCommand) reconcile(ctx context.Context, res resumable) reconcileResponse {
|
|
|
|
|
|
|
|
if !resume.shouldReconcile {
|
|
|
|
|
|
|
|
return reconcileResponse{}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if resumeArgs.wait || !resumeArgs.all {
|
|
|
|
namespacedName := types.NamespacedName{
|
|
|
|
namespacedName := types.NamespacedName{
|
|
|
|
Name: res.asClientObject().GetName(),
|
|
|
|
Name: resume.list.resumeItem(i).asClientObject().GetName(),
|
|
|
|
Namespace: resume.namespace,
|
|
|
|
Namespace: *kubeconfigArgs.Namespace,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
logger.Waitingf("waiting for %s reconciliation", resume.kind)
|
|
|
|
logger.Waitingf("waiting for %s reconciliation", resume.kind)
|
|
|
|
|
|
|
|
if err := wait.PollImmediate(rootArgs.pollInterval, rootArgs.timeout,
|
|
|
|
if err := wait.PollImmediate(rootArgs.pollInterval, rootArgs.timeout,
|
|
|
|
isReady(ctx, kubeClient, namespacedName, resume.list.resumeItem(i))); err != nil {
|
|
|
|
isReady(ctx, resume.client, namespacedName, res)); err != nil {
|
|
|
|
logger.Failuref(err.Error())
|
|
|
|
return reconcileResponse{
|
|
|
|
continue
|
|
|
|
resumable: res,
|
|
|
|
}
|
|
|
|
err: err,
|
|
|
|
logger.Successf("%s reconciliation completed", resume.kind)
|
|
|
|
|
|
|
|
logger.Successf(resume.list.resumeItem(i).successMessage())
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
return reconcileResponse{
|
|
|
|
|
|
|
|
resumable: res,
|
|
|
|
|
|
|
|
err: nil,
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Sorts the given reconcileResponses by resumable name and prints the success/error message for each response.
|
|
|
|
|
|
|
|
func (resume resumeCommand) printMessage(responses []reconcileResponse) {
|
|
|
|
|
|
|
|
sort.Slice(responses, func(i, j int) bool {
|
|
|
|
|
|
|
|
r1, r2 := responses[i], responses[j]
|
|
|
|
|
|
|
|
if r1.resumable == nil || r2.resumable == nil {
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return r1.asClientObject().GetName() <= r2.asClientObject().GetName()
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Print success/error message.
|
|
|
|
|
|
|
|
for _, r := range responses {
|
|
|
|
|
|
|
|
if r.resumable == nil {
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.err != nil {
|
|
|
|
|
|
|
|
logger.Failuref(r.err.Error())
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.Successf("%s %s reconciliation completed", resume.kind, r.asClientObject().GetName())
|
|
|
|
|
|
|
|
logger.Successf(r.successMessage())
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|