forked from toolshed/abra
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			upgrade-cl
			...
			app-volume
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6f7ee1ee94 | 
| @ -5,7 +5,6 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"log" | 	"log" | ||||||
| 	"os" | 	"os" | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| @ -13,7 +12,6 @@ import ( | |||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/docker/docker/api/types/volume" |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli" | ||||||
| ) | ) | ||||||
| @ -112,28 +110,19 @@ flag. | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		volumeListOptions := volume.ListOptions{fs} | 		volumeList, err := client.GetVolumes(cl, context.Background(), app.Server, fs) | ||||||
| 		volumeListOKBody, err := cl.VolumeList(context.Background(), volumeListOptions) |  | ||||||
| 		volumeList := volumeListOKBody.Volumes |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  | 		volumeNames := client.GetVolumeNames(volumeList) | ||||||
|  |  | ||||||
| 		var vols []string | 		if len(volumeNames) > 0 { | ||||||
| 		for _, vol := range volumeList { | 			err := client.RemoveVolumes(cl, context.Background(), volumeNames, internal.Force, 5) | ||||||
| 			vols = append(vols, vol.Name) | 			if err != nil { | ||||||
| 		} | 				log.Fatalf("removing volumes failed: %s", err) | ||||||
|  |  | ||||||
| 		if len(vols) > 0 { |  | ||||||
| 			for _, vol := range vols { |  | ||||||
| 				err = retryFunc(5, func() error { |  | ||||||
| 					return cl.VolumeRemove(context.Background(), vol, internal.Force) // last argument is for force removing |  | ||||||
| 				}) |  | ||||||
| 				if err != nil { |  | ||||||
| 					log.Fatalf("removing volumes failed: %s", err) |  | ||||||
| 				} |  | ||||||
| 				logrus.Info(fmt.Sprintf("volume %s removed", vol)) |  | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			logrus.Infof("%d volumes removed successfully", len(volumeNames)) | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Info("no volumes to remove") | 			logrus.Info("no volumes to remove") | ||||||
| 		} | 		} | ||||||
| @ -147,21 +136,3 @@ flag. | |||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| // retryFunc retries the given function for the given retries. After the nth |  | ||||||
| // retry it waits (n + 1)^2 seconds before the next retry (starting with n=0). |  | ||||||
| // It returns an error if the function still failed after the last retry. |  | ||||||
| func retryFunc(retries int, fn func() error) error { |  | ||||||
| 	for i := 0; i < retries; i++ { |  | ||||||
| 		err := fn() |  | ||||||
| 		if err == nil { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		if i+1 < retries { |  | ||||||
| 			sleep := time.Duration(i+1) * time.Duration(i+1) |  | ||||||
| 			logrus.Infof("%s: waiting %d seconds before next retry", err, sleep) |  | ||||||
| 			time.Sleep(sleep * time.Second) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return fmt.Errorf("%d retries failed", retries) |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -2,6 +2,7 @@ package app | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"log" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| @ -131,12 +132,12 @@ Passing "--force/-f" will select all volumes for removal. Be careful. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(volumesToRemove) > 0 { | 		if len(volumesToRemove) > 0 { | ||||||
| 			err = client.RemoveVolumes(cl, context.Background(), app.Server, volumesToRemove, internal.Force) | 			err := client.RemoveVolumes(cl, context.Background(), volumesToRemove, internal.Force, 5) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				log.Fatalf("removing volumes failed: %s", err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			logrus.Info("volumes removed successfully") | 			logrus.Infof("%d volumes removed successfully", len(volumesToRemove)) | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Info("no volumes removed") | 			logrus.Info("no volumes removed") | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -2,15 +2,17 @@ package client | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/docker/docker/api/types/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
| 	"github.com/docker/docker/api/types/volume" | 	"github.com/docker/docker/api/types/volume" | ||||||
| 	"github.com/docker/docker/client" | 	"github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func GetVolumes(cl *client.Client, ctx context.Context, server string, fs filters.Args) ([]*volume.Volume, error) { | func GetVolumes(cl *client.Client, ctx context.Context, server string, fs filters.Args) ([]*volume.Volume, error) { | ||||||
| 	volumeListOptions := volume.ListOptions{fs} | 	volumeListOKBody, err := cl.VolumeList(ctx, volume.ListOptions{Filters: fs}) | ||||||
| 	volumeListOKBody, err := cl.VolumeList(ctx, volumeListOptions) |  | ||||||
| 	volumeList := volumeListOKBody.Volumes | 	volumeList := volumeListOKBody.Volumes | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return volumeList, err | 		return volumeList, err | ||||||
| @ -29,13 +31,32 @@ func GetVolumeNames(volumes []*volume.Volume) []string { | |||||||
| 	return volumeNames | 	return volumeNames | ||||||
| } | } | ||||||
|  |  | ||||||
| func RemoveVolumes(cl *client.Client, ctx context.Context, server string, volumeNames []string, force bool) error { | func RemoveVolumes(cl *client.Client, ctx context.Context, volumeNames []string, force bool, retries int) error { | ||||||
| 	for _, volName := range volumeNames { | 	for _, volName := range volumeNames { | ||||||
| 		err := cl.VolumeRemove(ctx, volName, force) | 		err := retryFunc(5, func() error { | ||||||
|  | 			return cl.VolumeRemove(context.Background(), volName, force) | ||||||
|  | 		}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return fmt.Errorf("volume %s: %s", volName, err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // retryFunc retries the given function for the given retries. After the nth | ||||||
|  | // retry it waits (n + 1)^2 seconds before the next retry (starting with n=0). | ||||||
|  | // It returns an error if the function still failed after the last retry. | ||||||
|  | func retryFunc(retries int, fn func() error) error { | ||||||
|  | 	for i := 0; i < retries; i++ { | ||||||
|  | 		err := fn() | ||||||
|  | 		if err == nil { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		if i+1 < retries { | ||||||
|  | 			sleep := time.Duration(i+1) * time.Duration(i+1) | ||||||
|  | 			logrus.Infof("%s: waiting %d seconds before next retry", err, sleep) | ||||||
|  | 			time.Sleep(sleep * time.Second) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return fmt.Errorf("%d retries failed", retries) | ||||||
|  | } | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| package app | package client | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| @ -104,9 +104,6 @@ teardown(){ | |||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # TODO: should wait as long as volume is no longer in use |  | ||||||
|   sleep 10 |  | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --no-input |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --no-input | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|  | |||||||
| @ -78,9 +78,6 @@ teardown(){ | |||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |  | ||||||
|   sleep 10 |  | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'volumes removed successfully' |   assert_output --partial 'volumes removed successfully' | ||||||
| @ -92,9 +89,6 @@ teardown(){ | |||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |  | ||||||
|   sleep 10 |  | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'volumes removed successfully' |   assert_output --partial 'volumes removed successfully' | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user