All checks were successful
continuous-integration/drone/push Build is passing
See #478
243 lines
5.9 KiB
Go
243 lines
5.9 KiB
Go
package stack // https://github.com/docker/cli/blob/master/cli/command/stack/remove.go
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"os/signal"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"coopcloud.tech/abra/pkg/log"
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/api/types/network"
|
|
"github.com/docker/docker/api/types/swarm"
|
|
"github.com/docker/docker/api/types/versions"
|
|
"github.com/docker/docker/client"
|
|
apiclient "github.com/docker/docker/client"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// RunRemove is the swarm implementation of docker stack remove
|
|
func RunRemove(ctx context.Context, client *apiclient.Client, opts Remove) error {
|
|
sigIntCh := make(chan os.Signal, 1)
|
|
signal.Notify(sigIntCh, os.Interrupt)
|
|
defer signal.Stop(sigIntCh)
|
|
|
|
waitCh := make(chan struct{})
|
|
errCh := make(chan error)
|
|
|
|
go func() {
|
|
var errs []string
|
|
for _, namespace := range opts.Namespaces {
|
|
services, err := GetStackServices(ctx, client, namespace)
|
|
if err != nil {
|
|
errCh <- err
|
|
return
|
|
}
|
|
|
|
networks, err := getStackNetworks(ctx, client, namespace)
|
|
if err != nil {
|
|
errCh <- err
|
|
return
|
|
}
|
|
|
|
var secrets []swarm.Secret
|
|
if versions.GreaterThanOrEqualTo(client.ClientVersion(), "1.25") {
|
|
secrets, err = getStackSecrets(ctx, client, namespace)
|
|
if err != nil {
|
|
errCh <- err
|
|
return
|
|
}
|
|
}
|
|
|
|
var configs []swarm.Config
|
|
if versions.GreaterThanOrEqualTo(client.ClientVersion(), "1.30") {
|
|
configs, err = getStackConfigs(ctx, client, namespace)
|
|
if err != nil {
|
|
errCh <- err
|
|
return
|
|
}
|
|
}
|
|
|
|
if len(services)+len(networks)+len(secrets)+len(configs) == 0 {
|
|
log.Warnf("nothing found in stack: %s", namespace)
|
|
continue
|
|
}
|
|
|
|
hasError := removeServices(ctx, client, services)
|
|
hasError = removeSecrets(ctx, client, secrets) || hasError
|
|
hasError = removeConfigs(ctx, client, configs) || hasError
|
|
hasError = removeNetworks(ctx, client, networks) || hasError
|
|
|
|
if hasError {
|
|
errs = append(errs, fmt.Sprintf("failed to remove some resources from stack: %s", namespace))
|
|
continue
|
|
}
|
|
|
|
log.Info("polling undeploy status")
|
|
timeout, err := waitOnTasks(ctx, client, namespace)
|
|
if timeout {
|
|
errs = append(errs, err.Error())
|
|
} else {
|
|
if err != nil {
|
|
errs = append(errs, fmt.Sprintf("failed to wait on tasks of stack: %s: %s", namespace, err))
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(errs) > 0 {
|
|
errCh <- errors.Errorf(strings.Join(errs, "\n"))
|
|
return
|
|
}
|
|
|
|
close(waitCh)
|
|
}()
|
|
|
|
select {
|
|
case <-waitCh:
|
|
return nil
|
|
case <-sigIntCh:
|
|
return fmt.Errorf("skipping as requested, undeploy still in progress 🟠")
|
|
case err := <-errCh:
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func sortServiceByName(services []swarm.Service) func(i, j int) bool {
|
|
return func(i, j int) bool {
|
|
return services[i].Spec.Name < services[j].Spec.Name
|
|
}
|
|
}
|
|
|
|
func removeServices(
|
|
ctx context.Context,
|
|
client *apiclient.Client,
|
|
services []swarm.Service,
|
|
) bool {
|
|
var hasError bool
|
|
sort.Slice(services, sortServiceByName(services))
|
|
for _, service := range services {
|
|
log.Debugf("removing service %s", service.Spec.Name)
|
|
if err := client.ServiceRemove(ctx, service.ID); err != nil {
|
|
hasError = true
|
|
log.Fatalf("failed to remove service %s: %s", service.ID, err)
|
|
}
|
|
}
|
|
return hasError
|
|
}
|
|
|
|
func removeNetworks(
|
|
ctx context.Context,
|
|
client *apiclient.Client,
|
|
networks []network.Inspect,
|
|
) bool {
|
|
var hasError bool
|
|
for _, network := range networks {
|
|
log.Debugf("removing network %s", network.Name)
|
|
if err := client.NetworkRemove(ctx, network.ID); err != nil {
|
|
hasError = true
|
|
log.Fatalf("failed to remove network %s: %s", network.ID, err)
|
|
}
|
|
}
|
|
return hasError
|
|
}
|
|
|
|
func removeSecrets(
|
|
ctx context.Context,
|
|
client *apiclient.Client,
|
|
secrets []swarm.Secret,
|
|
) bool {
|
|
var hasError bool
|
|
for _, secret := range secrets {
|
|
log.Debugf("removing secret %s", secret.Spec.Name)
|
|
if err := client.SecretRemove(ctx, secret.ID); err != nil {
|
|
hasError = true
|
|
log.Fatalf("Failed to remove secret %s: %s", secret.ID, err)
|
|
}
|
|
}
|
|
return hasError
|
|
}
|
|
|
|
func removeConfigs(
|
|
ctx context.Context,
|
|
client *apiclient.Client,
|
|
configs []swarm.Config,
|
|
) bool {
|
|
var hasError bool
|
|
for _, config := range configs {
|
|
log.Debugf("removing config %s", config.Spec.Name)
|
|
if err := client.ConfigRemove(ctx, config.ID); err != nil {
|
|
hasError = true
|
|
log.Fatalf("failed to remove config %s: %s", config.ID, err)
|
|
}
|
|
}
|
|
return hasError
|
|
}
|
|
|
|
// https://github.com/docker/cli/pull/4259
|
|
func getStackTasks(ctx context.Context, apiclient client.APIClient, namespace string) ([]swarm.Task, error) {
|
|
return apiclient.TaskList(ctx, types.TaskListOptions{Filters: getStackFilter(namespace)})
|
|
}
|
|
|
|
var numberedStates = map[swarm.TaskState]int64{
|
|
swarm.TaskStateNew: 1,
|
|
swarm.TaskStateAllocated: 2,
|
|
swarm.TaskStatePending: 3,
|
|
swarm.TaskStateAssigned: 4,
|
|
swarm.TaskStateAccepted: 5,
|
|
swarm.TaskStatePreparing: 6,
|
|
swarm.TaskStateReady: 7,
|
|
swarm.TaskStateStarting: 8,
|
|
swarm.TaskStateRunning: 9,
|
|
swarm.TaskStateComplete: 10,
|
|
swarm.TaskStateShutdown: 11,
|
|
swarm.TaskStateFailed: 12,
|
|
swarm.TaskStateRejected: 13,
|
|
}
|
|
|
|
func terminalState(state swarm.TaskState) bool {
|
|
return numberedStates[state] > numberedStates[swarm.TaskStateRunning]
|
|
}
|
|
|
|
func waitOnTasks(ctx context.Context, client apiclient.APIClient, namespace string) (bool, error) {
|
|
var timedOut bool
|
|
|
|
log.Debugf("waiting on undeploy tasks (timeout=%v secs)", WaitTimeout)
|
|
|
|
go func() {
|
|
t := time.Duration(WaitTimeout) * time.Second
|
|
<-time.After(t)
|
|
log.Debug("timed out on undeploy")
|
|
timedOut = true
|
|
}()
|
|
|
|
terminalStatesReached := 0
|
|
for {
|
|
tasks, err := getStackTasks(ctx, client, namespace)
|
|
if err != nil {
|
|
return false, fmt.Errorf("failed to get tasks: %w", err)
|
|
}
|
|
|
|
for _, task := range tasks {
|
|
if terminalState(task.Status.State) {
|
|
terminalStatesReached++
|
|
break
|
|
}
|
|
}
|
|
|
|
if terminalStatesReached == len(tasks) {
|
|
break
|
|
}
|
|
|
|
if timedOut {
|
|
return true, fmt.Errorf("deployment timed out 🟠")
|
|
}
|
|
}
|
|
|
|
return false, nil
|
|
}
|