Compare commits
	
		
			289 Commits
		
	
	
		
			0.3.1-alph
			...
			0.4.0-alph
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| abf56f9054 | |||
| 4dec3c4646 | |||
| c900cebc30 | |||
| 30209de3e2 | |||
| 625747d048 | |||
| a71b070921 | |||
| 33ff04c686 | |||
| c69a3c23c5 | |||
| 0b46909961 | |||
| 832e8e5a96 | |||
| abf83aa641 | |||
| 1df69aa259 | |||
| 7596a67ad5 | |||
| 93c7612efc | |||
| 2c78ac22e0 | |||
| 13661c72ce | |||
| 454092644a | |||
| 224c0c38db | |||
| 560e0eab86 | |||
| b92fdbbd52 | |||
| 0a550363b8 | |||
| 3119220c21 | |||
| 49f565e5db | |||
| 94522178b1 | |||
| 810bc27967 | |||
| 35d95fb9fb | |||
| d26fabe8ef | |||
| 84bf3ffa50 | |||
| 575485ec7a | |||
| 0b17292219 | |||
| fffd8b2647 | |||
| c07128b308 | |||
| 929ff88013 | |||
| 0353427c71 | |||
| 7a0d18ceb6 | |||
| 8992050409 | |||
| abd094387f | |||
| a556ca625b | |||
| 1b7836009f | |||
| eb3509ab3f | |||
| 87851d26f7 | |||
| c4f344b50a | |||
| 60e4dfd9cb | |||
| d957adb675 | |||
| 5254af0fe4 | |||
| ce96269be0 | |||
| 299276c383 | |||
| 866cdd1f29 | |||
| 95d385c420 | |||
| 605e2553b8 | |||
| 1245827dff | |||
| 9bdb07463c | |||
| be26f80f03 | |||
| 930ff68bb2 | |||
| 62441acf03 | |||
| 7460668ef4 | |||
| 047d0e6fbc | |||
| 8785f66391 | |||
| 24882e95b4 | |||
| 1fd0941239 | |||
| 26a11533b4 | |||
| b4f48c3c59 | |||
| 43e68a99b0 | |||
| bac6fb0fa8 | |||
| dc9c9715ce | |||
| 1f91b3bb03 | |||
| a700aca23d | |||
| 5cacd09a04 | |||
| 6a98024a2b | |||
| e85117be22 | |||
| fb24357d38 | |||
| f5d2d3adf6 | |||
| 07119b0575 | |||
| d2a6e35986 | |||
| 0aa37fcee8 | |||
| eb1b6be4c5 | |||
| b98397144a | |||
| 4c186678b8 | |||
| b1d9d9d858 | |||
| a06043375d | |||
| 3eef1e8587 | |||
| 37e48f262b | |||
| 06cc5d1cc3 | |||
| c13f438580 | |||
| 5cd4317580 | |||
| 2ba1ec3df0 | |||
| 34cdb9c9d8 | |||
| 9c281d8608 | |||
| 321ba1e0ec | |||
| c5a74e9f6b | |||
| f8191ac248 | |||
| 027c8a1420 | |||
| cdc08ae95a | |||
| 3f35510507 | |||
| 9f70a69bbf | |||
| b0834925a3 | |||
| 86d87253c5 | |||
| 17340a79da | |||
| 779c810521 | |||
| 9cc2554846 | |||
| 9a1cf258a5 | |||
| ba8138079f | |||
| 8735a8f0ea | |||
| a84a5bc320 | |||
| ae0e7b8e4c | |||
| c0caf14d74 | |||
| d66c558b5c | |||
| c8541e1b9d | |||
| 653b6c6d49 | |||
| e2c3bc35c3 | |||
| 6937bfbb0d | |||
| decfe095fe | |||
| 4283f130a2 | |||
| 3b5354b2a5 | |||
| 14400d4ed8 | |||
| dddf84d92b | |||
| fefb042716 | |||
| ab8db8df64 | |||
| 20f7a18caa | |||
| 58a24a50e1 | |||
| e839f100df | |||
| 41a757b7ed | |||
| 4b4298caf1 | |||
| 8e8c241fdf | |||
| 9b8ff1ddcd | |||
| a85cfe40d0 | |||
| fc29ca6fce | |||
| cfb02f45ed | |||
| 696172ad48 | |||
| 4089949a3f | |||
| a75b01e78a | |||
| 014d32112e | |||
| a7894cbda9 | |||
| e03761f251 | |||
| 190c1033e6 | |||
| 15d1e9dee0 | |||
| 0362928840 | |||
| 844961d016 | |||
| d0cc51b829 | |||
| 606b5ac3e4 | |||
| 6f1bf258b3 | |||
| 7a5aa1b005 | |||
| db453f0ab1 | |||
| a07e71f7df | |||
| 4c6d52c426 | |||
| 327c5adef2 | |||
| 0dc8425a27 | |||
| 48c965bb21 | |||
| 5513754c22 | |||
| 3a27d9d9fb | |||
| 04b58230ea | |||
| 1b9097f9f3 | |||
| 3d100093dc | |||
| ef4383209e | |||
| 74f688350b | |||
| 737a22aacc | |||
| 56a1e7f8c4 | |||
| 6be2f36334 | |||
| a18d0e290d | |||
| 7e0feec311 | |||
| 29a4d05944 | |||
| b72bad955a | |||
| e9b4541c91 | |||
| 5b1b16d64a | |||
| ec7223146b | |||
| fa45264ea0 | |||
| f57222d6aa | |||
| 28d10928a4 | |||
| 0f4da38f98 | |||
| 11c2d1efe6 | |||
| 2b1cc9f6dd | |||
| 6100a636a6 | |||
| ddbf923338 | |||
| c1a00520dc | |||
| 0dc4b2beef | |||
| f75284364d | |||
| fbc3b48d39 | |||
| 6f0d8b190d | |||
| fc3742212c | |||
| fccbd7c7d7 | |||
| 2457b5fe95 | |||
| 72df640d99 | |||
| ae9e66c319 | |||
| 3589a7d56e | |||
| 8d499c0810 | |||
| cb2bb3f532 | |||
| 0a903f041f | |||
| 053a06ccba | |||
| 398deec272 | |||
| bf82bc9c7f | |||
| 217d4bc2cc | |||
| 9c8e6b63a6 | |||
| 5113db1612 | |||
| 66666e30b7 | |||
| 88d4984248 | |||
| bc34be4357 | |||
| 3d1aa55587 | |||
| e7469acf5b | |||
| a293179e89 | |||
| b912e73c5e | |||
| 4c66e44b3a | |||
| 033bad3d10 | |||
| a750344653 | |||
| f5caf5587a | |||
| fdc9e8b5fd | |||
| 75edcabb23 | |||
| fa0a63c11d | |||
| 3d3eefb2fe | |||
| 6998a87eef | |||
| b71a379788 | |||
| ba217dccbd | |||
| 45259b3266 | |||
| 59b80d5def | |||
| 8f6e1de1a1 | |||
| cd0d3b8892 | |||
| 0d1f65daac | |||
| cf1b46fa61 | |||
| 0fe0ffbafa | |||
| af3def7267 | |||
| c7de9c0719 | |||
| cf5ee4e682 | |||
| 9ddf69b988 | |||
| a925da8dee | |||
| 06f8078866 | |||
| 467947edf2 | |||
| 512cd9d85b | |||
| b8e2d1de67 | |||
| 3b7a8e6498 | |||
| 5bae262a79 | |||
| 6ad253b866 | |||
| b603069514 | |||
| d999cedd97 | |||
| 8215bb455b | |||
| 37ab9a9c08 | |||
| 48dd9cdeed | |||
| d02e1f247f | |||
| d087a60e09 | |||
| 48e16c414c | |||
| f3e55e5023 | |||
| ae6adace50 | |||
| 32dcddb631 | |||
| 3dbd343600 | |||
| 8393f4b134 | |||
| 8e56607cc9 | |||
| 85a543afac | |||
| 665396b679 | |||
| 870c561fee | |||
| 3fb43ffa2c | |||
| 2bc2f8630b | |||
| 6094dfaf92 | |||
| 3789e56404 | |||
| 2db5378418 | |||
| 7d8f3f1fab | |||
| 9be78bc5fa | |||
| 6c87d501e6 | |||
| 930c29f4a2 | |||
| 1d6c3e98e4 | |||
| a90f3b7463 | |||
| 962f566228 | |||
| 9896c57399 | |||
| 748d607ddc | |||
| 3901258a96 | |||
| 4347083f98 | |||
| 4641a942d8 | |||
| 759a00eeb3 | |||
| d1526fad21 | |||
| 6ef15e0a26 | |||
| dd0f328a65 | |||
| aea5cc69c3 | |||
| b02475eca5 | |||
| d0a30f6b7b | |||
| 8635922b9f | |||
| 9d62fff074 | |||
| 711c4e5ee8 | |||
| cb32e88cde | |||
| a18729bf98 | |||
| dbf84b7640 | |||
| 75db249053 | |||
| fdf4fc6737 | |||
| ef6a9abba9 | |||
| ce57d5ed54 | |||
| 3b01b1bb2e | |||
| fbdb792795 | |||
| 900f40f07a | |||
| ecd2a63f0a | |||
| 304b70639f | |||
| d821975aa2 | |||
| 1b836dbab6 | |||
| fc51cf7775 | 
							
								
								
									
										4
									
								
								.e2e.env.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.e2e.env.sample
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | GANDI_TOKEN=... | ||||||
|  | HCLOUD_TOKEN=... | ||||||
|  | REGISTRY_PASSWORD=... | ||||||
|  | REGISTRY_USERNAME=... | ||||||
							
								
								
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,6 +1,7 @@ | |||||||
| abra |  | ||||||
| .vscode/ |  | ||||||
| vendor/ |  | ||||||
| .envrc |  | ||||||
| dist/ |  | ||||||
| *fmtcoverage.html | *fmtcoverage.html | ||||||
|  | .e2e.env | ||||||
|  | .envrc | ||||||
|  | .vscode/ | ||||||
|  | abra | ||||||
|  | dist/ | ||||||
|  | vendor/ | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								Makefile
									
									
									
									
									
								
							| @ -5,7 +5,7 @@ LDFLAGS := "-X 'main.Commit=$(COMMIT)'" | |||||||
| DIST_LDFLAGS := $(LDFLAGS)" -s -w" | DIST_LDFLAGS := $(LDFLAGS)" -s -w" | ||||||
| export GOPRIVATE=coopcloud.tech | export GOPRIVATE=coopcloud.tech | ||||||
|  |  | ||||||
| all: run test install build clean format check static | all: format check static build test | ||||||
|  |  | ||||||
| run: | run: | ||||||
| 	@go run -ldflags=$(LDFLAGS) $(ABRA) | 	@go run -ldflags=$(LDFLAGS) $(ABRA) | ||||||
| @ -43,3 +43,15 @@ loc-author: | |||||||
|    	sort -f | \ |    	sort -f | \ | ||||||
|    	uniq -ic | \ |    	uniq -ic | \ | ||||||
|    	sort -n |    	sort -n | ||||||
|  |  | ||||||
|  | int-core: | ||||||
|  | 	@docker run \ | ||||||
|  | 		-v $$(pwd):/src \ | ||||||
|  | 		--env-file .e2e.env \ | ||||||
|  | 		debian:bullseye-slim \ | ||||||
|  | 		sh -c "\ | ||||||
|  | 			apt update && apt install -y wget curl git; echo ""; echo ""; \ | ||||||
|  | 			git config --global user.email 'e2e@coopcloud.tech'; \ | ||||||
|  | 			git config --global user.name 'e2e'; \ | ||||||
|  | 			cd /src/tests/integration && bash core.sh -- --dev \ | ||||||
|  | 		" | ||||||
|  | |||||||
| @ -7,7 +7,7 @@ import ( | |||||||
| // AppCommand defines the `abra app` command and ets subcommands | // AppCommand defines the `abra app` command and ets subcommands | ||||||
| var AppCommand = &cli.Command{ | var AppCommand = &cli.Command{ | ||||||
| 	Name:      "app", | 	Name:      "app", | ||||||
| 	Usage:     "Manage deployed apps", | 	Usage:     "Manage apps", | ||||||
| 	Aliases:   []string{"a"}, | 	Aliases:   []string{"a"}, | ||||||
| 	ArgsUsage: "<app>", | 	ArgsUsage: "<app>", | ||||||
| 	Description: ` | 	Description: ` | ||||||
| @ -35,5 +35,6 @@ to scaling apps up and spinning them down. | |||||||
| 		appSecretCommand, | 		appSecretCommand, | ||||||
| 		appVolumeCommand, | 		appVolumeCommand, | ||||||
| 		appVersionCommand, | 		appVersionCommand, | ||||||
|  | 		appErrorsCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -10,6 +10,7 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| @ -37,10 +38,10 @@ var appBackupCommand = &cli.Command{ | |||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use '<service>' and '--all' together")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use '<service>' and '--all' together")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		abraSh := path.Join(config.ABRA_DIR, "apps", app.Type, "abra.sh") | 		abraSh := path.Join(config.RECIPES_DIR, app.Type, "abra.sh") | ||||||
| 		if _, err := os.Stat(abraSh); err != nil { | 		if _, err := os.Stat(abraSh); err != nil { | ||||||
| 			if os.IsNotExist(err) { | 			if os.IsNotExist(err) { | ||||||
| 				logrus.Fatalf("'%s' does not exist?", abraSh) | 				logrus.Fatalf("%s does not exist?", abraSh) | ||||||
| 			} | 			} | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -61,7 +62,7 @@ var appBackupCommand = &cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		if !strings.Contains(string(bytes), execCmd) { | 		if !strings.Contains(string(bytes), execCmd) { | ||||||
| 			logrus.Fatalf("%s doesn't have a '%s' function", app.Type, execCmd) | 			logrus.Fatalf("%s doesn't have a %s function", app.Type, execCmd) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		sourceAndExec := fmt.Sprintf("%s; %s", sourceCmd, execCmd) | 		sourceAndExec := fmt.Sprintf("%s; %s", sourceCmd, execCmd) | ||||||
| @ -72,16 +73,5 @@ var appBackupCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,12 +1,12 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"os" | 	"os" | ||||||
| 	"path" | 	"path" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| @ -20,10 +20,10 @@ var appCheckCommand = &cli.Command{ | |||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		envSamplePath := path.Join(config.ABRA_DIR, "apps", app.Type, ".env.sample") | 		envSamplePath := path.Join(config.RECIPES_DIR, app.Type, ".env.sample") | ||||||
| 		if _, err := os.Stat(envSamplePath); err != nil { | 		if _, err := os.Stat(envSamplePath); err != nil { | ||||||
| 			if os.IsNotExist(err) { | 			if os.IsNotExist(err) { | ||||||
| 				logrus.Fatalf("'%s' does not exist?", envSamplePath) | 				logrus.Fatalf("%s does not exist?", envSamplePath) | ||||||
| 			} | 			} | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -45,20 +45,9 @@ var appCheckCommand = &cli.Command{ | |||||||
| 			logrus.Fatalf("%s is missing %s", app.Path, missingEnvVars) | 			logrus.Fatalf("%s is missing %s", app.Path, missingEnvVars) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Infof("all necessary environment variables defined for '%s'", app.Name) | 		logrus.Infof("all necessary environment variables defined for %s", app.Name) | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -2,11 +2,11 @@ package app | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" |  | ||||||
| 	"os" | 	"os" | ||||||
| 	"os/exec" | 	"os/exec" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| @ -31,7 +31,7 @@ var appConfigCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		appFile, exists := files[appName] | 		appFile, exists := files[appName] | ||||||
| 		if !exists { | 		if !exists { | ||||||
| 			logrus.Fatalf("cannot find app with name '%s'", appName) | 			logrus.Fatalf("cannot find app with name %s", appName) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		ed, ok := os.LookupEnv("EDITOR") | 		ed, ok := os.LookupEnv("EDITOR") | ||||||
| @ -55,16 +55,5 @@ var appConfigCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -6,7 +6,14 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
|  | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/container" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	"github.com/docker/docker/pkg/archive" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
| @ -76,23 +83,73 @@ And if you want to copy that file back to your current working directory locally | |||||||
| 				logrus.Fatalf("%s does not exist locally?", dstPath) | 				logrus.Fatalf("%s does not exist locally?", dstPath) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		err := internal.ConfigureAndCp(c, app, srcPath, dstPath, service, isToContainer) | 		err := configureAndCp(c, app, srcPath, dstPath, service, isToContainer) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		return nil | 		return nil | ||||||
|  |  | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() | } | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) | func configureAndCp( | ||||||
| 		} | 	c *cli.Context, | ||||||
| 		if c.NArg() > 0 { | 	app config.App, | ||||||
| 			return | 	srcPath string, | ||||||
| 		} | 	dstPath string, | ||||||
| 		for _, a := range appNames { | 	service string, | ||||||
| 			fmt.Println(a) | 	isToContainer bool) error { | ||||||
| 		} | 	appFiles, err := config.LoadAppFiles("") | ||||||
| 	}, | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	appEnv, err := config.GetApp(appFiles, app.Name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cl, err := client.New(app.Server) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	filters := filters.NewArgs() | ||||||
|  | 	filters.Add("name", fmt.Sprintf("%s_%s", appEnv.StackName(), service)) | ||||||
|  |  | ||||||
|  | 	container, err := container.GetContainer(c.Context, cl, filters, true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("retrieved %s as target container on %s", formatter.ShortenID(container.ID), app.Server) | ||||||
|  |  | ||||||
|  | 	if isToContainer { | ||||||
|  | 		if _, err := os.Stat(srcPath); err != nil { | ||||||
|  | 			logrus.Fatalf("%s does not exist?", srcPath) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		toTarOpts := &archive.TarOptions{NoOverwriteDirNonDir: true, Compression: archive.Gzip} | ||||||
|  | 		content, err := archive.TarWithOptions(srcPath, toTarOpts) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		copyOpts := types.CopyToContainerOptions{AllowOverwriteDirWithFile: false, CopyUIDGID: false} | ||||||
|  | 		if err := cl.CopyToContainer(c.Context, container.ID, dstPath, content, copyOpts); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		content, _, err := cl.CopyFromContainer(c.Context, container.ID, srcPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 		defer content.Close() | ||||||
|  | 		fromTarOpts := &archive.TarOptions{NoOverwriteDirNonDir: true, Compression: archive.Gzip} | ||||||
|  | 		if err := archive.Untar(content, dstPath, fromTarOpts); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,11 +1,8 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @ -16,11 +13,13 @@ var appDeployCommand = &cli.Command{ | |||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.ForceFlag, | 		internal.ForceFlag, | ||||||
| 		internal.ChaosFlag, | 		internal.ChaosFlag, | ||||||
|  | 		internal.NoDomainChecksFlag, | ||||||
|  | 		internal.DontWaitConvergeFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command deploys a new instance of an app. It does not support changing the | This command deploys an app. It does not support incrementing the version of a | ||||||
| version of an existing deployed app, for this you need to look at the "abra app | deployed app, for this you need to look at the "abra app upgrade <app>" | ||||||
| upgrade <app>" command. | command. | ||||||
|  |  | ||||||
| You may pass "--force" to re-deploy the same version again. This can be useful | You may pass "--force" to re-deploy the same version again. This can be useful | ||||||
| if the container runtime has gotten into a weird state. | if the container runtime has gotten into a weird state. | ||||||
| @ -29,17 +28,6 @@ Chas mode ("--chaos") will deploy your local checkout of a recipe as-is, | |||||||
| including unstaged changes and can be useful for live hacking and testing new | including unstaged changes and can be useful for live hacking and testing new | ||||||
| recipes. | recipes. | ||||||
| `, | `, | ||||||
| 	Action: internal.DeployAction, | 	Action:       internal.DeployAction, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										136
									
								
								cli/app/errors.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								cli/app/errors.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,136 @@ | |||||||
|  | package app | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
|  | 	"coopcloud.tech/abra/pkg/client" | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	dockerClient "github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | 	"github.com/urfave/cli/v2" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | var appErrorsCommand = &cli.Command{ | ||||||
|  | 	Name:  "errors", | ||||||
|  | 	Usage: "List errors for a deployed app", | ||||||
|  | 	Description: ` | ||||||
|  | This command lists errors for a deployed app. | ||||||
|  |  | ||||||
|  | This is a best-effort implementation and an attempt to gather a number of tips | ||||||
|  | & tricks for finding errors together into one convenient command. When an app | ||||||
|  | is failing to deploy or having issues, it could be a lot of things. | ||||||
|  |  | ||||||
|  | This command currently takes into account: | ||||||
|  |  | ||||||
|  | 		Is the service deployed? | ||||||
|  |     Is the service killed by an OOM error? | ||||||
|  | 		Is the service reporting an error (like in "ps --no-trunc" output) | ||||||
|  | 		Is the service healthcheck failing? what are the healthcheck logs? | ||||||
|  |  | ||||||
|  | Got any more ideas? Please let us know: | ||||||
|  |  | ||||||
|  |     https://git.coopcloud.tech/coop-cloud/organising/issues/new/choose | ||||||
|  |  | ||||||
|  | This command is best accompanied by "abra app logs <app>" which may reveal | ||||||
|  | further information which can help you debug the cause of an app failure via | ||||||
|  | the logs. | ||||||
|  |  | ||||||
|  | `, | ||||||
|  | 	Aliases:      []string{"e"}, | ||||||
|  | 	Flags:        []cli.Flag{internal.WatchFlag}, | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
|  | 		cl, err := client.New(app.Server) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		isDeployed, _, err := stack.IsDeployed(c.Context, cl, app.StackName()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !isDeployed { | ||||||
|  | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.Watch { | ||||||
|  | 			if err := checkErrors(c, cl, app); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for { | ||||||
|  | 			if err := checkErrors(c, cl, app); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			time.Sleep(2 * time.Second) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func checkErrors(c *cli.Context, cl *dockerClient.Client, app config.App) error { | ||||||
|  | 	recipe, err := recipe.Get(app.Type) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		filters := filters.NewArgs() | ||||||
|  | 		filters.Add("name", service.Name) | ||||||
|  | 		containers, err := cl.ContainerList(c.Context, types.ContainerListOptions{Filters: filters}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if len(containers) == 0 { | ||||||
|  | 			logrus.Warnf("%s is not up, something seems wrong", service.Name) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		container := containers[0] | ||||||
|  | 		containerState, err := cl.ContainerInspect(c.Context, container.ID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if containerState.State.OOMKilled { | ||||||
|  | 			logrus.Warnf("%s has been killed due to an out of memory error", service.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if containerState.State.Error != "" { | ||||||
|  | 			logrus.Warnf("%s reports this error: %s", service.Name, containerState.State.Error) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if containerState.State.Health != nil { | ||||||
|  | 			if containerState.State.Health.Status != "healthy" { | ||||||
|  | 				logrus.Warnf("%s healthcheck status is %s", service.Name, containerState.State.Health.Status) | ||||||
|  | 				logrus.Warnf("%s healthcheck has failed %s times", service.Name, strconv.Itoa(containerState.State.Health.FailingStreak)) | ||||||
|  | 				for _, log := range containerState.State.Health.Log { | ||||||
|  | 					logrus.Warnf("%s healthcheck logs: %s", service.Name, strings.TrimSpace(log.Output)) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func getServiceName(names []string) string { | ||||||
|  | 	containerName := strings.Join(names, " ") | ||||||
|  | 	trimmed := strings.TrimPrefix(containerName, "/") | ||||||
|  | 	return strings.Split(trimmed, ".")[0] | ||||||
|  | } | ||||||
							
								
								
									
										184
									
								
								cli/app/list.go
									
									
									
									
									
								
							
							
						
						
									
										184
									
								
								cli/app/list.go
									
									
									
									
									
								
							| @ -5,9 +5,10 @@ import ( | |||||||
| 	"sort" | 	"sort" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/ssh" | 	"coopcloud.tech/abra/pkg/ssh" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| @ -41,6 +42,25 @@ var listAppServerFlag = &cli.StringFlag{ | |||||||
| 	Destination: &listAppServer, | 	Destination: &listAppServer, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type appStatus struct { | ||||||
|  | 	server  string | ||||||
|  | 	recipe  string | ||||||
|  | 	appName string | ||||||
|  | 	domain  string | ||||||
|  | 	status  string | ||||||
|  | 	version string | ||||||
|  | 	upgrade string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type serverStatus struct { | ||||||
|  | 	apps             []appStatus | ||||||
|  | 	appCount         int | ||||||
|  | 	versionCount     int | ||||||
|  | 	unversionedCount int | ||||||
|  | 	latestCount      int | ||||||
|  | 	upgradeCount     int | ||||||
|  | } | ||||||
|  |  | ||||||
| var appListCommand = &cli.Command{ | var appListCommand = &cli.Command{ | ||||||
| 	Name:  "list", | 	Name:  "list", | ||||||
| 	Usage: "List all managed apps", | 	Usage: "List all managed apps", | ||||||
| @ -68,60 +88,79 @@ can take some time. | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		sort.Sort(config.ByServerAndType(apps)) | 		sort.Sort(config.ByServerAndType(apps)) | ||||||
|  |  | ||||||
| 		for _, app := range apps { | 		statuses := make(map[string]map[string]string) | ||||||
| 			if err := ssh.EnsureHostKey(app.Server); err != nil { | 		var catl recipe.RecipeCatalogue | ||||||
|  | 		if status { | ||||||
|  | 			alreadySeen := make(map[string]bool) | ||||||
|  | 			for _, app := range apps { | ||||||
|  | 				if _, ok := alreadySeen[app.Server]; !ok { | ||||||
|  | 					if err := ssh.EnsureHostKey(app.Server); err != nil { | ||||||
|  | 						logrus.Fatal(fmt.Sprintf(internal.SSHFailMsg, app.Server)) | ||||||
|  | 					} | ||||||
|  | 					alreadySeen[app.Server] = true | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			statuses, err = config.GetAppStatuses(appFiles) | ||||||
|  | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		statuses := make(map[string]map[string]string) | 			var err error | ||||||
| 		tableCol := []string{"Server", "Type", "Domain"} | 			catl, err = recipe.ReadRecipeCatalogue() | ||||||
| 		if status { |  | ||||||
| 			tableCol = append(tableCol, "Status", "Version", "Updates") |  | ||||||
| 			statuses, err = config.GetAppStatuses(appFiles) |  | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		var totalServersCount int | ||||||
| 		table.SetAutoMergeCellsByColumnIndex([]int{0}) | 		var totalAppsCount int | ||||||
|  | 		allStats := make(map[string]serverStatus) | ||||||
| 		var ( |  | ||||||
| 			versionedAppsCount   int |  | ||||||
| 			unversionedAppsCount int |  | ||||||
| 			onLatestCount        int |  | ||||||
| 			canUpgradeCount      int |  | ||||||
| 		) |  | ||||||
|  |  | ||||||
| 		for _, app := range apps { | 		for _, app := range apps { | ||||||
| 			var tableRow []string | 			var stats serverStatus | ||||||
|  | 			var ok bool | ||||||
|  | 			if stats, ok = allStats[app.Server]; !ok { | ||||||
|  | 				stats = serverStatus{} | ||||||
|  | 				if appType == "" { | ||||||
|  | 					// count server, no filtering | ||||||
|  | 					totalServersCount++ | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			if app.Type == appType || appType == "" { | 			if app.Type == appType || appType == "" { | ||||||
| 				// If type flag is set, check for it, if not, Type == "" | 				if appType != "" { | ||||||
| 				tableRow = []string{app.Server, app.Type, app.Domain} | 					// only count server if matches filter | ||||||
|  | 					totalServersCount++ | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				appStats := appStatus{} | ||||||
|  | 				stats.appCount++ | ||||||
|  | 				totalAppsCount++ | ||||||
|  |  | ||||||
| 				if status { | 				if status { | ||||||
| 					stackName := app.StackName() |  | ||||||
| 					status := "unknown" | 					status := "unknown" | ||||||
| 					version := "unknown" | 					version := "unknown" | ||||||
| 					if statusMeta, ok := statuses[stackName]; ok { | 					if statusMeta, ok := statuses[app.StackName()]; ok { | ||||||
| 						if currentVersion, exists := statusMeta["version"]; exists { | 						if currentVersion, exists := statusMeta["version"]; exists { | ||||||
| 							version = currentVersion | 							version = currentVersion | ||||||
| 						} | 						} | ||||||
| 						if statusMeta["status"] != "" { | 						if statusMeta["status"] != "" { | ||||||
| 							status = statusMeta["status"] | 							status = statusMeta["status"] | ||||||
| 						} | 						} | ||||||
| 						tableRow = append(tableRow, status, version) | 						stats.versionCount++ | ||||||
| 						versionedAppsCount++ |  | ||||||
| 					} else { | 					} else { | ||||||
| 						tableRow = append(tableRow, status, version) | 						stats.unversionedCount++ | ||||||
| 						unversionedAppsCount++ |  | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
|  | 					appStats.status = status | ||||||
|  | 					appStats.version = version | ||||||
|  |  | ||||||
| 					var newUpdates []string | 					var newUpdates []string | ||||||
| 					if version != "unknown" { | 					if version != "unknown" { | ||||||
| 						updates, err := catalogue.GetRecipeCatalogueVersions(app.Type) | 						updates, err := recipe.GetRecipeCatalogueVersions(app.Type, catl) | ||||||
| 						if err != nil { | 						if err != nil { | ||||||
| 							logrus.Fatal(err) | 							logrus.Fatal(err) | ||||||
| 						} | 						} | ||||||
| @ -145,35 +184,80 @@ can take some time. | |||||||
|  |  | ||||||
| 					if len(newUpdates) == 0 { | 					if len(newUpdates) == 0 { | ||||||
| 						if version == "unknown" { | 						if version == "unknown" { | ||||||
| 							tableRow = append(tableRow, "unknown") | 							appStats.upgrade = "unknown" | ||||||
| 						} else { | 						} else { | ||||||
| 							tableRow = append(tableRow, "on latest") | 							appStats.upgrade = "latest" | ||||||
| 							onLatestCount++ | 							stats.latestCount++ | ||||||
| 						} | 						} | ||||||
| 					} else { | 					} else { | ||||||
| 						// FIXME: jeezus golang why do you not have a list reverse function | 						newUpdates = internal.ReverseStringList(newUpdates) | ||||||
| 						for i, j := 0, len(newUpdates)-1; i < j; i, j = i+1, j-1 { | 						appStats.upgrade = strings.Join(newUpdates, "\n") | ||||||
| 							newUpdates[i], newUpdates[j] = newUpdates[j], newUpdates[i] | 						stats.upgradeCount++ | ||||||
| 						} |  | ||||||
| 						tableRow = append(tableRow, strings.Join(newUpdates, "\n")) |  | ||||||
| 						canUpgradeCount++ |  | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
|  | 				appStats.server = app.Server | ||||||
|  | 				appStats.recipe = app.Type | ||||||
|  | 				appStats.appName = app.Name | ||||||
|  | 				appStats.domain = app.Domain | ||||||
|  |  | ||||||
|  | 				stats.apps = append(stats.apps, appStats) | ||||||
| 			} | 			} | ||||||
| 			table.Append(tableRow) |  | ||||||
|  | 			allStats[app.Server] = stats | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		stats := fmt.Sprintf( | 		alreadySeen := make(map[string]bool) | ||||||
| 			"Total apps: %v | Versioned: %v | Unversioned: %v | On latest: %v | Can upgrade: %v", | 		for _, app := range apps { | ||||||
| 			len(apps), | 			if _, ok := alreadySeen[app.Server]; ok { | ||||||
| 			versionedAppsCount, | 				continue | ||||||
| 			unversionedAppsCount, | 			} | ||||||
| 			onLatestCount, |  | ||||||
| 			canUpgradeCount, |  | ||||||
| 		) |  | ||||||
|  |  | ||||||
| 		table.SetCaption(true, stats) | 			serverStat := allStats[app.Server] | ||||||
| 		table.Render() |  | ||||||
|  | 			tableCol := []string{"recipe", "domain", "app name"} | ||||||
|  | 			if status { | ||||||
|  | 				tableCol = append(tableCol, []string{"status", "version", "upgrade"}...) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
|  | 			for _, appStat := range serverStat.apps { | ||||||
|  | 				tableRow := []string{appStat.recipe, appStat.domain, appStat.appName} | ||||||
|  | 				if status { | ||||||
|  | 					tableRow = append(tableRow, []string{appStat.status, appStat.version, appStat.upgrade}...) | ||||||
|  | 				} | ||||||
|  | 				table.Append(tableRow) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if table.NumLines() > 0 { | ||||||
|  | 				table.Render() | ||||||
|  |  | ||||||
|  | 				if status { | ||||||
|  | 					fmt.Println(fmt.Sprintf( | ||||||
|  | 						"server: %s | total apps: %v | versioned: %v | unversioned: %v | latest: %v | upgrade: %v", | ||||||
|  | 						app.Server, | ||||||
|  | 						serverStat.appCount, | ||||||
|  | 						serverStat.versionCount, | ||||||
|  | 						serverStat.unversionedCount, | ||||||
|  | 						serverStat.latestCount, | ||||||
|  | 						serverStat.upgradeCount, | ||||||
|  | 					)) | ||||||
|  | 				} else { | ||||||
|  | 					fmt.Println(fmt.Sprintf("server: %s | total apps: %v", app.Server, serverStat.appCount)) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if len(allStats) > 1 && table.NumLines() > 0 { | ||||||
|  | 				fmt.Println() // newline separator for multiple servers | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			alreadySeen[app.Server] = true | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if len(allStats) > 1 { | ||||||
|  | 			fmt.Println(fmt.Sprintf("total servers: %v | total apps: %v ", totalServersCount, totalAppsCount)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
|  | |||||||
							
								
								
									
										106
									
								
								cli/app/logs.go
									
									
									
									
									
								
							
							
						
						
									
										106
									
								
								cli/app/logs.go
									
									
									
									
									
								
							| @ -7,8 +7,10 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/service" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/docker/docker/api/types/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
| 	dockerClient "github.com/docker/docker/client" | 	dockerClient "github.com/docker/docker/client" | ||||||
| @ -16,6 +18,15 @@ import ( | |||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | var logOpts = types.ContainerLogsOptions{ | ||||||
|  | 	Details:    false, | ||||||
|  | 	Follow:     true, | ||||||
|  | 	ShowStderr: true, | ||||||
|  | 	ShowStdout: true, | ||||||
|  | 	Tail:       "20", | ||||||
|  | 	Timestamps: true, | ||||||
|  | } | ||||||
|  |  | ||||||
| // stackLogs lists logs for all stack services | // stackLogs lists logs for all stack services | ||||||
| func stackLogs(c *cli.Context, stackName string, client *dockerClient.Client) { | func stackLogs(c *cli.Context, stackName string, client *dockerClient.Client) { | ||||||
| 	filters := filters.NewArgs() | 	filters := filters.NewArgs() | ||||||
| @ -30,19 +41,14 @@ func stackLogs(c *cli.Context, stackName string, client *dockerClient.Client) { | |||||||
| 	for _, service := range services { | 	for _, service := range services { | ||||||
| 		wg.Add(1) | 		wg.Add(1) | ||||||
| 		go func(s string) { | 		go func(s string) { | ||||||
| 			logOpts := types.ContainerLogsOptions{ | 			if internal.StdErrOnly { | ||||||
| 				Details:    true, | 				logOpts.ShowStdout = false | ||||||
| 				Follow:     true, |  | ||||||
| 				ShowStderr: true, |  | ||||||
| 				ShowStdout: true, |  | ||||||
| 				Tail:       "20", |  | ||||||
| 				Timestamps: true, |  | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			logs, err := client.ServiceLogs(c.Context, s, logOpts) | 			logs, err := client.ServiceLogs(c.Context, s, logOpts) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			// defer after err check as any err returns a nil io.ReadCloser |  | ||||||
| 			defer logs.Close() | 			defer logs.Close() | ||||||
|  |  | ||||||
| 			_, err = io.Copy(os.Stdout, logs) | 			_, err = io.Copy(os.Stdout, logs) | ||||||
| @ -51,7 +57,9 @@ func stackLogs(c *cli.Context, stackName string, client *dockerClient.Client) { | |||||||
| 			} | 			} | ||||||
| 		}(service.ID) | 		}(service.ID) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	wg.Wait() | 	wg.Wait() | ||||||
|  |  | ||||||
| 	os.Exit(0) | 	os.Exit(0) | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -60,6 +68,10 @@ var appLogsCommand = &cli.Command{ | |||||||
| 	Aliases:   []string{"l"}, | 	Aliases:   []string{"l"}, | ||||||
| 	ArgsUsage: "[<service>]", | 	ArgsUsage: "[<service>]", | ||||||
| 	Usage:     "Tail app logs", | 	Usage:     "Tail app logs", | ||||||
|  | 	Flags: []cli.Flag{ | ||||||
|  | 		internal.StdErrOnlyFlag, | ||||||
|  | 	}, | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| @ -70,55 +82,41 @@ var appLogsCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		serviceName := c.Args().Get(1) | 		serviceName := c.Args().Get(1) | ||||||
| 		if serviceName == "" { | 		if serviceName == "" { | ||||||
| 			logrus.Debug("tailing logs for all app services") | 			logrus.Debugf("tailing logs for all %s services", app.Type) | ||||||
| 			stackLogs(c, app.StackName(), cl) | 			stackLogs(c, app.StackName(), cl) | ||||||
| 		} | 		} else { | ||||||
| 		logrus.Debugf("tailing logs for '%s'", serviceName) | 			logrus.Debugf("tailing logs for %s", serviceName) | ||||||
|  | 			if err := tailServiceLogs(c, cl, app, serviceName); err != nil { | ||||||
| 		service := fmt.Sprintf("%s_%s", app.StackName(), serviceName) | 				logrus.Fatal(err) | ||||||
| 		filters := filters.NewArgs() | 			} | ||||||
| 		filters.Add("name", service) |  | ||||||
| 		serviceOpts := types.ServiceListOptions{Filters: filters} |  | ||||||
| 		services, err := cl.ServiceList(c.Context, serviceOpts) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		if len(services) != 1 { |  | ||||||
| 			logrus.Fatalf("expected 1 service but got %v", len(services)) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logOpts := types.ContainerLogsOptions{ |  | ||||||
| 			Details:    true, |  | ||||||
| 			Follow:     true, |  | ||||||
| 			ShowStderr: true, |  | ||||||
| 			ShowStdout: true, |  | ||||||
| 			Tail:       "20", |  | ||||||
| 			Timestamps: true, |  | ||||||
| 		} |  | ||||||
| 		logs, err := cl.ServiceLogs(c.Context, services[0].ID, logOpts) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		// defer after err check as any err returns a nil io.ReadCloser |  | ||||||
| 		defer logs.Close() |  | ||||||
|  |  | ||||||
| 		_, err = io.Copy(os.Stdout, logs) |  | ||||||
| 		if err != nil && err != io.EOF { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | } | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { | func tailServiceLogs(c *cli.Context, cl *dockerClient.Client, app config.App, serviceName string) error { | ||||||
| 			logrus.Warn(err) | 	filters := filters.NewArgs() | ||||||
| 		} | 	filters.Add("name", fmt.Sprintf("%s_%s", app.StackName(), serviceName)) | ||||||
| 		if c.NArg() > 0 { | 	chosenService, err := service.GetService(c.Context, cl, filters, internal.NoInput) | ||||||
| 			return | 	if err != nil { | ||||||
| 		} | 		logrus.Fatal(err) | ||||||
| 		for _, a := range appNames { | 	} | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} | 	if internal.StdErrOnly { | ||||||
| 	}, | 		logOpts.ShowStdout = false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logs, err := cl.ServiceLogs(c.Context, chosenService.ID, logOpts) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	defer logs.Close() | ||||||
|  |  | ||||||
|  | 	_, err = io.Copy(os.Stdout, logs) | ||||||
|  | 	if err != nil && err != io.EOF { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,11 +1,8 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @ -41,18 +38,7 @@ var appNewCommand = &cli.Command{ | |||||||
| 		internal.PassFlag, | 		internal.PassFlag, | ||||||
| 		internal.SecretsFlag, | 		internal.SecretsFlag, | ||||||
| 	}, | 	}, | ||||||
| 	ArgsUsage: "<recipe>", | 	ArgsUsage:    "<recipe>", | ||||||
| 	Action:    internal.NewAction, | 	Action:       internal.NewAction, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for name := range catl { |  | ||||||
| 			fmt.Println(name) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,72 +1,68 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/docker/cli/cli/command/formatter" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"coopcloud.tech/abra/pkg/service" | ||||||
|  | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
|  | 	"github.com/buger/goterm" | ||||||
|  | 	dockerFormatter "github.com/docker/cli/cli/command/formatter" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/docker/docker/api/types/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	dockerClient "github.com/docker/docker/client" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var watch bool |  | ||||||
| var watchFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "watch", |  | ||||||
| 	Aliases:     []string{"w"}, |  | ||||||
| 	Value:       false, |  | ||||||
| 	Usage:       "Watch status by polling repeatedly", |  | ||||||
| 	Destination: &watch, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var appPsCommand = &cli.Command{ | var appPsCommand = &cli.Command{ | ||||||
| 	Name:    "ps", | 	Name:        "ps", | ||||||
| 	Usage:   "Check app status", | 	Usage:       "Check app status", | ||||||
| 	Aliases: []string{"p"}, | 	Description: "This command shows a more detailed status output of a specific deployed app.", | ||||||
|  | 	Aliases:     []string{"p"}, | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		watchFlag, | 		internal.WatchFlag, | ||||||
| 	}, | 	}, | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		if !watch { | 		app := internal.ValidateApp(c) | ||||||
| 			showPSOutput(c) |  | ||||||
|  | 		cl, err := client.New(app.Server) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		isDeployed, _, err := stack.IsDeployed(c.Context, cl, app.StackName()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !isDeployed { | ||||||
|  | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.Watch { | ||||||
|  | 			showPSOutput(c, app, cl) | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// TODO: how do we make this update in-place in an x-platform way? | 		goterm.Clear() | ||||||
| 		for { | 		for { | ||||||
| 			showPSOutput(c) | 			goterm.MoveCursor(1, 1) | ||||||
|  | 			showPSOutput(c, app, cl) | ||||||
|  | 			goterm.Flush() | ||||||
| 			time.Sleep(2 * time.Second) | 			time.Sleep(2 * time.Second) | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { |  | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // showPSOutput renders ps output. | // showPSOutput renders ps output. | ||||||
| func showPSOutput(c *cli.Context) { | func showPSOutput(c *cli.Context, app config.App, cl *dockerClient.Client) { | ||||||
| 	app := internal.ValidateApp(c) |  | ||||||
|  |  | ||||||
| 	cl, err := client.New(app.Server) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	filters := filters.NewArgs() | 	filters := filters.NewArgs() | ||||||
| 	filters.Add("name", app.StackName()) | 	filters.Add("name", app.StackName()) | ||||||
|  |  | ||||||
| @ -75,8 +71,8 @@ func showPSOutput(c *cli.Context) { | |||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	tableCol := []string{"image", "created", "status", "ports", "names"} | 	tableCol := []string{"service name", "image", "created", "status", "state", "ports"} | ||||||
| 	table := abraFormatter.CreateTable(tableCol) | 	table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 	for _, container := range containers { | 	for _, container := range containers { | ||||||
| 		var containerNames []string | 		var containerNames []string | ||||||
| @ -86,11 +82,12 @@ func showPSOutput(c *cli.Context) { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableRow := []string{ | 		tableRow := []string{ | ||||||
| 			abraFormatter.RemoveSha(container.Image), | 			service.ContainerToServiceName(container.Names, app.StackName()), | ||||||
| 			abraFormatter.HumanDuration(container.Created), | 			formatter.RemoveSha(container.Image), | ||||||
|  | 			formatter.HumanDuration(container.Created), | ||||||
| 			container.Status, | 			container.Status, | ||||||
| 			formatter.DisplayablePorts(container.Ports), | 			container.State, | ||||||
| 			strings.Join(containerNames, "\n"), | 			dockerFormatter.DisplayablePorts(container.Ports), | ||||||
| 		} | 		} | ||||||
| 		table.Append(tableRow) | 		table.Append(tableRow) | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -5,8 +5,9 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	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/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
| @ -48,23 +49,18 @@ var appRemoveCommand = &cli.Command{ | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		appFiles, err := config.LoadAppFiles("") |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		cl, err := client.New(app.Server) | 		cl, err := client.New(app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !internal.Force { | 		if !internal.Force { | ||||||
| 			// FIXME: only query for app we are interested in, not all of them! | 			isDeployed, _, err := stack.IsDeployed(c.Context, cl, app.StackName()) | ||||||
| 			statuses, err := config.GetAppStatuses(appFiles) |  | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			if statuses[app.Name]["status"] == "deployed" { | 			if isDeployed { | ||||||
| 				logrus.Fatalf("'%s' is still deployed. Run \"abra app %s undeploy\" or pass --force", app.Name, app.Name) | 				logrus.Fatalf("%s is still deployed. Run \"abra app undeploy %s \" or pass --force", app.Name, app.Name) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -88,6 +84,8 @@ var appRemoveCommand = &cli.Command{ | |||||||
| 			if !internal.Force { | 			if !internal.Force { | ||||||
| 				secretsPrompt := &survey.MultiSelect{ | 				secretsPrompt := &survey.MultiSelect{ | ||||||
| 					Message: "which secrets do you want to remove?", | 					Message: "which secrets do you want to remove?", | ||||||
|  | 					Help:    "'x' indicates selected, enter / return to confirm, ctrl-c to exit, vim mode is enabled", | ||||||
|  | 					VimMode: true, | ||||||
| 					Options: secretNames, | 					Options: secretNames, | ||||||
| 					Default: secretNames, | 					Default: secretNames, | ||||||
| 				} | 				} | ||||||
| @ -124,6 +122,8 @@ var appRemoveCommand = &cli.Command{ | |||||||
| 				if !internal.Force { | 				if !internal.Force { | ||||||
| 					volumesPrompt := &survey.MultiSelect{ | 					volumesPrompt := &survey.MultiSelect{ | ||||||
| 						Message: "which volumes do you want to remove?", | 						Message: "which volumes do you want to remove?", | ||||||
|  | 						Help:    "'x' indicates selected, enter / return to confirm, ctrl-c to exit, vim mode is enabled", | ||||||
|  | 						VimMode: true, | ||||||
| 						Options: vols, | 						Options: vols, | ||||||
| 						Default: vols, | 						Default: vols, | ||||||
| 					} | 					} | ||||||
| @ -153,16 +153,5 @@ var appRemoveCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -3,28 +3,28 @@ package app | |||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	upstream "coopcloud.tech/abra/pkg/upstream/service" | ||||||
| 	"github.com/docker/docker/api/types" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/docker/docker/api/types/filters" |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appRestartCommand = &cli.Command{ | var appRestartCommand = &cli.Command{ | ||||||
| 	Name:        "restart", | 	Name:         "restart", | ||||||
| 	Usage:       "Restart an app", | 	Usage:        "Restart an app", | ||||||
| 	Aliases:     []string{"R"}, | 	Aliases:      []string{"re"}, | ||||||
| 	ArgsUsage:   "<service>", | 	ArgsUsage:    "<service>", | ||||||
| 	Description: `This command restarts a service within a deployed app.`, | 	Description:  `This command restarts a service within a deployed app.`, | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		serviceName := c.Args().Get(1) | 		serviceNameShort := c.Args().Get(1) | ||||||
| 		if serviceName == "" { | 		if serviceNameShort == "" { | ||||||
| 			err := errors.New("missing service?") | 			err := errors.New("missing service?") | ||||||
| 			internal.ShowSubcommandHelpAndError(c, err) | 			internal.ShowSubcommandHelpAndError(c, err) | ||||||
| 		} | 		} | ||||||
| @ -34,39 +34,32 @@ var appRestartCommand = &cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		serviceFilter := fmt.Sprintf("%s_%s", app.StackName(), serviceName) | 		serviceName := fmt.Sprintf("%s_%s", app.StackName(), serviceNameShort) | ||||||
| 		filters := filters.NewArgs() |  | ||||||
| 		filters.Add("name", serviceFilter) |  | ||||||
| 		containerOpts := types.ContainerListOptions{Filters: filters} |  | ||||||
| 		containers, err := cl.ContainerList(c.Context, containerOpts) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		if len(containers) != 1 { |  | ||||||
| 			logrus.Fatalf("expected 1 service but got %v", len(containers)) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("attempting to restart %s", serviceFilter) | 		logrus.Debugf("attempting to scale %s to 0 (restart logic)", serviceName) | ||||||
|  | 		if err := upstream.RunServiceScale(c.Context, cl, serviceName, 0); err != nil { | ||||||
| 		timeout := 30 * time.Second |  | ||||||
| 		if err := cl.ContainerRestart(c.Context, containers[0].ID, &timeout); err != nil { |  | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Infof("%s service restarted", serviceFilter) | 		if err := stack.WaitOnService(c.Context, cl, serviceName, app.Name); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Debugf("%s has been scaled to 0 (restart logic)", serviceName) | ||||||
|  |  | ||||||
|  | 		logrus.Debugf("attempting to scale %s to 1 (restart logic)", serviceName) | ||||||
|  | 		if err := upstream.RunServiceScale(c.Context, cl, serviceName, 1); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := stack.WaitOnService(c.Context, cl, serviceName, app.Name); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Debugf("%s has been scaled to 1 (restart logic)", serviceName) | ||||||
|  |  | ||||||
|  | 		logrus.Infof("%s service successfully restarted", serviceNameShort) | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { |  | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -27,7 +27,7 @@ var restoreAllServicesFlag = &cli.BoolFlag{ | |||||||
| var appRestoreCommand = &cli.Command{ | var appRestoreCommand = &cli.Command{ | ||||||
| 	Name:      "restore", | 	Name:      "restore", | ||||||
| 	Usage:     "Restore an app from a backup", | 	Usage:     "Restore an app from a backup", | ||||||
| 	Aliases:   []string{"r"}, | 	Aliases:   []string{"rs"}, | ||||||
| 	Flags:     []cli.Flag{restoreAllServicesFlag}, | 	Flags:     []cli.Flag{restoreAllServicesFlag}, | ||||||
| 	ArgsUsage: "<service> [<backup file>]", | 	ArgsUsage: "<service> [<backup file>]", | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| @ -37,10 +37,10 @@ var appRestoreCommand = &cli.Command{ | |||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use <service>/<backup file> and '--all' together")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use <service>/<backup file> and '--all' together")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		abraSh := path.Join(config.ABRA_DIR, "apps", app.Type, "abra.sh") | 		abraSh := path.Join(config.RECIPES_DIR, app.Type, "abra.sh") | ||||||
| 		if _, err := os.Stat(abraSh); err != nil { | 		if _, err := os.Stat(abraSh); err != nil { | ||||||
| 			if os.IsNotExist(err) { | 			if os.IsNotExist(err) { | ||||||
| 				logrus.Fatalf("'%s' does not exist?", abraSh) | 				logrus.Fatalf("%s does not exist?", abraSh) | ||||||
| 			} | 			} | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -60,7 +60,7 @@ var appRestoreCommand = &cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		if !strings.Contains(string(bytes), execCmd) { | 		if !strings.Contains(string(bytes), execCmd) { | ||||||
| 			logrus.Fatalf("%s doesn't have a '%s' function", app.Type, execCmd) | 			logrus.Fatalf("%s doesn't have a %s function", app.Type, execCmd) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		backupFile := c.Args().Get(2) | 		backupFile := c.Args().Get(2) | ||||||
|  | |||||||
| @ -3,8 +3,9 @@ package app | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/lint" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| @ -19,11 +20,12 @@ import ( | |||||||
| var appRollbackCommand = &cli.Command{ | var appRollbackCommand = &cli.Command{ | ||||||
| 	Name:      "rollback", | 	Name:      "rollback", | ||||||
| 	Usage:     "Roll an app back to a previous version", | 	Usage:     "Roll an app back to a previous version", | ||||||
| 	Aliases:   []string{"r", "downgrade"}, | 	Aliases:   []string{"rl"}, | ||||||
| 	ArgsUsage: "<app>", | 	ArgsUsage: "<app>", | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.ForceFlag, | 		internal.ForceFlag, | ||||||
| 		internal.ChaosFlag, | 		internal.ChaosFlag, | ||||||
|  | 		internal.DontWaitConvergeFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command rolls an app back to a previous version if one exists. | This command rolls an app back to a previous version if one exists. | ||||||
| @ -38,28 +40,30 @@ Chas mode ("--chaos") will deploy your local checkout of a recipe as-is, | |||||||
| including unstaged changes and can be useful for live hacking and testing new | including unstaged changes and can be useful for live hacking and testing new | ||||||
| recipes. | recipes. | ||||||
| `, | `, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
| 		stackName := app.StackName() | 		stackName := app.StackName() | ||||||
|  |  | ||||||
|  | 		if err := recipe.EnsureUpToDate(app.Type); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		r, err := recipe.Get(app.Type) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := lint.LintForErrors(r); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		cl, err := client.New(app.Server) | 		cl, err := client.New(app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("checking whether '%s' is already deployed", stackName) | 		logrus.Debugf("checking whether %s is already deployed", stackName) | ||||||
|  |  | ||||||
| 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -67,19 +71,27 @@ recipes. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !isDeployed { | 		if !isDeployed { | ||||||
| 			logrus.Fatalf("'%s' is not deployed?", app.Name) | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		versions, err := catalogue.GetRecipeCatalogueVersions(app.Type) | 		catl, err := recipe.ReadRecipeCatalogue() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		versions, err := recipe.GetRecipeCatalogueVersions(app.Type, catl) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if len(versions) == 0 && !internal.Chaos { | ||||||
|  | 			logrus.Fatalf("no published releases for %s in the recipe catalogue?", app.Type) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		var availableDowngrades []string | 		var availableDowngrades []string | ||||||
| 		if deployedVersion == "" { | 		if deployedVersion == "unknown" { | ||||||
| 			deployedVersion = "unknown" |  | ||||||
| 			availableDowngrades = versions | 			availableDowngrades = versions | ||||||
| 			logrus.Warnf("failed to determine version of deployed '%s'", app.Name) | 			logrus.Warnf("failed to determine version of deployed %s", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if deployedVersion != "unknown" && !internal.Chaos { | 		if deployedVersion != "unknown" && !internal.Chaos { | ||||||
| @ -98,23 +110,21 @@ recipes. | |||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if len(availableDowngrades) == 0 { | 			if len(availableDowngrades) == 0 { | ||||||
| 				logrus.Fatal("no available downgrades, you're on latest") | 				logrus.Info("no available downgrades, you're on oldest ✌️") | ||||||
|  | 				return nil | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// FIXME: jeezus golang why do you not have a list reverse function | 		availableDowngrades = internal.ReverseStringList(availableDowngrades) | ||||||
| 		for i, j := 0, len(availableDowngrades)-1; i < j; i, j = i+1, j-1 { |  | ||||||
| 			availableDowngrades[i], availableDowngrades[j] = availableDowngrades[j], availableDowngrades[i] |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		var chosenDowngrade string | 		var chosenDowngrade string | ||||||
| 		if !internal.Chaos { | 		if !internal.Chaos { | ||||||
| 			if internal.Force { | 			if internal.Force { | ||||||
| 				chosenDowngrade = availableDowngrades[0] | 				chosenDowngrade = availableDowngrades[0] | ||||||
| 				logrus.Debugf("choosing '%s' as version to downgrade to (--force)", chosenDowngrade) | 				logrus.Debugf("choosing %s as version to downgrade to (--force)", chosenDowngrade) | ||||||
| 			} else { | 			} else { | ||||||
| 				prompt := &survey.Select{ | 				prompt := &survey.Select{ | ||||||
| 					Message: fmt.Sprintf("Please select a downgrade (current version: '%s'):", deployedVersion), | 					Message: fmt.Sprintf("Please select a downgrade (current version: %s):", deployedVersion), | ||||||
| 					Options: availableDowngrades, | 					Options: availableDowngrades, | ||||||
| 				} | 				} | ||||||
| 				if err := survey.AskOne(prompt, &chosenDowngrade); err != nil { | 				if err := survey.AskOne(prompt, &chosenDowngrade); err != nil { | ||||||
| @ -138,7 +148,7 @@ recipes. | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		abraShPath := fmt.Sprintf("%s/%s/%s", config.APPS_DIR, app.Type, "abra.sh") | 		abraShPath := fmt.Sprintf("%s/%s/%s", config.RECIPES_DIR, app.Type, "abra.sh") | ||||||
| 		abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | 		abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| @ -163,12 +173,12 @@ recipes. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !internal.Force { | 		if !internal.Force { | ||||||
| 			if err := internal.NewVersionOverview(app, deployedVersion, chosenDowngrade); err != nil { | 			if err := internal.NewVersionOverview(app, deployedVersion, chosenDowngrade, ""); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := stack.RunDeploy(cl, deployOpts, compose); err != nil { | 		if err := stack.RunDeploy(cl, deployOpts, compose, app.StackName(), internal.DontWaitConverge); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | |||||||
| @ -5,8 +5,9 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	containerPkg "coopcloud.tech/abra/pkg/container" | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/container" | 	"coopcloud.tech/abra/pkg/upstream/container" | ||||||
| 	"github.com/docker/cli/cli/command" | 	"github.com/docker/cli/cli/command" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| @ -35,9 +36,10 @@ var appRunCommand = &cli.Command{ | |||||||
| 		noTTYFlag, | 		noTTYFlag, | ||||||
| 		userFlag, | 		userFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Aliases:   []string{"r"}, | 	Aliases:      []string{"r"}, | ||||||
| 	ArgsUsage: "<service> <args>...", | 	ArgsUsage:    "<service> <args>...", | ||||||
| 	Usage:     "Run a command in a service container", | 	Usage:        "Run a command in a service container", | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| @ -59,18 +61,11 @@ var appRunCommand = &cli.Command{ | |||||||
| 		filters := filters.NewArgs() | 		filters := filters.NewArgs() | ||||||
| 		filters.Add("name", stackAndServiceName) | 		filters.Add("name", stackAndServiceName) | ||||||
|  |  | ||||||
| 		containers, err := cl.ContainerList(c.Context, types.ContainerListOptions{Filters: filters}) | 		targetContainer, err := containerPkg.GetContainer(c.Context, cl, filters, true) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(containers) == 0 { |  | ||||||
| 			logrus.Fatalf("no containers matching '%s' found?", stackAndServiceName) |  | ||||||
| 		} |  | ||||||
| 		if len(containers) > 1 { |  | ||||||
| 			logrus.Fatalf("expected 1 container matching '%s' but got %d", stackAndServiceName, len(containers)) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		cmd := c.Args().Slice()[2:] | 		cmd := c.Args().Slice()[2:] | ||||||
| 		execCreateOpts := types.ExecConfig{ | 		execCreateOpts := types.ExecConfig{ | ||||||
| 			AttachStderr: true, | 			AttachStderr: true, | ||||||
| @ -88,41 +83,16 @@ var appRunCommand = &cli.Command{ | |||||||
| 			execCreateOpts.Tty = false | 			execCreateOpts.Tty = false | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// FIXME: an absolutely monumental hack to instantiate another command-line | 		// FIXME: avoid instantiating a new CLI | ||||||
| 		// client withing our command-line client so that we pass something down |  | ||||||
| 		// the tubes that satisfies the necessary interface requirements. We should |  | ||||||
| 		// refactor our vendored container code to not require all this cruft.  For |  | ||||||
| 		// now, It Works. |  | ||||||
| 		dcli, err := command.NewDockerCli() | 		dcli, err := command.NewDockerCli() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, containers[0].ID, &execCreateOpts); err != nil { | 		if err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { |  | ||||||
| 		switch c.NArg() { |  | ||||||
| 		case 0: |  | ||||||
| 			appNames, err := config.GetAppNames() |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Warn(err) |  | ||||||
| 			} |  | ||||||
| 			for _, a := range appNames { |  | ||||||
| 				fmt.Println(a) |  | ||||||
| 			} |  | ||||||
| 		case 1: |  | ||||||
| 			appName := c.Args().First() |  | ||||||
| 			serviceNames, err := config.GetAppServiceNames(appName) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Warn(err) |  | ||||||
| 			} |  | ||||||
| 			for _, s := range serviceNames { |  | ||||||
| 				fmt.Println(s) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -6,10 +6,10 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
| 	"strconv" | 	"strconv" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/abra/pkg/secret" | 	"coopcloud.tech/abra/pkg/secret" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/docker/docker/api/types/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
| @ -60,7 +60,7 @@ var appSecretGenerateCommand = &cli.Command{ | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			if !matches { | 			if !matches { | ||||||
| 				logrus.Fatalf("'%s' doesn't exist in the env config?", secretName) | 				logrus.Fatalf("%s doesn't exist in the env config?", secretName) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -83,7 +83,7 @@ var appSecretGenerateCommand = &cli.Command{ | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"name", "value"} | 		tableCol := []string{"name", "value"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
| 		for name, val := range secretVals { | 		for name, val := range secretVals { | ||||||
| 			table.Append([]string{name, val}) | 			table.Append([]string{name, val}) | ||||||
| 		} | 		} | ||||||
| @ -215,7 +215,7 @@ var appSecretLsCommand = &cli.Command{ | |||||||
| 		secrets := secret.ReadSecretEnvVars(app.Env) | 		secrets := secret.ReadSecretEnvVars(app.Env) | ||||||
|  |  | ||||||
| 		tableCol := []string{"Name", "Version", "Generated Name", "Created On Server"} | 		tableCol := []string{"Name", "Version", "Generated Name", "Created On Server"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 		cl, err := client.New(app.Server) | 		cl, err := client.New(app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -249,21 +249,15 @@ var appSecretLsCommand = &cli.Command{ | |||||||
| 			table.Append(tableRow) | 			table.Append(tableRow) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table.Render() | 		if table.NumLines() > 0 { | ||||||
|  | 			table.Render() | ||||||
|  | 		} else { | ||||||
|  | 			logrus.Warnf("no secrets stored for %s", app.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| var appSecretCommand = &cli.Command{ | var appSecretCommand = &cli.Command{ | ||||||
|  | |||||||
| @ -1,11 +1,9 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| @ -13,7 +11,7 @@ import ( | |||||||
|  |  | ||||||
| var appUndeployCommand = &cli.Command{ | var appUndeployCommand = &cli.Command{ | ||||||
| 	Name:    "undeploy", | 	Name:    "undeploy", | ||||||
| 	Aliases: []string{"u"}, | 	Aliases: []string{"un"}, | ||||||
| 	Usage:   "Undeploy an app", | 	Usage:   "Undeploy an app", | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This does not destroy any of the application data. However, you should remain | This does not destroy any of the application data. However, you should remain | ||||||
| @ -29,7 +27,7 @@ volumes as eligiblef or pruning once undeployed. | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("checking whether '%s' is already deployed", stackName) | 		logrus.Debugf("checking whether %s is already deployed", stackName) | ||||||
|  |  | ||||||
| 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -37,7 +35,7 @@ volumes as eligiblef or pruning once undeployed. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !isDeployed { | 		if !isDeployed { | ||||||
| 			logrus.Fatalf("'%s' is not deployed?", stackName) | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := internal.DeployOverview(app, deployedVersion, "continue with undeploy?"); err != nil { | 		if err := internal.DeployOverview(app, deployedVersion, "continue with undeploy?"); err != nil { | ||||||
| @ -51,16 +49,5 @@ volumes as eligiblef or pruning once undeployed. | |||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -4,9 +4,10 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/lint" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| @ -17,18 +18,19 @@ import ( | |||||||
|  |  | ||||||
| var appUpgradeCommand = &cli.Command{ | var appUpgradeCommand = &cli.Command{ | ||||||
| 	Name:      "upgrade", | 	Name:      "upgrade", | ||||||
| 	Aliases:   []string{"u"}, | 	Aliases:   []string{"up"}, | ||||||
| 	Usage:     "Upgrade an app", | 	Usage:     "Upgrade an app", | ||||||
| 	ArgsUsage: "<app>", | 	ArgsUsage: "<app>", | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.ForceFlag, | 		internal.ForceFlag, | ||||||
| 		internal.ChaosFlag, | 		internal.ChaosFlag, | ||||||
|  | 		internal.NoDomainChecksFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command supports upgrading an app. You can use it to choose and roll out a | This command supports upgrading an app. You can use it to choose and roll out a | ||||||
| new upgrade to an existing app. | new upgrade to an existing app. | ||||||
|  |  | ||||||
| This command specifically supports changing the version of running apps, as | This command specifically supports incrementing the version of running apps, as | ||||||
| opposed to "abra app deploy <app>" which will not change the version of a | opposed to "abra app deploy <app>" which will not change the version of a | ||||||
| deployed app. | deployed app. | ||||||
|  |  | ||||||
| @ -46,12 +48,25 @@ recipes. | |||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
| 		stackName := app.StackName() | 		stackName := app.StackName() | ||||||
|  |  | ||||||
|  | 		if err := recipe.EnsureUpToDate(app.Type); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		r, err := recipe.Get(app.Type) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := lint.LintForErrors(r); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		cl, err := client.New(app.Server) | 		cl, err := client.New(app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("checking whether '%s' is already deployed", stackName) | 		logrus.Debugf("checking whether %s is already deployed", stackName) | ||||||
|  |  | ||||||
| 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -59,23 +74,27 @@ recipes. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !isDeployed { | 		if !isDeployed { | ||||||
| 			logrus.Fatalf("'%s' is not deployed?", app.Name) | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		versions, err := catalogue.GetRecipeCatalogueVersions(app.Type) | 		catl, err := recipe.ReadRecipeCatalogue() | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		versions, err := recipe.GetRecipeCatalogueVersions(app.Type, catl) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(versions) == 0 && !internal.Chaos { | 		if len(versions) == 0 && !internal.Chaos { | ||||||
| 			logrus.Fatalf("no versions available '%s' in recipe catalogue?", app.Type) | 			logrus.Fatalf("no published releases for %s in the recipe catalogue?", app.Type) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		var availableUpgrades []string | 		var availableUpgrades []string | ||||||
| 		if deployedVersion == "" { | 		if deployedVersion == "uknown" { | ||||||
| 			deployedVersion = "unknown" |  | ||||||
| 			availableUpgrades = versions | 			availableUpgrades = versions | ||||||
| 			logrus.Warnf("failed to determine version of deployed '%s'", app.Name) | 			logrus.Warnf("failed to determine version of deployed %s", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if deployedVersion != "unknown" && !internal.Chaos { | 		if deployedVersion != "unknown" && !internal.Chaos { | ||||||
| @ -94,19 +113,21 @@ recipes. | |||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if len(availableUpgrades) == 0 && !internal.Force { | 			if len(availableUpgrades) == 0 && !internal.Force { | ||||||
| 				logrus.Fatal("no available upgrades, you're on latest") | 				logrus.Info("no available upgrades, you're on latest ✌️") | ||||||
| 				availableUpgrades = versions | 				return nil | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		availableUpgrades = internal.ReverseStringList(availableUpgrades) | ||||||
|  |  | ||||||
| 		var chosenUpgrade string | 		var chosenUpgrade string | ||||||
| 		if len(availableUpgrades) > 0 && !internal.Chaos { | 		if len(availableUpgrades) > 0 && !internal.Chaos { | ||||||
| 			if internal.Force { | 			if internal.Force { | ||||||
| 				chosenUpgrade = availableUpgrades[len(availableUpgrades)-1] | 				chosenUpgrade = availableUpgrades[len(availableUpgrades)-1] | ||||||
| 				logrus.Debugf("choosing '%s' as version to upgrade to", chosenUpgrade) | 				logrus.Debugf("choosing %s as version to upgrade to", chosenUpgrade) | ||||||
| 			} else { | 			} else { | ||||||
| 				prompt := &survey.Select{ | 				prompt := &survey.Select{ | ||||||
| 					Message: fmt.Sprintf("Please select an upgrade (current version: '%s'):", deployedVersion), | 					Message: fmt.Sprintf("Please select an upgrade (current version: %s):", deployedVersion), | ||||||
| 					Options: availableUpgrades, | 					Options: availableUpgrades, | ||||||
| 				} | 				} | ||||||
| 				if err := survey.AskOne(prompt, &chosenUpgrade); err != nil { | 				if err := survey.AskOne(prompt, &chosenUpgrade); err != nil { | ||||||
| @ -115,6 +136,14 @@ recipes. | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		// if release notes written after git tag published, read them before we | ||||||
|  | 		// check out the tag and then they'll appear to be missing. this covers | ||||||
|  | 		// when we obviously will forget to write release notes before publishing | ||||||
|  | 		releaseNotes, err := internal.GetReleaseNotes(app.Type, chosenUpgrade) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		if !internal.Chaos { | 		if !internal.Chaos { | ||||||
| 			if err := recipe.EnsureVersion(app.Type, chosenUpgrade); err != nil { | 			if err := recipe.EnsureVersion(app.Type, chosenUpgrade); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| @ -130,7 +159,7 @@ recipes. | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		abraShPath := fmt.Sprintf("%s/%s/%s", config.APPS_DIR, app.Type, "abra.sh") | 		abraShPath := fmt.Sprintf("%s/%s/%s", config.RECIPES_DIR, app.Type, "abra.sh") | ||||||
| 		abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | 		abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| @ -154,26 +183,15 @@ recipes. | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := internal.NewVersionOverview(app, deployedVersion, chosenUpgrade); err != nil { | 		if err := internal.NewVersionOverview(app, deployedVersion, chosenUpgrade, releaseNotes); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := stack.RunDeploy(cl, deployOpts, compose); err != nil { | 		if err := stack.RunDeploy(cl, deployOpts, compose, app.StackName(), internal.DontWaitConverge); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,14 +1,13 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" | 	"coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| @ -21,11 +20,14 @@ func getImagePath(image string) (string, error) { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	path := reference.Path(img) | 	path := reference.Path(img) | ||||||
| 	if strings.Contains(path, "library") { | 	if strings.Contains(path, "library") { | ||||||
| 		path = strings.Split(path, "/")[1] | 		path = strings.Split(path, "/")[1] | ||||||
| 	} | 	} | ||||||
| 	logrus.Debugf("parsed '%s' from '%s'", path, image) |  | ||||||
|  | 	logrus.Debugf("parsed %s from %s", path, image) | ||||||
|  |  | ||||||
| 	return path, nil | 	return path, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -47,27 +49,27 @@ Cloud recipe version. | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("checking whether '%s' is already deployed", stackName) | 		logrus.Debugf("checking whether %s is already deployed", stackName) | ||||||
|  |  | ||||||
| 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | 		isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if deployedVersion == "" { | 		if deployedVersion == "unknown" { | ||||||
| 			logrus.Fatalf("failed to determine version of deployed '%s'", app.Name) | 			logrus.Fatalf("failed to determine version of deployed %s", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !isDeployed { | 		if !isDeployed { | ||||||
| 			logrus.Fatalf("'%s' is not deployed?", app.Name) | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		recipeMeta, err := catalogue.GetRecipeMeta(app.Type) | 		recipeMeta, err := recipe.GetRecipeMeta(app.Type) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		versionsMeta := make(map[string]catalogue.ServiceMeta) | 		versionsMeta := make(map[string]recipe.ServiceMeta) | ||||||
| 		for _, recipeVersion := range recipeMeta.Versions { | 		for _, recipeVersion := range recipeMeta.Versions { | ||||||
| 			if currentVersion, exists := recipeVersion[deployedVersion]; exists { | 			if currentVersion, exists := recipeVersion[deployedVersion]; exists { | ||||||
| 				versionsMeta = currentVersion | 				versionsMeta = currentVersion | ||||||
| @ -75,30 +77,20 @@ Cloud recipe version. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(versionsMeta) == 0 { | 		if len(versionsMeta) == 0 { | ||||||
| 			logrus.Fatalf("PANIC: could not retrieve deployed version ('%s') from recipe catalogue?", deployedVersion) | 			logrus.Fatalf("could not retrieve deployed version (%s) from recipe catalogue?", deployedVersion) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"name", "image", "version", "tag", "digest"} | 		tableCol := []string{"version", "service", "image", "digest"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  | 		table.SetAutoMergeCellsByColumnIndex([]int{0}) | ||||||
|  |  | ||||||
| 		for serviceName, versionMeta := range versionsMeta { | 		for serviceName, versionMeta := range versionsMeta { | ||||||
| 			table.Append([]string{serviceName, versionMeta.Image, deployedVersion, versionMeta.Tag, versionMeta.Digest}) | 			table.Append([]string{deployedVersion, serviceName, versionMeta.Image, versionMeta.Digest}) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table.Render() | 		table.Render() | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,21 +1,20 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appVolumeListCommand = &cli.Command{ | var appVolumeListCommand = &cli.Command{ | ||||||
| 	Name:    "list", | 	Name:         "list", | ||||||
| 	Usage:   "List volumes associated with an app", | 	Usage:        "List volumes associated with an app", | ||||||
| 	Aliases: []string{"ls"}, | 	Aliases:      []string{"ls"}, | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| @ -24,7 +23,7 @@ var appVolumeListCommand = &cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table := abraFormatter.CreateTable([]string{"driver", "volume name"}) | 		table := formatter.CreateTable([]string{"driver", "volume name"}) | ||||||
| 		var volTable [][]string | 		var volTable [][]string | ||||||
| 		for _, volume := range volumeList { | 		for _, volume := range volumeList { | ||||||
| 			volRow := []string{ | 			volRow := []string{ | ||||||
| @ -35,16 +34,33 @@ var appVolumeListCommand = &cli.Command{ | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table.AppendBulk(volTable) | 		table.AppendBulk(volTable) | ||||||
| 		table.Render() |  | ||||||
|  | 		if table.NumLines() > 0 { | ||||||
|  | 			table.Render() | ||||||
|  | 		} else { | ||||||
|  | 			logrus.Warnf("no volumes created for %s", app.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| var appVolumeRemoveCommand = &cli.Command{ | var appVolumeRemoveCommand = &cli.Command{ | ||||||
| 	Name:    "remove", | 	Name:  "remove", | ||||||
| 	Usage:   "Remove volume(s) associated with an app", | 	Usage: "Remove volume(s) associated with an app", | ||||||
| 	Aliases: []string{"rm"}, | 	Description: ` | ||||||
|  | This command supports removing volumes associated with an app. The app in | ||||||
|  | question must be undeployed before you try to remove volumes. See "abra app | ||||||
|  | undeploy <app>" for more. | ||||||
|  |  | ||||||
|  | The command is interactive and will show a multiple select input which allows | ||||||
|  | you to make a seclection. Use the "?" key to see more help on navigating this | ||||||
|  | interface. | ||||||
|  |  | ||||||
|  | Passing "--force" will select all volumes for removal. Be careful. | ||||||
|  | `, | ||||||
|  | 	ArgsUsage: "<app>", | ||||||
|  | 	Aliases:   []string{"rm"}, | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.ForceFlag, | 		internal.ForceFlag, | ||||||
| 	}, | 	}, | ||||||
| @ -61,6 +77,8 @@ var appVolumeRemoveCommand = &cli.Command{ | |||||||
| 		if !internal.Force { | 		if !internal.Force { | ||||||
| 			volumesPrompt := &survey.MultiSelect{ | 			volumesPrompt := &survey.MultiSelect{ | ||||||
| 				Message: "which volumes do you want to remove?", | 				Message: "which volumes do you want to remove?", | ||||||
|  | 				Help:    "'x' indicates selected, enter / return to confirm, ctrl-c to exit, vim mode is enabled", | ||||||
|  | 				VimMode: true, | ||||||
| 				Options: volumeNames, | 				Options: volumeNames, | ||||||
| 				Default: volumeNames, | 				Default: volumeNames, | ||||||
| 			} | 			} | ||||||
| @ -80,18 +98,7 @@ var appVolumeRemoveCommand = &cli.Command{ | |||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 		appNames, err := config.GetAppNames() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for _, a := range appNames { |  | ||||||
| 			fmt.Println(a) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| var appVolumeCommand = &cli.Command{ | var appVolumeCommand = &cli.Command{ | ||||||
|  | |||||||
| @ -1,121 +0,0 @@ | |||||||
| package cli |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"net/http" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // downloadFile downloads a file brah |  | ||||||
| func downloadFile(filepath string, url string) (err error) { |  | ||||||
| 	out, err := os.Create(filepath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer out.Close() |  | ||||||
|  |  | ||||||
| 	resp, err := http.Get(url) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer resp.Body.Close() |  | ||||||
|  |  | ||||||
| 	if resp.StatusCode != http.StatusOK { |  | ||||||
| 		return fmt.Errorf("bad status: %s", resp.Status) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	_, err = io.Copy(out, resp.Body) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // AutoCompleteCommand helps people set up auto-complete in their shells |  | ||||||
| var AutoCompleteCommand = &cli.Command{ |  | ||||||
| 	Name:    "autocomplete", |  | ||||||
| 	Usage:   "Help set up shell autocompletion", |  | ||||||
| 	Aliases: []string{"ac"}, |  | ||||||
| 	Description: ` |  | ||||||
| This command helps set up autocompletion in your shell by downloading the |  | ||||||
| relevant autocompletion files and laying out what additional information must |  | ||||||
| be loaded. |  | ||||||
|  |  | ||||||
| Example: |  | ||||||
|  |  | ||||||
|     abra autocomplete bash |  | ||||||
|  |  | ||||||
| Supported shells are as follows: |  | ||||||
|  |  | ||||||
| 		fish |  | ||||||
| 		zsh |  | ||||||
|     bash |  | ||||||
| `, |  | ||||||
| 	ArgsUsage: "<shell>", |  | ||||||
| 	Action: func(c *cli.Context) error { |  | ||||||
| 		shellType := c.Args().First() |  | ||||||
|  |  | ||||||
| 		if shellType == "" { |  | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("no shell provided")) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		supportedShells := map[string]bool{ |  | ||||||
| 			"bash": true, |  | ||||||
| 			"zsh":  true, |  | ||||||
| 			"fish": true, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if _, ok := supportedShells[shellType]; !ok { |  | ||||||
| 			logrus.Fatalf("%s is not a supported shell right now, sorry", shellType) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if shellType == "fish" { |  | ||||||
| 			shellType = "zsh" // handled the same on the autocompletion side |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		autocompletionDir := path.Join(config.ABRA_DIR, "autocompletion") |  | ||||||
| 		if err := os.Mkdir(autocompletionDir, 0755); err != nil { |  | ||||||
| 			if !os.IsExist(err) { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			logrus.Debugf("'%s' already created, moving on...", autocompletionDir) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		autocompletionFile := path.Join(config.ABRA_DIR, "autocompletion", shellType) |  | ||||||
| 		if _, err := os.Stat(autocompletionFile); err != nil && os.IsNotExist(err) { |  | ||||||
| 			url := fmt.Sprintf("https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/autocomplete/%s", shellType) |  | ||||||
| 			logrus.Infof("fetching %s", url) |  | ||||||
| 			if err := downloadFile(autocompletionFile, url); err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		switch shellType { |  | ||||||
| 		case "bash": |  | ||||||
| 			fmt.Println(fmt.Sprintf(` |  | ||||||
| # Run the following commands to install autocompletion |  | ||||||
| sudo mkdir /etc/bash/completion.d/ |  | ||||||
| sudo cp %s /etc/bash_completion.d/abra |  | ||||||
| echo "source /etc/bash/completion.d/abra" >> ~/.bashrc |  | ||||||
| `, autocompletionFile)) |  | ||||||
| 		case "zsh": |  | ||||||
| 			fmt.Println(fmt.Sprintf(` |  | ||||||
| # Run the following commands to install autocompletion |  | ||||||
| sudo mkdir /etc/zsh/completion.d/ |  | ||||||
| sudo cp %s /etc/zsh/completion.d/abra |  | ||||||
| echo "PROG=abra\n_CLI_ZSH_AUTOCOMPLETE_HACK=1\nsource /etc/zsh/completion.d/abra" >> ~/.zshrc |  | ||||||
| `, autocompletionFile)) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		return nil |  | ||||||
| 	}, |  | ||||||
| } |  | ||||||
| @ -1,13 +1,260 @@ | |||||||
| package catalogue | package catalogue | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"path" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	gitPkg "coopcloud.tech/abra/pkg/git" | ||||||
|  | 	"coopcloud.tech/abra/pkg/limit" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | // CatalogueSkipList is all the repos that are not recipes. | ||||||
|  | var CatalogueSkipList = map[string]bool{ | ||||||
|  | 	"abra":                  true, | ||||||
|  | 	"abra-apps":             true, | ||||||
|  | 	"abra-aur":              true, | ||||||
|  | 	"abra-bash":             true, | ||||||
|  | 	"abra-capsul":           true, | ||||||
|  | 	"abra-gandi":            true, | ||||||
|  | 	"abra-hetzner":          true, | ||||||
|  | 	"apps":                  true, | ||||||
|  | 	"aur-abra-git":          true, | ||||||
|  | 	"auto-apps-json":        true, | ||||||
|  | 	"auto-mirror":           true, | ||||||
|  | 	"backup-bot":            true, | ||||||
|  | 	"backup-bot-two":        true, | ||||||
|  | 	"comrade-renovate-bot":  true, | ||||||
|  | 	"coopcloud.tech":        true, | ||||||
|  | 	"coturn":                true, | ||||||
|  | 	"docker-cp-deploy":      true, | ||||||
|  | 	"docker-dind-bats-kcov": true, | ||||||
|  | 	"docs.coopcloud.tech":   true, | ||||||
|  | 	"drone-abra":            true, | ||||||
|  | 	"example":               true, | ||||||
|  | 	"gardening":             true, | ||||||
|  | 	"go-abra":               true, | ||||||
|  | 	"organising":            true, | ||||||
|  | 	"outline-with-patch":    true, | ||||||
|  | 	"pyabra":                true, | ||||||
|  | 	"radicle-seed-node":     true, | ||||||
|  | 	"recipes":               true, | ||||||
|  | 	"stack-ssh-deploy":      true, | ||||||
|  | 	"swarm-cronjob":         true, | ||||||
|  | 	"tagcmp":                true, | ||||||
|  | 	"traefik-cert-dumper":   true, | ||||||
|  | 	"tyop":                  true, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var catalogueGenerateCommand = &cli.Command{ | ||||||
|  | 	Name:    "generate", | ||||||
|  | 	Aliases: []string{"g"}, | ||||||
|  | 	Usage:   "Generate the recipe catalogue", | ||||||
|  | 	Flags: []cli.Flag{ | ||||||
|  | 		internal.PublishFlag, | ||||||
|  | 		internal.DryFlag, | ||||||
|  | 		internal.SkipUpdatesFlag, | ||||||
|  | 		internal.RegistryUsernameFlag, | ||||||
|  | 		internal.RegistryPasswordFlag, | ||||||
|  | 	}, | ||||||
|  | 	Description: ` | ||||||
|  | This command generates a new copy of the recipe catalogue which can be found on: | ||||||
|  |  | ||||||
|  |     https://recipes.coopcloud.tech | ||||||
|  |  | ||||||
|  | It polls the entire git.coopcloud.tech/coop-cloud/... recipe repository | ||||||
|  | listing, parses README.md and git tags of those repositories to produce recipe | ||||||
|  | metadata and produces a recipes JSON file. | ||||||
|  |  | ||||||
|  | It is possible to generate new metadata for a single recipe by passing | ||||||
|  | <recipe>. The existing local catalogue will be updated, not overwritten. | ||||||
|  |  | ||||||
|  | It is quite easy to get rate limited by Docker Hub when running this command. | ||||||
|  | If you have a Hub account you can have Abra log you in to avoid this. Pass | ||||||
|  | "--user" and "--pass". | ||||||
|  |  | ||||||
|  | Push your new release git.coopcloud.tech with "-p/--publish". This requires | ||||||
|  | that you have permission to git push to these repositories and have your SSH | ||||||
|  | keys configured on your account. | ||||||
|  | `, | ||||||
|  | 	ArgsUsage: "[<recipe>]", | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		recipeName := c.Args().First() | ||||||
|  | 		if recipeName != "" { | ||||||
|  | 			internal.ValidateRecipe(c) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		catalogueDir := path.Join(config.ABRA_DIR, "catalogue") | ||||||
|  | 		url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, "recipes") | ||||||
|  | 		if err := gitPkg.Clone(catalogueDir, url); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		repos, err := recipe.ReadReposMetadata() | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		var barLength int | ||||||
|  | 		var logMsg string | ||||||
|  | 		if recipeName != "" { | ||||||
|  | 			barLength = 1 | ||||||
|  | 			logMsg = fmt.Sprintf("ensuring %v recipe is cloned & up-to-date", barLength) | ||||||
|  | 		} else { | ||||||
|  | 			barLength = len(repos) | ||||||
|  | 			logMsg = fmt.Sprintf("ensuring %v recipes are cloned & up-to-date, this could take some time...", barLength) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.SkipUpdates { | ||||||
|  | 			logrus.Warn(logMsg) | ||||||
|  | 			if err := updateRepositories(repos, recipeName); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		catl := make(recipe.RecipeCatalogue) | ||||||
|  | 		catlBar := formatter.CreateProgressbar(barLength, "generating catalogue metadata...") | ||||||
|  | 		for _, recipeMeta := range repos { | ||||||
|  | 			if recipeName != "" && recipeName != recipeMeta.Name { | ||||||
|  | 				catlBar.Add(1) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if _, exists := CatalogueSkipList[recipeMeta.Name]; exists { | ||||||
|  | 				catlBar.Add(1) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			versions, err := recipe.GetRecipeVersions( | ||||||
|  | 				recipeMeta.Name, | ||||||
|  | 				internal.RegistryUsername, | ||||||
|  | 				internal.RegistryPassword, | ||||||
|  | 			) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			features, category, err := recipe.GetRecipeFeaturesAndCategory(recipeMeta.Name) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Warn(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			catl[recipeMeta.Name] = recipe.RecipeMeta{ | ||||||
|  | 				Name:          recipeMeta.Name, | ||||||
|  | 				Repository:    recipeMeta.CloneURL, | ||||||
|  | 				SSHURL:        recipeMeta.SSHURL, | ||||||
|  | 				Icon:          recipeMeta.AvatarURL, | ||||||
|  | 				DefaultBranch: recipeMeta.DefaultBranch, | ||||||
|  | 				Description:   recipeMeta.Description, | ||||||
|  | 				Website:       recipeMeta.Website, | ||||||
|  | 				Versions:      versions, | ||||||
|  | 				Category:      category, | ||||||
|  | 				Features:      features, | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			catlBar.Add(1) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		recipesJSON, err := json.MarshalIndent(catl, "", "    ") | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if recipeName == "" { | ||||||
|  | 			if err := ioutil.WriteFile(config.RECIPES_JSON, recipesJSON, 0764); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			catlFS, err := recipe.ReadRecipeCatalogue() | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			catlFS[recipeName] = catl[recipeName] | ||||||
|  |  | ||||||
|  | 			updatedRecipesJSON, err := json.MarshalIndent(catlFS, "", "    ") | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if err := ioutil.WriteFile(config.RECIPES_JSON, updatedRecipesJSON, 0764); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Infof("generated new recipe catalogue in %s", config.RECIPES_JSON) | ||||||
|  |  | ||||||
|  | 		cataloguePath := path.Join(config.ABRA_DIR, "catalogue") | ||||||
|  | 		if internal.Publish { | ||||||
|  |  | ||||||
|  | 			isClean, err := gitPkg.IsClean(cataloguePath) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if isClean { | ||||||
|  | 				if !internal.Dry { | ||||||
|  | 					logrus.Fatalf("no changes discovered in %s, nothing to publish?", cataloguePath) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			msg := "chore: publish new catalogue release changes" | ||||||
|  | 			if err := gitPkg.Commit(cataloguePath, "**.json", msg, internal.Dry); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			repo, err := git.PlainOpen(cataloguePath) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			sshURL := fmt.Sprintf(config.SSH_URL_TEMPLATE, "recipes") | ||||||
|  | 			if err := gitPkg.CreateRemote(repo, "origin-ssh", sshURL, internal.Dry); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if err := gitPkg.Push(cataloguePath, "origin-ssh", false, internal.Dry); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		repo, err := git.PlainOpen(cataloguePath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		head, err := repo.Head() | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.Dry && internal.Publish { | ||||||
|  | 			url := fmt.Sprintf("%s/recipes/commit/%s", config.REPOS_BASE_URL, head.Hash()) | ||||||
|  | 			logrus.Infof("new changes published: %s", url) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if internal.Dry { | ||||||
|  | 			logrus.Info("dry run: no changes published") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
|  | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
|  | } | ||||||
|  |  | ||||||
| // CatalogueCommand defines the `abra catalogue` command and sub-commands. | // CatalogueCommand defines the `abra catalogue` command and sub-commands. | ||||||
| var CatalogueCommand = &cli.Command{ | var CatalogueCommand = &cli.Command{ | ||||||
| 	Name:        "catalogue", | 	Name:        "catalogue", | ||||||
| 	Usage:       "Manage the recipe catalogue (for maintainers)", | 	Usage:       "Manage the recipe catalogue", | ||||||
| 	Aliases:     []string{"c"}, | 	Aliases:     []string{"c"}, | ||||||
| 	ArgsUsage:   "<recipe>", | 	ArgsUsage:   "<recipe>", | ||||||
| 	Description: "This command helps recipe packagers interact with the recipe catalogue", | 	Description: "This command helps recipe packagers interact with the recipe catalogue", | ||||||
| @ -15,3 +262,62 @@ var CatalogueCommand = &cli.Command{ | |||||||
| 		catalogueGenerateCommand, | 		catalogueGenerateCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func updateRepositories(repos recipe.RepoCatalogue, recipeName string) error { | ||||||
|  | 	var barLength int | ||||||
|  | 	if recipeName != "" { | ||||||
|  | 		barLength = 1 | ||||||
|  | 	} else { | ||||||
|  | 		barLength = len(repos) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cloneLimiter := limit.New(10) | ||||||
|  |  | ||||||
|  | 	retrieveBar := formatter.CreateProgressbar(barLength, "ensuring recipes are cloned & up-to-date...") | ||||||
|  | 	ch := make(chan string, barLength) | ||||||
|  | 	for _, repoMeta := range repos { | ||||||
|  | 		go func(rm recipe.RepoMeta) { | ||||||
|  | 			cloneLimiter.Begin() | ||||||
|  | 			defer cloneLimiter.End() | ||||||
|  |  | ||||||
|  | 			if recipeName != "" && recipeName != rm.Name { | ||||||
|  | 				ch <- rm.Name | ||||||
|  | 				retrieveBar.Add(1) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			if _, exists := CatalogueSkipList[rm.Name]; exists { | ||||||
|  | 				ch <- rm.Name | ||||||
|  | 				retrieveBar.Add(1) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			recipeDir := path.Join(config.RECIPES_DIR, rm.Name) | ||||||
|  |  | ||||||
|  | 			if err := gitPkg.Clone(recipeDir, rm.CloneURL); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			isClean, err := gitPkg.IsClean(recipeDir) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if !isClean { | ||||||
|  | 				logrus.Fatalf("%s has locally unstaged changes", rm.Name) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if err := recipe.EnsureUpToDate(rm.Name); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			ch <- rm.Name | ||||||
|  | 			retrieveBar.Add(1) | ||||||
|  | 		}(repoMeta) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for range repos { | ||||||
|  | 		<-ch // wait for everything | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -1,262 +0,0 @@ | |||||||
| package catalogue |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io/ioutil" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" |  | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	gitPkg "coopcloud.tech/abra/pkg/git" |  | ||||||
| 	"coopcloud.tech/abra/pkg/limit" |  | ||||||
| 	"github.com/AlecAivazis/survey/v2" |  | ||||||
| 	"github.com/go-git/go-git/v5" |  | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // CatalogueSkipList is all the repos that are not recipes. |  | ||||||
| var CatalogueSkipList = map[string]bool{ |  | ||||||
| 	"abra":                  true, |  | ||||||
| 	"abra-apps":             true, |  | ||||||
| 	"abra-aur":              true, |  | ||||||
| 	"abra-bash":             true, |  | ||||||
| 	"abra-capsul":           true, |  | ||||||
| 	"abra-gandi":            true, |  | ||||||
| 	"abra-hetzner":          true, |  | ||||||
| 	"apps":                  true, |  | ||||||
| 	"aur-abra-git":          true, |  | ||||||
| 	"auto-apps-json":        true, |  | ||||||
| 	"auto-mirror":           true, |  | ||||||
| 	"backup-bot":            true, |  | ||||||
| 	"backup-bot-two":        true, |  | ||||||
| 	"comrade-renovate-bot":  true, |  | ||||||
| 	"coopcloud.tech":        true, |  | ||||||
| 	"coturn":                true, |  | ||||||
| 	"docker-cp-deploy":      true, |  | ||||||
| 	"docker-dind-bats-kcov": true, |  | ||||||
| 	"docs.coopcloud.tech":   true, |  | ||||||
| 	"example":               true, |  | ||||||
| 	"gardening":             true, |  | ||||||
| 	"go-abra":               true, |  | ||||||
| 	"organising":            true, |  | ||||||
| 	"pyabra":                true, |  | ||||||
| 	"radicle-seed-node":     true, |  | ||||||
| 	"stack-ssh-deploy":      true, |  | ||||||
| 	"swarm-cronjob":         true, |  | ||||||
| 	"tagcmp":                true, |  | ||||||
| 	"tyop":                  true, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var commit bool |  | ||||||
| var commitFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "commit", |  | ||||||
| 	Usage:       "Commits new generated catalogue changes", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"c"}, |  | ||||||
| 	Destination: &commit, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var catalogueGenerateCommand = &cli.Command{ |  | ||||||
| 	Name:    "generate", |  | ||||||
| 	Aliases: []string{"g"}, |  | ||||||
| 	Usage:   "Generate a new copy of the catalogue", |  | ||||||
| 	Flags: []cli.Flag{ |  | ||||||
| 		internal.PushFlag, |  | ||||||
| 		commitFlag, |  | ||||||
| 		internal.CommitMessageFlag, |  | ||||||
| 	}, |  | ||||||
| 	Description: ` |  | ||||||
| This command generates a new copy of the recipe catalogue which can be found on: |  | ||||||
|  |  | ||||||
|     https://recipes.coopcloud.tech |  | ||||||
|  |  | ||||||
| It polls the entire git.coopcloud.tech/coop-cloud/... recipe repository |  | ||||||
| listing, parses README and tags to produce recipe metadata and produces a |  | ||||||
| apps.json file which is placed in your ~/.abra/catalogue/recipes.json. |  | ||||||
|  |  | ||||||
| It is possible to generate new metadata for a single recipe by passing |  | ||||||
| <recipe>. The existing local catalogue will be updated, not overwritten. |  | ||||||
|  |  | ||||||
| A new catalogue copy can be published to the recipes repository by passing the |  | ||||||
| "--commit" and "--push" flags. The recipes repository is available here: |  | ||||||
|  |  | ||||||
|     https://git.coopcloud.tech/coop-cloud/recipes |  | ||||||
|  |  | ||||||
| `, |  | ||||||
| 	ArgsUsage: "[<recipe>]", |  | ||||||
| 	Action: func(c *cli.Context) error { |  | ||||||
| 		recipeName := c.Args().First() |  | ||||||
|  |  | ||||||
| 		catalogueDir := path.Join(config.ABRA_DIR, "catalogue") |  | ||||||
| 		url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, "recipes") |  | ||||||
| 		if err := gitPkg.Clone(catalogueDir, url); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		repos, err := catalogue.ReadReposMetadata() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("ensuring '%v' recipe(s) are locally present and up-to-date", len(repos)) |  | ||||||
|  |  | ||||||
| 		cloneLimiter := limit.New(10) |  | ||||||
| 		retrieveBar := formatter.CreateProgressbar(len(repos), "retrieving recipes...") |  | ||||||
| 		ch := make(chan string, len(repos)) |  | ||||||
| 		for _, repoMeta := range repos { |  | ||||||
| 			go func(rm catalogue.RepoMeta) { |  | ||||||
| 				cloneLimiter.Begin() |  | ||||||
| 				defer cloneLimiter.End() |  | ||||||
|  |  | ||||||
| 				if recipeName != "" && recipeName != rm.Name { |  | ||||||
| 					ch <- rm.Name |  | ||||||
| 					retrieveBar.Add(1) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				if _, exists := CatalogueSkipList[rm.Name]; exists { |  | ||||||
| 					ch <- rm.Name |  | ||||||
| 					retrieveBar.Add(1) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				recipeDir := path.Join(config.ABRA_DIR, "apps", rm.Name) |  | ||||||
|  |  | ||||||
| 				if err := gitPkg.Clone(recipeDir, rm.SSHURL); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				if err := gitPkg.EnsureUpToDate(recipeDir); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				ch <- rm.Name |  | ||||||
| 				retrieveBar.Add(1) |  | ||||||
| 			}(repoMeta) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		for range repos { |  | ||||||
| 			<-ch // wait for everything |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		catl := make(catalogue.RecipeCatalogue) |  | ||||||
| 		catlBar := formatter.CreateProgressbar(len(repos), "generating catalogue...") |  | ||||||
| 		for _, recipeMeta := range repos { |  | ||||||
| 			if recipeName != "" && recipeName != recipeMeta.Name { |  | ||||||
| 				catlBar.Add(1) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if _, exists := CatalogueSkipList[recipeMeta.Name]; exists { |  | ||||||
| 				catlBar.Add(1) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			versions, err := catalogue.GetRecipeVersions(recipeMeta.Name) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			catl[recipeMeta.Name] = catalogue.RecipeMeta{ |  | ||||||
| 				Name:          recipeMeta.Name, |  | ||||||
| 				Repository:    recipeMeta.CloneURL, |  | ||||||
| 				Icon:          recipeMeta.AvatarURL, |  | ||||||
| 				DefaultBranch: recipeMeta.DefaultBranch, |  | ||||||
| 				Description:   recipeMeta.Description, |  | ||||||
| 				Website:       recipeMeta.Website, |  | ||||||
| 				Versions:      versions, |  | ||||||
| 				// Category:      ..., // FIXME: parse & load |  | ||||||
| 				// Features:      ..., // FIXME: parse & load |  | ||||||
| 			} |  | ||||||
| 			catlBar.Add(1) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		recipesJSON, err := json.MarshalIndent(catl, "", "    ") |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if _, err := os.Stat(config.APPS_JSON); err != nil && os.IsNotExist(err) { |  | ||||||
| 			if err := ioutil.WriteFile(config.APPS_JSON, recipesJSON, 0644); err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if recipeName != "" { |  | ||||||
| 				catlFS, err := catalogue.ReadRecipeCatalogue() |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				catlFS[recipeName] = catl[recipeName] |  | ||||||
|  |  | ||||||
| 				updatedRecipesJSON, err := json.MarshalIndent(catlFS, "", "    ") |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				if err := ioutil.WriteFile(config.APPS_JSON, updatedRecipesJSON, 0644); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		cataloguePath := path.Join(config.ABRA_DIR, "catalogue", "recipes.json") |  | ||||||
| 		logrus.Infof("generated new recipe catalogue in %s", cataloguePath) |  | ||||||
|  |  | ||||||
| 		if commit { |  | ||||||
| 			repoPath := path.Join(config.ABRA_DIR, "catalogue") |  | ||||||
| 			commitRepo, err := git.PlainOpen(repoPath) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			commitWorktree, err := commitRepo.Worktree() |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if internal.CommitMessage == "" { |  | ||||||
| 				prompt := &survey.Input{ |  | ||||||
| 					Message: "commit message", |  | ||||||
| 					Default: "chore: publish new catalogue changes", |  | ||||||
| 				} |  | ||||||
| 				if err := survey.AskOne(prompt, &internal.CommitMessage); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			err = commitWorktree.AddGlob("**.json") |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			logrus.Debug("staged **.json for commit") |  | ||||||
|  |  | ||||||
| 			_, err = commitWorktree.Commit(internal.CommitMessage, &git.CommitOptions{}) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			logrus.Info("changes commited") |  | ||||||
|  |  | ||||||
| 			if err := commitRepo.Push(&git.PushOptions{}); err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			logrus.Info("changes pushed") |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		return nil |  | ||||||
| 	}, |  | ||||||
| 	BashComplete: func(c *cli.Context) { |  | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for name := range catl { |  | ||||||
| 			fmt.Println(name) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } |  | ||||||
							
								
								
									
										146
									
								
								cli/cli.go
									
									
									
									
									
								
							
							
						
						
									
										146
									
								
								cli/cli.go
									
									
									
									
									
								
							| @ -2,8 +2,10 @@ | |||||||
| package cli | package cli | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"os/exec" | ||||||
| 	"path" | 	"path" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/app" | 	"coopcloud.tech/abra/cli/app" | ||||||
| @ -13,35 +15,140 @@ import ( | |||||||
| 	"coopcloud.tech/abra/cli/record" | 	"coopcloud.tech/abra/cli/record" | ||||||
| 	"coopcloud.tech/abra/cli/server" | 	"coopcloud.tech/abra/cli/server" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/web" | ||||||
| 	logrusStack "github.com/Gurpartap/logrus-stack" | 	logrusStack "github.com/Gurpartap/logrus-stack" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Verbose stores the variable from VerboseFlag. | // AutoCompleteCommand helps people set up auto-complete in their shells | ||||||
| var Verbose bool | var AutoCompleteCommand = &cli.Command{ | ||||||
|  | 	Name:    "autocomplete", | ||||||
|  | 	Usage:   "Configure shell autocompletion (recommended)", | ||||||
|  | 	Aliases: []string{"ac"}, | ||||||
|  | 	Description: ` | ||||||
|  | This command helps set up autocompletion in your shell by downloading the | ||||||
|  | relevant autocompletion files and laying out what additional information must | ||||||
|  | be loaded. | ||||||
|  |  | ||||||
| // VerboseFlag turns on/off verbose logging down to the INFO level. | Example: | ||||||
| var VerboseFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "verbose", |     abra autocomplete bash | ||||||
| 	Aliases:     []string{"V"}, |  | ||||||
| 	Value:       false, | Supported shells are as follows: | ||||||
| 	Destination: &Verbose, |  | ||||||
| 	Usage:       "Show INFO messages", | 		fizsh | ||||||
|  | 		zsh | ||||||
|  |     bash | ||||||
|  |  | ||||||
|  | `, | ||||||
|  | 	ArgsUsage: "<shell>", | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		shellType := c.Args().First() | ||||||
|  |  | ||||||
|  | 		if shellType == "" { | ||||||
|  | 			internal.ShowSubcommandHelpAndError(c, errors.New("no shell provided")) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		supportedShells := map[string]bool{ | ||||||
|  | 			"bash":  true, | ||||||
|  | 			"zsh":   true, | ||||||
|  | 			"fizsh": true, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if _, ok := supportedShells[shellType]; !ok { | ||||||
|  | 			logrus.Fatalf("%s is not a supported shell right now, sorry", shellType) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if shellType == "fizsh" { | ||||||
|  | 			shellType = "zsh" // handled the same on the autocompletion side | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		autocompletionDir := path.Join(config.ABRA_DIR, "autocompletion") | ||||||
|  | 		if err := os.Mkdir(autocompletionDir, 0764); err != nil { | ||||||
|  | 			if !os.IsExist(err) { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			logrus.Debugf("%s already created", autocompletionDir) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		autocompletionFile := path.Join(config.ABRA_DIR, "autocompletion", shellType) | ||||||
|  | 		if _, err := os.Stat(autocompletionFile); err != nil && os.IsNotExist(err) { | ||||||
|  | 			url := fmt.Sprintf("https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/autocomplete/%s", shellType) | ||||||
|  | 			logrus.Infof("fetching %s", url) | ||||||
|  | 			if err := web.GetFile(autocompletionFile, url); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		switch shellType { | ||||||
|  | 		case "bash": | ||||||
|  | 			fmt.Println(fmt.Sprintf(` | ||||||
|  | # Run the following commands to install autocompletion | ||||||
|  | sudo mkdir /etc/bash_completion.d/ | ||||||
|  | sudo cp %s /etc/bash_completion.d/abra | ||||||
|  | echo "source /etc/bash_completion.d/abra" >> ~/.bashrc | ||||||
|  | # And finally run "abra app ps <hit tab key>" to test things are working, you should see app names listed! | ||||||
|  | `, autocompletionFile)) | ||||||
|  | 		case "zsh": | ||||||
|  | 			fmt.Println(fmt.Sprintf(` | ||||||
|  | # Run the following commands to install autocompletion | ||||||
|  | sudo mkdir /etc/zsh/completion.d/ | ||||||
|  | sudo cp %s /etc/zsh/completion.d/abra | ||||||
|  | echo "PROG=abra\n_CLI_ZSH_AUTOCOMPLETE_HACK=1\nsource /etc/zsh/completion.d/abra" >> ~/.zshrc | ||||||
|  | # And finally run "abra app ps <hit tab key>" to test things are working, you should see app names listed! | ||||||
|  | `, autocompletionFile)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // UpgradeCommand upgrades abra in-place. | ||||||
|  | var UpgradeCommand = &cli.Command{ | ||||||
|  | 	Name:    "upgrade", | ||||||
|  | 	Usage:   "Upgrade Abra itself", | ||||||
|  | 	Aliases: []string{"u"}, | ||||||
|  | 	Description: ` | ||||||
|  | This command allows you to upgrade Abra in-place with the latest stable or | ||||||
|  | release candidate. | ||||||
|  |  | ||||||
|  | If you would like to install the latest release candidate, please pass the | ||||||
|  | "--rc" option. Please bear in mind that the latest release candidate may have | ||||||
|  | some catastrophic bugs contained in it. In any case, thank you very much for | ||||||
|  | the testing efforts! | ||||||
|  | `, | ||||||
|  | 	Flags: []cli.Flag{internal.RCFlag}, | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		mainURL := "https://install.abra.coopcloud.tech" | ||||||
|  | 		cmd := exec.Command("bash", "-c", fmt.Sprintf("wget -q -O- %s | bash", mainURL)) | ||||||
|  |  | ||||||
|  | 		if internal.RC { | ||||||
|  | 			releaseCandidateURL := "https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/installer/installer" | ||||||
|  | 			cmd = exec.Command("bash", "-c", fmt.Sprintf("wget -q -O- %s | bash -s -- --rc", releaseCandidateURL)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Debugf("attempting to run %s", cmd) | ||||||
|  |  | ||||||
|  | 		if err := internal.RunCmd(cmd); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func newAbraApp(version, commit string) *cli.App { | func newAbraApp(version, commit string) *cli.App { | ||||||
| 	app := &cli.App{ | 	app := &cli.App{ | ||||||
| 		Name: "abra", | 		Name: "abra", | ||||||
| 		Usage: `The Co-op Cloud command-line utility belt 🎩🐇 | 		Usage: `The Co-op Cloud command-line utility belt 🎩🐇 | ||||||
|  |  | ||||||
|     ____                           ____ _                 _ |     ____                           ____ _                 _ | ||||||
|    / ___|___         ___  _ __    / ___| | ___  _   _  __| | |    / ___|___         ___  _ __    / ___| | ___  _   _  __| | | ||||||
|   | |   / _ \ _____ / _ \| '_ \  | |   | |/ _ \| | | |/ _' | |   | |   / _ \ _____ / _ \| '_ \  | |   | |/ _ \| | | |/ _' | | ||||||
|   | |__| (_) |_____| (_) | |_) | | |___| | (_) | |_| | (_| | |   | |__| (_) |_____| (_) | |_) | | |___| | (_) | |_| | (_| | | ||||||
|    \____\___/       \___/| .__/   \____|_|\___/ \__,_|\__,_| |    \____\___/       \___/| .__/   \____|_|\___/ \__,_|\__,_| | ||||||
|                          |_| |                          |_| | ||||||
| 		`, | `, | ||||||
| 		Version: fmt.Sprintf("%s-%s", version, commit[:7]), | 		Version: fmt.Sprintf("%s-%s", version, commit[:7]), | ||||||
| 		Commands: []*cli.Command{ | 		Commands: []*cli.Command{ | ||||||
| 			app.AppCommand, | 			app.AppCommand, | ||||||
| @ -53,13 +160,16 @@ func newAbraApp(version, commit string) *cli.App { | |||||||
| 			AutoCompleteCommand, | 			AutoCompleteCommand, | ||||||
| 		}, | 		}, | ||||||
| 		Flags: []cli.Flag{ | 		Flags: []cli.Flag{ | ||||||
| 			VerboseFlag, |  | ||||||
| 			internal.DebugFlag, | 			internal.DebugFlag, | ||||||
| 			internal.NoInputFlag, | 			internal.NoInputFlag, | ||||||
| 		}, | 		}, | ||||||
| 		Authors: []*cli.Author{ | 		Authors: []*cli.Author{ | ||||||
|  | 			// If you're looking at this and you hack on Abra and you're not listed | ||||||
|  | 			// here, please do add yourself! This is a community project, let's show | ||||||
|  | 			// some love | ||||||
| 			{Name: "3wordchant"}, | 			{Name: "3wordchant"}, | ||||||
| 			{Name: "decentral1se"}, | 			{Name: "decentral1se"}, | ||||||
|  | 			{Name: "kawaiipunk"}, | ||||||
| 			{Name: "knoflook"}, | 			{Name: "knoflook"}, | ||||||
| 			{Name: "roxxers"}, | 			{Name: "roxxers"}, | ||||||
| 		}, | 		}, | ||||||
| @ -77,23 +187,21 @@ func newAbraApp(version, commit string) *cli.App { | |||||||
|  |  | ||||||
| 		paths := []string{ | 		paths := []string{ | ||||||
| 			config.ABRA_DIR, | 			config.ABRA_DIR, | ||||||
| 			path.Join(config.ABRA_DIR, "servers"), | 			path.Join(config.SERVERS_DIR), | ||||||
| 			path.Join(config.ABRA_DIR, "apps"), | 			path.Join(config.RECIPES_DIR), | ||||||
| 			path.Join(config.ABRA_DIR, "vendor"), | 			path.Join(config.VENDOR_DIR), | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		for _, path := range paths { | 		for _, path := range paths { | ||||||
| 			if err := os.Mkdir(path, 0755); err != nil { | 			if err := os.Mkdir(path, 0764); err != nil { | ||||||
| 				if !os.IsExist(err) { | 				if !os.IsExist(err) { | ||||||
| 					logrus.Fatal(err) | 					logrus.Fatal(err) | ||||||
| 				} | 				} | ||||||
| 				logrus.Debugf("'%s' already created, moving on...", path) |  | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 			logrus.Debugf("'%s' is missing, creating...", path) |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("abra version '%s', commit '%s'", version, commit) | 		logrus.Debugf("abra version %s, commit %s", version, commit) | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -1,74 +0,0 @@ | |||||||
| package internal |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"os" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/pkg/client" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	"github.com/docker/docker/api/types" |  | ||||||
| 	"github.com/docker/docker/api/types/filters" |  | ||||||
| 	"github.com/docker/docker/pkg/archive" |  | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func ConfigureAndCp(c *cli.Context, app config.App, srcPath string, dstPath string, service string, isToContainer bool) error { |  | ||||||
| 	appFiles, err := config.LoadAppFiles("") |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	appEnv, err := config.GetApp(appFiles, app.Name) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	cl, err := client.New(app.Server) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	filters := filters.NewArgs() |  | ||||||
| 	filters.Add("name", fmt.Sprintf("%s_%s", appEnv.StackName(), service)) |  | ||||||
| 	containers, err := cl.ContainerList(c.Context, types.ContainerListOptions{Filters: filters}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if len(containers) != 1 { |  | ||||||
| 		logrus.Fatalf("expected 1 container but got %v", len(containers)) |  | ||||||
| 	} |  | ||||||
| 	container := containers[0] |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved '%s' as target container on '%s'", formatter.ShortenID(container.ID), app.Server) |  | ||||||
|  |  | ||||||
| 	if isToContainer { |  | ||||||
| 		if _, err := os.Stat(srcPath); err != nil { |  | ||||||
| 			logrus.Fatalf("'%s' does not exist?", srcPath) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		toTarOpts := &archive.TarOptions{NoOverwriteDirNonDir: true, Compression: archive.Gzip} |  | ||||||
| 		content, err := archive.TarWithOptions(srcPath, toTarOpts) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		copyOpts := types.CopyToContainerOptions{AllowOverwriteDirWithFile: false, CopyUIDGID: false} |  | ||||||
| 		if err := cl.CopyToContainer(c.Context, container.ID, dstPath, content, copyOpts); err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		content, _, err := cl.CopyFromContainer(c.Context, container.ID, srcPath) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		defer content.Close() |  | ||||||
| 		fromTarOpts := &archive.TarOptions{NoOverwriteDirNonDir: true, Compression: archive.Gzip} |  | ||||||
| 		if err := archive.Untar(content, dstPath, fromTarOpts); err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| @ -2,13 +2,17 @@ package internal | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" |  | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/dns" | 	"coopcloud.tech/abra/pkg/dns" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"coopcloud.tech/abra/pkg/git" | ||||||
|  | 	"coopcloud.tech/abra/pkg/lint" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" | 	"coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| @ -19,44 +23,62 @@ import ( | |||||||
| // DeployAction is the main command-line action for this package | // DeployAction is the main command-line action for this package | ||||||
| func DeployAction(c *cli.Context) error { | func DeployAction(c *cli.Context) error { | ||||||
| 	app := ValidateApp(c) | 	app := ValidateApp(c) | ||||||
| 	stackName := app.StackName() |  | ||||||
|  | 	if err := recipe.EnsureUpToDate(app.Type); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	r, err := recipe.Get(app.Type) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := lint.LintForErrors(r); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	cl, err := client.New(app.Server) | 	cl, err := client.New(app.Server) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("checking whether '%s' is already deployed", stackName) | 	logrus.Debugf("checking whether %s is already deployed", app.StackName()) | ||||||
|  |  | ||||||
| 	isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, stackName) | 	isDeployed, deployedVersion, err := stack.IsDeployed(c.Context, cl, app.StackName()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if isDeployed { | 	if isDeployed { | ||||||
| 		if Force { | 		if Force || Chaos { | ||||||
| 			logrus.Warnf("'%s' already deployed but continuing (--force)", stackName) | 			logrus.Warnf("%s is already deployed but continuing (--force/--chaos)", app.Name) | ||||||
| 		} else if Chaos { |  | ||||||
| 			logrus.Warnf("'%s' already deployed but continuing (--chaos)", stackName) |  | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Fatalf("'%s' is already deployed", stackName) | 			logrus.Fatalf("%s is already deployed", app.Name) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	version := deployedVersion | 	version := deployedVersion | ||||||
| 	if version == "" && !Chaos { | 	if version == "" && !Chaos { | ||||||
| 		versions, err := catalogue.GetRecipeCatalogueVersions(app.Type) | 		catl, err := recipe.ReadRecipeCatalogue() | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 		versions, err := recipe.GetRecipeCatalogueVersions(app.Type, catl) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		if len(versions) > 0 { | 		if len(versions) > 0 { | ||||||
| 			version = versions[len(versions)-1] | 			version = versions[len(versions)-1] | ||||||
| 			logrus.Debugf("choosing '%s' as version to deploy", version) | 			logrus.Debugf("choosing %s as version to deploy", version) | ||||||
| 			if err := recipe.EnsureVersion(app.Type, version); err != nil { | 			if err := recipe.EnsureVersion(app.Type, version); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			version = "latest commit" | 			head, err := git.GetRecipeHead(app.Type) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			version = formatter.SmallSHA(head.String()) | ||||||
| 			logrus.Warn("no versions detected, using latest commit") | 			logrus.Warn("no versions detected, using latest commit") | ||||||
| 			if err := recipe.EnsureLatest(app.Type); err != nil { | 			if err := recipe.EnsureLatest(app.Type); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| @ -65,7 +87,13 @@ func DeployAction(c *cli.Context) error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if version == "" && !Chaos { | 	if version == "" && !Chaos { | ||||||
| 		logrus.Debugf("choosing '%s' as version to deploy", version) | 		logrus.Debugf("choosing %s as version to deploy", version) | ||||||
|  | 		if err := recipe.EnsureVersion(app.Type, version); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if version != "" && !Chaos { | ||||||
| 		if err := recipe.EnsureVersion(app.Type, version); err != nil { | 		if err := recipe.EnsureVersion(app.Type, version); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -80,7 +108,7 @@ func DeployAction(c *cli.Context) error { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	abraShPath := fmt.Sprintf("%s/%s/%s", config.APPS_DIR, app.Type, "abra.sh") | 	abraShPath := fmt.Sprintf("%s/%s/%s", config.RECIPES_DIR, app.Type, "abra.sh") | ||||||
| 	abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | 	abraShEnv, err := config.ReadAbraShEnvVars(abraShPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| @ -95,7 +123,7 @@ func DeployAction(c *cli.Context) error { | |||||||
| 	} | 	} | ||||||
| 	deployOpts := stack.Deploy{ | 	deployOpts := stack.Deploy{ | ||||||
| 		Composefiles: composeFiles, | 		Composefiles: composeFiles, | ||||||
| 		Namespace:    stackName, | 		Namespace:    app.StackName(), | ||||||
| 		Prune:        false, | 		Prune:        false, | ||||||
| 		ResolveImage: stack.ResolveImageAlways, | 		ResolveImage: stack.ResolveImageAlways, | ||||||
| 	} | 	} | ||||||
| @ -108,17 +136,21 @@ func DeployAction(c *cli.Context) error { | |||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	domainName := app.Env["DOMAIN"] | 	if !NoDomainChecks { | ||||||
| 	ipv4, err := dns.EnsureIPv4(domainName) | 		domainName := app.Env["DOMAIN"] | ||||||
| 	if err != nil || ipv4 == "" { | 		ipv4, err := dns.EnsureIPv4(domainName) | ||||||
| 		logrus.Fatalf("could not find an IP address assigned to %s?", domainName) | 		if err != nil || ipv4 == "" { | ||||||
|  | 			logrus.Fatalf("could not find an IP address assigned to %s?", domainName) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if _, err = dns.EnsureDomainsResolveSameIPv4(domainName, app.Server); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		logrus.Warn("skipping domain checks as requested") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if _, err = dns.EnsureDomainsResolveSameIPv4(domainName, app.Server); err != nil { | 	if err := stack.RunDeploy(cl, deployOpts, compose, app.Name, DontWaitConverge); err != nil { | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := stack.RunDeploy(cl, deployOpts, compose); err != nil { |  | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @ -127,8 +159,8 @@ func DeployAction(c *cli.Context) error { | |||||||
|  |  | ||||||
| // DeployOverview shows a deployment overview | // DeployOverview shows a deployment overview | ||||||
| func DeployOverview(app config.App, version, message string) error { | func DeployOverview(app config.App, version, message string) error { | ||||||
| 	tableCol := []string{"server", "compose", "domain", "stack", "version"} | 	tableCol := []string{"server", "compose", "domain", "app name", "version"} | ||||||
| 	table := abraFormatter.CreateTable(tableCol) | 	table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 	deployConfig := "compose.yml" | 	deployConfig := "compose.yml" | ||||||
| 	if composeFiles, ok := app.Env["COMPOSE_FILE"]; ok { | 	if composeFiles, ok := app.Env["COMPOSE_FILE"]; ok { | ||||||
| @ -140,7 +172,7 @@ func DeployOverview(app config.App, version, message string) error { | |||||||
| 		server = "local" | 		server = "local" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	table.Append([]string{server, deployConfig, app.Domain, app.StackName(), version}) | 	table.Append([]string{server, deployConfig, app.Domain, app.Name, version}) | ||||||
| 	table.Render() | 	table.Render() | ||||||
|  |  | ||||||
| 	if NoInput { | 	if NoInput { | ||||||
| @ -164,9 +196,9 @@ func DeployOverview(app config.App, version, message string) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // NewVersionOverview shows an upgrade or downgrade overview | // NewVersionOverview shows an upgrade or downgrade overview | ||||||
| func NewVersionOverview(app config.App, currentVersion, newVersion string) error { | func NewVersionOverview(app config.App, currentVersion, newVersion, releaseNotes string) error { | ||||||
| 	tableCol := []string{"server", "compose", "domain", "stack", "current version", "to be deployed"} | 	tableCol := []string{"server", "compose", "domain", "app name", "current version", "to be deployed"} | ||||||
| 	table := abraFormatter.CreateTable(tableCol) | 	table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 	deployConfig := "compose.yml" | 	deployConfig := "compose.yml" | ||||||
| 	if composeFiles, ok := app.Env["COMPOSE_FILE"]; ok { | 	if composeFiles, ok := app.Env["COMPOSE_FILE"]; ok { | ||||||
| @ -178,9 +210,24 @@ func NewVersionOverview(app config.App, currentVersion, newVersion string) error | |||||||
| 		server = "local" | 		server = "local" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	table.Append([]string{server, deployConfig, app.Domain, app.StackName(), currentVersion, newVersion}) | 	table.Append([]string{server, deployConfig, app.Domain, app.Name, currentVersion, newVersion}) | ||||||
| 	table.Render() | 	table.Render() | ||||||
|  |  | ||||||
|  | 	if releaseNotes == "" { | ||||||
|  | 		var err error | ||||||
|  | 		releaseNotes, err = GetReleaseNotes(app.Type, newVersion) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if releaseNotes != "" { | ||||||
|  | 		fmt.Println() | ||||||
|  | 		fmt.Println(fmt.Sprintf("%s release notes:\n\n%s", newVersion, releaseNotes)) | ||||||
|  | 	} else { | ||||||
|  | 		logrus.Warnf("no release notes available for %s", newVersion) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if NoInput { | 	if NoInput { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -200,3 +247,18 @@ func NewVersionOverview(app config.App, currentVersion, newVersion string) error | |||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetReleaseNotes prints release notes for a recipe version | ||||||
|  | func GetReleaseNotes(recipeName, version string) (string, error) { | ||||||
|  | 	fpath := path.Join(config.RECIPES_DIR, recipeName, "release", version) | ||||||
|  |  | ||||||
|  | 	if _, err := os.Stat(fpath); !os.IsNotExist(err) { | ||||||
|  | 		releaseNotes, err := ioutil.ReadFile(fpath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return "", err | ||||||
|  | 		} | ||||||
|  | 		return string(releaseNotes), nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return "", nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -1,38 +0,0 @@ | |||||||
| package internal |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Testing functions that call os.Exit |  | ||||||
| // https://stackoverflow.com/questions/26225513/how-to-test-os-exit-scenarios-in-go |  | ||||||
| // https://talks.golang.org/2014/testing.slide#23 |  | ||||||
|  |  | ||||||
| var testapp = &cli.App{ |  | ||||||
| 	Name:  "abra", |  | ||||||
| 	Usage: `The Co-op Cloud command-line utility belt 🎩🐇`, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // not testing output as that changes. just if it exits with code 1 |  | ||||||
| // does not work because of some weird errors on cli's part. Its a hard lib to test effectively. |  | ||||||
| // func TestShowSubcommandHelpAndError(t *testing.T) { |  | ||||||
| // 	if os.Getenv("HelpAndError") == "1" { |  | ||||||
| // 		ShowSubcommandHelpAndError(cli.NewContext(testapp, nil, nil), errors.New("Test error")) |  | ||||||
| // 		return |  | ||||||
| // 	} |  | ||||||
| // 	cmd := exec.Command(os.Args[0], "-test.run=TestShowSubcommandHelpAndError") |  | ||||||
| // 	cmd.Env = append(os.Environ(), "HelpAndError=1") |  | ||||||
| // 	var out bytes.Buffer |  | ||||||
| // 	cmd.Stderr = &out |  | ||||||
| // 	err := cmd.Run() |  | ||||||
| // 	println(out.String()) |  | ||||||
| // 	if !strings.Contains(out.String(), "Test error") { |  | ||||||
|  |  | ||||||
| // 		t.Fatalf("expected command to show the error causing the exit, did not get correct stdout output") |  | ||||||
| // 	} |  | ||||||
|  |  | ||||||
| // 	if e, ok := err.(*exec.ExitError); ok && !e.Success() { |  | ||||||
| // 		return |  | ||||||
| // 	} |  | ||||||
| // 	t.Fatalf("process ran with err %v, want exit status 1", err) |  | ||||||
| // } |  | ||||||
| @ -10,7 +10,7 @@ var Secrets bool | |||||||
| // SecretsFlag turns on/off automatically generating secrets | // SecretsFlag turns on/off automatically generating secrets | ||||||
| var SecretsFlag = &cli.BoolFlag{ | var SecretsFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "secrets", | 	Name:        "secrets", | ||||||
| 	Aliases:     []string{"S"}, | 	Aliases:     []string{"ss"}, | ||||||
| 	Value:       false, | 	Value:       false, | ||||||
| 	Usage:       "Automatically generate secrets", | 	Usage:       "Automatically generate secrets", | ||||||
| 	Destination: &Secrets, | 	Destination: &Secrets, | ||||||
| @ -22,7 +22,7 @@ var Pass bool | |||||||
| // PassFlag turns on/off storing generated secrets in pass | // PassFlag turns on/off storing generated secrets in pass | ||||||
| var PassFlag = &cli.BoolFlag{ | var PassFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "pass", | 	Name:        "pass", | ||||||
| 	Aliases:     []string{"P"}, | 	Aliases:     []string{"p"}, | ||||||
| 	Value:       false, | 	Value:       false, | ||||||
| 	Usage:       "Store the generated secrets in a local pass store", | 	Usage:       "Store the generated secrets in a local pass store", | ||||||
| 	Destination: &Pass, | 	Destination: &Pass, | ||||||
| @ -47,6 +47,7 @@ var ForceFlag = &cli.BoolFlag{ | |||||||
| 	Name:        "force", | 	Name:        "force", | ||||||
| 	Value:       false, | 	Value:       false, | ||||||
| 	Aliases:     []string{"f"}, | 	Aliases:     []string{"f"}, | ||||||
|  | 	Usage:       "Perform action without further prompt. Use with care!", | ||||||
| 	Destination: &Force, | 	Destination: &Force, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -113,13 +114,12 @@ var DNSValueFlag = &cli.StringFlag{ | |||||||
| 	Destination: &DNSValue, | 	Destination: &DNSValue, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var DNSTTL int | var DNSTTL string | ||||||
| 
 | var DNSTTLFlag = &cli.StringFlag{ | ||||||
| var DNSTTLFlag = &cli.IntFlag{ |  | ||||||
| 	Name:        "ttl", | 	Name:        "ttl", | ||||||
| 	Value:       86400, | 	Value:       "600s", | ||||||
| 	Aliases:     []string{"T"}, | 	Aliases:     []string{"T"}, | ||||||
| 	Usage:       "Domain name TTL value)", | 	Usage:       "Domain name TTL value (seconds)", | ||||||
| 	Destination: &DNSTTL, | 	Destination: &DNSTTL, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -271,3 +271,218 @@ var DebugFlag = &cli.BoolFlag{ | |||||||
| 	Destination: &Debug, | 	Destination: &Debug, | ||||||
| 	Usage:       "Show DEBUG messages", | 	Usage:       "Show DEBUG messages", | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // RC signifies the latest release candidate | ||||||
|  | var RC bool | ||||||
|  | 
 | ||||||
|  | // RCFlag chooses the latest release candidate for install | ||||||
|  | var RCFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "rc", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Destination: &RC, | ||||||
|  | 	Usage:       "Insatll the latest release candidate", | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Major bool | ||||||
|  | var MajorFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "major", | ||||||
|  | 	Usage:       "Increase the major part of the version", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Aliases:     []string{"ma", "x"}, | ||||||
|  | 	Destination: &Major, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Minor bool | ||||||
|  | var MinorFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "minor", | ||||||
|  | 	Usage:       "Increase the minor part of the version", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Aliases:     []string{"mi", "y"}, | ||||||
|  | 	Destination: &Minor, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Patch bool | ||||||
|  | var PatchFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "patch", | ||||||
|  | 	Usage:       "Increase the patch part of the version", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Aliases:     []string{"pa", "z"}, | ||||||
|  | 	Destination: &Patch, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Dry bool | ||||||
|  | var DryFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "dry-run", | ||||||
|  | 	Usage:       "Only reports changes that would be made", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Aliases:     []string{"d"}, | ||||||
|  | 	Destination: &Dry, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Publish bool | ||||||
|  | var PublishFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "publish", | ||||||
|  | 	Usage:       "Publish changes to git.coopcloud.tech", | ||||||
|  | 	Value:       false, | ||||||
|  | 	Aliases:     []string{"p"}, | ||||||
|  | 	Destination: &Publish, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Domain string | ||||||
|  | var DomainFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "domain", | ||||||
|  | 	Aliases:     []string{"d"}, | ||||||
|  | 	Value:       "", | ||||||
|  | 	Usage:       "Choose a domain name", | ||||||
|  | 	Destination: &Domain, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var NewAppServer string | ||||||
|  | var NewAppServerFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "server", | ||||||
|  | 	Aliases:     []string{"s"}, | ||||||
|  | 	Value:       "", | ||||||
|  | 	Usage:       "Show apps of a specific server", | ||||||
|  | 	Destination: &NewAppServer, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var NewAppName string | ||||||
|  | var NewAppNameFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "app-name", | ||||||
|  | 	Aliases:     []string{"a"}, | ||||||
|  | 	Value:       "", | ||||||
|  | 	Usage:       "Choose an app name", | ||||||
|  | 	Destination: &NewAppName, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var NoDomainChecks bool | ||||||
|  | var NoDomainChecksFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "no-domain-checks", | ||||||
|  | 	Aliases:     []string{"nd"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Disable app domain sanity checks", | ||||||
|  | 	Destination: &NoDomainChecks, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var StdErrOnly bool | ||||||
|  | var StdErrOnlyFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "stderr", | ||||||
|  | 	Aliases:     []string{"s"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Only tail stderr", | ||||||
|  | 	Destination: &StdErrOnly, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var AutoDNSRecord bool | ||||||
|  | var AutoDNSRecordFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "auto", | ||||||
|  | 	Aliases:     []string{"a"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Automatically configure DNS records", | ||||||
|  | 	Destination: &AutoDNSRecord, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var DontWaitConverge bool | ||||||
|  | var DontWaitConvergeFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "no-converge-checks", | ||||||
|  | 	Aliases:     []string{"nc"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Don't wait for converge logic checks", | ||||||
|  | 	Destination: &DontWaitConverge, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var Watch bool | ||||||
|  | var WatchFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "watch", | ||||||
|  | 	Aliases:     []string{"w"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Watch status by polling repeatedly", | ||||||
|  | 	Destination: &Watch, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var OnlyErrors bool | ||||||
|  | var OnlyErrorFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "errors", | ||||||
|  | 	Aliases:     []string{"e"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Only show errors", | ||||||
|  | 	Destination: &OnlyErrors, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var SkipUpdates bool | ||||||
|  | var SkipUpdatesFlag = &cli.BoolFlag{ | ||||||
|  | 	Name:        "skip-updates", | ||||||
|  | 	Aliases:     []string{"s"}, | ||||||
|  | 	Value:       false, | ||||||
|  | 	Usage:       "Skip updating recipe repositories", | ||||||
|  | 	Destination: &SkipUpdates, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var RegistryUsername string | ||||||
|  | var RegistryUsernameFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "username", | ||||||
|  | 	Aliases:     []string{"user"}, | ||||||
|  | 	Value:       "", | ||||||
|  | 	Usage:       "Registry username", | ||||||
|  | 	EnvVars:     []string{"REGISTRY_USERNAME"}, | ||||||
|  | 	Destination: &RegistryUsername, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var RegistryPassword string | ||||||
|  | var RegistryPasswordFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "password", | ||||||
|  | 	Aliases:     []string{"pass"}, | ||||||
|  | 	Value:       "", | ||||||
|  | 	Usage:       "Registry password", | ||||||
|  | 	EnvVars:     []string{"REGISTRY_PASSWORD"}, | ||||||
|  | 	Destination: &RegistryUsername, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // SSHFailMsg is a hopefully helpful SSH failure message | ||||||
|  | var SSHFailMsg = ` | ||||||
|  | Woops, Abra is unable to connect to connect to %s. | ||||||
|  | 
 | ||||||
|  | Here are a few tips for debugging your local SSH config. Abra uses plain 'ol | ||||||
|  | SSH to make connections to servers, so if your SSH config is working, Abra is | ||||||
|  | working. | ||||||
|  | 
 | ||||||
|  | In the first place, Abra will always try to read your Docker context connection | ||||||
|  | string for SSH connection details. You can view your server context configs | ||||||
|  | with the following command. Are they correct? | ||||||
|  | 
 | ||||||
|  |     abra server ls | ||||||
|  | 
 | ||||||
|  | Is your ssh-agent running? You can start it by running the following command: | ||||||
|  | 
 | ||||||
|  |     eval "$(ssh-agent)" | ||||||
|  | 
 | ||||||
|  | If your SSH private key loaded? You can check by running the following command: | ||||||
|  | 
 | ||||||
|  |     ssh-add -L | ||||||
|  | 
 | ||||||
|  | If, you can add it with: | ||||||
|  | 
 | ||||||
|  |     ssh-add ~/.ssh/<private-key-part> | ||||||
|  | 
 | ||||||
|  | If you are using a non-default public/private key, you can configure this in | ||||||
|  | your ~/.ssh/config file which Abra will read in order to figure out connection | ||||||
|  | details: | ||||||
|  | 
 | ||||||
|  | Host foo.coopcloud.tech | ||||||
|  |   Hostname foo.coopcloud.tech | ||||||
|  |   User bar | ||||||
|  |   Port 12345 | ||||||
|  |   IdentityFile ~/.ssh/bar@foo.coopcloud.tech | ||||||
|  | 
 | ||||||
|  | If you're only using password authentication, you can use the following config: | ||||||
|  | 
 | ||||||
|  | Host foo.coopcloud.tech | ||||||
|  |   Hostname foo.coopcloud.tech | ||||||
|  |   User bar | ||||||
|  |   Port 12345 | ||||||
|  |   PreferredAuthentications=password | ||||||
|  |   PubkeyAuthentication=no | ||||||
|  | 
 | ||||||
|  | Good luck! | ||||||
|  | 
 | ||||||
|  | ` | ||||||
							
								
								
									
										10
									
								
								cli/internal/list.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								cli/internal/list.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | package internal | ||||||
|  |  | ||||||
|  | // ReverseStringList reverses a list of a strings. Roll on Go generics. | ||||||
|  | func ReverseStringList(strings []string) []string { | ||||||
|  | 	for i, j := 0, len(strings)-1; i < j; i, j = i+1, j-1 { | ||||||
|  | 		strings[i], strings[j] = strings[j], strings[i] | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return strings | ||||||
|  | } | ||||||
| @ -4,9 +4,10 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"path" | 	"path" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/secret" | 	"coopcloud.tech/abra/pkg/secret" | ||||||
| 	"coopcloud.tech/abra/pkg/ssh" | 	"coopcloud.tech/abra/pkg/ssh" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| @ -14,35 +15,9 @@ import ( | |||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | // AppSecrets represents all app secrest | ||||||
| type AppSecrets map[string]string | type AppSecrets map[string]string | ||||||
|  |  | ||||||
| var Domain string |  | ||||||
| var DomainFlag = &cli.StringFlag{ |  | ||||||
| 	Name:        "domain", |  | ||||||
| 	Aliases:     []string{"d"}, |  | ||||||
| 	Value:       "", |  | ||||||
| 	Usage:       "Choose a domain name", |  | ||||||
| 	Destination: &Domain, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var NewAppServer string |  | ||||||
| var NewAppServerFlag = &cli.StringFlag{ |  | ||||||
| 	Name:        "server", |  | ||||||
| 	Aliases:     []string{"s"}, |  | ||||||
| 	Value:       "", |  | ||||||
| 	Usage:       "Show apps of a specific server", |  | ||||||
| 	Destination: &NewAppServer, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var NewAppName string |  | ||||||
| var NewAppNameFlag = &cli.StringFlag{ |  | ||||||
| 	Name:        "app-name", |  | ||||||
| 	Aliases:     []string{"a"}, |  | ||||||
| 	Value:       "", |  | ||||||
| 	Usage:       "Choose an app name", |  | ||||||
| 	Destination: &NewAppName, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // RecipeName is used for configuring recipe name programmatically | // RecipeName is used for configuring recipe name programmatically | ||||||
| var RecipeName string | var RecipeName string | ||||||
|  |  | ||||||
| @ -119,7 +94,7 @@ func ensureAppNameFlag() error { | |||||||
| 	if NewAppName == "" && !NoInput { | 	if NewAppName == "" && !NoInput { | ||||||
| 		prompt := &survey.Input{ | 		prompt := &survey.Input{ | ||||||
| 			Message: "Specify app name:", | 			Message: "Specify app name:", | ||||||
| 			Default: config.SanitiseAppName(Domain), | 			Default: Domain, | ||||||
| 		} | 		} | ||||||
| 		if err := survey.AskOne(prompt, &NewAppName); err != nil { | 		if err := survey.AskOne(prompt, &NewAppName); err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -137,7 +112,7 @@ func ensureAppNameFlag() error { | |||||||
| func NewAction(c *cli.Context) error { | func NewAction(c *cli.Context) error { | ||||||
| 	recipe := ValidateRecipeWithPrompt(c) | 	recipe := ValidateRecipeWithPrompt(c) | ||||||
|  |  | ||||||
| 	if err := config.EnsureAbraDirExists(); err != nil { | 	if err := recipePkg.EnsureUpToDate(recipe.Name); err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @ -155,11 +130,11 @@ func NewAction(c *cli.Context) error { | |||||||
|  |  | ||||||
| 	sanitisedAppName := config.SanitiseAppName(NewAppName) | 	sanitisedAppName := config.SanitiseAppName(NewAppName) | ||||||
| 	if len(sanitisedAppName) > 45 { | 	if len(sanitisedAppName) > 45 { | ||||||
| 		logrus.Fatalf("'%s' cannot be longer than 45 characters", sanitisedAppName) | 		logrus.Fatalf("%s cannot be longer than 45 characters", sanitisedAppName) | ||||||
| 	} | 	} | ||||||
| 	logrus.Debugf("'%s' sanitised as '%s' for new app", NewAppName, sanitisedAppName) | 	logrus.Debugf("%s sanitised as %s for new app", NewAppName, sanitisedAppName) | ||||||
|  |  | ||||||
| 	if err := config.TemplateAppEnvSample(recipe.Name, NewAppName, NewAppServer, Domain, recipe.Name); err != nil { | 	if err := config.TemplateAppEnvSample(recipe.Name, NewAppName, NewAppServer, Domain); err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @ -174,7 +149,7 @@ func NewAction(c *cli.Context) error { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		secretCols := []string{"Name", "Value"} | 		secretCols := []string{"Name", "Value"} | ||||||
| 		secretTable := abraFormatter.CreateTable(secretCols) | 		secretTable := formatter.CreateTable(secretCols) | ||||||
| 		for secret := range secrets { | 		for secret := range secrets { | ||||||
| 			secretTable.Append([]string{secret, secrets[secret]}) | 			secretTable.Append([]string{secret, secrets[secret]}) | ||||||
| 		} | 		} | ||||||
| @ -189,7 +164,7 @@ func NewAction(c *cli.Context) error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	tableCol := []string{"Name", "Domain", "Type", "Server"} | 	tableCol := []string{"Name", "Domain", "Type", "Server"} | ||||||
| 	table := abraFormatter.CreateTable(tableCol) | 	table := formatter.CreateTable(tableCol) | ||||||
| 	table.Append([]string{sanitisedAppName, Domain, recipe.Name, NewAppServer}) | 	table.Append([]string{sanitisedAppName, Domain, recipe.Name, NewAppServer}) | ||||||
|  |  | ||||||
| 	fmt.Println("") | 	fmt.Println("") | ||||||
|  | |||||||
| @ -6,100 +6,49 @@ import ( | |||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
|  | 	"github.com/docker/distribution/reference" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var Major bool |  | ||||||
| var MajorFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "major", |  | ||||||
| 	Usage:       "Increase the major part of the version", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"ma", "x"}, |  | ||||||
| 	Destination: &Major, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var Minor bool |  | ||||||
| var MinorFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "minor", |  | ||||||
| 	Usage:       "Increase the minor part of the version", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"mi", "y"}, |  | ||||||
| 	Destination: &Minor, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var Patch bool |  | ||||||
| var PatchFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "patch", |  | ||||||
| 	Usage:       "Increase the patch part of the version", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"p", "z"}, |  | ||||||
| 	Destination: &Patch, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var Dry bool |  | ||||||
| var DryFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "dry-run", |  | ||||||
| 	Usage:       "No changes are made, only reports changes that would be made", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"d"}, |  | ||||||
| 	Destination: &Dry, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var Push bool |  | ||||||
| var PushFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "push", |  | ||||||
| 	Usage:       "Git push changes", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"P"}, |  | ||||||
| 	Destination: &Push, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var CommitMessage string |  | ||||||
| var CommitMessageFlag = &cli.StringFlag{ |  | ||||||
| 	Name:        "commit-message", |  | ||||||
| 	Usage:       "Commit message (implies --commit)", |  | ||||||
| 	Aliases:     []string{"cm"}, |  | ||||||
| 	Destination: &CommitMessage, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var Commit bool |  | ||||||
| var CommitFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "commit", |  | ||||||
| 	Usage:       "Commits compose.**yml file changes to recipe repository", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Aliases:     []string{"c"}, |  | ||||||
| 	Destination: &Commit, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var TagMessage string |  | ||||||
| var TagMessageFlag = &cli.StringFlag{ |  | ||||||
| 	Name:        "tag-comment", |  | ||||||
| 	Usage:       "Description for release tag", |  | ||||||
| 	Aliases:     []string{"t", "tm"}, |  | ||||||
| 	Destination: &TagMessage, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // PromptBumpType prompts for version bump type | // PromptBumpType prompts for version bump type | ||||||
| func PromptBumpType(tagString string) error { | func PromptBumpType(tagString string) error { | ||||||
| 	if (!Major && !Minor && !Patch) && tagString == "" { | 	if (!Major && !Minor && !Patch) && tagString == "" { | ||||||
| 		fmt.Printf(` | 		fmt.Printf(` | ||||||
| semver cheat sheet (more via semver.org): | You need to make a decision about what kind of an update this new recipe | ||||||
|   major: new features/bug fixes, backwards incompatible | version is. If someone else performs this upgrade, do they have to do some | ||||||
|   minor: new features/bug fixes, backwards compatible | migration work or take care of some breaking changes? This can be signaled in | ||||||
|   patch: bug fixes, backwards compatible | the version you specify on the recipe deploy label and is called a semantic | ||||||
|  | version. | ||||||
|  |  | ||||||
|  | Here is a semver cheat sheet (more on https://semver.org): | ||||||
|  |  | ||||||
|  |     major: new features/bug fixes, backwards incompatible (e.g 1.0.0 -> 2.0.0). | ||||||
|  |            the upgrade won't work without some preparation work and others need | ||||||
|  |            to take care when performing it. "it could go wrong". | ||||||
|  |  | ||||||
|  |     minor: new features/bug fixes, backwards compatible (e.g. 0.1.0 -> 0.2.0). | ||||||
|  |            the upgrade should Just Work and there are no breaking changes in | ||||||
|  |            the app and the recipe config. "it should go fine". | ||||||
|  |  | ||||||
|  |     patch: bug fixes, backwards compatible (e.g. 0.0.1 -> 0.0.2). this upgrade | ||||||
|  |            should also Just Work and is mostly to do with minor bug fixes | ||||||
|  |            and/or security patches. "nothing to worry about". | ||||||
|  |  | ||||||
| `) | `) | ||||||
|  |  | ||||||
| 		var chosenBumpType string | 		var chosenBumpType string | ||||||
| 		prompt := &survey.Select{ | 		prompt := &survey.Select{ | ||||||
| 			Message: fmt.Sprintf("select recipe version increment type"), | 			Message: fmt.Sprintf("select recipe version increment type"), | ||||||
| 			Options: []string{"major", "minor", "patch"}, | 			Options: []string{"major", "minor", "patch"}, | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := survey.AskOne(prompt, &chosenBumpType); err != nil { | 		if err := survey.AskOne(prompt, &chosenBumpType); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		SetBumpType(chosenBumpType) | 		SetBumpType(chosenBumpType) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -133,20 +82,29 @@ func SetBumpType(bumpType string) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetMainApp retrieves the main 'app' image name | // GetMainAppImage retrieves the main 'app' image name | ||||||
| func GetMainApp(recipe recipe.Recipe) string { | func GetMainAppImage(recipe recipe.Recipe) (string, error) { | ||||||
| 	var app string | 	var path string | ||||||
|  |  | ||||||
| 	for _, service := range recipe.Config.Services { | 	for _, service := range recipe.Config.Services { | ||||||
| 		name := service.Name | 		if service.Name == "app" { | ||||||
| 		if name == "app" { | 			img, err := reference.ParseNormalizedNamed(service.Image) | ||||||
| 			app = strings.Split(service.Image, ":")[0] | 			if err != nil { | ||||||
|  | 				return "", err | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			path = reference.Path(img) | ||||||
|  | 			if strings.Contains(path, "library") { | ||||||
|  | 				path = strings.Split(path, "/")[1] | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			return path, nil | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if app == "" { | 	if path == "" { | ||||||
| 		logrus.Fatalf("%s has no main 'app' service?", recipe.Name) | 		return path, fmt.Errorf("%s has no main 'app' service?", recipe.Name) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return app | 	return path, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,106 +0,0 @@ | |||||||
| package internal |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	"github.com/AlecAivazis/survey/v2" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // EnsureDNSProvider ensures a DNS provider is chosen. |  | ||||||
| func EnsureDNSProvider() error { |  | ||||||
| 	if DNSProvider == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Select{ |  | ||||||
| 			Message: "Select DNS provider", |  | ||||||
| 			Options: []string{"gandi"}, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := survey.AskOne(prompt, &DNSProvider); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if DNSProvider == "" { |  | ||||||
| 		return fmt.Errorf("missing DNS provider?") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureDNSTypeFlag ensures a DNS type flag is present. |  | ||||||
| func EnsureDNSTypeFlag(c *cli.Context) error { |  | ||||||
| 	if DNSType == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "Specify DNS record type", |  | ||||||
| 			Default: "A", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &DNSType); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if DNSType == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no record type provided")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureDNSNameFlag ensures a DNS name flag is present. |  | ||||||
| func EnsureDNSNameFlag(c *cli.Context) error { |  | ||||||
| 	if DNSName == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "Specify DNS record name", |  | ||||||
| 			Default: "mysubdomain", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &DNSName); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if DNSName == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no record name provided")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureDNSValueFlag ensures a DNS value flag is present. |  | ||||||
| func EnsureDNSValueFlag(c *cli.Context) error { |  | ||||||
| 	if DNSValue == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "Specify DNS record value", |  | ||||||
| 			Default: "192.168.1.2", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &DNSValue); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if DNSName == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no record value provided")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureZoneArgument ensures a zone argument is present. |  | ||||||
| func EnsureZoneArgument(c *cli.Context) (string, error) { |  | ||||||
| 	var zone string |  | ||||||
| 	if c.Args().First() == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "Specify a domain name zone", |  | ||||||
| 			Default: "example.com", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &zone); err != nil { |  | ||||||
| 			return zone, err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if zone == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no zone value provided")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return zone, nil |  | ||||||
| } |  | ||||||
| @ -1,208 +0,0 @@ | |||||||
| package internal |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"os" |  | ||||||
| 	"strings" |  | ||||||
|  |  | ||||||
| 	"github.com/AlecAivazis/survey/v2" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // EnsureServerProvider ensures a 3rd party server provider is chosen. |  | ||||||
| func EnsureServerProvider() error { |  | ||||||
| 	if ServerProvider == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Select{ |  | ||||||
| 			Message: "Select server provider", |  | ||||||
| 			Options: []string{"capsul", "hetzner-cloud"}, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := survey.AskOne(prompt, &ServerProvider); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if ServerProvider == "" { |  | ||||||
| 		return fmt.Errorf("missing server provider?") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureNewCapsulVPSFlags ensure all flags are present. |  | ||||||
| func EnsureNewCapsulVPSFlags(c *cli.Context) error { |  | ||||||
| 	if CapsulName == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify capsul name", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &CapsulName); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify capsul instance URL", |  | ||||||
| 			Default: CapsulInstanceURL, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &CapsulInstanceURL); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify capsul type", |  | ||||||
| 			Default: CapsulType, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &CapsulType); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify capsul image", |  | ||||||
| 			Default: CapsulImage, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &CapsulImage); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if len(CapsulSSHKeys.Value()) == 0 && !NoInput { |  | ||||||
| 		var sshKeys string |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify capsul SSH keys (e.g. me@foo.com)", |  | ||||||
| 			Default: "", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &CapsulSSHKeys); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		CapsulSSHKeys = *cli.NewStringSlice(strings.Split(sshKeys, ",")...) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if CapsulAPIToken == "" && !NoInput { |  | ||||||
| 		token, ok := os.LookupEnv("CAPSUL_TOKEN") |  | ||||||
| 		if !ok { |  | ||||||
| 			prompt := &survey.Input{ |  | ||||||
| 				Message: "specify capsul API token", |  | ||||||
| 			} |  | ||||||
| 			if err := survey.AskOne(prompt, &CapsulAPIToken); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			CapsulAPIToken = token |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if CapsulName == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul name?")) |  | ||||||
| 	} |  | ||||||
| 	if CapsulInstanceURL == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul instance url?")) |  | ||||||
| 	} |  | ||||||
| 	if CapsulType == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul type?")) |  | ||||||
| 	} |  | ||||||
| 	if CapsulImage == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul image?")) |  | ||||||
| 	} |  | ||||||
| 	if len(CapsulSSHKeys.Value()) == 0 { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul ssh keys?")) |  | ||||||
| 	} |  | ||||||
| 	if CapsulAPIToken == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul API token?")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EnsureNewHetznerCloudVPSFlags ensure all flags are present. |  | ||||||
| func EnsureNewHetznerCloudVPSFlags(c *cli.Context) error { |  | ||||||
| 	if HetznerCloudName == "" && !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify hetzner cloud VPS name", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &HetznerCloudName); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify hetzner cloud VPS type", |  | ||||||
| 			Default: HetznerCloudType, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &HetznerCloudType); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify hetzner cloud VPS image", |  | ||||||
| 			Default: HetznerCloudImage, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &HetznerCloudImage); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if len(HetznerCloudSSHKeys.Value()) == 0 && !NoInput { |  | ||||||
| 		var sshKeys string |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify hetzner cloud SSH keys (e.g. me@foo.com)", |  | ||||||
| 			Default: "", |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &sshKeys); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		HetznerCloudSSHKeys = *cli.NewStringSlice(strings.Split(sshKeys, ",")...) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !NoInput { |  | ||||||
| 		prompt := &survey.Input{ |  | ||||||
| 			Message: "specify hetzner cloud VPS location", |  | ||||||
| 			Default: HetznerCloudLocation, |  | ||||||
| 		} |  | ||||||
| 		if err := survey.AskOne(prompt, &HetznerCloudLocation); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if HetznerCloudAPIToken == "" && !NoInput { |  | ||||||
| 		token, ok := os.LookupEnv("HCLOUD_TOKEN") |  | ||||||
| 		if !ok { |  | ||||||
| 			prompt := &survey.Input{ |  | ||||||
| 				Message: "specify hetzner cloud API token", |  | ||||||
| 			} |  | ||||||
| 			if err := survey.AskOne(prompt, &HetznerCloudAPIToken); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			HetznerCloudAPIToken = token |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if HetznerCloudName == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS name?")) |  | ||||||
| 	} |  | ||||||
| 	if HetznerCloudType == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS type?")) |  | ||||||
| 	} |  | ||||||
| 	if HetznerCloudImage == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud image?")) |  | ||||||
| 	} |  | ||||||
| 	if len(HetznerCloudSSHKeys.Value()) == 0 { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud ssh keys?")) |  | ||||||
| 	} |  | ||||||
| 	if HetznerCloudLocation == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS location?")) |  | ||||||
| 	} |  | ||||||
| 	if HetznerCloudAPIToken == "" { |  | ||||||
| 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud API token?")) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| @ -2,10 +2,11 @@ package internal | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/app" | 	"coopcloud.tech/abra/pkg/app" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/ssh" | 	"coopcloud.tech/abra/pkg/ssh" | ||||||
| @ -22,17 +23,24 @@ func ValidateRecipe(c *cli.Context) recipe.Recipe { | |||||||
| 	recipeName := c.Args().First() | 	recipeName := c.Args().First() | ||||||
|  |  | ||||||
| 	if recipeName == "" { | 	if recipeName == "" { | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no recipe provided")) | 		ShowSubcommandHelpAndError(c, errors.New("no recipe name provided")) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	recipe, err := recipe.Get(recipeName) | 	chosenRecipe, err := recipe.Get(recipeName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		if c.Command.Name == "generate" { | ||||||
|  | 			if strings.Contains(err.Error(), "missing a compose") { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			logrus.Warn(err) | ||||||
|  | 		} else { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("validated '%s' as recipe argument", recipeName) | 	logrus.Debugf("validated %s as recipe argument", recipeName) | ||||||
|  |  | ||||||
| 	return recipe | 	return chosenRecipe | ||||||
| } | } | ||||||
|  |  | ||||||
| // ValidateRecipeWithPrompt ensures a recipe argument is present before | // ValidateRecipeWithPrompt ensures a recipe argument is present before | ||||||
| @ -41,14 +49,33 @@ func ValidateRecipeWithPrompt(c *cli.Context) recipe.Recipe { | |||||||
| 	recipeName := c.Args().First() | 	recipeName := c.Args().First() | ||||||
|  |  | ||||||
| 	if recipeName == "" && !NoInput { | 	if recipeName == "" && !NoInput { | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() | 		var recipes []string | ||||||
|  |  | ||||||
|  | 		catl, err := recipe.ReadRecipeCatalogue() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		var recipes []string |  | ||||||
|  | 		knownRecipes := make(map[string]bool) | ||||||
| 		for name := range catl { | 		for name := range catl { | ||||||
| 			recipes = append(recipes, name) | 			knownRecipes[name] = true | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		localRecipes, err := recipe.GetRecipesLocal() | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for _, recipeLocal := range localRecipes { | ||||||
|  | 			if _, ok := knownRecipes[recipeLocal]; !ok { | ||||||
|  | 				knownRecipes[recipeLocal] = true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for recipeName := range knownRecipes { | ||||||
|  | 			recipes = append(recipes, recipeName) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		prompt := &survey.Select{ | 		prompt := &survey.Select{ | ||||||
| 			Message: "Select recipe", | 			Message: "Select recipe", | ||||||
| 			Options: recipes, | 			Options: recipes, | ||||||
| @ -64,17 +91,17 @@ func ValidateRecipeWithPrompt(c *cli.Context) recipe.Recipe { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if recipeName == "" { | 	if recipeName == "" { | ||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no recipe provided")) | 		ShowSubcommandHelpAndError(c, errors.New("no recipe name provided")) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	recipe, err := recipe.Get(recipeName) | 	chosenRecipe, err := recipe.Get(recipeName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("validated '%s' as recipe argument", recipeName) | 	logrus.Debugf("validated %s as recipe argument", recipeName) | ||||||
|  |  | ||||||
| 	return recipe | 	return chosenRecipe | ||||||
| } | } | ||||||
|  |  | ||||||
| // ValidateApp ensures the app name arg is valid. | // ValidateApp ensures the app name arg is valid. | ||||||
| @ -103,7 +130,7 @@ func ValidateApp(c *cli.Context) config.App { | |||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("validated '%s' as app argument", appName) | 	logrus.Debugf("validated %s as app argument", appName) | ||||||
|  |  | ||||||
| 	return app | 	return app | ||||||
| } | } | ||||||
| @ -126,7 +153,7 @@ func ValidateDomain(c *cli.Context) (string, error) { | |||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no domain provided")) | 		ShowSubcommandHelpAndError(c, errors.New("no domain provided")) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("validated '%s' as domain argument", domainName) | 	logrus.Debugf("validated %s as domain argument", domainName) | ||||||
|  |  | ||||||
| 	return domainName, nil | 	return domainName, nil | ||||||
| } | } | ||||||
| @ -168,7 +195,301 @@ func ValidateServer(c *cli.Context) (string, error) { | |||||||
| 		ShowSubcommandHelpAndError(c, errors.New("no server provided")) | 		ShowSubcommandHelpAndError(c, errors.New("no server provided")) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("validated '%s' as server argument", serverName) | 	logrus.Debugf("validated %s as server argument", serverName) | ||||||
|  |  | ||||||
| 	return serverName, nil | 	return serverName, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // EnsureDNSProvider ensures a DNS provider is chosen. | ||||||
|  | func EnsureDNSProvider() error { | ||||||
|  | 	if DNSProvider == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Select{ | ||||||
|  | 			Message: "Select DNS provider", | ||||||
|  | 			Options: []string{"gandi"}, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &DNSProvider); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if DNSProvider == "" { | ||||||
|  | 		return fmt.Errorf("missing DNS provider?") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureDNSTypeFlag ensures a DNS type flag is present. | ||||||
|  | func EnsureDNSTypeFlag(c *cli.Context) error { | ||||||
|  | 	if DNSType == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "Specify DNS record type", | ||||||
|  | 			Default: "A", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &DNSType); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if DNSType == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, errors.New("no record type provided")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureDNSNameFlag ensures a DNS name flag is present. | ||||||
|  | func EnsureDNSNameFlag(c *cli.Context) error { | ||||||
|  | 	if DNSName == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "Specify DNS record name", | ||||||
|  | 			Default: "mysubdomain", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &DNSName); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if DNSName == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, errors.New("no record name provided")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureDNSValueFlag ensures a DNS value flag is present. | ||||||
|  | func EnsureDNSValueFlag(c *cli.Context) error { | ||||||
|  | 	if DNSValue == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "Specify DNS record value", | ||||||
|  | 			Default: "192.168.1.2", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &DNSValue); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if DNSValue == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, errors.New("no record value provided")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureZoneArgument ensures a zone argument is present. | ||||||
|  | func EnsureZoneArgument(c *cli.Context) (string, error) { | ||||||
|  | 	zone := c.Args().First() | ||||||
|  |  | ||||||
|  | 	if zone == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "Specify a domain name zone", | ||||||
|  | 			Default: "example.com", | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &zone); err != nil { | ||||||
|  | 			return zone, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if zone == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, errors.New("no zone value provided")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return zone, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureServerProvider ensures a 3rd party server provider is chosen. | ||||||
|  | func EnsureServerProvider() error { | ||||||
|  | 	if ServerProvider == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Select{ | ||||||
|  | 			Message: "Select server provider", | ||||||
|  | 			Options: []string{"capsul", "hetzner-cloud"}, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &ServerProvider); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if ServerProvider == "" { | ||||||
|  | 		return fmt.Errorf("missing server provider?") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureNewCapsulVPSFlags ensure all flags are present. | ||||||
|  | func EnsureNewCapsulVPSFlags(c *cli.Context) error { | ||||||
|  | 	if CapsulName == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify capsul name", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &CapsulName); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify capsul instance URL", | ||||||
|  | 			Default: CapsulInstanceURL, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &CapsulInstanceURL); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify capsul type", | ||||||
|  | 			Default: CapsulType, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &CapsulType); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify capsul image", | ||||||
|  | 			Default: CapsulImage, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &CapsulImage); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(CapsulSSHKeys.Value()) == 0 && !NoInput { | ||||||
|  | 		var sshKeys string | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify capsul SSH keys (e.g. me@foo.com)", | ||||||
|  | 			Default: "", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &CapsulSSHKeys); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		CapsulSSHKeys = *cli.NewStringSlice(strings.Split(sshKeys, ",")...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if CapsulAPIToken == "" && !NoInput { | ||||||
|  | 		token, ok := os.LookupEnv("CAPSUL_TOKEN") | ||||||
|  | 		if !ok { | ||||||
|  | 			prompt := &survey.Input{ | ||||||
|  | 				Message: "specify capsul API token", | ||||||
|  | 			} | ||||||
|  | 			if err := survey.AskOne(prompt, &CapsulAPIToken); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			CapsulAPIToken = token | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if CapsulName == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul name?")) | ||||||
|  | 	} | ||||||
|  | 	if CapsulInstanceURL == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul instance url?")) | ||||||
|  | 	} | ||||||
|  | 	if CapsulType == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul type?")) | ||||||
|  | 	} | ||||||
|  | 	if CapsulImage == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul image?")) | ||||||
|  | 	} | ||||||
|  | 	if len(CapsulSSHKeys.Value()) == 0 { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul ssh keys?")) | ||||||
|  | 	} | ||||||
|  | 	if CapsulAPIToken == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing capsul API token?")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // EnsureNewHetznerCloudVPSFlags ensure all flags are present. | ||||||
|  | func EnsureNewHetznerCloudVPSFlags(c *cli.Context) error { | ||||||
|  | 	if HetznerCloudName == "" && !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify hetzner cloud VPS name", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &HetznerCloudName); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify hetzner cloud VPS type", | ||||||
|  | 			Default: HetznerCloudType, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &HetznerCloudType); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify hetzner cloud VPS image", | ||||||
|  | 			Default: HetznerCloudImage, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &HetznerCloudImage); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(HetznerCloudSSHKeys.Value()) == 0 && !NoInput { | ||||||
|  | 		var sshKeys string | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify hetzner cloud SSH keys (e.g. me@foo.com)", | ||||||
|  | 			Default: "", | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &sshKeys); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		HetznerCloudSSHKeys = *cli.NewStringSlice(strings.Split(sshKeys, ",")...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !NoInput { | ||||||
|  | 		prompt := &survey.Input{ | ||||||
|  | 			Message: "specify hetzner cloud VPS location", | ||||||
|  | 			Default: HetznerCloudLocation, | ||||||
|  | 		} | ||||||
|  | 		if err := survey.AskOne(prompt, &HetznerCloudLocation); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if HetznerCloudAPIToken == "" && !NoInput { | ||||||
|  | 		token, ok := os.LookupEnv("HCLOUD_TOKEN") | ||||||
|  | 		if !ok { | ||||||
|  | 			prompt := &survey.Input{ | ||||||
|  | 				Message: "specify hetzner cloud API token", | ||||||
|  | 			} | ||||||
|  | 			if err := survey.AskOne(prompt, &HetznerCloudAPIToken); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			HetznerCloudAPIToken = token | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if HetznerCloudName == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS name?")) | ||||||
|  | 	} | ||||||
|  | 	if HetznerCloudType == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS type?")) | ||||||
|  | 	} | ||||||
|  | 	if HetznerCloudImage == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud image?")) | ||||||
|  | 	} | ||||||
|  | 	if HetznerCloudLocation == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud VPS location?")) | ||||||
|  | 	} | ||||||
|  | 	if HetznerCloudAPIToken == "" { | ||||||
|  | 		ShowSubcommandHelpAndError(c, fmt.Errorf("missing hetzner cloud API token?")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -2,112 +2,74 @@ package recipe | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"os" |  | ||||||
| 	"strconv" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/abra/pkg/lint" | ||||||
| 	"github.com/docker/distribution/reference" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeLintCommand = &cli.Command{ | var recipeLintCommand = &cli.Command{ | ||||||
| 	Name:      "lint", | 	Name:         "lint", | ||||||
| 	Usage:     "Lint a recipe", | 	Usage:        "Lint a recipe", | ||||||
| 	Aliases:   []string{"l"}, | 	Aliases:      []string{"l"}, | ||||||
| 	ArgsUsage: "<recipe>", | 	ArgsUsage:    "<recipe>", | ||||||
|  | 	Flags:        []cli.Flag{internal.OnlyErrorFlag}, | ||||||
|  | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipe := internal.ValidateRecipe(c) | 		recipe := internal.ValidateRecipe(c) | ||||||
|  |  | ||||||
| 		expectedVersion := false | 		if err := recipePkg.EnsureUpToDate(recipe.Name); err != nil { | ||||||
| 		if recipe.Config.Version == "3.8" { |  | ||||||
| 			expectedVersion = true |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		envSampleProvided := false |  | ||||||
| 		envSample := fmt.Sprintf("%s/%s/.env.sample", config.APPS_DIR, recipe.Name) |  | ||||||
| 		if _, err := os.Stat(envSample); !os.IsNotExist(err) { |  | ||||||
| 			envSampleProvided = true |  | ||||||
| 		} else if err != nil { |  | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		serviceNamedApp := false | 		tableCol := []string{"ref", "rule", "satisfied", "severity", "resolve"} | ||||||
| 		traefikEnabled := false | 		table := formatter.CreateTable(tableCol) | ||||||
| 		healthChecksForAllServices := true |  | ||||||
| 		allImagesTagged := true |  | ||||||
| 		noUnstableTags := true |  | ||||||
| 		semverLikeTags := true |  | ||||||
| 		for _, service := range recipe.Config.Services { |  | ||||||
| 			if service.Name == "app" { |  | ||||||
| 				serviceNamedApp = true |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			for label := range service.Deploy.Labels { | 		hasError := false | ||||||
| 				if label == "traefik.enable" { | 		bar := formatter.CreateProgressbar(-1, "running recipe lint rules...") | ||||||
| 					if service.Deploy.Labels[label] == "true" { | 		for level := range lint.LintRules { | ||||||
| 						traefikEnabled = true | 			for _, rule := range lint.LintRules[level] { | ||||||
| 					} | 				ok, err := rule.Function(recipe) | ||||||
|  | 				if err != nil { | ||||||
|  | 					logrus.Warn(err) | ||||||
| 				} | 				} | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			img, err := reference.ParseNormalizedNamed(service.Image) | 				if !ok && rule.Level == "error" { | ||||||
| 			if err != nil { | 					hasError = true | ||||||
| 				logrus.Fatal(err) | 				} | ||||||
| 			} |  | ||||||
| 			if reference.IsNameOnly(img) { |  | ||||||
| 				allImagesTagged = false |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			var tag string | 				var result string | ||||||
| 			switch img.(type) { | 				if ok { | ||||||
| 			case reference.NamedTagged: | 					result = "yes" | ||||||
| 				tag = img.(reference.NamedTagged).Tag() | 				} else { | ||||||
| 			case reference.Named: | 					result = "NO" | ||||||
| 				noUnstableTags = false | 				} | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if tag == "latest" { | 				if internal.OnlyErrors { | ||||||
| 				noUnstableTags = false | 					if !ok && rule.Level == "error" { | ||||||
| 			} | 						table.Append([]string{rule.Ref, rule.Description, result, rule.Level, rule.HowToResolve}) | ||||||
|  | 						bar.Add(1) | ||||||
| 			if !tagcmp.IsParsable(tag) { | 					} | ||||||
| 				semverLikeTags = false | 				} else { | ||||||
| 			} | 					table.Append([]string{rule.Ref, rule.Description, result, rule.Level, rule.HowToResolve}) | ||||||
|  | 					bar.Add(1) | ||||||
| 			if service.HealthCheck == nil { | 				} | ||||||
| 				healthChecksForAllServices = false |  | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"rule", "satisfied"} | 		if table.NumLines() > 0 { | ||||||
| 		table := formatter.CreateTable(tableCol) | 			fmt.Println() | ||||||
| 		table.Append([]string{"compose files have the expected version", strconv.FormatBool(expectedVersion)}) | 			table.Render() | ||||||
| 		table.Append([]string{"environment configuration is provided", strconv.FormatBool(envSampleProvided)}) | 		} | ||||||
| 		table.Append([]string{"recipe contains a service named 'app'", strconv.FormatBool(serviceNamedApp)}) |  | ||||||
| 		table.Append([]string{"traefik routing enabled on at least one service", strconv.FormatBool(traefikEnabled)}) | 		if hasError { | ||||||
| 		table.Append([]string{"all services have a healthcheck enabled", strconv.FormatBool(healthChecksForAllServices)}) | 			logrus.Warn("watch out, some critical errors are present in your recipe config") | ||||||
| 		table.Append([]string{"all images are using a tag", strconv.FormatBool(allImagesTagged)}) | 		} | ||||||
| 		table.Append([]string{"no usage of unstable 'latest' tags", strconv.FormatBool(noUnstableTags)}) |  | ||||||
| 		table.Append([]string{"all tags are using a semver-like format", strconv.FormatBool(semverLikeTags)}) |  | ||||||
| 		table.Render() |  | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { |  | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for name := range catl { |  | ||||||
| 			fmt.Println(name) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -2,37 +2,82 @@ package recipe | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"path" | ||||||
| 	"sort" | 	"sort" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	gitPkg "coopcloud.tech/abra/pkg/git" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | var pattern string | ||||||
|  | var patternFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "pattern", | ||||||
|  | 	Value:       "", | ||||||
|  | 	Aliases:     []string{"p"}, | ||||||
|  | 	Usage:       "Simple string to filter recipes", | ||||||
|  | 	Destination: &pattern, | ||||||
|  | } | ||||||
|  |  | ||||||
| var recipeListCommand = &cli.Command{ | var recipeListCommand = &cli.Command{ | ||||||
| 	Name:    "list", | 	Name:    "list", | ||||||
| 	Usage:   "List available recipes", | 	Usage:   "List available recipes", | ||||||
| 	Aliases: []string{"ls"}, | 	Aliases: []string{"ls"}, | ||||||
|  | 	Flags: []cli.Flag{ | ||||||
|  | 		patternFlag, | ||||||
|  | 	}, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() | 		catalogueDir := path.Join(config.ABRA_DIR, "catalogue") | ||||||
|  | 		url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, "recipes") | ||||||
|  | 		if err := gitPkg.Clone(catalogueDir, url); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		catl, err := recipe.ReadRecipeCatalogue() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err.Error()) | 			logrus.Fatal(err.Error()) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		recipes := catl.Flatten() | 		recipes := catl.Flatten() | ||||||
| 		sort.Sort(catalogue.ByRecipeName(recipes)) | 		sort.Sort(recipe.ByRecipeName(recipes)) | ||||||
|  |  | ||||||
| 		tableCol := []string{"name", "category", "status"} | 		tableCol := []string{"name", "category", "status", "healthcheck", "backups", "email", "tests", "SSO"} | ||||||
| 		table := formatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
|  | 		len := 0 | ||||||
| 		for _, recipe := range recipes { | 		for _, recipe := range recipes { | ||||||
| 			status := fmt.Sprintf("%v", recipe.Features.Status) | 			tableRow := []string{ | ||||||
| 			tableRow := []string{recipe.Name, recipe.Category, status} | 				recipe.Name, | ||||||
| 			table.Append(tableRow) | 				recipe.Category, | ||||||
|  | 				strconv.Itoa(recipe.Features.Status), | ||||||
|  | 				recipe.Features.Healthcheck, | ||||||
|  | 				recipe.Features.Backups, | ||||||
|  | 				recipe.Features.Email, | ||||||
|  | 				recipe.Features.Tests, | ||||||
|  | 				recipe.Features.SSO, | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if pattern != "" { | ||||||
|  | 				if strings.Contains(recipe.Name, pattern) { | ||||||
|  | 					table.Append(tableRow) | ||||||
|  | 					len++ | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				table.Append(tableRow) | ||||||
|  | 				len++ | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table.Render() | 		table.SetCaption(true, fmt.Sprintf("total recipes: %v", len)) | ||||||
|  |  | ||||||
|  | 		if table.NumLines() > 0 { | ||||||
|  | 			table.Render() | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
|  | |||||||
| @ -1,8 +1,10 @@ | |||||||
| package recipe | package recipe | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"io/ioutil" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path" | 	"path" | ||||||
| 	"text/template" | 	"text/template" | ||||||
| @ -14,6 +16,20 @@ import ( | |||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | // recipeMetadata is the recipe metadata for the README.md | ||||||
|  | type recipeMetadata struct { | ||||||
|  | 	Name        string | ||||||
|  | 	Description string | ||||||
|  | 	Category    string | ||||||
|  | 	Status      string | ||||||
|  | 	Image       string | ||||||
|  | 	Healthcheck string | ||||||
|  | 	Backups     string | ||||||
|  | 	Email       string | ||||||
|  | 	Tests       string | ||||||
|  | 	SSO         string | ||||||
|  | } | ||||||
|  |  | ||||||
| var recipeNewCommand = &cli.Command{ | var recipeNewCommand = &cli.Command{ | ||||||
| 	Name:      "new", | 	Name:      "new", | ||||||
| 	Usage:     "Create a new recipe", | 	Usage:     "Create a new recipe", | ||||||
| @ -29,19 +45,17 @@ Abra uses our built-in example repository which is available here: | |||||||
| Files within the example repository make use of the Golang templating system | Files within the example repository make use of the Golang templating system | ||||||
| which Abra uses to inject values into the generated recipe folder (e.g. name of | which Abra uses to inject values into the generated recipe folder (e.g. name of | ||||||
| recipe and domain in the sample environment config). | recipe and domain in the sample environment config). | ||||||
|  |  | ||||||
| The new example repository is cloned to ~/.abra/apps/<recipe>. |  | ||||||
| `, | `, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipeName := c.Args().First() | 		recipeName := c.Args().First() | ||||||
|  |  | ||||||
| 		if recipeName == "" { | 		if recipeName == "" { | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("no recipe provided")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("no recipe name provided")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		directory := path.Join(config.APPS_DIR, recipeName) | 		directory := path.Join(config.RECIPES_DIR, recipeName) | ||||||
| 		if _, err := os.Stat(directory); !os.IsNotExist(err) { | 		if _, err := os.Stat(directory); !os.IsNotExist(err) { | ||||||
| 			logrus.Fatalf("'%s' recipe directory already exists?", directory) | 			logrus.Fatalf("%s recipe directory already exists?", directory) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		url := fmt.Sprintf("%s/example.git", config.REPOS_BASE_URL) | 		url := fmt.Sprintf("%s/example.git", config.REPOS_BASE_URL) | ||||||
| @ -49,49 +63,73 @@ The new example repository is cloned to ~/.abra/apps/<recipe>. | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		gitRepo := path.Join(config.APPS_DIR, recipeName, ".git") | 		gitRepo := path.Join(config.RECIPES_DIR, recipeName, ".git") | ||||||
| 		if err := os.RemoveAll(gitRepo); err != nil { | 		if err := os.RemoveAll(gitRepo); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		logrus.Debugf("removed git repo in '%s'", gitRepo) | 		logrus.Debugf("removed example git repo in %s", gitRepo) | ||||||
|  |  | ||||||
|  | 		meta := newRecipeMeta(recipeName) | ||||||
|  |  | ||||||
| 		toParse := []string{ | 		toParse := []string{ | ||||||
| 			path.Join(config.APPS_DIR, recipeName, "README.md"), | 			path.Join(config.RECIPES_DIR, recipeName, "README.md"), | ||||||
| 			path.Join(config.APPS_DIR, recipeName, ".env.sample"), | 			path.Join(config.RECIPES_DIR, recipeName, ".env.sample"), | ||||||
| 			path.Join(config.APPS_DIR, recipeName, ".drone.yml"), |  | ||||||
| 		} | 		} | ||||||
| 		for _, path := range toParse { | 		for _, path := range toParse { | ||||||
| 			file, err := os.OpenFile(path, os.O_RDWR, 0755) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			tpl, err := template.ParseFiles(path) | 			tpl, err := template.ParseFiles(path) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			// TODO: ask for description and probably other things so that the | 			var templated bytes.Buffer | ||||||
| 			// template repository is more "ready" to go than the current best-guess | 			if err := tpl.Execute(&templated, meta); err != nil { | ||||||
| 			// mode of templating |  | ||||||
| 			if err := tpl.Execute(file, struct { |  | ||||||
| 				Name        string |  | ||||||
| 				Description string |  | ||||||
| 			}{recipeName, "TODO"}); err != nil { |  | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			if err := ioutil.WriteFile(path, templated.Bytes(), 0644); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		newGitRepo := path.Join(config.APPS_DIR, recipeName) | 		newGitRepo := path.Join(config.RECIPES_DIR, recipeName) | ||||||
| 		if err := git.Init(newGitRepo, true); err != nil { | 		if err := git.Init(newGitRepo, true); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Infof( | 		fmt.Print(fmt.Sprintf(` | ||||||
| 			"new recipe '%s' created in %s, happy hacking!\n", | Your new %s recipe has been created in %s. | ||||||
| 			recipeName, path.Join(config.APPS_DIR, recipeName), |  | ||||||
| 		) | In order to share your recipe, you can upload it the git repository to: | ||||||
|  |  | ||||||
|  |     https://git.coopcloud.tech/coop-cloud/%s | ||||||
|  |  | ||||||
|  | If you're not sure how to do that, come chat with us: | ||||||
|  |  | ||||||
|  |     https://docs.coopcloud.tech/contact | ||||||
|  |  | ||||||
|  | See "abra recipe -h" for additional recipe maintainer commands. | ||||||
|  |  | ||||||
|  | Happy Hacking! | ||||||
|  |  | ||||||
|  | `, recipeName, path.Join(config.RECIPES_DIR, recipeName), recipeName)) | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // newRecipeMeta creates a new recipeMetadata instance with defaults | ||||||
|  | func newRecipeMeta(recipeName string) recipeMetadata { | ||||||
|  | 	return recipeMetadata{ | ||||||
|  | 		Name:        recipeName, | ||||||
|  | 		Description: "> One line description of the recipe", | ||||||
|  | 		Category:    "Apps", | ||||||
|  | 		Status:      "0", | ||||||
|  | 		Image:       fmt.Sprintf("[`%s`](https://hub.docker.com/r/%s), 4, upstream", recipeName, recipeName), | ||||||
|  | 		Healthcheck: "No", | ||||||
|  | 		Backups:     "No", | ||||||
|  | 		Email:       "No", | ||||||
|  | 		Tests:       "No", | ||||||
|  | 		SSO:         "No", | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | |||||||
| @ -7,13 +7,18 @@ import ( | |||||||
| // RecipeCommand defines all recipe related sub-commands. | // RecipeCommand defines all recipe related sub-commands. | ||||||
| var RecipeCommand = &cli.Command{ | var RecipeCommand = &cli.Command{ | ||||||
| 	Name:      "recipe", | 	Name:      "recipe", | ||||||
| 	Usage:     "Manage recipes (for maintainers)", | 	Usage:     "Manage recipes", | ||||||
| 	ArgsUsage: "<recipe>", | 	ArgsUsage: "<recipe>", | ||||||
| 	Aliases:   []string{"r"}, | 	Aliases:   []string{"r"}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| A recipe is a blueprint for an app. It is a bunch of configuration files which | A recipe is a blueprint for an app. It is a bunch of config files which | ||||||
| describe how to deploy and maintain an app. Recipes are maintained by the Co-op | describe how to deploy and maintain an app. Recipes are maintained by the Co-op | ||||||
| Cloud community and you can use Abra to read them and create apps for you. | Cloud community and you can use Abra to read them and create apps for you. | ||||||
|  |  | ||||||
|  | Anyone who uses a recipe can become a maintainer. Maintainers typically make | ||||||
|  | sure the recipe is in good working order and the config upgraded in a timely | ||||||
|  | manner. Abra supports convenient automation for recipe maintainenace, see the | ||||||
|  | "abra recipe upgrade", "abra recipe sync" and "abra recipe release" commands. | ||||||
| `, | `, | ||||||
| 	Subcommands: []*cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		recipeListCommand, | 		recipeListCommand, | ||||||
|  | |||||||
| @ -6,16 +6,17 @@ import ( | |||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	gitPkg "coopcloud.tech/abra/pkg/git" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
| @ -26,270 +27,96 @@ var recipeReleaseCommand = &cli.Command{ | |||||||
| 	Aliases:   []string{"rl"}, | 	Aliases:   []string{"rl"}, | ||||||
| 	ArgsUsage: "<recipe> [<version>]", | 	ArgsUsage: "<recipe> [<version>]", | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command is used to specify a new tag for a recipe. These tags are used to | This command is used to specify a new version of a recipe. These versions are | ||||||
| identify different versions of the recipe and are published on the Co-op Cloud | then published on the Co-op Cloud recipe catalogue. These versions take the | ||||||
| recipe catalogue. | following form: | ||||||
|  |  | ||||||
| These tags take the following form: |  | ||||||
|  |  | ||||||
|     a.b.c+x.y.z |     a.b.c+x.y.z | ||||||
|  |  | ||||||
| Where the "a.b.c" part is maintained as a semantic version of the recipe by the | Where the "a.b.c" part is a semantic version determined by the maintainer. And | ||||||
| recipe maintainer. And the "x.y.z" part is the image tag of the recipe "app" | the "x.y.z" part is the image tag of the recipe "app" service (the main | ||||||
| service (the main container which contains the software to be used). | container which contains the software to be used). | ||||||
|  |  | ||||||
| We maintain a semantic versioning scheme ("a.b.c") alongside the libre app | We maintain a semantic versioning scheme ("a.b.c") alongside the libre app | ||||||
| versioning scheme in order to maximise the chances that the nature of recipe | versioning scheme ("x.y.z") in order to maximise the chances that the nature of | ||||||
| updates are properly communicated. | recipe updates are properly communicated. I.e. developers of an app might | ||||||
|  | publish a minor version but that might lead to changes in the recipe which are | ||||||
| Abra does its best to read the "a.b.c" version scheme and communicate what | major and therefore require intervention while doing the upgrade work. | ||||||
| action needs to be taken when performing different operations such as an update |  | ||||||
| or a rollback of an app. |  | ||||||
|  |  | ||||||
| You may invoke this command in "wizard" mode and be prompted for input: |  | ||||||
|  |  | ||||||
|     abra recipe release gitea |  | ||||||
|  |  | ||||||
|  | Publish your new release to git.coopcloud.tech with "-p/--publish". This | ||||||
|  | requires that you have permission to git push to these repositories and have | ||||||
|  | your SSH keys configured on your account. | ||||||
| `, | `, | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.DryFlag, | 		internal.DryFlag, | ||||||
| 		internal.MajorFlag, | 		internal.MajorFlag, | ||||||
| 		internal.MinorFlag, | 		internal.MinorFlag, | ||||||
| 		internal.PatchFlag, | 		internal.PatchFlag, | ||||||
| 		internal.PushFlag, | 		internal.PublishFlag, | ||||||
| 		internal.CommitFlag, |  | ||||||
| 		internal.CommitMessageFlag, |  | ||||||
| 		internal.TagMessageFlag, |  | ||||||
| 	}, | 	}, | ||||||
|  | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipe := internal.ValidateRecipeWithPrompt(c) | 		recipe := internal.ValidateRecipeWithPrompt(c) | ||||||
| 		directory := path.Join(config.APPS_DIR, recipe.Name) |  | ||||||
| 		tagString := c.Args().Get(1) |  | ||||||
| 		mainApp := internal.GetMainApp(recipe) |  | ||||||
|  |  | ||||||
| 		imagesTmp, err := getImageVersions(recipe) | 		imagesTmp, err := getImageVersions(recipe) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		mainAppVersion := imagesTmp[mainApp] |  | ||||||
|  |  | ||||||
| 		if err := recipePkg.EnsureExists(recipe.Name); err != nil { | 		mainApp, err := internal.GetMainAppImage(recipe) | ||||||
|  | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		mainAppVersion := imagesTmp[mainApp] | ||||||
| 		if mainAppVersion == "" { | 		if mainAppVersion == "" { | ||||||
| 			logrus.Fatalf("main 'app' service version for %s is empty?", recipe.Name) | 			logrus.Fatalf("main app service version for %s is empty?", recipe.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		tagString := c.Args().Get(1) | ||||||
| 		if tagString != "" { | 		if tagString != "" { | ||||||
| 			if _, err := tagcmp.Parse(tagString); err != nil { | 			if _, err := tagcmp.Parse(tagString); err != nil { | ||||||
| 				logrus.Fatal("invalid tag specified") | 				logrus.Fatalf("cannot parse %s, invalid tag specified?", tagString) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (!internal.Major && !internal.Minor && !internal.Patch) && tagString != "" { |  | ||||||
| 			logrus.Fatal("please specify <version> or bump type (--major/--minor/--patch)") |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (internal.Major || internal.Minor || internal.Patch) && tagString != "" { | 		if (internal.Major || internal.Minor || internal.Patch) && tagString != "" { | ||||||
| 			logrus.Fatal("cannot specify tag and bump type at the same time") | 			logrus.Fatal("cannot specify tag and bump type at the same time") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// bumpType is used to decide what part of the tag should be incremented |  | ||||||
| 		bumpType := btoi(internal.Major)*4 + btoi(internal.Minor)*2 + btoi(internal.Patch) |  | ||||||
| 		if bumpType != 0 { |  | ||||||
| 			// a bitwise check if the number is a power of 2 |  | ||||||
| 			if (bumpType & (bumpType - 1)) != 0 { |  | ||||||
| 				logrus.Fatal("you can only use one of: --major, --minor, --patch.") |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := internal.PromptBumpType(tagString); err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if internal.TagMessage == "" { |  | ||||||
| 			prompt := &survey.Input{ |  | ||||||
| 				Message: "tag message", |  | ||||||
| 				Default: fmt.Sprintf("chore: publish new %s version", internal.GetBumpType()), |  | ||||||
| 			} |  | ||||||
| 			if err := survey.AskOne(prompt, &internal.TagMessage); err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		var createTagOptions git.CreateTagOptions |  | ||||||
| 		createTagOptions.Message = internal.TagMessage |  | ||||||
|  |  | ||||||
| 		if !internal.Commit { |  | ||||||
| 			prompt := &survey.Confirm{ |  | ||||||
| 				Message: "git commit changes also?", |  | ||||||
| 			} |  | ||||||
| 			if err := survey.AskOne(prompt, &internal.Commit); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if !internal.Push { |  | ||||||
| 			prompt := &survey.Confirm{ |  | ||||||
| 				Message: "git push changes also?", |  | ||||||
| 			} |  | ||||||
| 			if err := survey.AskOne(prompt, &internal.Push); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if internal.Commit || internal.CommitMessage != "" { |  | ||||||
| 			commitRepo, err := git.PlainOpen(directory) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			commitWorktree, err := commitRepo.Worktree() |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if internal.CommitMessage == "" { |  | ||||||
| 				prompt := &survey.Input{ |  | ||||||
| 					Message: "commit message", |  | ||||||
| 					Default: fmt.Sprintf("chore: publish new %s version", internal.GetBumpType()), |  | ||||||
| 				} |  | ||||||
| 				if err := survey.AskOne(prompt, &internal.CommitMessage); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			err = commitWorktree.AddGlob("compose.**yml") |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
| 			logrus.Debug("staged compose.**yml for commit") |  | ||||||
|  |  | ||||||
| 			if !internal.Dry { |  | ||||||
| 				_, err = commitWorktree.Commit(internal.CommitMessage, &git.CommitOptions{}) |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				logrus.Info("changes commited") |  | ||||||
| 			} else { |  | ||||||
| 				logrus.Info("dry run only: NOT committing changes") |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		repo, err := git.PlainOpen(directory) |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		head, err := repo.Head() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if tagString != "" { | 		if tagString != "" { | ||||||
| 			tag, err := tagcmp.Parse(tagString) | 			if err := createReleaseFromTag(recipe, tagString, mainAppVersion); err != nil { | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			if tag.MissingMinor { |  | ||||||
| 				tag.Minor = "0" |  | ||||||
| 				tag.MissingMinor = false |  | ||||||
| 			} |  | ||||||
| 			if tag.MissingPatch { |  | ||||||
| 				tag.Patch = "0" |  | ||||||
| 				tag.MissingPatch = false |  | ||||||
| 			} |  | ||||||
| 			tagString = fmt.Sprintf("%s+%s", tag.String(), mainAppVersion) |  | ||||||
| 			if internal.Dry { |  | ||||||
| 				hash := abraFormatter.SmallSHA(head.Hash().String()) |  | ||||||
| 				logrus.Info(fmt.Sprintf("dry run only: NOT creating tag %s at %s", tagString, hash)) |  | ||||||
| 				return nil |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			repo.CreateTag(tagString, head.Hash(), &createTagOptions) |  | ||||||
| 			hash := abraFormatter.SmallSHA(head.Hash().String()) |  | ||||||
| 			logrus.Info(fmt.Sprintf("created tag %s at %s", tagString, hash)) |  | ||||||
| 			if internal.Push && !internal.Dry { |  | ||||||
| 				if err := repo.Push(&git.PushOptions{}); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				logrus.Info(fmt.Sprintf("pushed tag %s to remote", tagString)) |  | ||||||
| 			} else { |  | ||||||
| 				logrus.Info("dry run only: NOT pushing changes") |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			return nil |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// get the latest tag with its hash, name etc | 		tags, err := recipe.Tags() | ||||||
| 		var lastGitTag tagcmp.Tag |  | ||||||
| 		iter, err := repo.Tags() |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 		if err := iter.ForEach(func(ref *plumbing.Reference) error { |  | ||||||
| 			obj, err := repo.TagObject(ref.Hash()) | 		if tagString == "" && (!internal.Major && !internal.Minor && !internal.Patch) { | ||||||
|  | 			var err error | ||||||
|  | 			tagString, err = getLabelVersion(recipe, false) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 			tagcmpTag, err := tagcmp.Parse(obj.Name) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 			if (lastGitTag == tagcmp.Tag{}) { |  | ||||||
| 				lastGitTag = tagcmpTag |  | ||||||
| 			} else if tagcmpTag.IsGreaterThan(lastGitTag) { |  | ||||||
| 				lastGitTag = tagcmpTag |  | ||||||
| 			} |  | ||||||
| 			return nil |  | ||||||
| 		}); err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		newTag := lastGitTag |  | ||||||
| 		var newtagString string |  | ||||||
| 		if bumpType > 0 { |  | ||||||
| 			if internal.Patch { |  | ||||||
| 				now, err := strconv.Atoi(newTag.Patch) |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				newTag.Patch = strconv.Itoa(now + 1) |  | ||||||
| 			} else if internal.Minor { |  | ||||||
| 				now, err := strconv.Atoi(newTag.Minor) |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				newTag.Patch = "0" |  | ||||||
| 				newTag.Minor = strconv.Itoa(now + 1) |  | ||||||
| 			} else if internal.Major { |  | ||||||
| 				now, err := strconv.Atoi(newTag.Major) |  | ||||||
| 				if err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 				newTag.Patch = "0" |  | ||||||
| 				newTag.Minor = "0" |  | ||||||
| 				newTag.Major = strconv.Itoa(now + 1) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		newTag.Metadata = mainAppVersion |  | ||||||
| 		newtagString = newTag.String() |  | ||||||
| 		if internal.Dry { |  | ||||||
| 			hash := abraFormatter.SmallSHA(head.Hash().String()) |  | ||||||
| 			logrus.Info(fmt.Sprintf("dry run only: NOT creating tag %s at %s", newtagString, hash)) |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		repo.CreateTag(newtagString, head.Hash(), &createTagOptions) |  | ||||||
| 		hash := abraFormatter.SmallSHA(head.Hash().String()) |  | ||||||
| 		logrus.Info(fmt.Sprintf("created tag %s at %s", newtagString, hash)) |  | ||||||
| 		if internal.Push && !internal.Dry { |  | ||||||
| 			if err := repo.Push(&git.PushOptions{}); err != nil { |  | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			logrus.Info(fmt.Sprintf("pushed tag %s to remote", newtagString)) | 		} | ||||||
|  |  | ||||||
|  | 		if len(tags) > 0 { | ||||||
|  | 			logrus.Warnf("previous git tags detected, assuming this is a new semver release") | ||||||
|  | 			if err := createReleaseFromPreviousTag(tagString, mainAppVersion, recipe, tags); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Info("gry run only: NOT pushing changes") | 			logrus.Warnf("no tag specified and no previous tag available for %s, assuming this is the initial release", recipe.Name) | ||||||
|  |  | ||||||
|  | 			if err := createReleaseFromTag(recipe, tagString, mainAppVersion); err != nil { | ||||||
|  | 				if cleanUpErr := cleanUpTag(tagString, recipe.Name); err != nil { | ||||||
|  | 					logrus.Fatal(cleanUpErr) | ||||||
|  | 				} | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| @ -320,7 +147,7 @@ func getImageVersions(recipe recipe.Recipe) (map[string]string, error) { | |||||||
| 		case reference.NamedTagged: | 		case reference.NamedTagged: | ||||||
| 			tag = img.(reference.NamedTagged).Tag() | 			tag = img.(reference.NamedTagged).Tag() | ||||||
| 		case reference.Named: | 		case reference.Named: | ||||||
| 			logrus.Fatalf("%s service is missing image tag?", path) | 			return services, fmt.Errorf("%s service is missing image tag?", path) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		services[path] = tag | 		services[path] = tag | ||||||
| @ -329,6 +156,50 @@ func getImageVersions(recipe recipe.Recipe) (map[string]string, error) { | |||||||
| 	return services, nil | 	return services, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // createReleaseFromTag creates a new release based on a supplied recipe version string | ||||||
|  | func createReleaseFromTag(recipe recipe.Recipe, tagString, mainAppVersion string) error { | ||||||
|  | 	var err error | ||||||
|  |  | ||||||
|  | 	directory := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
|  | 	repo, err := git.PlainOpen(directory) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tag, err := tagcmp.Parse(tagString) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if tag.MissingMinor { | ||||||
|  | 		tag.Minor = "0" | ||||||
|  | 		tag.MissingMinor = false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if tag.MissingPatch { | ||||||
|  | 		tag.Patch = "0" | ||||||
|  | 		tag.MissingPatch = false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if tagString == "" { | ||||||
|  | 		tagString = fmt.Sprintf("%s+%s", tag.String(), mainAppVersion) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := commitRelease(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := tagRelease(tagString, repo); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := pushRelease(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| // btoi converts a boolean value into an integer | // btoi converts a boolean value into an integer | ||||||
| func btoi(b bool) int { | func btoi(b bool) int { | ||||||
| 	if b { | 	if b { | ||||||
| @ -337,3 +208,242 @@ func btoi(b bool) int { | |||||||
|  |  | ||||||
| 	return 0 | 	return 0 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // getTagCreateOptions constructs git tag create options | ||||||
|  | func getTagCreateOptions(tag string) (git.CreateTagOptions, error) { | ||||||
|  | 	msg := fmt.Sprintf("chore: publish %s release", tag) | ||||||
|  | 	return git.CreateTagOptions{Message: msg}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func commitRelease(recipe recipe.Recipe, tag string) error { | ||||||
|  | 	if internal.Dry { | ||||||
|  | 		logrus.Debugf("dry run: no changes committed") | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	isClean, err := gitPkg.IsClean(recipe.Dir()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if isClean { | ||||||
|  | 		if !internal.Dry { | ||||||
|  | 			return fmt.Errorf("no changes discovered in %s, nothing to publish?", recipe.Dir()) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if internal.Publish { | ||||||
|  | 		msg := fmt.Sprintf("chore: publish %s release", tag) | ||||||
|  | 		repoPath := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
|  | 		if err := gitPkg.Commit(repoPath, "compose.**yml", msg, internal.Dry); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func tagRelease(tagString string, repo *git.Repository) error { | ||||||
|  | 	if internal.Dry { | ||||||
|  | 		logrus.Debugf("dry run: no git tag created (%s)", tagString) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	head, err := repo.Head() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	createTagOptions, err := getTagCreateOptions(tagString) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	_, err = repo.CreateTag(tagString, head.Hash(), &createTagOptions) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	hash := formatter.SmallSHA(head.Hash().String()) | ||||||
|  | 	logrus.Debugf(fmt.Sprintf("created tag %s at %s", tagString, hash)) | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func pushRelease(recipe recipe.Recipe, tagString string) error { | ||||||
|  | 	if internal.Dry { | ||||||
|  | 		logrus.Info("dry run: no changes published") | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !internal.Publish && !internal.NoInput { | ||||||
|  | 		prompt := &survey.Confirm{ | ||||||
|  | 			Message: "publish new release?", | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &internal.Publish); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if internal.Publish { | ||||||
|  | 		if err := recipe.Push(internal.Dry); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.Dry { | ||||||
|  | 			url := fmt.Sprintf("%s/%s/src/tag/%s", config.REPOS_BASE_URL, recipe.Name, tagString) | ||||||
|  | 			logrus.Infof("new release published: %s", url) | ||||||
|  | 		} else { | ||||||
|  | 			logrus.Info("dry run: no changes published") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func createReleaseFromPreviousTag(tagString, mainAppVersion string, recipe recipe.Recipe, tags []string) error { | ||||||
|  | 	directory := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
|  | 	repo, err := git.PlainOpen(directory) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bumpType := btoi(internal.Major)*4 + btoi(internal.Minor)*2 + btoi(internal.Patch) | ||||||
|  | 	if bumpType != 0 { | ||||||
|  | 		if (bumpType & (bumpType - 1)) != 0 { | ||||||
|  | 			return fmt.Errorf("you can only use one of: --major, --minor, --patch") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var lastGitTag tagcmp.Tag | ||||||
|  | 	if tagString == "" { | ||||||
|  | 		if err := internal.PromptBumpType(tagString); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tag := range tags { | ||||||
|  | 		parsed, err := tagcmp.Parse(tag) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (lastGitTag == tagcmp.Tag{}) { | ||||||
|  | 			lastGitTag = parsed | ||||||
|  | 		} else if parsed.IsGreaterThan(lastGitTag) { | ||||||
|  | 			lastGitTag = parsed | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	newTag := lastGitTag | ||||||
|  | 	if internal.Patch { | ||||||
|  | 		now, err := strconv.Atoi(newTag.Patch) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		newTag.Patch = strconv.Itoa(now + 1) | ||||||
|  | 	} else if internal.Minor { | ||||||
|  | 		now, err := strconv.Atoi(newTag.Minor) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		newTag.Patch = "0" | ||||||
|  | 		newTag.Minor = strconv.Itoa(now + 1) | ||||||
|  | 	} else if internal.Major { | ||||||
|  | 		now, err := strconv.Atoi(newTag.Major) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		newTag.Patch = "0" | ||||||
|  | 		newTag.Minor = "0" | ||||||
|  | 		newTag.Major = strconv.Itoa(now + 1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if internal.Major || internal.Minor || internal.Patch { | ||||||
|  | 		newTag.Metadata = mainAppVersion | ||||||
|  | 		tagString = newTag.String() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if lastGitTag.String() == tagString { | ||||||
|  | 		logrus.Fatalf("latest git tag (%s) and synced lable (%s) are the same?", lastGitTag, tagString) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !internal.NoInput { | ||||||
|  | 		prompt := &survey.Confirm{ | ||||||
|  | 			Message: fmt.Sprintf("current: %s, new: %s, correct?", lastGitTag, tagString), | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		var ok bool | ||||||
|  | 		if err := survey.AskOne(prompt, &ok); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !ok { | ||||||
|  | 			logrus.Fatal("exiting as requested") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := commitRelease(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := tagRelease(tagString, repo); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := pushRelease(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // cleanUpTag removes a freshly created tag | ||||||
|  | func cleanUpTag(tag, recipeName string) error { | ||||||
|  | 	directory := path.Join(config.RECIPES_DIR, recipeName) | ||||||
|  | 	repo, err := git.PlainOpen(directory) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := repo.DeleteTag(tag); err != nil { | ||||||
|  | 		if !strings.Contains(err.Error(), "not found") { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("removed freshly created tag %s", tag) | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func getLabelVersion(recipe recipe.Recipe, prompt bool) (string, error) { | ||||||
|  | 	initTag, err := recipePkg.GetVersionLabelLocal(recipe) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if initTag == "" { | ||||||
|  | 		logrus.Fatalf("unable to read version for %s from synced label. Did you try running \"abra recipe sync %s\" already?", recipe.Name, recipe.Name) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Warnf("discovered %s as currently synced recipe label", initTag) | ||||||
|  |  | ||||||
|  | 	if prompt && !internal.NoInput { | ||||||
|  | 		var response bool | ||||||
|  | 		prompt := &survey.Confirm{Message: fmt.Sprintf("use %s as the new version?", initTag)} | ||||||
|  | 		if err := survey.AskOne(prompt, &response); err != nil { | ||||||
|  | 			return "", err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !response { | ||||||
|  | 			return "", fmt.Errorf("please fix your synced label for %s and re-run this command", recipe.Name) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return initTag, nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ import ( | |||||||
| 	"strconv" | 	"strconv" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| @ -18,7 +18,7 @@ import ( | |||||||
|  |  | ||||||
| var recipeSyncCommand = &cli.Command{ | var recipeSyncCommand = &cli.Command{ | ||||||
| 	Name:      "sync", | 	Name:      "sync", | ||||||
| 	Usage:     "Ensure recipe version labels are up-to-date", | 	Usage:     "Sync recipe version label", | ||||||
| 	Aliases:   []string{"s"}, | 	Aliases:   []string{"s"}, | ||||||
| 	ArgsUsage: "<recipe> [<version>]", | 	ArgsUsage: "<recipe> [<version>]", | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| @ -29,28 +29,27 @@ var recipeSyncCommand = &cli.Command{ | |||||||
| 	}, | 	}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command will generate labels for the main recipe service (i.e. by | This command will generate labels for the main recipe service (i.e. by | ||||||
| convention, the service named "app") which corresponds to the following format: | convention, the service named 'app') which corresponds to the following format: | ||||||
|  |  | ||||||
|     coop-cloud.${STACK_NAME}.version=<version> |     coop-cloud.${STACK_NAME}.version=<version> | ||||||
|  |  | ||||||
| The <version> is determined by the recipe maintainer and is specified on the | Where <version> can be specifed on the command-line or Abra can attempt to | ||||||
| command-line. The <recipe> configuration will be updated on the local file | auto-generate it for you. The <recipe> configuration will be updated on the | ||||||
| system. | local file system. | ||||||
|  |  | ||||||
| You may invoke this command in "wizard" mode and be prompted for input: |  | ||||||
|  |  | ||||||
|     abra recipe sync |  | ||||||
|  |  | ||||||
| `, | `, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipe := internal.ValidateRecipeWithPrompt(c) | 		recipe := internal.ValidateRecipeWithPrompt(c) | ||||||
|  |  | ||||||
| 		mainApp := internal.GetMainApp(recipe) | 		mainApp, err := internal.GetMainAppImage(recipe) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		imagesTmp, err := getImageVersions(recipe) | 		imagesTmp, err := getImageVersions(recipe) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		mainAppVersion := imagesTmp[mainApp] | 		mainAppVersion := imagesTmp[mainApp] | ||||||
|  |  | ||||||
| 		tags, err := recipe.Tags() | 		tags, err := recipe.Tags() | ||||||
| @ -60,15 +59,35 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
|  |  | ||||||
| 		nextTag := c.Args().Get(1) | 		nextTag := c.Args().Get(1) | ||||||
| 		if len(tags) == 0 && nextTag == "" { | 		if len(tags) == 0 && nextTag == "" { | ||||||
| 			logrus.Warnf("no tags found for %s", recipe.Name) | 			logrus.Warnf("no git tags found for %s", recipe.Name) | ||||||
|  | 			fmt.Println(fmt.Sprintf(` | ||||||
|  | The following options are two types of initial semantic version that you can | ||||||
|  | pick for %s that will be published in the recipe catalogue. This follows the | ||||||
|  | semver convention (more on https://semver.org), here is a short cheatsheet | ||||||
|  |  | ||||||
|  |     0.1.0: development release, still hacking. when you make a major upgrade | ||||||
|  |            you increment the "y" part (i.e. 0.1.0 -> 0.2.0) and only move to | ||||||
|  |            using the "x" part when things are stable. | ||||||
|  |  | ||||||
|  |     1.0.0: public release, assumed to be working. you already have a stable | ||||||
|  |            and reliable deployment of this app and feel relatively confident | ||||||
|  |            about it. | ||||||
|  |  | ||||||
|  | If you want people to be able alpha test your current config for %s but don't | ||||||
|  | think it is quite reliable, go with 0.1.0 and people will know that things are | ||||||
|  | likely to change. | ||||||
|  |  | ||||||
|  | `, recipe.Name, recipe.Name)) | ||||||
| 			var chosenVersion string | 			var chosenVersion string | ||||||
| 			edPrompt := &survey.Select{ | 			edPrompt := &survey.Select{ | ||||||
| 				Message: "which version do you want to begin with?", | 				Message: "which version do you want to begin with?", | ||||||
| 				Options: []string{"0.1.0", "1.0.0"}, | 				Options: []string{"0.1.0", "1.0.0"}, | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if err := survey.AskOne(edPrompt, &chosenVersion); err != nil { | 			if err := survey.AskOne(edPrompt, &chosenVersion); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			nextTag = fmt.Sprintf("%s+%s", chosenVersion, mainAppVersion) | 			nextTag = fmt.Sprintf("%s+%s", chosenVersion, mainAppVersion) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -79,30 +98,35 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if nextTag == "" { | 		if nextTag == "" { | ||||||
| 			recipeDir := path.Join(config.APPS_DIR, recipe.Name) | 			recipeDir := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
| 			repo, err := git.PlainOpen(recipeDir) | 			repo, err := git.PlainOpen(recipeDir) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			var lastGitTag tagcmp.Tag | 			var lastGitTag tagcmp.Tag | ||||||
| 			iter, err := repo.Tags() | 			iter, err := repo.Tags() | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if err := iter.ForEach(func(ref *plumbing.Reference) error { | 			if err := iter.ForEach(func(ref *plumbing.Reference) error { | ||||||
| 				obj, err := repo.TagObject(ref.Hash()) | 				obj, err := repo.TagObject(ref.Hash()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				tagcmpTag, err := tagcmp.Parse(obj.Name) | 				tagcmpTag, err := tagcmp.Parse(obj.Name) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				if (lastGitTag == tagcmp.Tag{}) { | 				if (lastGitTag == tagcmp.Tag{}) { | ||||||
| 					lastGitTag = tagcmpTag | 					lastGitTag = tagcmpTag | ||||||
| 				} else if tagcmpTag.IsGreaterThan(lastGitTag) { | 				} else if tagcmpTag.IsGreaterThan(lastGitTag) { | ||||||
| 					lastGitTag = tagcmpTag | 					lastGitTag = tagcmpTag | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				return nil | 				return nil | ||||||
| 			}); err != nil { | 			}); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| @ -113,7 +137,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 			if bumpType != 0 { | 			if bumpType != 0 { | ||||||
| 				// a bitwise check if the number is a power of 2 | 				// a bitwise check if the number is a power of 2 | ||||||
| 				if (bumpType & (bumpType - 1)) != 0 { | 				if (bumpType & (bumpType - 1)) != 0 { | ||||||
| 					logrus.Fatal("you can only use one of: --major, --minor, --patch.") | 					logrus.Fatal("you can only use one version flag: --major, --minor or --patch") | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @ -124,12 +148,14 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 					if err != nil { | 					if err != nil { | ||||||
| 						logrus.Fatal(err) | 						logrus.Fatal(err) | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
| 					newTag.Patch = strconv.Itoa(now + 1) | 					newTag.Patch = strconv.Itoa(now + 1) | ||||||
| 				} else if internal.Minor { | 				} else if internal.Minor { | ||||||
| 					now, err := strconv.Atoi(newTag.Minor) | 					now, err := strconv.Atoi(newTag.Minor) | ||||||
| 					if err != nil { | 					if err != nil { | ||||||
| 						logrus.Fatal(err) | 						logrus.Fatal(err) | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
| 					newTag.Patch = "0" | 					newTag.Patch = "0" | ||||||
| 					newTag.Minor = strconv.Itoa(now + 1) | 					newTag.Minor = strconv.Itoa(now + 1) | ||||||
| 				} else if internal.Major { | 				} else if internal.Major { | ||||||
| @ -137,6 +163,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 					if err != nil { | 					if err != nil { | ||||||
| 						logrus.Fatal(err) | 						logrus.Fatal(err) | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
| 					newTag.Patch = "0" | 					newTag.Patch = "0" | ||||||
| 					newTag.Minor = "0" | 					newTag.Minor = "0" | ||||||
| 					newTag.Major = strconv.Itoa(now + 1) | 					newTag.Major = strconv.Itoa(now + 1) | ||||||
| @ -153,44 +180,16 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		mainService := "app" | 		mainService := "app" | ||||||
| 		var services []string |  | ||||||
| 		hasAppService := false |  | ||||||
| 		for _, service := range recipe.Config.Services { |  | ||||||
| 			services = append(services, service.Name) |  | ||||||
| 			if service.Name == "app" { |  | ||||||
| 				hasAppService = true |  | ||||||
| 				logrus.Debugf("detected app service in %s", recipe.Name) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if !hasAppService { |  | ||||||
| 			logrus.Fatalf("%s has no main 'app' service?", recipe.Name) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("selecting %s as the service to sync version label", mainService) |  | ||||||
|  |  | ||||||
| 		label := fmt.Sprintf("coop-cloud.${STACK_NAME}.version=%s", nextTag) | 		label := fmt.Sprintf("coop-cloud.${STACK_NAME}.version=%s", nextTag) | ||||||
| 		if !internal.Dry { | 		if !internal.Dry { | ||||||
| 			if err := recipe.UpdateLabel(mainService, label); err != nil { | 			if err := recipe.UpdateLabel("compose.y*ml", mainService, label); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			logrus.Infof("synced label '%s' to service '%s'", label, mainService) |  | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Infof("dry run only: NOT syncing label %s for recipe %s", nextTag, recipe.Name) | 			logrus.Infof("dry run: not syncing label %s for recipe %s", nextTag, recipe.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| 	BashComplete: func(c *cli.Context) { | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 		catl, err := catalogue.ReadRecipeCatalogue() |  | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Warn(err) |  | ||||||
| 		} |  | ||||||
| 		if c.NArg() > 0 { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		for name := range catl { |  | ||||||
| 			fmt.Println(name) |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -9,9 +9,10 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
| @ -37,12 +38,17 @@ Some image tags cannot be parsed because they do not follow some sort of | |||||||
| semver-like convention. In this case, all possible tags will be listed and it | semver-like convention. In this case, all possible tags will be listed and it | ||||||
| is up to the end-user to decide. | is up to the end-user to decide. | ||||||
|  |  | ||||||
|  | The command is interactive and will show a select input which allows you to | ||||||
|  | make a seclection. Use the "?" key to see more help on navigating this | ||||||
|  | interface. | ||||||
|  |  | ||||||
| You may invoke this command in "wizard" mode and be prompted for input: | You may invoke this command in "wizard" mode and be prompted for input: | ||||||
|  |  | ||||||
|     abra recipe upgrade |     abra recipe upgrade | ||||||
|  |  | ||||||
| `, | `, | ||||||
| 	ArgsUsage: "<recipe>", | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
|  | 	ArgsUsage:    "<recipe>", | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.PatchFlag, | 		internal.PatchFlag, | ||||||
| 		internal.MinorFlag, | 		internal.MinorFlag, | ||||||
| @ -61,7 +67,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
|  |  | ||||||
| 		// check for versions file and load pinned versions | 		// check for versions file and load pinned versions | ||||||
| 		versionsPresent := false | 		versionsPresent := false | ||||||
| 		recipeDir := path.Join(config.ABRA_DIR, "apps", recipe.Name) | 		recipeDir := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
| 		versionsPath := path.Join(recipeDir, "versions") | 		versionsPath := path.Join(recipeDir, "versions") | ||||||
| 		var servicePins = make(map[string]imgPin) | 		var servicePins = make(map[string]imgPin) | ||||||
| 		if _, err := os.Stat(versionsPath); err == nil { | 		if _, err := os.Stat(versionsPath); err == nil { | ||||||
| @ -97,11 +103,6 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		for _, service := range recipe.Config.Services { | 		for _, service := range recipe.Config.Services { | ||||||
| 			catlVersions, err := catalogue.VersionsOfService(recipe.Name, service.Name) |  | ||||||
| 			if err != nil { |  | ||||||
| 				logrus.Fatal(err) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			img, err := reference.ParseNormalizedNamed(service.Image) | 			img, err := reference.ParseNormalizedNamed(service.Image) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| @ -112,7 +113,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			logrus.Debugf("retrieved '%s' from remote registry for '%s'", regVersions, image) | 			logrus.Debugf("retrieved %s from remote registry for %s", regVersions, image) | ||||||
|  |  | ||||||
| 			if strings.Contains(image, "library") { | 			if strings.Contains(image, "library") { | ||||||
| 				// ParseNormalizedNamed prepends 'library' to images like nginx:<tag>, | 				// ParseNormalizedNamed prepends 'library' to images like nginx:<tag>, | ||||||
| @ -122,7 +123,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 			} | 			} | ||||||
| 			semverLikeTag := true | 			semverLikeTag := true | ||||||
| 			if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) { | 			if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) { | ||||||
| 				logrus.Debugf("'%s' not considered semver-like", img.(reference.NamedTagged).Tag()) | 				logrus.Debugf("%s not considered semver-like", img.(reference.NamedTagged).Tag()) | ||||||
| 				semverLikeTag = false | 				semverLikeTag = false | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @ -130,7 +131,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 			if err != nil && semverLikeTag { | 			if err != nil && semverLikeTag { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			logrus.Debugf("parsed '%s' for '%s'", tag, service.Name) | 			logrus.Debugf("parsed %s for %s", tag, service.Name) | ||||||
| 			var compatible []tagcmp.Tag | 			var compatible []tagcmp.Tag | ||||||
| 			for _, regVersion := range regVersions { | 			for _, regVersion := range regVersions { | ||||||
| 				other, err := tagcmp.Parse(regVersion.Name) | 				other, err := tagcmp.Parse(regVersion.Name) | ||||||
| @ -143,16 +144,21 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			logrus.Debugf("detected potential upgradable tags '%s' for '%s'", compatible, service.Name) | 			logrus.Debugf("detected potential upgradable tags %s for %s", compatible, service.Name) | ||||||
|  |  | ||||||
| 			sort.Sort(tagcmp.ByTagDesc(compatible)) | 			sort.Sort(tagcmp.ByTagDesc(compatible)) | ||||||
|  |  | ||||||
| 			if len(compatible) == 0 && semverLikeTag { | 			if len(compatible) == 0 && semverLikeTag { | ||||||
| 				logrus.Info(fmt.Sprintf("no new versions available for '%s', '%s' is the latest", image, tag)) | 				logrus.Info(fmt.Sprintf("no new versions available for %s, %s is the latest", image, tag)) | ||||||
| 				continue // skip on to the next tag and don't update any compose files | 				continue // skip on to the next tag and don't update any compose files | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			var compatibleStrings []string | 			catlVersions, err := recipePkg.VersionsOfService(recipe.Name, service.Name) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			compatibleStrings := []string{"skip"} | ||||||
| 			for _, compat := range compatible { | 			for _, compat := range compatible { | ||||||
| 				skip := false | 				skip := false | ||||||
| 				for _, catlVersion := range catlVersions { | 				for _, catlVersion := range catlVersions { | ||||||
| @ -165,7 +171,7 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			logrus.Debugf("detected compatible upgradable tags '%s' for '%s'", compatibleStrings, service.Name) | 			logrus.Debugf("detected compatible upgradable tags %s for %s", compatibleStrings, service.Name) | ||||||
|  |  | ||||||
| 			var upgradeTag string | 			var upgradeTag string | ||||||
| 			_, ok := servicePins[service.Name] | 			_, ok := servicePins[service.Name] | ||||||
| @ -205,14 +211,14 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 					if upgradeTag == "" { | 					if upgradeTag == "" { | ||||||
| 						logrus.Warnf("not upgrading from '%s' to '%s' for '%s', because the upgrade type is more serious than what user wants.", tag.String(), compatible[0].String(), image) | 						logrus.Warnf("not upgrading from %s to %s for %s, because the upgrade type is more serious than what user wants.", tag.String(), compatible[0].String(), image) | ||||||
| 						continue | 						continue | ||||||
| 					} | 					} | ||||||
| 				} else { | 				} else { | ||||||
| 					msg := fmt.Sprintf("upgrade to which tag? (service: %s, tag: %s)", service.Name, tag) | 					msg := fmt.Sprintf("upgrade to which tag? (service: %s, image: %s, tag: %s)", service.Name, image, tag) | ||||||
| 					if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) { | 					if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) { | ||||||
| 						tag := img.(reference.NamedTagged).Tag() | 						tag := img.(reference.NamedTagged).Tag() | ||||||
| 						logrus.Warning(fmt.Sprintf("unable to determine versioning semantics of '%s', listing all tags", tag)) | 						logrus.Warning(fmt.Sprintf("unable to determine versioning semantics of %s, listing all tags", tag)) | ||||||
| 						msg = fmt.Sprintf("upgrade to which tag? (service: %s, tag: %s)", service.Name, tag) | 						msg = fmt.Sprintf("upgrade to which tag? (service: %s, tag: %s)", service.Name, tag) | ||||||
| 						compatibleStrings = []string{} | 						compatibleStrings = []string{} | ||||||
| 						for _, regVersion := range regVersions { | 						for _, regVersion := range regVersions { | ||||||
| @ -222,6 +228,8 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
|  |  | ||||||
| 					prompt := &survey.Select{ | 					prompt := &survey.Select{ | ||||||
| 						Message: msg, | 						Message: msg, | ||||||
|  | 						Help:    "enter / return to confirm, choose 'skip' to not upgrade this tag, vim mode is enabled", | ||||||
|  | 						VimMode: true, | ||||||
| 						Options: compatibleStrings, | 						Options: compatibleStrings, | ||||||
| 					} | 					} | ||||||
| 					if err := survey.AskOne(prompt, &upgradeTag); err != nil { | 					if err := survey.AskOne(prompt, &upgradeTag); err != nil { | ||||||
| @ -229,10 +237,14 @@ You may invoke this command in "wizard" mode and be prompted for input: | |||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			if err := recipe.UpdateTag(image, upgradeTag); err != nil { | 			if upgradeTag != "skip" { | ||||||
| 				logrus.Fatal(err) | 				if err := recipe.UpdateTag(image, upgradeTag); err != nil { | ||||||
|  | 					logrus.Fatal(err) | ||||||
|  | 				} | ||||||
|  | 				logrus.Infof("tag upgraded from %s to %s for %s", tag.String(), upgradeTag, image) | ||||||
|  | 			} else { | ||||||
|  | 				logrus.Warnf("not upgrading %s, skipping as requested", image) | ||||||
| 			} | 			} | ||||||
| 			logrus.Infof("tag upgraded from '%s' to '%s' for '%s'", tag.String(), upgradeTag, image) |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
|  | |||||||
| @ -1,36 +1,49 @@ | |||||||
| package recipe | package recipe | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"coopcloud.tech/abra/cli/formatter" | 	"fmt" | ||||||
|  | 	"path" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/catalogue" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	gitPkg "coopcloud.tech/abra/pkg/git" | ||||||
|  | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeVersionCommand = &cli.Command{ | var recipeVersionCommand = &cli.Command{ | ||||||
| 	Name:      "versions", | 	Name:         "versions", | ||||||
| 	Usage:     "List recipe versions", | 	Usage:        "List recipe versions", | ||||||
| 	Aliases:   []string{"v"}, | 	Aliases:      []string{"v"}, | ||||||
| 	ArgsUsage: "<recipe>", | 	ArgsUsage:    "<recipe>", | ||||||
|  | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipe := internal.ValidateRecipe(c) | 		recipe := internal.ValidateRecipe(c) | ||||||
|  |  | ||||||
| 		catalogue, err := catalogue.ReadRecipeCatalogue() | 		catalogueDir := path.Join(config.ABRA_DIR, "catalogue") | ||||||
|  | 		url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, "recipes") | ||||||
|  | 		if err := gitPkg.Clone(catalogueDir, url); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		catalogue, err := recipePkg.ReadRecipeCatalogue() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		recipeMeta, ok := catalogue[recipe.Name] | 		recipeMeta, ok := catalogue[recipe.Name] | ||||||
| 		if !ok { | 		if !ok { | ||||||
| 			logrus.Fatalf("'%s' recipe doesn't exist?", recipe.Name) | 			logrus.Fatalf("%s recipe doesn't exist?", recipe.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"Version", "Service", "Image", "Tag", "Digest"} | 		tableCol := []string{"Version", "Service", "Image", "Tag", "Digest"} | ||||||
| 		table := formatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 		for _, serviceVersion := range recipeMeta.Versions { | 		for i := len(recipeMeta.Versions) - 1; i >= 0; i-- { | ||||||
| 			for tag, meta := range serviceVersion { | 			for tag, meta := range recipeMeta.Versions[i] { | ||||||
| 				for service, serviceMeta := range meta { | 				for service, serviceMeta := range meta { | ||||||
| 					table.Append([]string{tag, service, serviceMeta.Image, serviceMeta.Tag, serviceMeta.Digest}) | 					table.Append([]string{tag, service, serviceMeta.Image, serviceMeta.Tag, serviceMeta.Digest}) | ||||||
| 				} | 				} | ||||||
| @ -38,7 +51,12 @@ var recipeVersionCommand = &cli.Command{ | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		table.SetAutoMergeCells(true) | 		table.SetAutoMergeCells(true) | ||||||
| 		table.Render() |  | ||||||
|  | 		if table.NumLines() > 0 { | ||||||
|  | 			table.Render() | ||||||
|  | 		} else { | ||||||
|  | 			logrus.Fatalf("%s has no published versions?", recipe.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
|  | |||||||
| @ -4,9 +4,9 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"strconv" | 	"strconv" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/libdns/gandi" | 	"github.com/libdns/gandi" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| @ -46,7 +46,7 @@ are listed. This zone must already be created on your provider account. | |||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		default: | 		default: | ||||||
| 			logrus.Fatalf("'%s' is not a supported DNS provider", internal.DNSProvider) | 			logrus.Fatalf("%s is not a supported DNS provider", internal.DNSProvider) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		records, err := provider.GetRecords(c.Context, zone) | 		records, err := provider.GetRecords(c.Context, zone) | ||||||
| @ -55,7 +55,7 @@ are listed. This zone must already be created on your provider account. | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 		for _, record := range records { | 		for _, record := range records { | ||||||
| 			value := record.Value | 			value := record.Value | ||||||
|  | |||||||
| @ -3,11 +3,11 @@ package record | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/dns" | ||||||
| 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/libdns/gandi" | 	"github.com/libdns/gandi" | ||||||
| 	"github.com/libdns/libdns" | 	"github.com/libdns/libdns" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| @ -27,6 +27,7 @@ var RecordNewCommand = &cli.Command{ | |||||||
| 		internal.DNSValueFlag, | 		internal.DNSValueFlag, | ||||||
| 		internal.DNSTTLFlag, | 		internal.DNSTTLFlag, | ||||||
| 		internal.DNSPriorityFlag, | 		internal.DNSPriorityFlag, | ||||||
|  | 		internal.AutoDNSRecordFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command creates a new domain name record for a specific zone. | This command creates a new domain name record for a specific zone. | ||||||
| @ -38,9 +39,16 @@ Example: | |||||||
|  |  | ||||||
|     abra record new foo.com -p gandi -t A -n myapp -v 192.168.178.44 |     abra record new foo.com -p gandi -t A -n myapp -v 192.168.178.44 | ||||||
|  |  | ||||||
|  | Typically, you need two records, an A record which points at the zone (@.) and | ||||||
|  | a wildcard record for your apps (*.). Pass "--auto" to have Abra automatically | ||||||
|  | set this up. | ||||||
|  |  | ||||||
|  |     abra record new --auto foo.com -p gandi -v 192.168.178.44 | ||||||
|  |  | ||||||
| You may also invoke this command in "wizard" mode and be prompted for input | You may also invoke this command in "wizard" mode and be prompted for input | ||||||
|  |  | ||||||
|     abra record new |     abra record new | ||||||
|  |  | ||||||
| `, | `, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		zone, err := internal.EnsureZoneArgument(c) | 		zone, err := internal.EnsureZoneArgument(c) | ||||||
| @ -60,7 +68,26 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		default: | 		default: | ||||||
| 			logrus.Fatalf("'%s' is not a supported DNS provider", internal.DNSProvider) | 			logrus.Fatalf("%s is not a supported DNS provider", internal.DNSProvider) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if internal.AutoDNSRecord { | ||||||
|  | 			ipv4, err := dns.EnsureIPv4(zone) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Debugf("no ipv4 associated with %s, prompting for input", zone) | ||||||
|  | 				if err := internal.EnsureDNSValueFlag(c); err != nil { | ||||||
|  | 					logrus.Fatal(err) | ||||||
|  | 				} | ||||||
|  | 				ipv4 = internal.DNSValue | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			logrus.Infof("automatically configuring @./*. A records for %s for %s (--auto)", zone, ipv4) | ||||||
|  |  | ||||||
|  | 			if err := autoConfigure(c, &provider, zone, ipv4); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			return nil | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := internal.EnsureDNSTypeFlag(c); err != nil { | 		if err := internal.EnsureDNSTypeFlag(c); err != nil { | ||||||
| @ -75,11 +102,16 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		ttl, err := dns.GetTTL(internal.DNSTTL) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		record := libdns.Record{ | 		record := libdns.Record{ | ||||||
| 			Type:  internal.DNSType, | 			Type:  internal.DNSType, | ||||||
| 			Name:  internal.DNSName, | 			Name:  internal.DNSName, | ||||||
| 			Value: internal.DNSValue, | 			Value: internal.DNSValue, | ||||||
| 			TTL:   time.Duration(internal.DNSTTL), | 			TTL:   ttl, | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if internal.DNSType == "MX" || internal.DNSType == "SRV" || internal.DNSType == "URI" { | 		if internal.DNSType == "MX" || internal.DNSType == "SRV" || internal.DNSType == "URI" { | ||||||
| @ -95,7 +127,7 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 			if existingRecord.Type == record.Type && | 			if existingRecord.Type == record.Type && | ||||||
| 				existingRecord.Name == record.Name && | 				existingRecord.Name == record.Name && | ||||||
| 				existingRecord.Value == record.Value { | 				existingRecord.Value == record.Value { | ||||||
| 				logrus.Fatal("provider library reports that this record already exists?") | 				logrus.Fatalf("%s record for %s already exists?", record.Type, zone) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -104,6 +136,9 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 			zone, | 			zone, | ||||||
| 			[]libdns.Record{record}, | 			[]libdns.Record{record}, | ||||||
| 		) | 		) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		if len(createdRecords) == 0 { | 		if len(createdRecords) == 0 { | ||||||
| 			logrus.Fatal("provider library reports that no record was created?") | 			logrus.Fatal("provider library reports that no record was created?") | ||||||
| @ -112,7 +147,7 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 		createdRecord := createdRecords[0] | 		createdRecord := createdRecords[0] | ||||||
|  |  | ||||||
| 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 		value := createdRecord.Value | 		value := createdRecord.Value | ||||||
| 		if len(createdRecord.Value) > 30 { | 		if len(createdRecord.Value) > 30 { | ||||||
| @ -134,3 +169,84 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func autoConfigure(c *cli.Context, provider *gandi.Provider, zone, ipv4 string) error { | ||||||
|  | 	ttl, err := dns.GetTTL(internal.DNSTTL) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	atRecord := libdns.Record{ | ||||||
|  | 		Type:  "A", | ||||||
|  | 		Name:  "@", | ||||||
|  | 		Value: ipv4, | ||||||
|  | 		TTL:   ttl, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	wildcardRecord := libdns.Record{ | ||||||
|  | 		Type:  "A", | ||||||
|  | 		Name:  "*", | ||||||
|  | 		Value: ipv4, | ||||||
|  | 		TTL:   ttl, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	records := []libdns.Record{atRecord, wildcardRecord} | ||||||
|  |  | ||||||
|  | 	tableCol := []string{"type", "name", "value", "TTL", "priority"} | ||||||
|  | 	table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
|  | 	for _, record := range records { | ||||||
|  | 		existingRecords, err := provider.GetRecords(c.Context, zone) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		discovered := false | ||||||
|  | 		for _, existingRecord := range existingRecords { | ||||||
|  | 			if existingRecord.Type == record.Type && | ||||||
|  | 				existingRecord.Name == record.Name && | ||||||
|  | 				existingRecord.Value == record.Value { | ||||||
|  | 				logrus.Warnf("%s record: %s %s for %s already exists?", record.Type, record.Name, record.Value, zone) | ||||||
|  | 				discovered = true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if discovered { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		createdRecords, err := provider.SetRecords( | ||||||
|  | 			c.Context, | ||||||
|  | 			zone, | ||||||
|  | 			[]libdns.Record{record}, | ||||||
|  | 		) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if len(createdRecords) == 0 { | ||||||
|  | 			return fmt.Errorf("provider library reports that no record was created?") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		createdRecord := createdRecords[0] | ||||||
|  |  | ||||||
|  | 		value := createdRecord.Value | ||||||
|  | 		if len(createdRecord.Value) > 30 { | ||||||
|  | 			value = fmt.Sprintf("%s...", createdRecord.Value[:30]) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		table.Append([]string{ | ||||||
|  | 			createdRecord.Type, | ||||||
|  | 			createdRecord.Name, | ||||||
|  | 			value, | ||||||
|  | 			createdRecord.TTL.String(), | ||||||
|  | 			strconv.Itoa(createdRecord.Priority), | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if table.NumLines() > 0 { | ||||||
|  | 		table.Render() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -7,7 +7,7 @@ import ( | |||||||
| // RecordCommand supports managing DNS entries. | // RecordCommand supports managing DNS entries. | ||||||
| var RecordCommand = &cli.Command{ | var RecordCommand = &cli.Command{ | ||||||
| 	Name:      "record", | 	Name:      "record", | ||||||
| 	Usage:     "Manage domain name records via 3rd party providers", | 	Usage:     "Manage domain name records", | ||||||
| 	Aliases:   []string{"rc"}, | 	Aliases:   []string{"rc"}, | ||||||
| 	ArgsUsage: "<record>", | 	ArgsUsage: "<record>", | ||||||
| 	Description: ` | 	Description: ` | ||||||
|  | |||||||
| @ -4,9 +4,9 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"strconv" | 	"strconv" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | 	gandiPkg "coopcloud.tech/abra/pkg/dns/gandi" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/libdns/gandi" | 	"github.com/libdns/gandi" | ||||||
| 	"github.com/libdns/libdns" | 	"github.com/libdns/libdns" | ||||||
| @ -59,7 +59,7 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		default: | 		default: | ||||||
| 			logrus.Fatalf("'%s' is not a supported DNS provider", internal.DNSProvider) | 			logrus.Fatalf("%s is not a supported DNS provider", internal.DNSProvider) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := internal.EnsureDNSTypeFlag(c); err != nil { | 		if err := internal.EnsureDNSTypeFlag(c); err != nil { | ||||||
| @ -88,7 +88,7 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | 		tableCol := []string{"type", "name", "value", "TTL", "priority"} | ||||||
| 		table := abraFormatter.CreateTable(tableCol) | 		table := formatter.CreateTable(tableCol) | ||||||
|  |  | ||||||
| 		value := toDelete.Value | 		value := toDelete.Value | ||||||
| 		if len(toDelete.Value) > 30 { | 		if len(toDelete.Value) > 30 { | ||||||
| @ -105,17 +105,19 @@ You may also invoke this command in "wizard" mode and be prompted for input | |||||||
|  |  | ||||||
| 		table.Render() | 		table.Render() | ||||||
|  |  | ||||||
| 		response := false | 		if !internal.NoInput { | ||||||
| 		prompt := &survey.Confirm{ | 			response := false | ||||||
| 			Message: "continue with record deletion?", | 			prompt := &survey.Confirm{ | ||||||
| 		} | 				Message: "continue with record deletion?", | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 		if err := survey.AskOne(prompt, &response); err != nil { | 			if err := survey.AskOne(prompt, &response); err != nil { | ||||||
| 			return err | 				return err | ||||||
| 		} | 			} | ||||||
|  |  | ||||||
| 		if !response { | 			if !response { | ||||||
| 			logrus.Fatal("exiting as requested") | 				logrus.Fatal("exiting as requested") | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		_, err = provider.DeleteRecords(c.Context, zone, []libdns.Record{toDelete}) | 		_, err = provider.DeleteRecords(c.Context, zone, []libdns.Record{toDelete}) | ||||||
|  | |||||||
| @ -10,7 +10,6 @@ import ( | |||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	abraFormatter "coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| @ -98,7 +97,7 @@ func cleanUp(domainName string) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Warnf("cleaning up server directory for %s", domainName) | 	logrus.Warnf("cleaning up server directory for %s", domainName) | ||||||
| 	if err := os.RemoveAll(filepath.Join(config.ABRA_SERVER_FOLDER, domainName)); err != nil { | 	if err := os.RemoveAll(filepath.Join(config.SERVERS_DIR, domainName)); err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -117,7 +116,17 @@ func installDockerLocal(c *cli.Context) error { | |||||||
| 		logrus.Fatal("exiting as requested") | 		logrus.Fatal("exiting as requested") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	cmd := exec.Command("bash", "-c", "curl -s https://get.docker.com | bash") | 	for _, exe := range []string{"wget", "bash"} { | ||||||
|  | 		exists, err := ensureLocalExecutable(exe) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if !exists { | ||||||
|  | 			return fmt.Errorf("%s missing, please install it", exe) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cmd := exec.Command("bash", "-c", "wget -O- https://get.docker.com | bash") | ||||||
| 	if err := internal.RunCmd(cmd); err != nil { | 	if err := internal.RunCmd(cmd); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| @ -136,15 +145,17 @@ func newLocalServer(c *cli.Context, domainName string) error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if provision { | 	if provision { | ||||||
| 		out, err := exec.Command("which", "docker").Output() | 		exists, err := ensureLocalExecutable("docker") | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		if string(out) == "" { |  | ||||||
|  | 		if !exists { | ||||||
| 			if err := installDockerLocal(c); err != nil { | 			if err := installDockerLocal(c); err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := initSwarmLocal(c, cl, domainName); err != nil { | 		if err := initSwarmLocal(c, cl, domainName); err != nil { | ||||||
| 			if !strings.Contains(err.Error(), "proxy already exists") { | 			if !strings.Contains(err.Error(), "proxy already exists") { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| @ -195,59 +206,127 @@ func newClient(c *cli.Context, domainName string) (*dockerClient.Client, error) | |||||||
| } | } | ||||||
|  |  | ||||||
| func installDocker(c *cli.Context, cl *dockerClient.Client, sshCl *ssh.Client, domainName string) error { | func installDocker(c *cli.Context, cl *dockerClient.Client, sshCl *ssh.Client, domainName string) error { | ||||||
| 	result, err := sshCl.Exec("which docker") | 	exists, err := ensureRemoteExecutable("docker", sshCl) | ||||||
| 	if err != nil && string(result) != "" { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if string(result) == "" { | 	if !exists { | ||||||
| 		fmt.Println(fmt.Sprintf(dockerInstallMsg, domainName)) | 		fmt.Println(fmt.Sprintf(dockerInstallMsg, domainName)) | ||||||
|  |  | ||||||
| 		response := false | 		response := false | ||||||
| 		prompt := &survey.Confirm{ | 		prompt := &survey.Confirm{ | ||||||
| 			Message: fmt.Sprintf("attempt install docker on %s?", domainName), | 			Message: fmt.Sprintf("attempt install docker on %s?", domainName), | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := survey.AskOne(prompt, &response); err != nil { | 		if err := survey.AskOne(prompt, &response); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !response { | 		if !response { | ||||||
| 			logrus.Fatal("exiting as requested") | 			logrus.Fatal("exiting as requested") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		cmd := "curl -s https://get.docker.com | bash" | 		exes := []string{"wget", "bash"} | ||||||
|  | 		if askSudoPass { | ||||||
|  | 			exes = append(exes, "ssh-askpass") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for _, exe := range exes { | ||||||
|  | 			exists, err := ensureRemoteExecutable(exe, sshCl) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if !exists { | ||||||
|  | 				return fmt.Errorf("%s missing on remote, please install it", exe) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		var sudoPass string | 		var sudoPass string | ||||||
| 		if askSudoPass { | 		if askSudoPass { | ||||||
|  | 			cmd := "wget -O- https://get.docker.com | bash" | ||||||
|  |  | ||||||
| 			prompt := &survey.Password{ | 			prompt := &survey.Password{ | ||||||
| 				Message: "sudo password?", | 				Message: "sudo password?", | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if err := survey.AskOne(prompt, &sudoPass); err != nil { | 			if err := survey.AskOne(prompt, &sudoPass); err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 			logrus.Debugf("running '%s' on %s now with sudo password", cmd, domainName) |  | ||||||
|  | 			logrus.Debugf("running %s on %s now with sudo password", cmd, domainName) | ||||||
|  |  | ||||||
|  | 			if sudoPass == "" { | ||||||
|  | 				return fmt.Errorf("missing sudo password but requested --ask-sudo-pass?") | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			logrus.Warn("installing docker, this could take some time...") | ||||||
|  |  | ||||||
| 			if err := ssh.RunSudoCmd(cmd, sudoPass, sshCl); err != nil { | 			if err := ssh.RunSudoCmd(cmd, sudoPass, sshCl); err != nil { | ||||||
|  | 				fmt.Print(fmt.Sprintf(` | ||||||
|  | Abra was unable to bootstrap Docker, see below for logs: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | %s | ||||||
|  |  | ||||||
|  | If nothing works, you try running the Docker install script manually on your server: | ||||||
|  |  | ||||||
|  |     wget -O- https://get.docker.com | bash | ||||||
|  |  | ||||||
|  | `, string(err.Error()))) | ||||||
|  | 				logrus.Fatal("Process exited with status 1") | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			logrus.Infof("docker is installed on %s", domainName) | ||||||
|  |  | ||||||
|  | 			remoteUser := sshCl.SSHClient.Conn.User() | ||||||
|  | 			logrus.Infof("adding %s to docker group", remoteUser) | ||||||
|  | 			permsCmd := fmt.Sprintf("sudo usermod -aG docker %s", remoteUser) | ||||||
|  | 			if err := ssh.RunSudoCmd(permsCmd, sudoPass, sshCl); err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			logrus.Debugf("running '%s' on %s now without sudo password", cmd, domainName) | 			cmd := "wget -O- https://get.docker.com | bash" | ||||||
| 			if err := ssh.Exec(cmd, sshCl); err != nil { |  | ||||||
| 				return err | 			logrus.Debugf("running %s on %s now without sudo password", cmd, domainName) | ||||||
|  |  | ||||||
|  | 			logrus.Warn("installing docker, this could take some time...") | ||||||
|  |  | ||||||
|  | 			if out, err := sshCl.Exec(cmd); err != nil { | ||||||
|  | 				fmt.Print(fmt.Sprintf(` | ||||||
|  | Abra was unable to bootstrap Docker, see below for logs: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | %s | ||||||
|  |  | ||||||
|  | This could be due to a number of things but one of the most common is that your | ||||||
|  | server user account does not have sudo access, and if it does, you need to pass | ||||||
|  | "--ask-sudo-pass" in order to supply Abra with your password. | ||||||
|  |  | ||||||
|  | If nothing works, you try running the Docker install script manually on your server: | ||||||
|  |  | ||||||
|  |     wget -O- https://get.docker.com | bash | ||||||
|  |  | ||||||
|  | `, string(out))) | ||||||
|  | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			logrus.Infof("docker is installed on %s", domainName) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Infof("docker is installed on %s", domainName) |  | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func initSwarmLocal(c *cli.Context, cl *dockerClient.Client, domainName string) error { | func initSwarmLocal(c *cli.Context, cl *dockerClient.Client, domainName string) error { | ||||||
| 	initReq := swarm.InitRequest{ListenAddr: "0.0.0.0:2377"} | 	initReq := swarm.InitRequest{ListenAddr: "0.0.0.0:2377"} | ||||||
| 	if _, err := cl.SwarmInit(c.Context, initReq); err != nil { | 	if _, err := cl.SwarmInit(c.Context, initReq); err != nil { | ||||||
| 		if !strings.Contains(err.Error(), "is already part of a swarm") { | 		if strings.Contains(err.Error(), "is already part of a swarm") || | ||||||
|  | 			strings.Contains(err.Error(), "must specify a listening address") { | ||||||
|  | 			logrus.Infof("swarm mode already initialised on %s", domainName) | ||||||
|  | 		} else { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		logrus.Info("swarm mode already initialised on local server") |  | ||||||
| 	} else { | 	} else { | ||||||
| 		logrus.Infof("initialised swarm mode on local server") | 		logrus.Infof("initialised swarm mode on local server") | ||||||
| 	} | 	} | ||||||
| @ -276,10 +355,12 @@ func initSwarm(c *cli.Context, cl *dockerClient.Client, domainName string) error | |||||||
| 		AdvertiseAddr: ipv4, | 		AdvertiseAddr: ipv4, | ||||||
| 	} | 	} | ||||||
| 	if _, err := cl.SwarmInit(c.Context, initReq); err != nil { | 	if _, err := cl.SwarmInit(c.Context, initReq); err != nil { | ||||||
| 		if !strings.Contains(err.Error(), "is already part of a swarm") { | 		if strings.Contains(err.Error(), "is already part of a swarm") || | ||||||
|  | 			strings.Contains(err.Error(), "must specify a listening address") { | ||||||
|  | 			logrus.Infof("swarm mode already initialised on %s", domainName) | ||||||
|  | 		} else { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		logrus.Infof("swarm mode already initialised on %s", domainName) |  | ||||||
| 	} else { | 	} else { | ||||||
| 		logrus.Infof("initialised swarm mode on %s", domainName) | 		logrus.Infof("initialised swarm mode on %s", domainName) | ||||||
| 	} | 	} | ||||||
| @ -316,16 +397,8 @@ func deployTraefik(c *cli.Context, cl *dockerClient.Client, domainName string) e | |||||||
| 	internal.NewAppName = fmt.Sprintf("%s_%s", "traefik", config.SanitiseAppName(domainName)) | 	internal.NewAppName = fmt.Sprintf("%s_%s", "traefik", config.SanitiseAppName(domainName)) | ||||||
|  |  | ||||||
| 	appEnvPath := path.Join(config.ABRA_DIR, "servers", internal.Domain, fmt.Sprintf("%s.env", internal.NewAppName)) | 	appEnvPath := path.Join(config.ABRA_DIR, "servers", internal.Domain, fmt.Sprintf("%s.env", internal.NewAppName)) | ||||||
| 	if _, err := os.Stat(appEnvPath); !os.IsNotExist(err) { | 	if _, err := os.Stat(appEnvPath); os.IsNotExist(err) { | ||||||
| 		fmt.Println(fmt.Sprintf(` | 		logrus.Info(fmt.Sprintf("-t/--traefik specified, automatically deploying traefik to %s", internal.NewAppServer)) | ||||||
| 	You specified "--traefik/-t" and that means that Abra will now try to |  | ||||||
| 	automatically create a new Traefik app on %s. |  | ||||||
| 	`, internal.NewAppServer)) |  | ||||||
|  |  | ||||||
| 		tableCol := []string{"recipe", "domain", "server", "name"} |  | ||||||
| 		table := abraFormatter.CreateTable(tableCol) |  | ||||||
| 		table.Append([]string{internal.RecipeName, internal.Domain, internal.NewAppServer, internal.NewAppName}) |  | ||||||
|  |  | ||||||
| 		if err := internal.NewAction(c); err != nil { | 		if err := internal.NewAction(c); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -405,12 +478,12 @@ You may omit flags to avoid performing this provisioning logic. | |||||||
| 	ArgsUsage: "<domain> [<user>] [<port>]", | 	ArgsUsage: "<domain> [<user>] [<port>]", | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		if c.Args().Len() > 0 && local || !internal.ValidateSubCmdFlags(c) { | 		if c.Args().Len() > 0 && local || !internal.ValidateSubCmdFlags(c) { | ||||||
| 			err := errors.New("cannot use '<domain>' and '--local' together") | 			err := errors.New("cannot use <domain> and --local together") | ||||||
| 			internal.ShowSubcommandHelpAndError(c, err) | 			internal.ShowSubcommandHelpAndError(c, err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if sshAuth != "password" && sshAuth != "identity-file" { | 		if sshAuth != "password" && sshAuth != "identity-file" { | ||||||
| 			err := errors.New("--ssh-auth only accepts 'identity-file' or 'password'") | 			err := errors.New("--ssh-auth only accepts identity-file or password") | ||||||
| 			internal.ShowSubcommandHelpAndError(c, err) | 			internal.ShowSubcommandHelpAndError(c, err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -484,3 +557,23 @@ You may omit flags to avoid performing this provisioning logic. | |||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ensureLocalExecutable ensures that an executable is present on the local machine | ||||||
|  | func ensureLocalExecutable(exe string) (bool, error) { | ||||||
|  | 	out, err := exec.Command("which", exe).Output() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return string(out) != "", nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ensureRemoteExecutable ensures that an executable is present on a remote machine | ||||||
|  | func ensureRemoteExecutable(exe string, sshCl *ssh.Client) (bool, error) { | ||||||
|  | 	out, err := sshCl.Exec(fmt.Sprintf("which %s", exe)) | ||||||
|  | 	if err != nil && string(out) != "" { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return string(out) != "", nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -3,9 +3,9 @@ package server | |||||||
| import ( | import ( | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/context" | 	"coopcloud.tech/abra/pkg/context" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/docker/cli/cli/connhelper/ssh" | 	"github.com/docker/cli/cli/connhelper/ssh" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
|  | |||||||
| @ -4,8 +4,8 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/libcapsul" | 	"coopcloud.tech/libcapsul" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/hetznercloud/hcloud-go/hcloud" | 	"github.com/hetznercloud/hcloud-go/hcloud" | ||||||
| @ -43,13 +43,18 @@ func newHetznerCloudVPS(c *cli.Context) error { | |||||||
| 		Location:   &hcloud.Location{Name: internal.HetznerCloudLocation}, | 		Location:   &hcloud.Location{Name: internal.HetznerCloudLocation}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	sshKeyIDs := strings.Join(sshKeysRaw, "\n") | ||||||
|  | 	if sshKeyIDs == "" { | ||||||
|  | 		sshKeyIDs = "N/A (password auth)" | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	tableColumns := []string{"name", "type", "image", "ssh-keys", "location"} | 	tableColumns := []string{"name", "type", "image", "ssh-keys", "location"} | ||||||
| 	table := formatter.CreateTable(tableColumns) | 	table := formatter.CreateTable(tableColumns) | ||||||
| 	table.Append([]string{ | 	table.Append([]string{ | ||||||
| 		internal.HetznerCloudName, | 		internal.HetznerCloudName, | ||||||
| 		internal.HetznerCloudType, | 		internal.HetznerCloudType, | ||||||
| 		internal.HetznerCloudImage, | 		internal.HetznerCloudImage, | ||||||
| 		strings.Join(sshKeysRaw, "\n"), | 		sshKeyIDs, | ||||||
| 		internal.HetznerCloudLocation, | 		internal.HetznerCloudLocation, | ||||||
| 	}) | 	}) | ||||||
| 	table.Render() | 	table.Render() | ||||||
| @ -96,9 +101,21 @@ Please note, this server is not managed by Abra yet (i.e. "abra server ls" will | |||||||
| not list this server)! You will need to assign a domain name record ("abra | not list this server)! You will need to assign a domain name record ("abra | ||||||
| record new") and add the server to your Abra configuration ("abra server add") | record new") and add the server to your Abra configuration ("abra server add") | ||||||
| to have a working server that you can deploy Co-op Cloud apps to. | to have a working server that you can deploy Co-op Cloud apps to. | ||||||
|  |  | ||||||
|  | When setting up domain name records, you probably want to set up the following | ||||||
|  | 2 A records. This supports deploying apps to your root domain (e.g. | ||||||
|  | example.com) and other apps on sub-domains (e.g. foo.example.com, | ||||||
|  | bar.example.com). | ||||||
|  |  | ||||||
|  |     @  1800 IN A %s | ||||||
|  |     *  1800 IN A %s | ||||||
|  |  | ||||||
|  | "abra record new --auto" can help you do this quickly if you use a supported | ||||||
|  | DNS provider. | ||||||
|  |  | ||||||
| 	`, | 	`, | ||||||
| 		internal.HetznerCloudName, ip, rootPassword, | 		internal.HetznerCloudName, ip, rootPassword, | ||||||
| 		ip, | 		ip, ip, ip, | ||||||
| 	)) | 	)) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| @ -169,6 +186,15 @@ Please note, this server is not managed by Abra yet (i.e. "abra server ls" will | |||||||
| not list this server)! You will need to assign a domain name record ("abra | not list this server)! You will need to assign a domain name record ("abra | ||||||
| record new") and add the server to your Abra configuration ("abra server add") | record new") and add the server to your Abra configuration ("abra server add") | ||||||
| to have a working server that you can deploy Co-op Cloud apps to. | to have a working server that you can deploy Co-op Cloud apps to. | ||||||
|  |  | ||||||
|  | When setting up domain name records, you probably want to set up the following | ||||||
|  | 2 A records. This supports deploying apps to your root domain (e.g. | ||||||
|  | example.com) and other apps on sub-domains (e.g. foo.example.com, | ||||||
|  | bar.example.com). | ||||||
|  |  | ||||||
|  |     @  1800 IN A <your-capsul-ip> | ||||||
|  |     *  1800 IN A <your-capsul-ip> | ||||||
|  |  | ||||||
| 	`, internal.CapsulName, resp.ID, internal.CapsulInstanceURL)) | 	`, internal.CapsulName, resp.ID, internal.CapsulInstanceURL)) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| @ -196,7 +222,6 @@ API tokens are read from the environment if specified, e.g. | |||||||
|  |  | ||||||
| Where "$provider_TOKEN" is the expected env var format. | Where "$provider_TOKEN" is the expected env var format. | ||||||
| `, | `, | ||||||
| 	ArgsUsage: "<provider>", |  | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.ServerProviderFlag, | 		internal.ServerProviderFlag, | ||||||
|  |  | ||||||
|  | |||||||
| @ -5,10 +5,10 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/hetznercloud/hcloud-go/hcloud" | 	"github.com/hetznercloud/hcloud-go/hcloud" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| @ -102,7 +102,7 @@ destroyed. | |||||||
| var serverRemoveCommand = &cli.Command{ | var serverRemoveCommand = &cli.Command{ | ||||||
| 	Name:      "remove", | 	Name:      "remove", | ||||||
| 	Aliases:   []string{"rm"}, | 	Aliases:   []string{"rm"}, | ||||||
| 	ArgsUsage: "<server>", | 	ArgsUsage: "[<server>]", | ||||||
| 	Usage:     "Remove a managed server", | 	Usage:     "Remove a managed server", | ||||||
| 	Description: ` | 	Description: ` | ||||||
| This command removes a server from Abra management. | This command removes a server from Abra management. | ||||||
| @ -117,15 +117,36 @@ like tears in rain. | |||||||
| `, | `, | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		rmServerFlag, | 		rmServerFlag, | ||||||
|  | 		internal.ServerProviderFlag, | ||||||
|  |  | ||||||
| 		// Hetzner | 		// Hetzner | ||||||
| 		internal.HetznerCloudNameFlag, | 		internal.HetznerCloudNameFlag, | ||||||
| 		internal.HetznerCloudAPITokenFlag, | 		internal.HetznerCloudAPITokenFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		serverName, err := internal.ValidateServer(c) | 		serverName := c.Args().Get(1) | ||||||
| 		if err != nil { | 		if serverName != "" { | ||||||
| 			logrus.Fatal(err) | 			var err error | ||||||
|  | 			serverName, err = internal.ValidateServer(c) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !rmServer { | ||||||
|  | 			logrus.Warn("did not pass -s/--server for actual server deletion, prompting") | ||||||
|  |  | ||||||
|  | 			response := false | ||||||
|  | 			prompt := &survey.Confirm{ | ||||||
|  | 				Message: "prompt to actual server deletion?", | ||||||
|  | 			} | ||||||
|  | 			if err := survey.AskOne(prompt, &response); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			if response { | ||||||
|  | 				logrus.Info("setting -s/--server and attempting to remove actual server") | ||||||
|  | 				rmServer = true | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if rmServer { | 		if rmServer { | ||||||
| @ -144,15 +165,17 @@ like tears in rain. | |||||||
|  |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := client.DeleteContext(serverName); err != nil { | 		if serverName != "" { | ||||||
| 			logrus.Fatal(err) | 			if err := client.DeleteContext(serverName); err != nil { | ||||||
| 		} | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 		if err := os.RemoveAll(filepath.Join(config.ABRA_SERVER_FOLDER, serverName)); err != nil { | 			if err := os.RemoveAll(filepath.Join(config.SERVERS_DIR, serverName)); err != nil { | ||||||
| 			logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 		} | 			} | ||||||
|  |  | ||||||
| 		logrus.Infof("server at '%s' has been lost in time, like tears in rain", serverName) | 			logrus.Infof("server at %s has been lost in time, like tears in rain", serverName) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
| var ServerCommand = &cli.Command{ | var ServerCommand = &cli.Command{ | ||||||
| 	Name:    "server", | 	Name:    "server", | ||||||
| 	Aliases: []string{"s"}, | 	Aliases: []string{"s"}, | ||||||
| 	Usage:   "Manage servers via 3rd party providers", | 	Usage:   "Manage servers", | ||||||
| 	Description: ` | 	Description: ` | ||||||
| These commands support creating, managing and removing servers using 3rd party | These commands support creating, managing and removing servers using 3rd party | ||||||
| integrations.  | integrations.  | ||||||
|  | |||||||
| @ -1,35 +0,0 @@ | |||||||
| package cli |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"os/exec" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" |  | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| 	"github.com/urfave/cli/v2" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var RC bool |  | ||||||
| var RCFlag = &cli.BoolFlag{ |  | ||||||
| 	Name:        "rc", |  | ||||||
| 	Value:       false, |  | ||||||
| 	Destination: &RC, |  | ||||||
| 	Usage:       "Insatll the latest Release Candidate", |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // UpgradeCommand upgrades abra in-place. |  | ||||||
| var UpgradeCommand = &cli.Command{ |  | ||||||
| 	Name:  "upgrade", |  | ||||||
| 	Usage: "Upgrade abra", |  | ||||||
| 	Flags: []cli.Flag{RCFlag}, |  | ||||||
| 	Action: func(c *cli.Context) error { |  | ||||||
| 		cmd := exec.Command("bash", "-c", "curl -s https://install.abra.coopcloud.tech | bash") |  | ||||||
| 		if RC { |  | ||||||
| 			cmd = exec.Command("bash", "-c", "curl -s https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/installer/installer | bash -s -- --rc") |  | ||||||
| 		} |  | ||||||
| 		logrus.Debugf("attempting to run '%s'", cmd) |  | ||||||
| 		if err := internal.RunCmd(cmd); err != nil { |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	}, |  | ||||||
| } |  | ||||||
| @ -5,14 +5,13 @@ import ( | |||||||
| 	"coopcloud.tech/abra/cli" | 	"coopcloud.tech/abra/cli" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Version is the current version of abra. | // Version is the current version of Abra | ||||||
| var Version string | var Version string | ||||||
|  |  | ||||||
| // Commit is the current commit of abra. | // Commit is the current git commit of Abra | ||||||
| var Commit string | var Commit string | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	// If not set in the ld-flags |  | ||||||
| 	if Version == "" { | 	if Version == "" { | ||||||
| 		Version = "dev" | 		Version = "dev" | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								go.mod
									
									
									
									
									
								
							| @ -7,9 +7,9 @@ require ( | |||||||
| 	github.com/AlecAivazis/survey/v2 v2.3.2 | 	github.com/AlecAivazis/survey/v2 v2.3.2 | ||||||
| 	github.com/Autonomic-Cooperative/godotenv v1.3.1-0.20210731170023-c37c0920d1a4 | 	github.com/Autonomic-Cooperative/godotenv v1.3.1-0.20210731170023-c37c0920d1a4 | ||||||
| 	github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 | 	github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 | ||||||
| 	github.com/docker/cli v20.10.11+incompatible | 	github.com/docker/cli v20.10.12+incompatible | ||||||
| 	github.com/docker/distribution v2.7.1+incompatible | 	github.com/docker/distribution v2.7.1+incompatible | ||||||
| 	github.com/docker/docker v20.10.11+incompatible | 	github.com/docker/docker v20.10.12+incompatible | ||||||
| 	github.com/docker/go-units v0.4.0 | 	github.com/docker/go-units v0.4.0 | ||||||
| 	github.com/go-git/go-git/v5 v5.4.2 | 	github.com/go-git/go-git/v5 v5.4.2 | ||||||
| 	github.com/hetznercloud/hcloud-go v1.33.1 | 	github.com/hetznercloud/hcloud-go v1.33.1 | ||||||
| @ -17,7 +17,7 @@ require ( | |||||||
| 	github.com/moby/term v0.0.0-20210619224110-3f7ff695adc6 | 	github.com/moby/term v0.0.0-20210619224110-3f7ff695adc6 | ||||||
| 	github.com/olekukonko/tablewriter v0.0.5 | 	github.com/olekukonko/tablewriter v0.0.5 | ||||||
| 	github.com/pkg/errors v0.9.1 | 	github.com/pkg/errors v0.9.1 | ||||||
| 	github.com/schollz/progressbar/v3 v3.8.3 | 	github.com/schollz/progressbar/v3 v3.8.5 | ||||||
| 	github.com/schultz-is/passgen v1.0.1 | 	github.com/schultz-is/passgen v1.0.1 | ||||||
| 	github.com/sirupsen/logrus v1.8.1 | 	github.com/sirupsen/logrus v1.8.1 | ||||||
| 	github.com/urfave/cli/v2 v2.3.0 | 	github.com/urfave/cli/v2 v2.3.0 | ||||||
| @ -27,6 +27,7 @@ require ( | |||||||
| require ( | require ( | ||||||
| 	coopcloud.tech/libcapsul v0.0.0-20211022074848-c35e78fe3f3e | 	coopcloud.tech/libcapsul v0.0.0-20211022074848-c35e78fe3f3e | ||||||
| 	github.com/Microsoft/hcsshim v0.8.21 // indirect | 	github.com/Microsoft/hcsshim v0.8.21 // indirect | ||||||
|  | 	github.com/buger/goterm v1.0.3 | ||||||
| 	github.com/containerd/containerd v1.5.5 // indirect | 	github.com/containerd/containerd v1.5.5 // indirect | ||||||
| 	github.com/docker/docker-credential-helpers v0.6.4 // indirect | 	github.com/docker/docker-credential-helpers v0.6.4 // indirect | ||||||
| 	github.com/facebookgo/stack v0.0.0-20160209184415-751773369052 // indirect | 	github.com/facebookgo/stack v0.0.0-20160209184415-751773369052 // indirect | ||||||
| @ -34,6 +35,7 @@ require ( | |||||||
| 	github.com/gliderlabs/ssh v0.3.3 | 	github.com/gliderlabs/ssh v0.3.3 | ||||||
| 	github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect | 	github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect | ||||||
| 	github.com/gorilla/mux v1.8.0 // indirect | 	github.com/gorilla/mux v1.8.0 // indirect | ||||||
|  | 	github.com/hashicorp/go-retryablehttp v0.7.0 | ||||||
| 	github.com/kevinburke/ssh_config v1.1.0 | 	github.com/kevinburke/ssh_config v1.1.0 | ||||||
| 	github.com/libdns/gandi v1.0.2 | 	github.com/libdns/gandi v1.0.2 | ||||||
| 	github.com/libdns/libdns v0.2.1 | 	github.com/libdns/libdns v0.2.1 | ||||||
| @ -42,6 +44,6 @@ require ( | |||||||
| 	github.com/opencontainers/runc v1.0.2 // indirect | 	github.com/opencontainers/runc v1.0.2 // indirect | ||||||
| 	github.com/theupdateframework/notary v0.7.0 // indirect | 	github.com/theupdateframework/notary v0.7.0 // indirect | ||||||
| 	github.com/xeipuuv/gojsonschema v1.2.0 // indirect | 	github.com/xeipuuv/gojsonschema v1.2.0 // indirect | ||||||
| 	golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 | 	golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3 | ||||||
| 	golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359 | 	golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e | ||||||
| ) | ) | ||||||
|  | |||||||
							
								
								
									
										40
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								go.sum
									
									
									
									
									
								
							| @ -110,6 +110,8 @@ github.com/blang/semver v3.1.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnweb | |||||||
| github.com/blang/semver v3.5.1+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk= | github.com/blang/semver v3.5.1+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk= | ||||||
| github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869/go.mod h1:Ekp36dRnpXw/yCqJaO+ZrUyxD+3VXMFFr56k5XYrpB4= | github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869/go.mod h1:Ekp36dRnpXw/yCqJaO+ZrUyxD+3VXMFFr56k5XYrpB4= | ||||||
| github.com/bshuster-repo/logrus-logstash-hook v0.4.1/go.mod h1:zsTqEiSzDgAa/8GZR7E1qaXrhYNDKBYy5/dWPTIflbk= | github.com/bshuster-repo/logrus-logstash-hook v0.4.1/go.mod h1:zsTqEiSzDgAa/8GZR7E1qaXrhYNDKBYy5/dWPTIflbk= | ||||||
|  | github.com/buger/goterm v1.0.3 h1:7V/HeAQHrzPk/U4BvyH2g9u+xbUW9nr4yRPyG59W4fM= | ||||||
|  | github.com/buger/goterm v1.0.3/go.mod h1:HiFWV3xnkolgrBV3mY8m0X0Pumt4zg4QhbdOzQtB8tE= | ||||||
| github.com/buger/jsonparser v0.0.0-20180808090653-f4dd9f5a6b44/go.mod h1:bbYlZJ7hK1yFx9hf58LP0zeX7UjIGs20ufpu3evjr+s= | github.com/buger/jsonparser v0.0.0-20180808090653-f4dd9f5a6b44/go.mod h1:bbYlZJ7hK1yFx9hf58LP0zeX7UjIGs20ufpu3evjr+s= | ||||||
| github.com/bugsnag/bugsnag-go v0.0.0-20141110184014-b1d153021fcd/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8= | github.com/bugsnag/bugsnag-go v0.0.0-20141110184014-b1d153021fcd/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8= | ||||||
| github.com/bugsnag/bugsnag-go v1.0.5-0.20150529004307-13fd6b8acda0 h1:s7+5BfS4WFJoVF9pnB8kBk03S7pZXRdKamnV0FOl5Sc= | github.com/bugsnag/bugsnag-go v1.0.5-0.20150529004307-13fd6b8acda0 h1:s7+5BfS4WFJoVF9pnB8kBk03S7pZXRdKamnV0FOl5Sc= | ||||||
| @ -260,14 +262,14 @@ github.com/dgrijalva/jwt-go v0.0.0-20170104182250-a601269ab70c/go.mod h1:E3ru+11 | |||||||
| github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= | github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= | ||||||
| github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= | github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= | ||||||
| github.com/dnaeon/go-vcr v1.0.1/go.mod h1:aBB1+wY4s93YsC3HHjMBMrwTj2R9FHDzUr9KyGc8n1E= | github.com/dnaeon/go-vcr v1.0.1/go.mod h1:aBB1+wY4s93YsC3HHjMBMrwTj2R9FHDzUr9KyGc8n1E= | ||||||
| github.com/docker/cli v20.10.11+incompatible h1:tXU1ezXcruZQRrMP8RN2z9N91h+6egZTS1gsPsKantc= | github.com/docker/cli v20.10.12+incompatible h1:lZlz0uzG+GH+c0plStMUdF/qk3ppmgnswpR5EbqzVGA= | ||||||
| github.com/docker/cli v20.10.11+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= | github.com/docker/cli v20.10.12+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= | ||||||
| github.com/docker/distribution v0.0.0-20190905152932-14b96e55d84c/go.mod h1:0+TTO4EOBfRPhZXAeF1Vu+W3hHZ8eLp8PgKVZlcvtFY= | github.com/docker/distribution v0.0.0-20190905152932-14b96e55d84c/go.mod h1:0+TTO4EOBfRPhZXAeF1Vu+W3hHZ8eLp8PgKVZlcvtFY= | ||||||
| github.com/docker/distribution v2.7.1-0.20190205005809-0d3efadf0154+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= | github.com/docker/distribution v2.7.1-0.20190205005809-0d3efadf0154+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= | ||||||
| github.com/docker/distribution v2.7.1+incompatible h1:a5mlkVzth6W5A4fOsS3D2EO5BUmsJpcB+cRlLU7cSug= | github.com/docker/distribution v2.7.1+incompatible h1:a5mlkVzth6W5A4fOsS3D2EO5BUmsJpcB+cRlLU7cSug= | ||||||
| github.com/docker/distribution v2.7.1+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= | github.com/docker/distribution v2.7.1+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= | ||||||
| github.com/docker/docker v20.10.11+incompatible h1:OqzI/g/W54LczvhnccGqniFoQghHx3pklbLuhfXpqGo= | github.com/docker/docker v20.10.12+incompatible h1:CEeNmFM0QZIsJCZKMkZx0ZcahTiewkrgiwfYD+dfl1U= | ||||||
| github.com/docker/docker v20.10.11+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | github.com/docker/docker v20.10.12+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | ||||||
| github.com/docker/docker-credential-helpers v0.6.4 h1:axCks+yV+2MR3/kZhAmy07yC56WZ2Pwu/fKWtKuZB0o= | github.com/docker/docker-credential-helpers v0.6.4 h1:axCks+yV+2MR3/kZhAmy07yC56WZ2Pwu/fKWtKuZB0o= | ||||||
| github.com/docker/docker-credential-helpers v0.6.4/go.mod h1:ofX3UI0Gz1TteYBjtgs07O36Pyasyp66D2uKT7H8W1c= | github.com/docker/docker-credential-helpers v0.6.4/go.mod h1:ofX3UI0Gz1TteYBjtgs07O36Pyasyp66D2uKT7H8W1c= | ||||||
| github.com/docker/go v1.5.1-1.0.20160303222718-d30aec9fd63c h1:lzqkGL9b3znc+ZUgi7FlLnqjQhcXxkNM/quxIjBVMD0= | github.com/docker/go v1.5.1-1.0.20160303222718-d30aec9fd63c h1:lzqkGL9b3znc+ZUgi7FlLnqjQhcXxkNM/quxIjBVMD0= | ||||||
| @ -444,8 +446,14 @@ github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed h1:5upAirOpQc | |||||||
| github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed/go.mod h1:tMWxXQ9wFIaZeTI9F+hmhFiGpFmhOHzyShyFUhRm0H4= | github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed/go.mod h1:tMWxXQ9wFIaZeTI9F+hmhFiGpFmhOHzyShyFUhRm0H4= | ||||||
| github.com/hashicorp/errwrap v0.0.0-20141028054710-7554cd9344ce/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | github.com/hashicorp/errwrap v0.0.0-20141028054710-7554cd9344ce/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||||
| github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||||
|  | github.com/hashicorp/go-cleanhttp v0.5.1 h1:dH3aiDG9Jvb5r5+bYHsikaOUIpcM0xvgMXVoDkXMzJM= | ||||||
|  | github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||||
|  | github.com/hashicorp/go-hclog v0.9.2 h1:CG6TE5H9/JXsFWJCfoIVpKFIkFe6ysEuHirp4DxCsHI= | ||||||
|  | github.com/hashicorp/go-hclog v0.9.2/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= | ||||||
| github.com/hashicorp/go-multierror v0.0.0-20161216184304-ed905158d874/go.mod h1:JMRHfdO9jKNzS/+BTlxCjKNQHg/jZAft8U7LloJvN7I= | github.com/hashicorp/go-multierror v0.0.0-20161216184304-ed905158d874/go.mod h1:JMRHfdO9jKNzS/+BTlxCjKNQHg/jZAft8U7LloJvN7I= | ||||||
| github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= | ||||||
|  | github.com/hashicorp/go-retryablehttp v0.7.0 h1:eu1EI/mbirUgP5C8hVsTNaGZreBDlYiwC1FZWkvQPQ4= | ||||||
|  | github.com/hashicorp/go-retryablehttp v0.7.0/go.mod h1:vAew36LZh98gCBJNLH42IQ1ER/9wtLZZ8meHqQvEYWY= | ||||||
| github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||||
| github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||||
| github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= | github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= | ||||||
| @ -689,8 +697,8 @@ github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0 | |||||||
| github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | ||||||
| github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4= | github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4= | ||||||
| github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= | github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= | ||||||
| github.com/schollz/progressbar/v3 v3.8.3 h1:FnLGl3ewlDUP+YdSwveXBaXs053Mem/du+wr7XSYKl8= | github.com/schollz/progressbar/v3 v3.8.5 h1:VcmmNRO+eFN3B0m5dta6FXYXY+MEJmXdWoIS+jjssQM= | ||||||
| github.com/schollz/progressbar/v3 v3.8.3/go.mod h1:pWnVCjSBZsT2X3nx9HfRdnCDrpbevliMeoEVhStwHko= | github.com/schollz/progressbar/v3 v3.8.5/go.mod h1:ewO25kD7ZlaJFTvMeOItkOZa8kXu1UvFs379htE8HMQ= | ||||||
| github.com/schultz-is/passgen v1.0.1 h1:wUINzqW1Xmmy3yREHR6YTj+83VlFYjj2DIDMHzIi5TQ= | github.com/schultz-is/passgen v1.0.1 h1:wUINzqW1Xmmy3yREHR6YTj+83VlFYjj2DIDMHzIi5TQ= | ||||||
| github.com/schultz-is/passgen v1.0.1/go.mod h1:NnqzT2aSfvyheNQvBtlLUa0YlPFLDj60Jw2DZVwqiJk= | github.com/schultz-is/passgen v1.0.1/go.mod h1:NnqzT2aSfvyheNQvBtlLUa0YlPFLDj60Jw2DZVwqiJk= | ||||||
| github.com/seccomp/libseccomp-golang v0.9.1/go.mod h1:GbW5+tmTXfcxTToHLXlScSlAvWlF4P2Ca7zGrPiEpWo= | github.com/seccomp/libseccomp-golang v0.9.1/go.mod h1:GbW5+tmTXfcxTToHLXlScSlAvWlF4P2Ca7zGrPiEpWo= | ||||||
| @ -823,8 +831,8 @@ golang.org/x/crypto v0.0.0-20201117144127-c1f2f97bffc9/go.mod h1:jdWPYTVW3xRLrWP | |||||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||||
| golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||||
| golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||||
| golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 h1:HWj/xjIHfjYU5nVXpTM0s39J9CbLn7Cc5a7IC5rwsMQ= | golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3 h1:0es+/5331RGQPcXlMfP+WrnIIS6dNnNRe0WB02W0F4M= | ||||||
| golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||||
| golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||||
| golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= | golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= | ||||||
| @ -890,8 +898,9 @@ golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwY | |||||||
| golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||||
| golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||||
| golang.org/x/net v0.0.0-20210326060303-6b1517762897 h1:KrsHThm5nFk34YtATK1LsThyGhGbGe1olrte/HInHvs= |  | ||||||
| golang.org/x/net v0.0.0-20210326060303-6b1517762897/go.mod h1:uSPa2vr4CLtc/ILN5odXGNXS6mhrKVzTaCXzk9m6W3k= | golang.org/x/net v0.0.0-20210326060303-6b1517762897/go.mod h1:uSPa2vr4CLtc/ILN5odXGNXS6mhrKVzTaCXzk9m6W3k= | ||||||
|  | golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2 h1:CIJ76btIcR3eFI5EgSo6k1qKw9KJexJuRLI9G7Hp5wE= | ||||||
|  | golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||||
| @ -969,27 +978,30 @@ golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7w | |||||||
| golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210324051608-47abb6519492/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210324051608-47abb6519492/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
|  | golang.org/x/sys v0.0.0-20210331175145-43e1dd70ce54/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
|  | golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210426230700-d19ff857e887/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210426230700-d19ff857e887/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210502180810-71e4cd670f79/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210502180810-71e4cd670f79/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20210910150752-751e447fb3d0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359 h1:2B5p2L5IfGiD7+b9BOoRMC6DgObAVZV+Fsp050NqXik= |  | ||||||
| golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
|  | golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e h1:fLOSk5Q00efkSvAm+4xcoXD+RRmLmmulPn5I3Y9F2EM= | ||||||
|  | golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | ||||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||||
| golang.org/x/term v0.0.0-20210503060354-a79de5458b56/go.mod h1:tfny5GFUkzUvx4ps4ajbZsCe5lw1metzhBm9T3x7oIY= | golang.org/x/term v0.0.0-20210503060354-a79de5458b56/go.mod h1:tfny5GFUkzUvx4ps4ajbZsCe5lw1metzhBm9T3x7oIY= | ||||||
| golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b h1:9zKuko04nR4gjZ4+DNjHqRlAJqbJETHwiNKDqTfOjfE= | golang.org/x/term v0.0.0-20210927222741-03fcf44c2211 h1:JGgROgKl9N8DuW20oFS5gxc+lE67/N3FcwmBPMe7ArY= | ||||||
| golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||||
| golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.4 h1:0YWbFKbhXG/wIiuHDSKpS0Iy7FSA+u45VtBMfQcFTTc= |  | ||||||
| golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
|  | golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M= | ||||||
|  | golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||||
| golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||||
|  | |||||||
| @ -23,7 +23,7 @@ func Get(appName string) (config.App, error) { | |||||||
| 		return config.App{}, err | 		return config.App{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved '%s' for '%s'", app, appName) | 	logrus.Debugf("retrieved %s for %s", app, appName) | ||||||
|  |  | ||||||
| 	return app, nil | 	return app, nil | ||||||
| } | } | ||||||
| @ -57,9 +57,9 @@ func DeployedVersions(ctx context.Context, cl *apiclient.Client, app config.App) | |||||||
| 	deployed := len(services) > 0 | 	deployed := len(services) > 0 | ||||||
|  |  | ||||||
| 	if deployed { | 	if deployed { | ||||||
| 		logrus.Debugf("detected '%s' as deployed versions of '%s'", appSpec, app.Name) | 		logrus.Debugf("detected %s as deployed versions of %s", appSpec, app.Name) | ||||||
| 	} else { | 	} else { | ||||||
| 		logrus.Debugf("detected '%s' as not deployed", app.Name) | 		logrus.Debugf("detected %s as not deployed", app.Name) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return appSpec, len(services) > 0, nil | 	return appSpec, len(services) > 0, nil | ||||||
| @ -71,15 +71,15 @@ func ParseVersionLabel(label string) (string, string) { | |||||||
| 	idx := strings.LastIndex(label, "-") | 	idx := strings.LastIndex(label, "-") | ||||||
| 	version := label[:idx] | 	version := label[:idx] | ||||||
| 	digest := label[idx+1:] | 	digest := label[idx+1:] | ||||||
| 	logrus.Debugf("parsed '%s' as version from '%s'", version, label) | 	logrus.Debugf("parsed %s as version from %s", version, label) | ||||||
| 	logrus.Debugf("parsed '%s' as digest from '%s'", digest, label) | 	logrus.Debugf("parsed %s as digest from %s", digest, label) | ||||||
| 	return version, digest | 	return version, digest | ||||||
| } | } | ||||||
|  |  | ||||||
| // ParseVersionName parses a $STACK_NAME_$SERVICE_NAME service label. | // ParseServiceName parses a $STACK_NAME_$SERVICE_NAME service label. | ||||||
| func ParseServiceName(label string) string { | func ParseServiceName(label string) string { | ||||||
| 	idx := strings.LastIndex(label, "_") | 	idx := strings.LastIndex(label, "_") | ||||||
| 	serviceName := label[idx+1:] | 	serviceName := label[idx+1:] | ||||||
| 	logrus.Debugf("parsed '%s' as service name from '%s'", serviceName, label) | 	logrus.Debugf("parsed %s as service name from %s", serviceName, label) | ||||||
| 	return serviceName | 	return serviceName | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										42
									
								
								pkg/autocomplete/autocomplete.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								pkg/autocomplete/autocomplete.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | package autocomplete | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | 	"github.com/urfave/cli/v2" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // AppNameComplete copletes app names | ||||||
|  | func AppNameComplete(c *cli.Context) { | ||||||
|  | 	appNames, err := config.GetAppNames() | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Warn(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if c.NArg() > 0 { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, a := range appNames { | ||||||
|  | 		fmt.Println(a) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // RecipeNameComplete completes recipe names | ||||||
|  | func RecipeNameComplete(c *cli.Context) { | ||||||
|  | 	catl, err := recipe.ReadRecipeCatalogue() | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Warn(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if c.NArg() > 0 { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for name := range catl { | ||||||
|  | 		fmt.Println(name) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -1,500 +0,0 @@ | |||||||
| // Package catalogue provides ways of interacting with recipe catalogues which |  | ||||||
| // are JSON data structures which contain meta information about recipes (e.g. |  | ||||||
| // what versions of the Nextcloud recipe are available?). |  | ||||||
| package catalogue |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io/ioutil" |  | ||||||
| 	"net/http" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/client" |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" |  | ||||||
| 	"coopcloud.tech/abra/pkg/web" |  | ||||||
| 	"github.com/docker/distribution/reference" |  | ||||||
| 	"github.com/go-git/go-git/v5" |  | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" |  | ||||||
| 	"github.com/sirupsen/logrus" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // RecipeCatalogueURL is the only current recipe catalogue available. |  | ||||||
| const RecipeCatalogueURL = "https://apps.coopcloud.tech" |  | ||||||
|  |  | ||||||
| // ReposMetadataURL is the recipe repository metadata |  | ||||||
| const ReposMetadataURL = "https://git.coopcloud.tech/api/v1/orgs/coop-cloud/repos" |  | ||||||
|  |  | ||||||
| // image represents a recipe container image. |  | ||||||
| type image struct { |  | ||||||
| 	Image  string `json:"image"` |  | ||||||
| 	Rating string `json:"rating"` |  | ||||||
| 	Source string `json:"source"` |  | ||||||
| 	URL    string `json:"url"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // features represent what top-level features a recipe supports (e.g. does this |  | ||||||
| // recipe support backups?). |  | ||||||
| type features struct { |  | ||||||
| 	Backups     string `json:"backups"` |  | ||||||
| 	Email       string `json:"email"` |  | ||||||
| 	Healthcheck string `json:"healthcheck"` |  | ||||||
| 	Image       image  `json:"image"` |  | ||||||
| 	Status      int    `json:"status"` |  | ||||||
| 	Tests       string `json:"tests"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // tag represents a git tag. |  | ||||||
| type tag = string |  | ||||||
|  |  | ||||||
| // service represents a service within a recipe. |  | ||||||
| type service = string |  | ||||||
|  |  | ||||||
| // ServiceMeta represents meta info associated with a service. |  | ||||||
| type ServiceMeta struct { |  | ||||||
| 	Digest string `json:"digest"` |  | ||||||
| 	Image  string `json:"image"` |  | ||||||
| 	Tag    string `json:"tag"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // RecipeVersions are the versions associated with a recipe. |  | ||||||
| type RecipeVersions []map[tag]map[service]ServiceMeta |  | ||||||
|  |  | ||||||
| // RecipeMeta represents metadata for a recipe in the abra catalogue. |  | ||||||
| type RecipeMeta struct { |  | ||||||
| 	Category      string         `json:"category"` |  | ||||||
| 	DefaultBranch string         `json:"default_branch"` |  | ||||||
| 	Description   string         `json:"description"` |  | ||||||
| 	Features      features       `json:"features"` |  | ||||||
| 	Icon          string         `json:"icon"` |  | ||||||
| 	Name          string         `json:"name"` |  | ||||||
| 	Repository    string         `json:"repository"` |  | ||||||
| 	Versions      RecipeVersions `json:"versions"` |  | ||||||
| 	Website       string         `json:"website"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // LatestVersion returns the latest version of a recipe. |  | ||||||
| func (r RecipeMeta) LatestVersion() string { |  | ||||||
| 	var version string |  | ||||||
|  |  | ||||||
| 	// apps.json versions are sorted so the last key is latest |  | ||||||
| 	latest := r.Versions[len(r.Versions)-1] |  | ||||||
|  |  | ||||||
| 	for tag := range latest { |  | ||||||
| 		version = tag |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("choosing '%s' as latest version of '%s'", version, r.Name) |  | ||||||
|  |  | ||||||
| 	return version |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Name represents a recipe name. |  | ||||||
| type Name = string |  | ||||||
|  |  | ||||||
| // RecipeCatalogue represents the entire recipe catalogue. |  | ||||||
| type RecipeCatalogue map[Name]RecipeMeta |  | ||||||
|  |  | ||||||
| // Flatten converts AppCatalogue to slice |  | ||||||
| func (r RecipeCatalogue) Flatten() []RecipeMeta { |  | ||||||
| 	recipes := make([]RecipeMeta, 0, len(r)) |  | ||||||
|  |  | ||||||
| 	for name := range r { |  | ||||||
| 		recipes = append(recipes, r[name]) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return recipes |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ByRecipeName sorts recipes by name. |  | ||||||
| type ByRecipeName []RecipeMeta |  | ||||||
|  |  | ||||||
| func (r ByRecipeName) Len() int      { return len(r) } |  | ||||||
| func (r ByRecipeName) Swap(i, j int) { r[i], r[j] = r[j], r[i] } |  | ||||||
| func (r ByRecipeName) Less(i, j int) bool { |  | ||||||
| 	return strings.ToLower(r[i].Name) < strings.ToLower(r[j].Name) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // recipeCatalogueFSIsLatest checks whether the recipe catalogue stored locally |  | ||||||
| // is up to date. |  | ||||||
| func recipeCatalogueFSIsLatest() (bool, error) { |  | ||||||
| 	httpClient := &http.Client{Timeout: web.Timeout} |  | ||||||
| 	res, err := httpClient.Head(RecipeCatalogueURL) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lastModified := res.Header["Last-Modified"][0] |  | ||||||
| 	parsed, err := time.Parse(time.RFC1123, lastModified) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	info, err := os.Stat(config.APPS_JSON) |  | ||||||
| 	if err != nil { |  | ||||||
| 		if os.IsNotExist(err) { |  | ||||||
| 			logrus.Debugf("no recipe catalogue found in file system cache") |  | ||||||
| 			return false, nil |  | ||||||
| 		} |  | ||||||
| 		return false, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	localModifiedTime := info.ModTime().Unix() |  | ||||||
| 	remoteModifiedTime := parsed.Unix() |  | ||||||
|  |  | ||||||
| 	if localModifiedTime < remoteModifiedTime { |  | ||||||
| 		logrus.Debug("file system cached recipe catalogue is out-of-date") |  | ||||||
| 		return false, nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debug("file system cached recipe catalogue is up-to-date") |  | ||||||
|  |  | ||||||
| 	return true, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ReadRecipeCatalogue reads the recipe catalogue. |  | ||||||
| func ReadRecipeCatalogue() (RecipeCatalogue, error) { |  | ||||||
| 	recipes := make(RecipeCatalogue) |  | ||||||
|  |  | ||||||
| 	recipeFSIsLatest, err := recipeCatalogueFSIsLatest() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !recipeFSIsLatest { |  | ||||||
| 		logrus.Debugf("reading recipe catalogue from web to get latest") |  | ||||||
| 		if err := readRecipeCatalogueWeb(&recipes); err != nil { |  | ||||||
| 			return nil, err |  | ||||||
| 		} |  | ||||||
| 		return recipes, nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("reading recipe catalogue from file system cache to get latest") |  | ||||||
| 	if err := readRecipeCatalogueFS(&recipes); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return recipes, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // readRecipeCatalogueFS reads the catalogue from the file system. |  | ||||||
| func readRecipeCatalogueFS(target interface{}) error { |  | ||||||
| 	recipesJSONFS, err := ioutil.ReadFile(config.APPS_JSON) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := json.Unmarshal(recipesJSONFS, &target); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("read recipe catalogue from file system cache in '%s'", config.APPS_JSON) |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // readRecipeCatalogueWeb reads the catalogue from the web. |  | ||||||
| func readRecipeCatalogueWeb(target interface{}) error { |  | ||||||
| 	if err := web.ReadJSON(RecipeCatalogueURL, &target); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	recipesJSON, err := json.MarshalIndent(target, "", "    ") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := ioutil.WriteFile(config.APPS_JSON, recipesJSON, 0644); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("read recipe catalogue from web at '%s'", RecipeCatalogueURL) |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // VersionsOfService lists the version of a service. |  | ||||||
| func VersionsOfService(recipe, serviceName string) ([]string, error) { |  | ||||||
| 	catalogue, err := ReadRecipeCatalogue() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	rec, ok := catalogue[recipe] |  | ||||||
| 	if !ok { |  | ||||||
| 		return nil, fmt.Errorf("recipe '%s' does not exist?", recipe) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	versions := []string{} |  | ||||||
| 	alreadySeen := make(map[string]bool) |  | ||||||
| 	for _, serviceVersion := range rec.Versions { |  | ||||||
| 		for tag := range serviceVersion { |  | ||||||
| 			if _, ok := alreadySeen[tag]; !ok { |  | ||||||
| 				alreadySeen[tag] = true |  | ||||||
| 				versions = append(versions, tag) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("detected versions '%s' for '%s'", strings.Join(versions, ", "), recipe) |  | ||||||
|  |  | ||||||
| 	return versions, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetRecipeMeta retrieves the recipe metadata from the recipe catalogue. |  | ||||||
| func GetRecipeMeta(recipeName string) (RecipeMeta, error) { |  | ||||||
| 	catl, err := ReadRecipeCatalogue() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return RecipeMeta{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	recipeMeta, ok := catl[recipeName] |  | ||||||
| 	if !ok { |  | ||||||
| 		err := fmt.Errorf("recipe '%s' does not exist?", recipeName) |  | ||||||
| 		return RecipeMeta{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := recipe.EnsureExists(recipeName); err != nil { |  | ||||||
| 		return RecipeMeta{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("recipe metadata retrieved for '%s'", recipeName) |  | ||||||
|  |  | ||||||
| 	return recipeMeta, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // RepoMeta is a single recipe repo metadata. |  | ||||||
| type RepoMeta struct { |  | ||||||
| 	ID                        int `json:"id"` |  | ||||||
| 	Owner                     Owner |  | ||||||
| 	Name                      string      `json:"name"` |  | ||||||
| 	FullName                  string      `json:"full_name"` |  | ||||||
| 	Description               string      `json:"description"` |  | ||||||
| 	Empty                     bool        `json:"empty"` |  | ||||||
| 	Private                   bool        `json:"private"` |  | ||||||
| 	Fork                      bool        `json:"fork"` |  | ||||||
| 	Template                  bool        `json:"template"` |  | ||||||
| 	Parent                    interface{} `json:"parent"` |  | ||||||
| 	Mirror                    bool        `json:"mirror"` |  | ||||||
| 	Size                      int         `json:"size"` |  | ||||||
| 	HTMLURL                   string      `json:"html_url"` |  | ||||||
| 	SSHURL                    string      `json:"ssh_url"` |  | ||||||
| 	CloneURL                  string      `json:"clone_url"` |  | ||||||
| 	OriginalURL               string      `json:"original_url"` |  | ||||||
| 	Website                   string      `json:"website"` |  | ||||||
| 	StarsCount                int         `json:"stars_count"` |  | ||||||
| 	ForksCount                int         `json:"forks_count"` |  | ||||||
| 	WatchersCount             int         `json:"watchers_count"` |  | ||||||
| 	OpenIssuesCount           int         `json:"open_issues_count"` |  | ||||||
| 	OpenPRCount               int         `json:"open_pr_counter"` |  | ||||||
| 	ReleaseCounter            int         `json:"release_counter"` |  | ||||||
| 	DefaultBranch             string      `json:"default_branch"` |  | ||||||
| 	Archived                  bool        `json:"archived"` |  | ||||||
| 	CreatedAt                 string      `json:"created_at"` |  | ||||||
| 	UpdatedAt                 string      `json:"updated_at"` |  | ||||||
| 	Permissions               Permissions |  | ||||||
| 	HasIssues                 bool `json:"has_issues"` |  | ||||||
| 	InternalTracker           InternalTracker |  | ||||||
| 	HasWiki                   bool   `json:"has_wiki"` |  | ||||||
| 	HasPullRequests           bool   `json:"has_pull_requests"` |  | ||||||
| 	HasProjects               bool   `json:"has_projects"` |  | ||||||
| 	IgnoreWhitespaceConflicts bool   `json:"ignore_whitespace_conflicts"` |  | ||||||
| 	AllowMergeCommits         bool   `json:"allow_merge_commits"` |  | ||||||
| 	AllowRebase               bool   `json:"allow_rebase"` |  | ||||||
| 	AllowRebaseExplicit       bool   `json:"allow_rebase_explicit"` |  | ||||||
| 	AllowSquashMerge          bool   `json:"allow_squash_merge"` |  | ||||||
| 	AvatarURL                 string `json:"avatar_url"` |  | ||||||
| 	Internal                  bool   `json:"internal"` |  | ||||||
| 	MirrorInterval            string `json:"mirror_interval"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Owner is the repo organisation owner metadata. |  | ||||||
| type Owner struct { |  | ||||||
| 	ID         int    `json:"id"` |  | ||||||
| 	Login      string `json:"login"` |  | ||||||
| 	FullName   string `json:"full_name"` |  | ||||||
| 	Email      string `json:"email"` |  | ||||||
| 	AvatarURL  string `json:"avatar_url"` |  | ||||||
| 	Language   string `json:"language"` |  | ||||||
| 	IsAdmin    bool   `json:"is_admin"` |  | ||||||
| 	LastLogin  string `json:"last_login"` |  | ||||||
| 	Created    string `json:"created"` |  | ||||||
| 	Restricted bool   `json:"restricted"` |  | ||||||
| 	Username   string `json:"username"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Permissions is perms metadata for a repo. |  | ||||||
| type Permissions struct { |  | ||||||
| 	Admin bool `json:"admin"` |  | ||||||
| 	Push  bool `json:"push"` |  | ||||||
| 	Pull  bool `json:"pull"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // InternalTracker is issue tracker metadata for a repo. |  | ||||||
| type InternalTracker struct { |  | ||||||
| 	EnableTimeTracker                bool `json:"enable_time_tracker"` |  | ||||||
| 	AllowOnlyContributorsToTrackTime bool `json:"allow_only_contributors_to_track_time"` |  | ||||||
| 	EnableIssuesDependencies         bool `json:"enable_issue_dependencies"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // RepoCatalogue represents all the recipe repo metadata. |  | ||||||
| type RepoCatalogue map[string]RepoMeta |  | ||||||
|  |  | ||||||
| // ReadReposMetadata retrieves coop-cloud/... repo metadata from Gitea. |  | ||||||
| func ReadReposMetadata() (RepoCatalogue, error) { |  | ||||||
| 	reposMeta := make(RepoCatalogue) |  | ||||||
|  |  | ||||||
| 	pageIdx := 1 |  | ||||||
| 	for { |  | ||||||
| 		var reposList []RepoMeta |  | ||||||
|  |  | ||||||
| 		pagedURL := fmt.Sprintf("%s?page=%v", ReposMetadataURL, pageIdx) |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("fetching repo metadata from '%s'", pagedURL) |  | ||||||
|  |  | ||||||
| 		if err := web.ReadJSON(pagedURL, &reposList); err != nil { |  | ||||||
| 			return reposMeta, err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if len(reposList) == 0 { |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		for idx, repo := range reposList { |  | ||||||
| 			reposMeta[repo.Name] = reposList[idx] |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		pageIdx++ |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return reposMeta, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetRecipeVersions retrieves all recipe versions. |  | ||||||
| func GetRecipeVersions(recipeName string) (RecipeVersions, error) { |  | ||||||
| 	versions := RecipeVersions{} |  | ||||||
|  |  | ||||||
| 	recipeDir := path.Join(config.ABRA_DIR, "apps", recipeName) |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("attempting to open git repository in '%s'", recipeDir) |  | ||||||
|  |  | ||||||
| 	repo, err := git.PlainOpen(recipeDir) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return versions, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	worktree, err := repo.Worktree() |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	gitTags, err := repo.Tags() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return versions, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := gitTags.ForEach(func(ref *plumbing.Reference) (err error) { |  | ||||||
| 		tag := strings.TrimPrefix(string(ref.Name()), "refs/tags/") |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("processing '%s' for '%s'", tag, recipeName) |  | ||||||
|  |  | ||||||
| 		checkOutOpts := &git.CheckoutOptions{ |  | ||||||
| 			Create: false, |  | ||||||
| 			Force:  true, |  | ||||||
| 			Branch: plumbing.ReferenceName(ref.Name()), |  | ||||||
| 		} |  | ||||||
| 		if err := worktree.Checkout(checkOutOpts); err != nil { |  | ||||||
| 			logrus.Debugf("failed to check out '%s' in '%s'", tag, recipeDir) |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("successfully checked out '%s' in '%s'", ref.Name(), recipeDir) |  | ||||||
|  |  | ||||||
| 		recipe, err := recipe.Get(recipeName) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		versionMeta := make(map[string]ServiceMeta) |  | ||||||
| 		for _, service := range recipe.Config.Services { |  | ||||||
|  |  | ||||||
| 			img, err := reference.ParseNormalizedNamed(service.Image) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			path := reference.Path(img) |  | ||||||
| 			if strings.Contains(path, "library") { |  | ||||||
| 				path = strings.Split(path, "/")[1] |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			digest, err := client.GetTagDigest(img) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			versionMeta[service.Name] = ServiceMeta{ |  | ||||||
| 				Digest: digest, |  | ||||||
| 				Image:  path, |  | ||||||
| 				Tag:    img.(reference.NamedTagged).Tag(), |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			logrus.Debugf("collecting digest: '%s', image: '%s', tag: '%s'", digest, path, tag) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		versions = append(versions, map[string]map[string]ServiceMeta{tag: versionMeta}) |  | ||||||
|  |  | ||||||
| 		return nil |  | ||||||
| 	}); err != nil { |  | ||||||
| 		return versions, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	branch := "master" |  | ||||||
| 	if _, err := repo.Branch("master"); err != nil { |  | ||||||
| 		if _, err := repo.Branch("main"); err != nil { |  | ||||||
| 			logrus.Debugf("failed to select branch in '%s'", recipeDir) |  | ||||||
| 			logrus.Fatal(err) |  | ||||||
| 		} |  | ||||||
| 		branch = "main" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	refName := fmt.Sprintf("refs/heads/%s", branch) |  | ||||||
| 	checkOutOpts := &git.CheckoutOptions{ |  | ||||||
| 		Create: false, |  | ||||||
| 		Force:  true, |  | ||||||
| 		Branch: plumbing.ReferenceName(refName), |  | ||||||
| 	} |  | ||||||
| 	if err := worktree.Checkout(checkOutOpts); err != nil { |  | ||||||
| 		logrus.Debugf("failed to check out '%s' in '%s'", branch, recipeDir) |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("switched back to '%s' in '%s'", branch, recipeDir) |  | ||||||
| 	logrus.Debugf("collected '%s' for '%s'", versions, recipeName) |  | ||||||
|  |  | ||||||
| 	return versions, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetRecipeCatalogueVersions list the recipe versions listed in the recipe catalogue. |  | ||||||
| func GetRecipeCatalogueVersions(recipeName string) ([]string, error) { |  | ||||||
| 	var versions []string |  | ||||||
|  |  | ||||||
| 	catl, err := ReadRecipeCatalogue() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return versions, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if recipeMeta, exists := catl[recipeName]; exists { |  | ||||||
| 		for _, versionMeta := range recipeMeta.Versions { |  | ||||||
| 			for tag := range versionMeta { |  | ||||||
| 				versions = append(versions, tag) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return versions, nil |  | ||||||
| } |  | ||||||
| @ -55,7 +55,7 @@ func New(contextName string) (*client.Client, error) { | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("created client for '%s'", contextName) | 	logrus.Debugf("created client for %s", contextName) | ||||||
|  |  | ||||||
| 	return cl, nil | 	return cl, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -26,7 +26,7 @@ func CreateContext(contextName string, user string, port string) error { | |||||||
| 	if err := createContext(contextName, host); err != nil { | 	if err := createContext(contextName, host); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	logrus.Debugf("created the '%s' context", contextName) | 	logrus.Debugf("created the %s context", contextName) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -72,8 +72,6 @@ func DeleteContext(name string) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// remove any context that might be loaded |  | ||||||
| 	// TODO: Check if the context we are removing is the active one rather than doing it all the time |  | ||||||
| 	cfg := dConfig.LoadDefaultConfigFile(nil) | 	cfg := dConfig.LoadDefaultConfigFile(nil) | ||||||
| 	cfg.CurrentContext = "" | 	cfg.CurrentContext = "" | ||||||
| 	if err := cfg.Save(); err != nil { | 	if err := cfg.Save(); err != nil { | ||||||
|  | |||||||
| @ -1,6 +1,7 @@ | |||||||
| package client | package client | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/base64" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| @ -9,6 +10,9 @@ import ( | |||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/web" | 	"coopcloud.tech/abra/pkg/web" | ||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
|  | 	"github.com/docker/docker/client" | ||||||
|  | 	"github.com/hashicorp/go-retryablehttp" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type RawTag struct { | type RawTag struct { | ||||||
| @ -31,16 +35,30 @@ func GetRegistryTags(image string) (RawTags, error) { | |||||||
| 	return tags, nil | 	return tags, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func basicAuth(username, password string) string { | ||||||
|  | 	auth := username + ":" + password | ||||||
|  | 	return base64.StdEncoding.EncodeToString([]byte(auth)) | ||||||
|  | } | ||||||
|  |  | ||||||
| // getRegv2Token retrieves a registry v2 authentication token. | // getRegv2Token retrieves a registry v2 authentication token. | ||||||
| func getRegv2Token(image reference.Named) (string, error) { | func getRegv2Token(cl *client.Client, image reference.Named, registryUsername, registryPassword string) (string, error) { | ||||||
| 	img := reference.Path(image) | 	img := reference.Path(image) | ||||||
| 	authTokenURL := fmt.Sprintf("https://auth.docker.io/token?service=registry.docker.io&scope=repository:%s:pull", img) | 	tokenURL := "https://auth.docker.io/token" | ||||||
| 	req, err := http.NewRequest("GET", authTokenURL, nil) | 	values := fmt.Sprintf("service=registry.docker.io&scope=repository:%s:pull", img) | ||||||
|  |  | ||||||
|  | 	fullURL := fmt.Sprintf("%s?%s", tokenURL, values) | ||||||
|  | 	req, err := retryablehttp.NewRequest("GET", fullURL, nil) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	client := &http.Client{Timeout: web.Timeout} | 	if registryUsername != "" && registryPassword != "" { | ||||||
|  | 		logrus.Debugf("using registry log in credentials for token request") | ||||||
|  | 		auth := basicAuth(registryUsername, registryPassword) | ||||||
|  | 		req.Header.Add("Authorization", fmt.Sprintf("Basic %s", auth)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	client := web.NewHTTPRetryClient() | ||||||
| 	res, err := client.Do(req) | 	res, err := client.Do(req) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| @ -60,9 +78,10 @@ func getRegv2Token(image reference.Named) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	tokenRes := struct { | 	tokenRes := struct { | ||||||
| 		Token  string | 		AccessToken string `json:"access_token"` | ||||||
| 		Expiry string | 		Expiry      int    `json:"expires_in"` | ||||||
| 		Issued string | 		Issued      string `json:"issued_at"` | ||||||
|  | 		Token       string `json:"token"` | ||||||
| 	}{} | 	}{} | ||||||
|  |  | ||||||
| 	if err := json.Unmarshal(body, &tokenRes); err != nil { | 	if err := json.Unmarshal(body, &tokenRes); err != nil { | ||||||
| @ -73,21 +92,25 @@ func getRegv2Token(image reference.Named) (string, error) { | |||||||
| } | } | ||||||
|  |  | ||||||
| // GetTagDigest retrieves an image digest from a v2 registry | // GetTagDigest retrieves an image digest from a v2 registry | ||||||
| func GetTagDigest(image reference.Named) (string, error) { | func GetTagDigest(cl *client.Client, image reference.Named, registryUsername, registryPassword string) (string, error) { | ||||||
| 	img := reference.Path(image) | 	img := reference.Path(image) | ||||||
| 	tag := image.(reference.NamedTagged).Tag() | 	tag := image.(reference.NamedTagged).Tag() | ||||||
| 	manifestURL := fmt.Sprintf("https://index.docker.io/v2/%s/manifests/%s", img, tag) | 	manifestURL := fmt.Sprintf("https://index.docker.io/v2/%s/manifests/%s", img, tag) | ||||||
|  |  | ||||||
| 	req, err := http.NewRequest("GET", manifestURL, nil) | 	req, err := retryablehttp.NewRequest("GET", manifestURL, nil) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	token, err := getRegv2Token(image) | 	token, err := getRegv2Token(cl, image, registryUsername, registryPassword) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if token == "" { | ||||||
|  | 		return "", fmt.Errorf("unable to retrieve registry token?") | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	req.Header = http.Header{ | 	req.Header = http.Header{ | ||||||
| 		"Accept": []string{ | 		"Accept": []string{ | ||||||
| 			"application/vnd.docker.distribution.manifest.v2+json", | 			"application/vnd.docker.distribution.manifest.v2+json", | ||||||
| @ -96,7 +119,7 @@ func GetTagDigest(image reference.Named) (string, error) { | |||||||
| 		"Authorization": []string{fmt.Sprintf("Bearer %s", token)}, | 		"Authorization": []string{fmt.Sprintf("Bearer %s", token)}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	client := &http.Client{Timeout: web.Timeout} | 	client := web.NewHTTPRetryClient() | ||||||
| 	res, err := client.Do(req) | 	res, err := client.Do(req) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| @ -163,7 +186,7 @@ func GetTagDigest(image reference.Named) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if digest == "" { | 	if digest == "" { | ||||||
| 		return "", fmt.Errorf("Unable to retrieve amd64 digest for '%s'", image) | 		return "", fmt.Errorf("Unable to retrieve amd64 digest for %s", image) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return digest, nil | 	return digest, nil | ||||||
|  | |||||||
| @ -22,12 +22,12 @@ func UpdateTag(pattern, image, tag, recipeName string) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("considering '%s' config(s) for tag update", strings.Join(composeFiles, ", ")) | 	logrus.Debugf("considering %s config(s) for tag update", strings.Join(composeFiles, ", ")) | ||||||
|  |  | ||||||
| 	for _, composeFile := range composeFiles { | 	for _, composeFile := range composeFiles { | ||||||
| 		opts := stack.Deploy{Composefiles: []string{composeFile}} | 		opts := stack.Deploy{Composefiles: []string{composeFile}} | ||||||
|  |  | ||||||
| 		envSamplePath := path.Join(config.ABRA_DIR, "apps", recipeName, ".env.sample") | 		envSamplePath := path.Join(config.RECIPES_DIR, recipeName, ".env.sample") | ||||||
| 		sampleEnv, err := config.ReadEnv(envSamplePath) | 		sampleEnv, err := config.ReadEnv(envSamplePath) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -57,7 +57,7 @@ func UpdateTag(pattern, image, tag, recipeName string) error { | |||||||
| 			} | 			} | ||||||
| 			composeTag := img.(reference.NamedTagged).Tag() | 			composeTag := img.(reference.NamedTagged).Tag() | ||||||
|  |  | ||||||
| 			logrus.Debugf("parsed '%s' from '%s'", composeTag, service.Image) | 			logrus.Debugf("parsed %s from %s", composeTag, service.Image) | ||||||
|  |  | ||||||
| 			if image == composeImage { | 			if image == composeImage { | ||||||
| 				bytes, err := ioutil.ReadFile(composeFile) | 				bytes, err := ioutil.ReadFile(composeFile) | ||||||
| @ -69,9 +69,9 @@ func UpdateTag(pattern, image, tag, recipeName string) error { | |||||||
| 				new := fmt.Sprintf("%s:%s", composeImage, tag) | 				new := fmt.Sprintf("%s:%s", composeImage, tag) | ||||||
| 				replacedBytes := strings.Replace(string(bytes), old, new, -1) | 				replacedBytes := strings.Replace(string(bytes), old, new, -1) | ||||||
|  |  | ||||||
| 				logrus.Debugf("updating '%s' to '%s' in '%s'", old, new, compose.Filename) | 				logrus.Debugf("updating %s to %s in %s", old, new, compose.Filename) | ||||||
|  |  | ||||||
| 				if err := ioutil.WriteFile(compose.Filename, []byte(replacedBytes), 0644); err != nil { | 				if err := ioutil.WriteFile(compose.Filename, []byte(replacedBytes), 0764); err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @ -88,12 +88,12 @@ func UpdateLabel(pattern, serviceName, label, recipeName string) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("considering '%s' config(s) for label update", strings.Join(composeFiles, ", ")) | 	logrus.Debugf("considering %s config(s) for label update", strings.Join(composeFiles, ", ")) | ||||||
|  |  | ||||||
| 	for _, composeFile := range composeFiles { | 	for _, composeFile := range composeFiles { | ||||||
| 		opts := stack.Deploy{Composefiles: []string{composeFile}} | 		opts := stack.Deploy{Composefiles: []string{composeFile}} | ||||||
|  |  | ||||||
| 		envSamplePath := path.Join(config.ABRA_DIR, "apps", recipeName, ".env.sample") | 		envSamplePath := path.Join(config.RECIPES_DIR, recipeName, ".env.sample") | ||||||
| 		sampleEnv, err := config.ReadEnv(envSamplePath) | 		sampleEnv, err := config.ReadEnv(envSamplePath) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -130,19 +130,25 @@ func UpdateLabel(pattern, serviceName, label, recipeName string) error { | |||||||
| 				old := fmt.Sprintf("coop-cloud.${STACK_NAME}.version=%s", value) | 				old := fmt.Sprintf("coop-cloud.${STACK_NAME}.version=%s", value) | ||||||
| 				replacedBytes := strings.Replace(string(bytes), old, label, -1) | 				replacedBytes := strings.Replace(string(bytes), old, label, -1) | ||||||
|  |  | ||||||
|  | 				if old == label { | ||||||
|  | 					logrus.Warnf("%s is already set, nothing to do?", label) | ||||||
|  | 					return nil | ||||||
|  | 				} | ||||||
|  |  | ||||||
| 				logrus.Debugf("updating %s to %s in %s", old, label, compose.Filename) | 				logrus.Debugf("updating %s to %s in %s", old, label, compose.Filename) | ||||||
|  |  | ||||||
| 				if err := ioutil.WriteFile(compose.Filename, []byte(replacedBytes), 0644); err != nil { | 				if err := ioutil.WriteFile(compose.Filename, []byte(replacedBytes), 0764); err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
|  | 				logrus.Infof("synced label %s to service %s", label, serviceName) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !discovered { | 		if !discovered { | ||||||
| 			logrus.Warn("no existing label found, cannot continue...") | 			logrus.Warn("no existing label found, automagic insertion not supported yet") | ||||||
| 			logrus.Fatalf("add '%s' manually, automagic insertion not supported yet", label) | 			logrus.Fatalf("add '- \"%s\"' manually to the 'app' service in %s", label, composeFile) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
|  | |||||||
| @ -1,14 +1,14 @@ | |||||||
| package config | package config | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"html/template" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path" | 	"path" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/formatter" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/convert" | 	"coopcloud.tech/abra/pkg/upstream/convert" | ||||||
| 	loader "coopcloud.tech/abra/pkg/upstream/stack" | 	loader "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| @ -43,13 +43,17 @@ type App struct { | |||||||
| 	Path   string | 	Path   string | ||||||
| } | } | ||||||
|  |  | ||||||
| // StackName gets what the docker safe stack name is for the app | // StackName gets what the docker safe stack name is for the app. This should | ||||||
|  | // not not shown to the user, use a.Name for that. Give the output of this | ||||||
|  | // command to Docker only. | ||||||
| func (a App) StackName() string { | func (a App) StackName() string { | ||||||
| 	if _, exists := a.Env["STACK_NAME"]; exists { | 	if _, exists := a.Env["STACK_NAME"]; exists { | ||||||
| 		return a.Env["STACK_NAME"] | 		return a.Env["STACK_NAME"] | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	stackName := SanitiseAppName(a.Name) | 	stackName := SanitiseAppName(a.Name) | ||||||
| 	a.Env["STACK_NAME"] = stackName | 	a.Env["STACK_NAME"] = stackName | ||||||
|  |  | ||||||
| 	return stackName | 	return stackName | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -97,14 +101,14 @@ func (a ByName) Less(i, j int) bool { | |||||||
| func readAppEnvFile(appFile AppFile, name AppName) (App, error) { | func readAppEnvFile(appFile AppFile, name AppName) (App, error) { | ||||||
| 	env, err := ReadEnv(appFile.Path) | 	env, err := ReadEnv(appFile.Path) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return App{}, fmt.Errorf("env file for '%s' couldn't be read: %s", name, err.Error()) | 		return App{}, fmt.Errorf("env file for %s couldn't be read: %s", name, err.Error()) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("read env '%s' from '%s'", env, appFile.Path) | 	logrus.Debugf("read env %s from %s", env, appFile.Path) | ||||||
|  |  | ||||||
| 	app, err := newApp(env, name, appFile) | 	app, err := newApp(env, name, appFile) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return App{}, fmt.Errorf("env file for '%s' has issues: %s", name, err.Error()) | 		return App{}, fmt.Errorf("env file for %s has issues: %s", name, err.Error()) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return app, nil | 	return app, nil | ||||||
| @ -112,17 +116,17 @@ func readAppEnvFile(appFile AppFile, name AppName) (App, error) { | |||||||
|  |  | ||||||
| // newApp creates new App object | // newApp creates new App object | ||||||
| func newApp(env AppEnv, name string, appFile AppFile) (App, error) { | func newApp(env AppEnv, name string, appFile AppFile) (App, error) { | ||||||
| 	// Checking for type as it is required - apps wont work without it |  | ||||||
| 	domain := env["DOMAIN"] | 	domain := env["DOMAIN"] | ||||||
| 	apptype, ok := env["TYPE"] |  | ||||||
| 	if !ok { | 	appType, exists := env["TYPE"] | ||||||
| 		return App{}, errors.New("missing TYPE variable") | 	if !exists { | ||||||
|  | 		return App{}, fmt.Errorf("%s is missing the TYPE env var", name) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return App{ | 	return App{ | ||||||
| 		Name:   name, | 		Name:   name, | ||||||
| 		Domain: domain, | 		Domain: domain, | ||||||
| 		Type:   apptype, | 		Type:   appType, | ||||||
| 		Env:    env, | 		Env:    env, | ||||||
| 		Server: appFile.Server, | 		Server: appFile.Server, | ||||||
| 		Path:   appFile.Path, | 		Path:   appFile.Path, | ||||||
| @ -136,24 +140,24 @@ func LoadAppFiles(servers ...string) (AppFiles, error) { | |||||||
| 		if servers[0] == "" { | 		if servers[0] == "" { | ||||||
| 			// Empty servers flag, one string will always be passed | 			// Empty servers flag, one string will always be passed | ||||||
| 			var err error | 			var err error | ||||||
| 			servers, err = getAllFoldersInDirectory(ABRA_SERVER_FOLDER) | 			servers, err = GetAllFoldersInDirectory(SERVERS_DIR) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return nil, err | 				return nil, err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("collecting metadata from '%v' servers: '%s'", len(servers), strings.Join(servers, ", ")) | 	logrus.Debugf("collecting metadata from %v servers: %s", len(servers), strings.Join(servers, ", ")) | ||||||
|  |  | ||||||
| 	for _, server := range servers { | 	for _, server := range servers { | ||||||
| 		serverDir := path.Join(ABRA_SERVER_FOLDER, server) | 		serverDir := path.Join(SERVERS_DIR, server) | ||||||
| 		files, err := getAllFilesInDirectory(serverDir) | 		files, err := getAllFilesInDirectory(serverDir) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		for _, file := range files { | 		for _, file := range files { | ||||||
| 			appName := strings.TrimSuffix(file.Name(), ".env") | 			appName := strings.TrimSuffix(file.Name(), ".env") | ||||||
| 			appFilePath := path.Join(ABRA_SERVER_FOLDER, server, file.Name()) | 			appFilePath := path.Join(SERVERS_DIR, server, file.Name()) | ||||||
| 			appFiles[appName] = AppFile{ | 			appFiles[appName] = AppFile{ | ||||||
| 				Path:   appFilePath, | 				Path:   appFilePath, | ||||||
| 				Server: server, | 				Server: server, | ||||||
| @ -169,7 +173,7 @@ func LoadAppFiles(servers ...string) (AppFiles, error) { | |||||||
| func GetApp(apps AppFiles, name AppName) (App, error) { | func GetApp(apps AppFiles, name AppName) (App, error) { | ||||||
| 	appFile, exists := apps[name] | 	appFile, exists := apps[name] | ||||||
| 	if !exists { | 	if !exists { | ||||||
| 		return App{}, fmt.Errorf("cannot find app with name '%s'", name) | 		return App{}, fmt.Errorf("cannot find app with name %s", name) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	app, err := readAppEnvFile(appFile, name) | 	app, err := readAppEnvFile(appFile, name) | ||||||
| @ -249,27 +253,39 @@ func GetAppNames() ([]string, error) { | |||||||
| } | } | ||||||
|  |  | ||||||
| // TemplateAppEnvSample copies the example env file for the app into the users env files | // TemplateAppEnvSample copies the example env file for the app into the users env files | ||||||
| func TemplateAppEnvSample(appType, appName, server, domain, recipe string) error { | func TemplateAppEnvSample(recipeName, appName, server, domain string) error { | ||||||
| 	envSamplePath := path.Join(ABRA_DIR, "apps", appType, ".env.sample") | 	envSamplePath := path.Join(RECIPES_DIR, recipeName, ".env.sample") | ||||||
| 	envSample, err := ioutil.ReadFile(envSamplePath) | 	envSample, err := ioutil.ReadFile(envSamplePath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	appEnvPath := path.Join(ABRA_DIR, "servers", server, fmt.Sprintf("%s.env", appName)) | 	appEnvPath := path.Join(ABRA_DIR, "servers", server, fmt.Sprintf("%s.env", appName)) | ||||||
| 	if _, err := os.Stat(appEnvPath); err == nil { | 	if _, err := os.Stat(appEnvPath); os.IsExist(err) { | ||||||
| 		return fmt.Errorf("%s already exists?", appEnvPath) | 		return fmt.Errorf("%s already exists?", appEnvPath) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	envSample = []byte(strings.Replace(string(envSample), fmt.Sprintf("%s.example.com", recipe), domain, -1)) | 	err = ioutil.WriteFile(appEnvPath, envSample, 0664) | ||||||
| 	envSample = []byte(strings.Replace(string(envSample), "example.com", domain, -1)) |  | ||||||
|  |  | ||||||
| 	err = ioutil.WriteFile(appEnvPath, envSample, 0755) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("copied '%s' to '%s'", envSamplePath, appEnvPath) | 	file, err := os.OpenFile(appEnvPath, os.O_RDWR, 0664) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  |  | ||||||
|  | 	tpl, err := template.ParseFiles(appEnvPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := tpl.Execute(file, struct{ Name string }{recipeName}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("copied & templated %s to %s", envSamplePath, appEnvPath) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -315,9 +331,6 @@ func GetAppStatuses(appFiles AppFiles) (map[string]map[string]string, error) { | |||||||
| 			if version, ok := service.Spec.Labels[labelKey]; ok { | 			if version, ok := service.Spec.Labels[labelKey]; ok { | ||||||
| 				result["version"] = version | 				result["version"] = version | ||||||
| 			} else { | 			} else { | ||||||
| 				//FIXME: we only need to check containers with the version label not |  | ||||||
| 				//       every single container and then skip when we see no label perf gains |  | ||||||
| 				//       to be had here |  | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @ -325,7 +338,7 @@ func GetAppStatuses(appFiles AppFiles) (map[string]map[string]string, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved app statuses: '%s'", statuses) | 	logrus.Debugf("retrieved app statuses: %s", statuses) | ||||||
|  |  | ||||||
| 	return statuses, nil | 	return statuses, nil | ||||||
| } | } | ||||||
| @ -337,20 +350,20 @@ func GetAppComposeFiles(recipe string, appEnv AppEnv) ([]string, error) { | |||||||
|  |  | ||||||
| 	if _, ok := appEnv["COMPOSE_FILE"]; !ok { | 	if _, ok := appEnv["COMPOSE_FILE"]; !ok { | ||||||
| 		logrus.Debug("no COMPOSE_FILE detected, loading compose.yml") | 		logrus.Debug("no COMPOSE_FILE detected, loading compose.yml") | ||||||
| 		path := fmt.Sprintf("%s/%s/compose.yml", APPS_DIR, recipe) | 		path := fmt.Sprintf("%s/%s/compose.yml", RECIPES_DIR, recipe) | ||||||
| 		composeFiles = append(composeFiles, path) | 		composeFiles = append(composeFiles, path) | ||||||
| 		return composeFiles, nil | 		return composeFiles, nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	composeFileEnvVar := appEnv["COMPOSE_FILE"] | 	composeFileEnvVar := appEnv["COMPOSE_FILE"] | ||||||
| 	envVars := strings.Split(composeFileEnvVar, ":") | 	envVars := strings.Split(composeFileEnvVar, ":") | ||||||
| 	logrus.Debugf("COMPOSE_FILE detected ('%s'), loading '%s'", composeFileEnvVar, strings.Join(envVars, ", ")) | 	logrus.Debugf("COMPOSE_FILE detected (%s), loading %s", composeFileEnvVar, strings.Join(envVars, ", ")) | ||||||
| 	for _, file := range strings.Split(composeFileEnvVar, ":") { | 	for _, file := range strings.Split(composeFileEnvVar, ":") { | ||||||
| 		path := fmt.Sprintf("%s/%s/%s", APPS_DIR, recipe, file) | 		path := fmt.Sprintf("%s/%s/%s", RECIPES_DIR, recipe, file) | ||||||
| 		composeFiles = append(composeFiles, path) | 		composeFiles = append(composeFiles, path) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved '%s' configs for '%s'", strings.Join(composeFiles, ", "), recipe) | 	logrus.Debugf("retrieved %s configs for %s", strings.Join(composeFiles, ", "), recipe) | ||||||
|  |  | ||||||
| 	return composeFiles, nil | 	return composeFiles, nil | ||||||
| } | } | ||||||
| @ -364,7 +377,7 @@ func GetAppComposeConfig(recipe string, opts stack.Deploy, appEnv AppEnv) (*comp | |||||||
| 		return &composetypes.Config{}, err | 		return &composetypes.Config{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved '%s' for '%s'", compose.Filename, recipe) | 	logrus.Debugf("retrieved %s for %s", compose.Filename, recipe) | ||||||
|  |  | ||||||
| 	return compose, nil | 	return compose, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -26,7 +26,6 @@ func TestReadAppEnvFile(t *testing.T) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func TestGetApp(t *testing.T) { | func TestGetApp(t *testing.T) { | ||||||
| 	// TODO: Test failures as well as successes |  | ||||||
| 	app, err := GetApp(expectedAppFiles, appName) | 	app, err := GetApp(expectedAppFiles, appName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
|  | |||||||
| @ -15,21 +15,23 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| var ABRA_DIR = os.ExpandEnv("$HOME/.abra") | var ABRA_DIR = os.ExpandEnv("$HOME/.abra") | ||||||
| var ABRA_SERVER_FOLDER = path.Join(ABRA_DIR, "servers") | var SERVERS_DIR = path.Join(ABRA_DIR, "servers") | ||||||
| var APPS_JSON = path.Join(ABRA_DIR, "apps.json") | var RECIPES_DIR = path.Join(ABRA_DIR, "apps") | ||||||
| var APPS_DIR = path.Join(ABRA_DIR, "apps") | var VENDOR_DIR = path.Join(ABRA_DIR, "vendor") | ||||||
|  | var RECIPES_JSON = path.Join(ABRA_DIR, "catalogue", "recipes.json") | ||||||
| var REPOS_BASE_URL = "https://git.coopcloud.tech/coop-cloud" | var REPOS_BASE_URL = "https://git.coopcloud.tech/coop-cloud" | ||||||
|  | var SSH_URL_TEMPLATE = "ssh://git@git.coopcloud.tech:2222/coop-cloud/%s.git" | ||||||
|  |  | ||||||
| // GetServers retrieves all servers. | // GetServers retrieves all servers. | ||||||
| func GetServers() ([]string, error) { | func GetServers() ([]string, error) { | ||||||
| 	var servers []string | 	var servers []string | ||||||
|  |  | ||||||
| 	servers, err := getAllFoldersInDirectory(ABRA_SERVER_FOLDER) | 	servers, err := GetAllFoldersInDirectory(SERVERS_DIR) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return servers, err | 		return servers, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("retrieved '%v' servers: '%s'", len(servers), servers) | 	logrus.Debugf("retrieved %v servers: %s", len(servers), servers) | ||||||
|  |  | ||||||
| 	return servers, nil | 	return servers, nil | ||||||
| } | } | ||||||
| @ -43,20 +45,20 @@ func ReadEnv(filePath string) (AppEnv, error) { | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("read '%s' from '%s'", envFile, filePath) | 	logrus.Debugf("read %s from %s", envFile, filePath) | ||||||
|  |  | ||||||
| 	return envFile, nil | 	return envFile, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // ReadServerNames retrieves all server names. | // ReadServerNames retrieves all server names. | ||||||
| func ReadServerNames() ([]string, error) { | func ReadServerNames() ([]string, error) { | ||||||
| 	serverNames, err := getAllFoldersInDirectory(ABRA_SERVER_FOLDER) | 	serverNames, err := GetAllFoldersInDirectory(SERVERS_DIR) | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("read '%s' from '%s'", strings.Join(serverNames, ","), ABRA_SERVER_FOLDER) | 	logrus.Debugf("read %s from %s", strings.Join(serverNames, ","), SERVERS_DIR) | ||||||
|  |  | ||||||
| 	return serverNames, nil | 	return serverNames, nil | ||||||
| } | } | ||||||
| @ -80,7 +82,7 @@ func getAllFilesInDirectory(directory string) ([]fs.FileInfo, error) { | |||||||
|  |  | ||||||
| 		realPath, err := filepath.EvalSymlinks(filePath) | 		realPath, err := filepath.EvalSymlinks(filePath) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Warningf("broken symlink in your abra config folders: '%s'", filePath) | 			logrus.Warningf("broken symlink in your abra config folders: %s", filePath) | ||||||
| 		} else { | 		} else { | ||||||
| 			realFile, err := os.Stat(realPath) | 			realFile, err := os.Stat(realPath) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| @ -95,8 +97,8 @@ func getAllFilesInDirectory(directory string) ([]fs.FileInfo, error) { | |||||||
| 	return realFiles, nil | 	return realFiles, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // getAllFoldersInDirectory returns both folder and symlink paths | // GetAllFoldersInDirectory returns both folder and symlink paths | ||||||
| func getAllFoldersInDirectory(directory string) ([]string, error) { | func GetAllFoldersInDirectory(directory string) ([]string, error) { | ||||||
| 	var folders []string | 	var folders []string | ||||||
|  |  | ||||||
| 	files, err := ioutil.ReadDir(directory) | 	files, err := ioutil.ReadDir(directory) | ||||||
| @ -104,7 +106,7 @@ func getAllFoldersInDirectory(directory string) ([]string, error) { | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if len(files) == 0 { | 	if len(files) == 0 { | ||||||
| 		return nil, fmt.Errorf("directory is empty: '%s'", directory) | 		return nil, fmt.Errorf("directory is empty: %s", directory) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, file := range files { | 	for _, file := range files { | ||||||
| @ -113,7 +115,7 @@ func getAllFoldersInDirectory(directory string) ([]string, error) { | |||||||
| 			filePath := path.Join(directory, file.Name()) | 			filePath := path.Join(directory, file.Name()) | ||||||
| 			realDir, err := filepath.EvalSymlinks(filePath) | 			realDir, err := filepath.EvalSymlinks(filePath) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				logrus.Warningf("broken symlink in your abra config folders: '%s'", filePath) | 				logrus.Warningf("broken symlink in your abra config folders: %s", filePath) | ||||||
| 			} else if stat, err := os.Stat(realDir); err == nil && stat.IsDir() { | 			} else if stat, err := os.Stat(realDir); err == nil && stat.IsDir() { | ||||||
| 				// path is a directory | 				// path is a directory | ||||||
| 				folders = append(folders, file.Name()) | 				folders = append(folders, file.Name()) | ||||||
| @ -124,17 +126,6 @@ func getAllFoldersInDirectory(directory string) ([]string, error) { | |||||||
| 	return folders, nil | 	return folders, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // EnsureAbraDirExists checks for the abra config folder and throws error if not |  | ||||||
| func EnsureAbraDirExists() error { |  | ||||||
| 	if _, err := os.Stat(ABRA_DIR); os.IsNotExist(err) { |  | ||||||
| 		logrus.Debugf("'%s' does not exist, creating it", ABRA_DIR) |  | ||||||
| 		if err := os.Mkdir(ABRA_DIR, 0777); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ReadAbraShEnvVars reads env vars from an abra.sh recipe file. | // ReadAbraShEnvVars reads env vars from an abra.sh recipe file. | ||||||
| func ReadAbraShEnvVars(abraSh string) (map[string]string, error) { | func ReadAbraShEnvVars(abraSh string) (map[string]string, error) { | ||||||
| 	envVars := make(map[string]string) | 	envVars := make(map[string]string) | ||||||
| @ -161,7 +152,7 @@ func ReadAbraShEnvVars(abraSh string) (map[string]string, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("read '%s' from '%s'", envVars, abraSh) | 	logrus.Debugf("read %s from %s", envVars, abraSh) | ||||||
|  |  | ||||||
| 	return envVars, nil | 	return envVars, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -44,7 +44,7 @@ var expectedAppFiles = map[string]AppFile{ | |||||||
| // var expectedServerNames = []string{"evil.corp"} | // var expectedServerNames = []string{"evil.corp"} | ||||||
|  |  | ||||||
| func TestGetAllFoldersInDirectory(t *testing.T) { | func TestGetAllFoldersInDirectory(t *testing.T) { | ||||||
| 	folders, err := getAllFoldersInDirectory(testFolder) | 	folders, err := GetAllFoldersInDirectory(testFolder) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										70
									
								
								pkg/container/container.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								pkg/container/container.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | |||||||
|  | package container | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"github.com/AlecAivazis/survey/v2" | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	"github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // GetContainer retrieves a container. If prompt is true and the retrievd count | ||||||
|  | // of containers does not match 1, then a prompt is presented to let the user | ||||||
|  | // choose. A count of 0 is handled gracefully. | ||||||
|  | func GetContainer(c context.Context, cl *client.Client, filters filters.Args, prompt bool) (types.Container, error) { | ||||||
|  | 	containerOpts := types.ContainerListOptions{Filters: filters} | ||||||
|  | 	containers, err := cl.ContainerList(c, containerOpts) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return types.Container{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(containers) == 0 { | ||||||
|  | 		filter := filters.Get("name")[0] | ||||||
|  | 		return types.Container{}, fmt.Errorf("no containers matching the %v filter found?", filter) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(containers) != 1 { | ||||||
|  | 		var containersRaw []string | ||||||
|  | 		for _, container := range containers { | ||||||
|  | 			containerName := strings.Join(container.Names, " ") | ||||||
|  | 			trimmed := strings.TrimPrefix(containerName, "/") | ||||||
|  | 			created := formatter.HumanDuration(container.Created) | ||||||
|  | 			containersRaw = append(containersRaw, fmt.Sprintf("%s (created %v)", trimmed, created)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !prompt { | ||||||
|  | 			err := fmt.Errorf("expected 1 container but found %v: %s", len(containers), strings.Join(containersRaw, " ")) | ||||||
|  | 			return types.Container{}, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Warnf("ambiguous container list received, prompting for input") | ||||||
|  |  | ||||||
|  | 		var response string | ||||||
|  | 		prompt := &survey.Select{ | ||||||
|  | 			Message: "which container are you looking for?", | ||||||
|  | 			Options: containersRaw, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &response); err != nil { | ||||||
|  | 			return types.Container{}, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		chosenContainer := strings.TrimSpace(strings.Split(response, " ")[0]) | ||||||
|  | 		for _, container := range containers { | ||||||
|  | 			containerName := strings.TrimSpace(strings.Join(container.Names, " ")) | ||||||
|  | 			trimmed := strings.TrimPrefix(containerName, "/") | ||||||
|  | 			if trimmed == chosenContainer { | ||||||
|  | 				return container, nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Panic("failed to match chosen container") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return containers[0], nil | ||||||
|  | } | ||||||
| @ -47,7 +47,7 @@ func EnsureIPv4(domainName string) (string, error) { | |||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("created DNS resolver via '%s'", freifunkDNS) | 	logrus.Debugf("created DNS resolver via %s", freifunkDNS) | ||||||
|  |  | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
| 	ips, err := resolver.LookupIPAddr(ctx, domainName) | 	ips, err := resolver.LookupIPAddr(ctx, domainName) | ||||||
| @ -94,3 +94,12 @@ func EnsureDomainsResolveSameIPv4(domainName, server string) (string, error) { | |||||||
|  |  | ||||||
| 	return ipv4, nil | 	return ipv4, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetTTL parses a ttl string into a duration | ||||||
|  | func GetTTL(ttl string) (time.Duration, error) { | ||||||
|  | 	val, err := time.ParseDuration(ttl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return val, err | ||||||
|  | 	} | ||||||
|  | 	return val, nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -1,12 +1,10 @@ | |||||||
| package formatter | package formatter | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"os" | 	"os" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/docker/cli/cli/command/formatter" |  | ||||||
| 	"github.com/docker/go-units" | 	"github.com/docker/go-units" | ||||||
| 	"github.com/olekukonko/tablewriter" | 	"github.com/olekukonko/tablewriter" | ||||||
| 	"github.com/schollz/progressbar/v3" | 	"github.com/schollz/progressbar/v3" | ||||||
| @ -16,10 +14,6 @@ func ShortenID(str string) string { | |||||||
| 	return str[:12] | 	return str[:12] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Truncate(str string) string { |  | ||||||
| 	return fmt.Sprintf(`"%s"`, formatter.Ellipsis(str, 19)) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func SmallSHA(hash string) string { | func SmallSHA(hash string) string { | ||||||
| 	return hash[:8] | 	return hash[:8] | ||||||
| } | } | ||||||
| @ -39,6 +33,7 @@ func HumanDuration(timestamp int64) string { | |||||||
| // CreateTable prepares a table layout for output. | // CreateTable prepares a table layout for output. | ||||||
| func CreateTable(columns []string) *tablewriter.Table { | func CreateTable(columns []string) *tablewriter.Table { | ||||||
| 	table := tablewriter.NewWriter(os.Stdout) | 	table := tablewriter.NewWriter(os.Stdout) | ||||||
|  | 	table.SetAutoWrapText(false) | ||||||
| 	table.SetHeader(columns) | 	table.SetHeader(columns) | ||||||
| 	return table | 	return table | ||||||
| } | } | ||||||
							
								
								
									
										35
									
								
								pkg/git/branch.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								pkg/git/branch.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | package git | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/go-git/go-git/v5/plumbing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // GetCurrentBranch retrieves the current branch of a repository | ||||||
|  | func GetCurrentBranch(repository *git.Repository) (string, error) { | ||||||
|  | 	branchRefs, err := repository.Branches() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	headRef, err := repository.Head() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var currentBranchName string | ||||||
|  | 	err = branchRefs.ForEach(func(branchRef *plumbing.Reference) error { | ||||||
|  | 		if branchRef.Hash() == headRef.Hash() { | ||||||
|  | 			currentBranchName = branchRef.Name().String() | ||||||
|  |  | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return currentBranchName, nil | ||||||
|  | } | ||||||
| @ -7,7 +7,6 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
| 	"github.com/go-git/go-git/v5/config" |  | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" | 	"github.com/go-git/go-git/v5/plumbing" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
| @ -15,10 +14,10 @@ import ( | |||||||
| // Clone runs a git clone which accounts for different default branches. | // Clone runs a git clone which accounts for different default branches. | ||||||
| func Clone(dir, url string) error { | func Clone(dir, url string) error { | ||||||
| 	if _, err := os.Stat(dir); os.IsNotExist(err) { | 	if _, err := os.Stat(dir); os.IsNotExist(err) { | ||||||
| 		logrus.Debugf("'%s' does not exist, attempting to git clone from '%s'", dir, url) | 		logrus.Debugf("%s does not exist, attempting to git clone from %s", dir, url) | ||||||
| 		_, err := git.PlainClone(dir, false, &git.CloneOptions{URL: url, Tags: git.AllTags}) | 		_, err := git.PlainClone(dir, false, &git.CloneOptions{URL: url, Tags: git.AllTags}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Debugf("cloning '%s' default branch failed, attempting from main branch", url) | 			logrus.Debugf("cloning %s default branch failed, attempting from main branch", url) | ||||||
| 			_, err := git.PlainClone(dir, false, &git.CloneOptions{ | 			_, err := git.PlainClone(dir, false, &git.CloneOptions{ | ||||||
| 				URL:           url, | 				URL:           url, | ||||||
| 				Tags:          git.AllTags, | 				Tags:          git.AllTags, | ||||||
| @ -32,67 +31,10 @@ func Clone(dir, url string) error { | |||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		logrus.Debugf("'%s' has been git cloned successfully", dir) | 		logrus.Debugf("%s has been git cloned successfully", dir) | ||||||
| 	} else { | 	} else { | ||||||
| 		logrus.Debugf("'%s' already exists, doing nothing", dir) | 		logrus.Debugf("%s already exists", dir) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // EnsureUpToDate ensures that a git repo on disk has the latest changes (git-fetch). |  | ||||||
| func EnsureUpToDate(dir string) error { |  | ||||||
| 	repo, err := git.PlainOpen(dir) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	branch := "master" |  | ||||||
| 	if _, err := repo.Branch("master"); err != nil { |  | ||||||
| 		if _, err := repo.Branch("main"); err != nil { |  | ||||||
| 			logrus.Debugf("failed to select branch in '%s'", dir) |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		branch = "main" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("choosing '%s' as main git branch in '%s'", branch, dir) |  | ||||||
|  |  | ||||||
| 	worktree, err := repo.Worktree() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	refName := fmt.Sprintf("refs/heads/%s", branch) |  | ||||||
| 	checkOutOpts := &git.CheckoutOptions{ |  | ||||||
| 		Create: false, |  | ||||||
| 		Force:  true, |  | ||||||
| 		Branch: plumbing.ReferenceName(refName), |  | ||||||
| 	} |  | ||||||
| 	if err := worktree.Checkout(checkOutOpts); err != nil { |  | ||||||
| 		logrus.Debugf("failed to check out '%s' in '%s'", refName, dir) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("successfully checked out '%s' in '%s'", branch, dir) |  | ||||||
|  |  | ||||||
| 	remote, err := repo.Remote("origin") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	fetchOpts := &git.FetchOptions{ |  | ||||||
| 		RemoteName: "origin", |  | ||||||
| 		RefSpecs:   []config.RefSpec{"refs/heads/*:refs/remotes/origin/*"}, |  | ||||||
| 		Force:      true, |  | ||||||
| 	} |  | ||||||
| 	if err := remote.Fetch(fetchOpts); err != nil { |  | ||||||
| 		if !strings.Contains(err.Error(), "already up-to-date") { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Debugf("successfully fetched all changes in '%s'", dir) |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  | |||||||
							
								
								
									
										56
									
								
								pkg/git/commit.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								pkg/git/commit.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | |||||||
|  | package git | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Commit runs a git commit | ||||||
|  | func Commit(repoPath, glob, commitMessage string, dryRun bool) error { | ||||||
|  | 	if commitMessage == "" { | ||||||
|  | 		return fmt.Errorf("no commit message specified?") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	commitRepo, err := git.PlainOpen(repoPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	commitWorktree, err := commitRepo.Worktree() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	patterns, err := GetExcludesFiles() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(patterns) > 0 { | ||||||
|  | 		commitWorktree.Excludes = append(patterns, commitWorktree.Excludes...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !dryRun { | ||||||
|  | 		err = commitWorktree.AddGlob(glob) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		logrus.Debugf("staged %s for commit", glob) | ||||||
|  | 	} else { | ||||||
|  | 		logrus.Debugf("dry run: did not stage %s for commit", glob) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !dryRun { | ||||||
|  | 		_, err = commitWorktree.Commit(commitMessage, &git.CommitOptions{}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		logrus.Debug("git changes commited") | ||||||
|  | 	} else { | ||||||
|  | 		logrus.Debug("dry run: no changes commited") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @ -24,7 +24,7 @@ func Init(repoPath string, commit bool) error { | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := commitWorktree.AddGlob("**"); err != nil { | 		if err := commitWorktree.AddWithOptions(&git.AddOptions{All: true}); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										43
									
								
								pkg/git/push.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								pkg/git/push.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | |||||||
|  | package git | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/go-git/go-git/v5/config" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Push pushes the latest changes & optionally tags to the default remote | ||||||
|  | func Push(repoDir string, remote string, tags bool, dryRun bool) error { | ||||||
|  | 	if dryRun { | ||||||
|  | 		logrus.Debugf("dry run: no git changes pushed in %s", repoDir) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	commitRepo, err := git.PlainOpen(repoDir) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	opts := &git.PushOptions{} | ||||||
|  | 	if remote != "" { | ||||||
|  | 		opts.RemoteName = remote | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := commitRepo.Push(opts); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("git changes pushed") | ||||||
|  |  | ||||||
|  | 	if tags { | ||||||
|  | 		opts.RefSpecs = append(opts.RefSpecs, config.RefSpec("+refs/tags/*:refs/tags/*")) | ||||||
|  |  | ||||||
|  | 		if err := commitRepo.Push(opts); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Debugf("git tags pushed") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										146
									
								
								pkg/git/read.go
									
									
									
									
									
								
							
							
						
						
									
										146
									
								
								pkg/git/read.go
									
									
									
									
									
								
							| @ -1,17 +1,24 @@ | |||||||
| package git | package git | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"os" | ||||||
|  | 	"os/user" | ||||||
| 	"path" | 	"path" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	gitConfigPkg "github.com/go-git/go-git/v5/config" | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" | 	"github.com/go-git/go-git/v5/plumbing" | ||||||
|  | 	"github.com/go-git/go-git/v5/plumbing/format/gitignore" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // GetRecipeHead retrieves latest HEAD metadata. | // GetRecipeHead retrieves latest HEAD metadata. | ||||||
| func GetRecipeHead(recipeName string) (*plumbing.Reference, error) { | func GetRecipeHead(recipeName string) (*plumbing.Reference, error) { | ||||||
| 	recipeDir := path.Join(config.ABRA_DIR, "apps", recipeName) | 	recipeDir := path.Join(config.RECIPES_DIR, recipeName) | ||||||
|  |  | ||||||
| 	repo, err := git.PlainOpen(recipeDir) | 	repo, err := git.PlainOpen(recipeDir) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -27,10 +34,8 @@ func GetRecipeHead(recipeName string) (*plumbing.Reference, error) { | |||||||
| } | } | ||||||
|  |  | ||||||
| // IsClean checks if a repo has unstaged changes | // IsClean checks if a repo has unstaged changes | ||||||
| func IsClean(recipeName string) (bool, error) { | func IsClean(repoPath string) (bool, error) { | ||||||
| 	recipeDir := path.Join(config.ABRA_DIR, "apps", recipeName) | 	repo, err := git.PlainOpen(repoPath) | ||||||
|  |  | ||||||
| 	repo, err := git.PlainOpen(recipeDir) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return false, err | 		return false, err | ||||||
| 	} | 	} | ||||||
| @ -40,16 +45,143 @@ func IsClean(recipeName string) (bool, error) { | |||||||
| 		return false, err | 		return false, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	patterns, err := GetExcludesFiles() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(patterns) > 0 { | ||||||
|  | 		worktree.Excludes = append(patterns, worktree.Excludes...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	status, err := worktree.Status() | 	status, err := worktree.Status() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return false, err | 		return false, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if status.String() != "" { | 	if status.String() != "" { | ||||||
| 		logrus.Debugf("discovered git status for %s repository: %s", recipeName, status.String()) | 		logrus.Debugf("discovered git status in %s: %s", repoPath, status.String()) | ||||||
| 	} else { | 	} else { | ||||||
| 		logrus.Debugf("discovered clean git status for %s repository", recipeName) | 		logrus.Debugf("discovered clean git status in %s", repoPath) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return status.IsClean(), nil | 	return status.IsClean(), nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetExcludesFiles reads the exlude files from a global gitignore | ||||||
|  | func GetExcludesFiles() ([]gitignore.Pattern, error) { | ||||||
|  | 	var err error | ||||||
|  | 	var patterns []gitignore.Pattern | ||||||
|  |  | ||||||
|  | 	cfg, err := parseGitConfig() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return patterns, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	excludesfile := getExcludesFile(cfg) | ||||||
|  | 	patterns, err = parseExcludesFile(excludesfile) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return patterns, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return patterns, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func parseGitConfig() (*gitConfigPkg.Config, error) { | ||||||
|  | 	cfg := gitConfigPkg.NewConfig() | ||||||
|  |  | ||||||
|  | 	usr, err := user.Current() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	globalGitConfig := filepath.Join(usr.HomeDir, ".gitconfig") | ||||||
|  | 	if _, err := os.Stat(globalGitConfig); err != nil { | ||||||
|  | 		if os.IsNotExist(err) { | ||||||
|  | 			logrus.Debugf("no %s exists, not reading any global gitignore config", globalGitConfig) | ||||||
|  | 			return cfg, nil | ||||||
|  | 		} | ||||||
|  | 		return cfg, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	b, err := ioutil.ReadFile(globalGitConfig) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := cfg.Unmarshal(b); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return cfg, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func getExcludesFile(cfg *gitConfigPkg.Config) string { | ||||||
|  | 	for _, sec := range cfg.Raw.Sections { | ||||||
|  | 		if sec.Name == "core" { | ||||||
|  | 			for _, opt := range sec.Options { | ||||||
|  | 				if opt.Key == "excludesfile" { | ||||||
|  | 					return opt.Value | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return "~/.gitignore" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func parseExcludesFile(excludesfile string) ([]gitignore.Pattern, error) { | ||||||
|  | 	var ps []gitignore.Pattern | ||||||
|  |  | ||||||
|  | 	excludesfile, err := expandTilde(excludesfile) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := os.Stat(excludesfile); err != nil { | ||||||
|  | 		if os.IsNotExist(err) { | ||||||
|  | 			logrus.Debugf("no %s exists, skipping reading gitignore paths", excludesfile) | ||||||
|  | 			return ps, nil | ||||||
|  | 		} | ||||||
|  | 		return ps, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	data, err := ioutil.ReadFile(excludesfile) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var pathsRaw []string | ||||||
|  | 	for _, s := range strings.Split(string(data), "\n") { | ||||||
|  | 		if !strings.HasPrefix(s, "#") && len(strings.TrimSpace(s)) > 0 { | ||||||
|  | 			pathsRaw = append(pathsRaw, s) | ||||||
|  | 			ps = append(ps, gitignore.ParsePattern(s, nil)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("read global ignore paths: %s", strings.Join(pathsRaw, " ")) | ||||||
|  |  | ||||||
|  | 	return ps, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func expandTilde(path string) (string, error) { | ||||||
|  | 	if !strings.HasPrefix(path, "~") { | ||||||
|  | 		return path, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var paths []string | ||||||
|  | 	u, err := user.Current() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, p := range strings.Split(path, string(filepath.Separator)) { | ||||||
|  | 		if p == "~" { | ||||||
|  | 			paths = append(paths, u.HomeDir) | ||||||
|  | 		} else { | ||||||
|  | 			paths = append(paths, p) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return filepath.Join(paths...), nil | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										28
									
								
								pkg/git/remote.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								pkg/git/remote.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | package git | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/go-git/go-git/v5/config" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // CreateRemote creates a new git remote in a repository | ||||||
|  | func CreateRemote(repo *git.Repository, name, url string, dryRun bool) error { | ||||||
|  | 	if dryRun { | ||||||
|  | 		logrus.Debugf("dry run: remote %s (%s) not created", name, url) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := repo.CreateRemote(&config.RemoteConfig{ | ||||||
|  | 		Name: name, | ||||||
|  | 		URLs: []string{url}, | ||||||
|  | 	}); err != nil { | ||||||
|  | 		if !strings.Contains(err.Error(), "remote already exists") { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								pkg/integration/integration.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								pkg/integration/integration.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | package integration | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"os" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func skipIfNotIntegration(t *testing.T) { | ||||||
|  | 	if os.Getenv("ABRA_INTEGRATION") == "" { | ||||||
|  | 		t.Skip("missing 'ABRA_INTEGRATION', not running integration tests") | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										338
									
								
								pkg/lint/recipe.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										338
									
								
								pkg/lint/recipe.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,338 @@ | |||||||
|  | package lint | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	"coopcloud.tech/tagcmp" | ||||||
|  | 	"github.com/docker/distribution/reference" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | var Warn = "warn" | ||||||
|  | var Critical = "critical" | ||||||
|  |  | ||||||
|  | type LintFunction func(recipe.Recipe) (bool, error) | ||||||
|  |  | ||||||
|  | type LintRule struct { | ||||||
|  | 	Ref          string | ||||||
|  | 	Level        string | ||||||
|  | 	Description  string | ||||||
|  | 	HowToResolve string | ||||||
|  | 	Function     LintFunction | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var LintRules = map[string][]LintRule{ | ||||||
|  | 	"warn": { | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R001", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "compose config has expected version", | ||||||
|  | 			HowToResolve: "ensure 'version: \"3.8\"' in compose configs", | ||||||
|  | 			Function:     LintComposeVersion, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R002", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "healthcheck enabled for all services", | ||||||
|  | 			HowToResolve: "wire up healthchecks", | ||||||
|  | 			Function:     LintHealthchecks, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R003", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "all images use a tag", | ||||||
|  | 			HowToResolve: "use a tag for all images", | ||||||
|  | 			Function:     LintAllImagesTagged, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R004", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "no unstable tags", | ||||||
|  | 			HowToResolve: "tag all images with stable tags", | ||||||
|  | 			Function:     LintNoUnstableTags, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R005", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "tags use semver-like format", | ||||||
|  | 			HowToResolve: "use semver-like tags", | ||||||
|  | 			Function:     LintSemverLikeTags, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R006", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "has published catalogue version", | ||||||
|  | 			HowToResolve: "publish a recipe version to the catalogue", | ||||||
|  | 			Function:     LintHasPublishedVersion, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R007", | ||||||
|  | 			Level:        "warn", | ||||||
|  | 			Description:  "README.md metadata filled in", | ||||||
|  | 			HowToResolve: "fill out all the metadata", | ||||||
|  | 			Function:     LintMetadataFilledIn, | ||||||
|  | 		}, | ||||||
|  | 	}, | ||||||
|  | 	"error": { | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R008", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  ".env.sample provided", | ||||||
|  | 			HowToResolve: "create an example .env.sample", | ||||||
|  | 			Function:     LintEnvConfigPresent, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R009", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  "one service named 'app'", | ||||||
|  | 			HowToResolve: "name a servce 'app'", | ||||||
|  | 			Function:     LintAppService, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R010", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  "traefik routing enabled", | ||||||
|  | 			HowToResolve: "include \"traefik.enable=true\" deploy label", | ||||||
|  | 			Function:     LintTraefikEnabled, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R011", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  "all services have images", | ||||||
|  | 			HowToResolve: "ensure \"image: ...\" set on all services", | ||||||
|  | 			Function:     LintImagePresent, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R012", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  "config version are vendored", | ||||||
|  | 			HowToResolve: "vendor config versions in an abra.sh", | ||||||
|  | 			Function:     LintAbraShVendors, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Ref:          "R013", | ||||||
|  | 			Level:        "error", | ||||||
|  | 			Description:  "git.coopcloud.tech repo exists", | ||||||
|  | 			HowToResolve: "upload your recipe to git.coopcloud.tech/coop-cloud/...", | ||||||
|  | 			Function:     LintHasRecipeRepo, | ||||||
|  | 		}, | ||||||
|  | 	}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintForErrors(recipe recipe.Recipe) error { | ||||||
|  | 	logrus.Debugf("linting for critical errors in %s configs", recipe.Name) | ||||||
|  |  | ||||||
|  | 	for level := range LintRules { | ||||||
|  | 		if level != "error" { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		for _, rule := range LintRules[level] { | ||||||
|  | 			ok, err := rule.Function(recipe) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if !ok { | ||||||
|  | 				return fmt.Errorf("lint error in %s configs: \"%s\" failed lint checks (%s)", recipe.Name, rule.Description, rule.Ref) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("linting successful, %s is well configured", recipe.Name) | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintComposeVersion(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	if recipe.Config.Version == "3.8" { | ||||||
|  | 		return true, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintEnvConfigPresent(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	envSample := fmt.Sprintf("%s/%s/.env.sample", config.RECIPES_DIR, recipe.Name) | ||||||
|  | 	if _, err := os.Stat(envSample); !os.IsNotExist(err) { | ||||||
|  | 		return true, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return false, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintAppService(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		if service.Name == "app" { | ||||||
|  | 			return true, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return false, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintTraefikEnabled(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		for label := range service.Deploy.Labels { | ||||||
|  | 			if label == "traefik.enable" { | ||||||
|  | 				if service.Deploy.Labels[label] == "true" { | ||||||
|  | 					return true, nil | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return false, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintHealthchecks(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		if service.HealthCheck == nil { | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintAllImagesTagged(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		img, err := reference.ParseNormalizedNamed(service.Image) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return false, err | ||||||
|  | 		} | ||||||
|  | 		if reference.IsNameOnly(img) { | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintNoUnstableTags(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		img, err := reference.ParseNormalizedNamed(service.Image) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return false, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		var tag string | ||||||
|  | 		switch img.(type) { | ||||||
|  | 		case reference.NamedTagged: | ||||||
|  | 			tag = img.(reference.NamedTagged).Tag() | ||||||
|  | 		case reference.Named: | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if tag == "latest" { | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintSemverLikeTags(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		img, err := reference.ParseNormalizedNamed(service.Image) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return false, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		var tag string | ||||||
|  | 		switch img.(type) { | ||||||
|  | 		case reference.NamedTagged: | ||||||
|  | 			tag = img.(reference.NamedTagged).Tag() | ||||||
|  | 		case reference.Named: | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !tagcmp.IsParsable(tag) { | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintImagePresent(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		if service.Image == "" { | ||||||
|  | 			return false, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintHasPublishedVersion(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	catl, err := recipePkg.ReadRecipeCatalogue() | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	versions, err := recipePkg.GetRecipeCatalogueVersions(recipe.Name, catl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(versions) == 0 { | ||||||
|  | 		return false, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintMetadataFilledIn(r recipe.Recipe) (bool, error) { | ||||||
|  | 	features, category, err := recipe.GetRecipeFeaturesAndCategory(r.Name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if category == "" { | ||||||
|  | 		return false, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if features.Backups == "" || | ||||||
|  | 		features.Email == "" || | ||||||
|  | 		features.Healthcheck == "" || | ||||||
|  | 		features.Image.Image == "" || | ||||||
|  | 		features.SSO == "" { | ||||||
|  | 		return false, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintAbraShVendors(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	for _, service := range recipe.Config.Services { | ||||||
|  | 		if len(service.Configs) > 0 { | ||||||
|  | 			abraSh := path.Join(config.RECIPES_DIR, recipe.Name, "abra.sh") | ||||||
|  | 			if _, err := os.Stat(abraSh); err != nil { | ||||||
|  | 				if os.IsNotExist(err) { | ||||||
|  | 					return false, err | ||||||
|  | 				} | ||||||
|  | 				return false, err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func LintHasRecipeRepo(recipe recipe.Recipe) (bool, error) { | ||||||
|  | 	url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, recipe.Name) | ||||||
|  |  | ||||||
|  | 	res, err := http.Get(url) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if res.StatusCode != 200 { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true, nil | ||||||
|  | } | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -19,13 +19,13 @@ func PassInsertSecret(secretValue, secretName, appName, server string) error { | |||||||
| 		secretValue, server, appName, secretName, | 		secretValue, server, appName, secretName, | ||||||
| 	) | 	) | ||||||
|  |  | ||||||
| 	logrus.Debugf("attempting to run '%s'", cmd) | 	logrus.Debugf("attempting to run %s", cmd) | ||||||
|  |  | ||||||
| 	if err := exec.Command("bash", "-c", cmd).Run(); err != nil { | 	if err := exec.Command("bash", "-c", cmd).Run(); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Infof("'%s' inserted into pass store", secretName) | 	logrus.Infof("%s inserted into pass store", secretName) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -41,13 +41,13 @@ func PassRmSecret(secretName, appName, server string) error { | |||||||
| 		server, appName, secretName, | 		server, appName, secretName, | ||||||
| 	) | 	) | ||||||
|  |  | ||||||
| 	logrus.Debugf("attempting to run '%s'", cmd) | 	logrus.Debugf("attempting to run %s", cmd) | ||||||
|  |  | ||||||
| 	if err := exec.Command("bash", "-c", cmd).Run(); err != nil { | 	if err := exec.Command("bash", "-c", cmd).Run(); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Infof("'%s' removed from pass store", secretName) | 	logrus.Infof("%s removed from pass store", secretName) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -34,7 +34,7 @@ func GeneratePasswords(count, length uint) ([]string, error) { | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("generated '%s'", strings.Join(passwords, ", ")) | 	logrus.Debugf("generated %s", strings.Join(passwords, ", ")) | ||||||
|  |  | ||||||
| 	return passwords, nil | 	return passwords, nil | ||||||
| } | } | ||||||
| @ -53,7 +53,7 @@ func GeneratePassphrases(count uint) ([]string, error) { | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("generated '%s'", strings.Join(passphrases, ", ")) | 	logrus.Debugf("generated %s", strings.Join(passphrases, ", ")) | ||||||
|  |  | ||||||
| 	return passphrases, nil | 	return passphrases, nil | ||||||
| } | } | ||||||
| @ -69,35 +69,32 @@ func ReadSecretEnvVars(appEnv config.AppEnv) map[string]string { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("read '%s' as secrets from '%s'", secretEnvVars, appEnv) | 	logrus.Debugf("read %s as secrets from %s", secretEnvVars, appEnv) | ||||||
|  |  | ||||||
| 	return secretEnvVars | 	return secretEnvVars | ||||||
| } | } | ||||||
|  |  | ||||||
| // TODO: should probably go in the config/app package? |  | ||||||
| func ParseSecretEnvVarName(secretEnvVar string) string { | func ParseSecretEnvVarName(secretEnvVar string) string { | ||||||
| 	withoutPrefix := strings.TrimPrefix(secretEnvVar, "SECRET_") | 	withoutPrefix := strings.TrimPrefix(secretEnvVar, "SECRET_") | ||||||
| 	withoutSuffix := strings.TrimSuffix(withoutPrefix, "_VERSION") | 	withoutSuffix := strings.TrimSuffix(withoutPrefix, "_VERSION") | ||||||
| 	name := strings.ToLower(withoutSuffix) | 	name := strings.ToLower(withoutSuffix) | ||||||
| 	logrus.Debugf("parsed '%s' as name from '%s'", name, secretEnvVar) | 	logrus.Debugf("parsed %s as name from %s", name, secretEnvVar) | ||||||
| 	return name | 	return name | ||||||
| } | } | ||||||
|  |  | ||||||
| // TODO: should probably go in the config/app package? |  | ||||||
| func ParseGeneratedSecretName(secret string, appEnv config.App) string { | func ParseGeneratedSecretName(secret string, appEnv config.App) string { | ||||||
| 	name := fmt.Sprintf("%s_", appEnv.StackName()) | 	name := fmt.Sprintf("%s_", appEnv.StackName()) | ||||||
| 	withoutAppName := strings.TrimPrefix(secret, name) | 	withoutAppName := strings.TrimPrefix(secret, name) | ||||||
| 	idx := strings.LastIndex(withoutAppName, "_") | 	idx := strings.LastIndex(withoutAppName, "_") | ||||||
| 	parsed := withoutAppName[:idx] | 	parsed := withoutAppName[:idx] | ||||||
| 	logrus.Debugf("parsed '%s' as name from '%s'", parsed, secret) | 	logrus.Debugf("parsed %s as name from %s", parsed, secret) | ||||||
| 	return parsed | 	return parsed | ||||||
| } | } | ||||||
|  |  | ||||||
| // TODO: should probably go in the config/app package? |  | ||||||
| func ParseSecretEnvVarValue(secret string) (secretValue, error) { | func ParseSecretEnvVarValue(secret string) (secretValue, error) { | ||||||
| 	values := strings.Split(secret, "#") | 	values := strings.Split(secret, "#") | ||||||
| 	if len(values) == 0 { | 	if len(values) == 0 { | ||||||
| 		return secretValue{}, fmt.Errorf("unable to parse '%s'", secret) | 		return secretValue{}, fmt.Errorf("unable to parse %s", secret) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if len(values) == 1 { | 	if len(values) == 1 { | ||||||
| @ -113,7 +110,7 @@ func ParseSecretEnvVarValue(secret string) (secretValue, error) { | |||||||
| 	} | 	} | ||||||
| 	version := strings.ReplaceAll(values[0], " ", "") | 	version := strings.ReplaceAll(values[0], " ", "") | ||||||
|  |  | ||||||
| 	logrus.Debugf("parsed version '%s' and length '%v' from '%s'", version, length, secret) | 	logrus.Debugf("parsed version %s and length '%v' from %s", version, length, secret) | ||||||
|  |  | ||||||
| 	return secretValue{Version: version, Length: length}, nil | 	return secretValue{Version: version, Length: length}, nil | ||||||
| } | } | ||||||
| @ -132,7 +129,7 @@ func GenerateSecrets(secretEnvVars map[string]string, appName, server string) (m | |||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			secretRemoteName := fmt.Sprintf("%s_%s_%s", appName, secretName, secretValue.Version) | 			secretRemoteName := fmt.Sprintf("%s_%s_%s", appName, secretName, secretValue.Version) | ||||||
| 			logrus.Debugf("attempting to generate and store '%s' on '%s'", secretRemoteName, server) | 			logrus.Debugf("attempting to generate and store %s on %s", secretRemoteName, server) | ||||||
| 			if secretValue.Length > 0 { | 			if secretValue.Length > 0 { | ||||||
| 				passwords, err := GeneratePasswords(1, uint(secretValue.Length)) | 				passwords, err := GeneratePasswords(1, uint(secretValue.Length)) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| @ -140,7 +137,12 @@ func GenerateSecrets(secretEnvVars map[string]string, appName, server string) (m | |||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 				if err := client.StoreSecret(secretRemoteName, passwords[0], server); err != nil { | 				if err := client.StoreSecret(secretRemoteName, passwords[0], server); err != nil { | ||||||
| 					ch <- err | 					if strings.Contains(err.Error(), "AlreadyExists") { | ||||||
|  | 						logrus.Warnf("%s already exists, moving on...", secretRemoteName) | ||||||
|  | 						ch <- nil | ||||||
|  | 					} else { | ||||||
|  | 						ch <- err | ||||||
|  | 					} | ||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 				secrets[secretName] = passwords[0] | 				secrets[secretName] = passwords[0] | ||||||
| @ -151,7 +153,13 @@ func GenerateSecrets(secretEnvVars map[string]string, appName, server string) (m | |||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 				if err := client.StoreSecret(secretRemoteName, passphrases[0], server); err != nil { | 				if err := client.StoreSecret(secretRemoteName, passphrases[0], server); err != nil { | ||||||
| 					ch <- err | 					if strings.Contains(err.Error(), "AlreadyExists") { | ||||||
|  | 						logrus.Warnf("%s already exists, moving on...", secretRemoteName) | ||||||
|  | 						ch <- nil | ||||||
|  | 					} else { | ||||||
|  | 						ch <- err | ||||||
|  | 					} | ||||||
|  | 					return | ||||||
| 				} | 				} | ||||||
| 				secrets[secretName] = passphrases[0] | 				secrets[secretName] = passphrases[0] | ||||||
| 			} | 			} | ||||||
| @ -166,7 +174,7 @@ func GenerateSecrets(secretEnvVars map[string]string, appName, server string) (m | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("generated and stored '%s' on '%s'", secrets, server) | 	logrus.Debugf("generated and stored %s on %s", secrets, server) | ||||||
|  |  | ||||||
| 	return secrets, nil | 	return secrets, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -12,7 +12,7 @@ import ( | |||||||
| func CreateServerDir(serverName string) error { | func CreateServerDir(serverName string) error { | ||||||
| 	serverPath := path.Join(config.ABRA_DIR, "servers", serverName) | 	serverPath := path.Join(config.ABRA_DIR, "servers", serverName) | ||||||
|  |  | ||||||
| 	if err := os.Mkdir(serverPath, 0755); err != nil { | 	if err := os.Mkdir(serverPath, 0764); err != nil { | ||||||
| 		if !os.IsExist(err) { | 		if !os.IsExist(err) { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  | |||||||
							
								
								
									
										78
									
								
								pkg/service/service.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								pkg/service/service.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,78 @@ | |||||||
|  | package service | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
|  | 	"github.com/AlecAivazis/survey/v2" | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	"github.com/docker/docker/api/types/swarm" | ||||||
|  | 	"github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // GetService retrieves a service container. If prompt is true and the retrievd | ||||||
|  | // count of service containers does not match 1, then a prompt is presented to | ||||||
|  | // let the user choose. A count of 0 is handled gracefully. | ||||||
|  | func GetService(c context.Context, cl *client.Client, filters filters.Args, prompt bool) (swarm.Service, error) { | ||||||
|  | 	serviceOpts := types.ServiceListOptions{Filters: filters} | ||||||
|  | 	services, err := cl.ServiceList(c, serviceOpts) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return swarm.Service{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(services) == 0 { | ||||||
|  | 		filter := filters.Get("name")[0] | ||||||
|  | 		return swarm.Service{}, fmt.Errorf("no services matching the %v filter found?", filter) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(services) != 1 { | ||||||
|  | 		var servicesRaw []string | ||||||
|  | 		for _, service := range services { | ||||||
|  | 			serviceName := service.Spec.Name | ||||||
|  | 			created := formatter.HumanDuration(service.CreatedAt.Unix()) | ||||||
|  | 			servicesRaw = append(servicesRaw, fmt.Sprintf("%s (created %v)", serviceName, created)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !prompt { | ||||||
|  | 			err := fmt.Errorf("expected 1 service but found %v: %s", len(services), strings.Join(servicesRaw, " ")) | ||||||
|  | 			return swarm.Service{}, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Warnf("ambiguous service list received, prompting for input") | ||||||
|  |  | ||||||
|  | 		var response string | ||||||
|  | 		prompt := &survey.Select{ | ||||||
|  | 			Message: "which service are you looking for?", | ||||||
|  | 			Options: servicesRaw, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := survey.AskOne(prompt, &response); err != nil { | ||||||
|  | 			return swarm.Service{}, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		chosenService := strings.TrimSpace(strings.Split(response, " ")[0]) | ||||||
|  | 		for _, service := range services { | ||||||
|  | 			serviceName := strings.ToLower(service.Spec.Name) | ||||||
|  | 			if serviceName == chosenService { | ||||||
|  | 				return service, nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Panic("failed to match chosen service") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return services[0], nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ContainerToServiceName converts a container name to a service name. | ||||||
|  | func ContainerToServiceName(containerNames []string, stackName string) string { | ||||||
|  | 	containerName := strings.Join(containerNames, "") | ||||||
|  | 	trimmed := strings.TrimPrefix(containerName, "/") | ||||||
|  | 	stackNameServiceName := strings.Split(trimmed, ".")[0] | ||||||
|  | 	splitter := fmt.Sprintf("%s_", stackName) | ||||||
|  | 	return strings.Split(stackNameServiceName, splitter)[1] | ||||||
|  | } | ||||||
							
								
								
									
										107
									
								
								pkg/ssh/ssh.go
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								pkg/ssh/ssh.go
									
									
									
									
									
								
							| @ -111,7 +111,7 @@ type sudoWriter struct { | |||||||
|  |  | ||||||
| // Write satisfies the write interface for sudoWriter | // Write satisfies the write interface for sudoWriter | ||||||
| func (w *sudoWriter) Write(p []byte) (int, error) { | func (w *sudoWriter) Write(p []byte) (int, error) { | ||||||
| 	if string(p) == "sudo_password" { | 	if strings.Contains(string(p), "sudo_password") { | ||||||
| 		w.stdin.Write([]byte(w.pw + "\n")) | 		w.stdin.Write([]byte(w.pw + "\n")) | ||||||
| 		w.pw = "" | 		w.pw = "" | ||||||
| 		return len(p), nil | 		return len(p), nil | ||||||
| @ -131,11 +131,9 @@ func RunSudoCmd(cmd, passwd string, cl *Client) error { | |||||||
| 	} | 	} | ||||||
| 	defer session.Close() | 	defer session.Close() | ||||||
|  |  | ||||||
| 	cmd = "sudo -p " + "sudo_password" + " -S " + cmd | 	sudoCmd := fmt.Sprintf("SSH_ASKPASS=/usr/bin/ssh-askpass; sudo -p sudo_password -S %s", cmd) | ||||||
|  |  | ||||||
| 	w := &sudoWriter{ | 	w := &sudoWriter{pw: passwd} | ||||||
| 		pw: passwd, |  | ||||||
| 	} |  | ||||||
| 	w.stdin, err = session.StdinPipe() | 	w.stdin, err = session.StdinPipe() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -144,79 +142,19 @@ func RunSudoCmd(cmd, passwd string, cl *Client) error { | |||||||
| 	session.Stdout = w | 	session.Stdout = w | ||||||
| 	session.Stderr = w | 	session.Stderr = w | ||||||
|  |  | ||||||
| 	done := make(chan struct{}) | 	modes := ssh.TerminalModes{ | ||||||
| 	scanner := bufio.NewScanner(session.Stdin) | 		ssh.ECHO:          0, | ||||||
|  | 		ssh.TTY_OP_ISPEED: 14400, | ||||||
| 	go func() { | 		ssh.TTY_OP_OSPEED: 14400, | ||||||
| 		for scanner.Scan() { |  | ||||||
| 			line := scanner.Text() |  | ||||||
| 			fmt.Println(line) |  | ||||||
| 		} |  | ||||||
| 		done <- struct{}{} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	if err := session.Start(cmd); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	<-done | 	err = session.RequestPty("xterm", 80, 40, modes) | ||||||
|  |  | ||||||
| 	if err := session.Wait(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Exec runs a command on a remote and streams output |  | ||||||
| func Exec(cmd string, cl *Client) error { |  | ||||||
| 	session, err := cl.SSHClient.NewSession() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer session.Close() |  | ||||||
|  |  | ||||||
| 	stdout, err := session.StdoutPipe() |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	stderr, err := session.StdoutPipe() | 	if err := session.Run(sudoCmd); err != nil { | ||||||
| 	if err != nil { | 		return fmt.Errorf("%s", string(w.b.Bytes())) | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	stdoutDone := make(chan struct{}) |  | ||||||
| 	stdoutScanner := bufio.NewScanner(stdout) |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		for stdoutScanner.Scan() { |  | ||||||
| 			line := stdoutScanner.Text() |  | ||||||
| 			fmt.Println(line) |  | ||||||
| 		} |  | ||||||
| 		stdoutDone <- struct{}{} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	stderrDone := make(chan struct{}) |  | ||||||
| 	stderrScanner := bufio.NewScanner(stderr) |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		for stderrScanner.Scan() { |  | ||||||
| 			line := stderrScanner.Text() |  | ||||||
| 			fmt.Println(line) |  | ||||||
| 		} |  | ||||||
| 		stderrDone <- struct{}{} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	if err := session.Start(cmd); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	<-stdoutDone |  | ||||||
| 	<-stderrDone |  | ||||||
|  |  | ||||||
| 	if err := session.Wait(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| @ -320,7 +258,7 @@ func HostKeyAddCallback(hostnameAndPort string, remote net.Addr, pubKey ssh.Publ | |||||||
|  |  | ||||||
| 	if exists { | 	if exists { | ||||||
| 		hostname := strings.Split(hostnameAndPort, ":")[0] | 		hostname := strings.Split(hostnameAndPort, ":")[0] | ||||||
| 		logrus.Debugf("server SSH host key found for %s, moving on", hostname) | 		logrus.Debugf("server SSH host key found for %s", hostname) | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @ -330,9 +268,9 @@ func HostKeyAddCallback(hostnameAndPort string, remote net.Addr, pubKey ssh.Publ | |||||||
|  |  | ||||||
| 		fmt.Printf(fmt.Sprintf(` | 		fmt.Printf(fmt.Sprintf(` | ||||||
| You are attempting to make an SSH connection to a server but there is no entry | You are attempting to make an SSH connection to a server but there is no entry | ||||||
| in your ~/.ssh/known_hosts file which confirms that this is indeed the server | in your ~/.ssh/known_hosts file which confirms that you have already validated | ||||||
| you want to connect to. Please take a moment to validate the following SSH host | that this is indeed the server you want to connect to. Please take a moment to | ||||||
| key, it is important. | validate the following SSH host key, it is important. | ||||||
|  |  | ||||||
|     Host:        %s |     Host:        %s | ||||||
|     Fingerprint: %s |     Fingerprint: %s | ||||||
| @ -537,11 +475,10 @@ func GetContextConnDetails(serverName string) (*dockerSSHPkg.Spec, error) { | |||||||
| func GetHostConfig(hostname, username, port string) (HostConfig, error) { | func GetHostConfig(hostname, username, port string) (HostConfig, error) { | ||||||
| 	var hostConfig HostConfig | 	var hostConfig HostConfig | ||||||
|  |  | ||||||
| 	var host, idf string | 	if hostname == "" { | ||||||
|  | 		if hostname = ssh_config.Get(hostname, "Hostname"); hostname == "" { | ||||||
| 	if host = ssh_config.Get(hostname, "Hostname"); host == "" { | 			logrus.Debugf("no hostname found in SSH config, assuming %s", hostname) | ||||||
| 		logrus.Debugf("no hostname found in SSH config, assuming %s", hostname) | 		} | ||||||
| 		host = hostname |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if username == "" { | 	if username == "" { | ||||||
| @ -562,17 +499,19 @@ func GetHostConfig(hostname, username, port string) (HostConfig, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	idf = ssh_config.Get(hostname, "IdentityFile") | 	if idf := ssh_config.Get(hostname, "IdentityFile"); idf != "" && idf != "~/.ssh/identity" { | ||||||
| 	if idf != "" { |  | ||||||
| 		var err error | 		var err error | ||||||
| 		idf, err = identityFileAbsPath(idf) | 		idf, err = identityFileAbsPath(idf) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return hostConfig, err | 			return hostConfig, err | ||||||
| 		} | 		} | ||||||
| 		hostConfig.IdentityFile = idf | 		hostConfig.IdentityFile = idf | ||||||
|  | 	} else { | ||||||
|  | 		logrus.Debugf("no identity file found in SSH config for %s", hostname) | ||||||
|  | 		hostConfig.IdentityFile = "" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	hostConfig.Host = host | 	hostConfig.Host = hostname | ||||||
| 	hostConfig.Port = port | 	hostConfig.Port = port | ||||||
| 	hostConfig.User = username | 	hostConfig.User = username | ||||||
|  |  | ||||||
|  | |||||||
| @ -188,14 +188,14 @@ func ignorableCloseError(err error) bool { | |||||||
| func (c *commandConn) CloseRead() error { | func (c *commandConn) CloseRead() error { | ||||||
| 	// NOTE: maybe already closed here | 	// NOTE: maybe already closed here | ||||||
| 	if err := c.stdout.Close(); err != nil && !ignorableCloseError(err) { | 	if err := c.stdout.Close(); err != nil && !ignorableCloseError(err) { | ||||||
| 		// TODO: muted because https://github.com/docker/compose/issues/8544 | 		// muted because https://github.com/docker/compose/issues/8544 | ||||||
| 		// logrus.Warnf("commandConn.CloseRead: %v", err) | 		// logrus.Warnf("commandConn.CloseRead: %v", err) | ||||||
| 	} | 	} | ||||||
| 	c.stdioClosedMu.Lock() | 	c.stdioClosedMu.Lock() | ||||||
| 	c.stdoutClosed = true | 	c.stdoutClosed = true | ||||||
| 	c.stdioClosedMu.Unlock() | 	c.stdioClosedMu.Unlock() | ||||||
| 	if err := c.killIfStdioClosed(); err != nil { | 	if err := c.killIfStdioClosed(); err != nil { | ||||||
| 		// TODO: muted because https://github.com/docker/compose/issues/8544 | 		// muted because https://github.com/docker/compose/issues/8544 | ||||||
| 		// logrus.Warnf("commandConn.CloseRead: %v", err) | 		// logrus.Warnf("commandConn.CloseRead: %v", err) | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -212,14 +212,14 @@ func (c *commandConn) Read(p []byte) (int, error) { | |||||||
| func (c *commandConn) CloseWrite() error { | func (c *commandConn) CloseWrite() error { | ||||||
| 	// NOTE: maybe already closed here | 	// NOTE: maybe already closed here | ||||||
| 	if err := c.stdin.Close(); err != nil && !ignorableCloseError(err) { | 	if err := c.stdin.Close(); err != nil && !ignorableCloseError(err) { | ||||||
| 		// TODO: muted because https://github.com/docker/compose/issues/8544 | 		// muted because https://github.com/docker/compose/issues/8544 | ||||||
| 		// logrus.Warnf("commandConn.CloseWrite: %v", err) | 		// logrus.Warnf("commandConn.CloseWrite: %v", err) | ||||||
| 	} | 	} | ||||||
| 	c.stdioClosedMu.Lock() | 	c.stdioClosedMu.Lock() | ||||||
| 	c.stdinClosed = true | 	c.stdinClosed = true | ||||||
| 	c.stdioClosedMu.Unlock() | 	c.stdioClosedMu.Unlock() | ||||||
| 	if err := c.killIfStdioClosed(); err != nil { | 	if err := c.killIfStdioClosed(); err != nil { | ||||||
| 		// TODO: muted because https://github.com/docker/compose/issues/8544 | 		// muted because https://github.com/docker/compose/issues/8544 | ||||||
| 		// logrus.Warnf("commandConn.CloseWrite: %v", err) | 		// logrus.Warnf("commandConn.CloseWrite: %v", err) | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -239,7 +239,7 @@ func (c *commandConn) Close() error { | |||||||
| 		logrus.Warnf("commandConn.Close: CloseRead: %v", err) | 		logrus.Warnf("commandConn.Close: CloseRead: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err = c.CloseWrite(); err != nil { | 	if err = c.CloseWrite(); err != nil { | ||||||
| 		// TODO: muted because https://github.com/docker/compose/issues/8544 | 		// muted because https://github.com/docker/compose/issues/8544 | ||||||
| 		// logrus.Warnf("commandConn.Close: CloseWrite: %v", err) | 		// logrus.Warnf("commandConn.Close: CloseWrite: %v", err) | ||||||
| 	} | 	} | ||||||
| 	return err | 	return err | ||||||
|  | |||||||
| @ -16,7 +16,6 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| // The default escape key sequence: ctrl-p, ctrl-q | // The default escape key sequence: ctrl-p, ctrl-q | ||||||
| // TODO: This could be moved to `pkg/term`. |  | ||||||
| var defaultEscapeKeys = []byte{16, 17} | var defaultEscapeKeys = []byte{16, 17} | ||||||
|  |  | ||||||
| // A hijackedIOStreamer handles copying input to and output from streams to the | // A hijackedIOStreamer handles copying input to and output from streams to the | ||||||
|  | |||||||
| @ -399,7 +399,6 @@ func convertServiceNetworks( | |||||||
| 	return nets, nil | 	return nets, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // TODO: fix secrets API so that SecretAPIClient is not required here |  | ||||||
| func convertServiceSecrets( | func convertServiceSecrets( | ||||||
| 	client client.SecretAPIClient, | 	client client.SecretAPIClient, | ||||||
| 	namespace Namespace, | 	namespace Namespace, | ||||||
| @ -442,8 +441,6 @@ func convertServiceSecrets( | |||||||
| // required by the serivce. Unlike convertServiceSecrets, this takes the whole | // required by the serivce. Unlike convertServiceSecrets, this takes the whole | ||||||
| // ServiceConfig, because some Configs may be needed as a result of other | // ServiceConfig, because some Configs may be needed as a result of other | ||||||
| // fields (like CredentialSpecs). | // fields (like CredentialSpecs). | ||||||
| // |  | ||||||
| // TODO: fix configs API so that ConfigsAPIClient is not required here |  | ||||||
| func convertServiceConfigObjs( | func convertServiceConfigObjs( | ||||||
| 	client client.ConfigAPIClient, | 	client client.ConfigAPIClient, | ||||||
| 	namespace Namespace, | 	namespace Namespace, | ||||||
| @ -626,7 +623,6 @@ func convertHealthcheck(healthcheck *composetypes.HealthCheckConfig) (*container | |||||||
| } | } | ||||||
|  |  | ||||||
| func convertRestartPolicy(restart string, source *composetypes.RestartPolicy) (*swarm.RestartPolicy, error) { | func convertRestartPolicy(restart string, source *composetypes.RestartPolicy) (*swarm.RestartPolicy, error) { | ||||||
| 	// TODO: log if restart is being ignored |  | ||||||
| 	if source == nil { | 	if source == nil { | ||||||
| 		policy, err := opts.ParseRestartPolicy(restart) | 		policy, err := opts.ParseRestartPolicy(restart) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
|  | |||||||
							
								
								
									
										44
									
								
								pkg/upstream/service/scale.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								pkg/upstream/service/scale.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | |||||||
|  | package upstream | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // RunServiceScale scales a service (useful for restart action) | ||||||
|  | func RunServiceScale(ctx context.Context, cl *client.Client, serviceID string, scale uint64) error { | ||||||
|  | 	service, _, err := cl.ServiceInspectWithRaw(ctx, serviceID, types.ServiceInspectOptions{}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	serviceMode := &service.Spec.Mode | ||||||
|  | 	if serviceMode.Replicated != nil { | ||||||
|  | 		serviceMode.Replicated.Replicas = &scale | ||||||
|  | 	} else if serviceMode.ReplicatedJob != nil { | ||||||
|  | 		serviceMode.ReplicatedJob.TotalCompletions = &scale | ||||||
|  | 	} else { | ||||||
|  | 		return fmt.Errorf("scale can only be used with replicated or replicated-job mode") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	response, err := cl.ServiceUpdate( | ||||||
|  | 		ctx, | ||||||
|  | 		service.ID, | ||||||
|  | 		service.Version, | ||||||
|  | 		service.Spec, | ||||||
|  | 		types.ServiceUpdateOptions{}, | ||||||
|  | 	) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, warning := range response.Warnings { | ||||||
|  | 		logrus.Warn(warning) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @ -13,6 +13,11 @@ import ( | |||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | // DontSkipValidation ensures validation is done for compose file loading | ||||||
|  | func DontSkipValidation(opts *loader.Options) { | ||||||
|  | 	opts.SkipValidation = false | ||||||
|  | } | ||||||
|  |  | ||||||
| // LoadComposefile parse the composefile specified in the cli and returns its Config and version. | // LoadComposefile parse the composefile specified in the cli and returns its Config and version. | ||||||
| func LoadComposefile(opts Deploy, appEnv map[string]string) (*composetypes.Config, error) { | func LoadComposefile(opts Deploy, appEnv map[string]string) (*composetypes.Config, error) { | ||||||
| 	configDetails, err := getConfigDetails(opts.Composefiles, appEnv) | 	configDetails, err := getConfigDetails(opts.Composefiles, appEnv) | ||||||
| @ -21,26 +26,25 @@ func LoadComposefile(opts Deploy, appEnv map[string]string) (*composetypes.Confi | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	dicts := getDictsFrom(configDetails.ConfigFiles) | 	dicts := getDictsFrom(configDetails.ConfigFiles) | ||||||
| 	config, err := loader.Load(configDetails) | 	config, err := loader.Load(configDetails, DontSkipValidation) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if fpe, ok := err.(*loader.ForbiddenPropertiesError); ok { | 		if fpe, ok := err.(*loader.ForbiddenPropertiesError); ok { | ||||||
| 			return nil, fmt.Errorf("compose file contains unsupported options:\n\n%s", | 			return nil, fmt.Errorf("compose file contains unsupported options: %s", | ||||||
| 				propertyWarnings(fpe.Properties)) | 				propertyWarnings(fpe.Properties)) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	unsupportedProperties := loader.GetUnsupportedProperties(dicts...) | 	unsupportedProperties := loader.GetUnsupportedProperties(dicts...) | ||||||
| 	if len(unsupportedProperties) > 0 { | 	if len(unsupportedProperties) > 0 { | ||||||
| 		logrus.Warnf("Ignoring unsupported options: %s\n\n", | 		logrus.Warnf("%s: ignoring unsupported options: %s", | ||||||
| 			strings.Join(unsupportedProperties, ", ")) | 			appEnv["TYPE"], strings.Join(unsupportedProperties, ", ")) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	deprecatedProperties := loader.GetDeprecatedProperties(dicts...) | 	deprecatedProperties := loader.GetDeprecatedProperties(dicts...) | ||||||
| 	if len(deprecatedProperties) > 0 { | 	if len(deprecatedProperties) > 0 { | ||||||
| 		logrus.Warnf("Ignoring deprecated options:\n\n%s\n\n", | 		logrus.Warnf("%s: ignoring deprecated options: %s", | ||||||
| 			propertyWarnings(deprecatedProperties)) | 			appEnv["TYPE"], propertyWarnings(deprecatedProperties)) | ||||||
| 	} | 	} | ||||||
| 	return config, nil | 	return config, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -112,7 +112,7 @@ func GetDeployedServicesByName(ctx context.Context, cl *dockerclient.Client, sta | |||||||
|  |  | ||||||
| // IsDeployed chekcks whether an appp is deployed or not. | // IsDeployed chekcks whether an appp is deployed or not. | ||||||
| func IsDeployed(ctx context.Context, cl *dockerclient.Client, stackName string) (bool, string, error) { | func IsDeployed(ctx context.Context, cl *dockerclient.Client, stackName string) (bool, string, error) { | ||||||
| 	version := "" | 	version := "unknown" | ||||||
| 	isDeployed := false | 	isDeployed := false | ||||||
|  |  | ||||||
| 	filter := filters.NewArgs() | 	filter := filters.NewArgs() | ||||||
| @ -132,12 +132,12 @@ func IsDeployed(ctx context.Context, cl *dockerclient.Client, stackName string) | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Debugf("'%s' has been detected as deployed with version '%s'", stackName, version) | 		logrus.Debugf("%s has been detected as deployed with version %s", stackName, version) | ||||||
|  |  | ||||||
| 		return true, version, nil | 		return true, version, nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Debugf("'%s' has been detected as not deployed", stackName) | 	logrus.Debugf("%s has been detected as not deployed", stackName) | ||||||
| 	return isDeployed, version, nil | 	return isDeployed, version, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -158,7 +158,7 @@ func pruneServices(ctx context.Context, cl *dockerclient.Client, namespace conve | |||||||
| } | } | ||||||
|  |  | ||||||
| // RunDeploy is the swarm implementation of docker stack deploy | // RunDeploy is the swarm implementation of docker stack deploy | ||||||
| func RunDeploy(cl *dockerclient.Client, opts Deploy, cfg *composetypes.Config) error { | func RunDeploy(cl *dockerclient.Client, opts Deploy, cfg *composetypes.Config, appName string, dontWait bool) error { | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
|  |  | ||||||
| 	if err := validateResolveImageFlag(&opts); err != nil { | 	if err := validateResolveImageFlag(&opts); err != nil { | ||||||
| @ -170,7 +170,7 @@ func RunDeploy(cl *dockerclient.Client, opts Deploy, cfg *composetypes.Config) e | |||||||
| 		opts.ResolveImage = ResolveImageNever | 		opts.ResolveImage = ResolveImageNever | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return deployCompose(ctx, cl, opts, cfg) | 	return deployCompose(ctx, cl, opts, cfg, appName, dontWait) | ||||||
| } | } | ||||||
|  |  | ||||||
| // validateResolveImageFlag validates the opts.resolveImage command line option | // validateResolveImageFlag validates the opts.resolveImage command line option | ||||||
| @ -183,7 +183,7 @@ func validateResolveImageFlag(opts *Deploy) error { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func deployCompose(ctx context.Context, cl *dockerclient.Client, opts Deploy, config *composetypes.Config) error { | func deployCompose(ctx context.Context, cl *dockerclient.Client, opts Deploy, config *composetypes.Config, appName string, dontWait bool) error { | ||||||
| 	namespace := convert.NewNamespace(opts.Namespace) | 	namespace := convert.NewNamespace(opts.Namespace) | ||||||
|  |  | ||||||
| 	if opts.Prune { | 	if opts.Prune { | ||||||
| @ -224,7 +224,7 @@ func deployCompose(ctx context.Context, cl *dockerclient.Client, opts Deploy, co | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return deployServices(ctx, cl, services, namespace, opts.SendRegistryAuth, opts.ResolveImage) | 	return deployServices(ctx, cl, services, namespace, opts.SendRegistryAuth, opts.ResolveImage, appName, dontWait) | ||||||
| } | } | ||||||
|  |  | ||||||
| func getServicesDeclaredNetworks(serviceConfigs []composetypes.ServiceConfig) map[string]struct{} { | func getServicesDeclaredNetworks(serviceConfigs []composetypes.ServiceConfig) map[string]struct{} { | ||||||
| @ -339,7 +339,9 @@ func deployServices( | |||||||
| 	services map[string]swarm.ServiceSpec, | 	services map[string]swarm.ServiceSpec, | ||||||
| 	namespace convert.Namespace, | 	namespace convert.Namespace, | ||||||
| 	sendAuth bool, | 	sendAuth bool, | ||||||
| 	resolveImage string) error { | 	resolveImage string, | ||||||
|  | 	appName string, | ||||||
|  | 	dontWait bool) error { | ||||||
| 	existingServices, err := GetStackServices(ctx, cl, namespace.Name()) | 	existingServices, err := GetStackServices(ctx, cl, namespace.Name()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -350,7 +352,7 @@ func deployServices( | |||||||
| 		existingServiceMap[service.Spec.Name] = service | 		existingServiceMap[service.Spec.Name] = service | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var serviceIDs []string | 	serviceIDs := make(map[string]string) | ||||||
| 	for internalName, serviceSpec := range services { | 	for internalName, serviceSpec := range services { | ||||||
| 		var ( | 		var ( | ||||||
| 			name        = namespace.Scope(internalName) | 			name        = namespace.Scope(internalName) | ||||||
| @ -358,18 +360,6 @@ func deployServices( | |||||||
| 			encodedAuth string | 			encodedAuth string | ||||||
| 		) | 		) | ||||||
|  |  | ||||||
| 		// FIXME: disable for now as not sure how to avoid having a `dockerCli` |  | ||||||
| 		// instance here and would rather not copy/pasta that entire module in |  | ||||||
| 		// right now for something that we don't even support right now. Will skip |  | ||||||
| 		// this for now. |  | ||||||
| 		if sendAuth { |  | ||||||
| 			// Retrieve encoded auth token from the image reference |  | ||||||
| 			// encodedAuth, err = command.RetrieveAuthTokenFromImage(ctx, dockerCli, image) |  | ||||||
| 			// if err != nil { |  | ||||||
| 			// 	return err |  | ||||||
| 			// } |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if service, exists := existingServiceMap[name]; exists { | 		if service, exists := existingServiceMap[name]; exists { | ||||||
| 			logrus.Infof("Updating service %s (id: %s)\n", name, service.ID) | 			logrus.Infof("Updating service %s (id: %s)\n", name, service.ID) | ||||||
|  |  | ||||||
| @ -402,7 +392,6 @@ func deployServices( | |||||||
|  |  | ||||||
| 			// Stack deploy does not have a `--force` option. Preserve existing | 			// Stack deploy does not have a `--force` option. Preserve existing | ||||||
| 			// ForceUpdate value so that tasks are not re-deployed if not updated. | 			// ForceUpdate value so that tasks are not re-deployed if not updated. | ||||||
| 			// TODO move this to API client? |  | ||||||
| 			serviceSpec.TaskTemplate.ForceUpdate = service.Spec.TaskTemplate.ForceUpdate | 			serviceSpec.TaskTemplate.ForceUpdate = service.Spec.TaskTemplate.ForceUpdate | ||||||
|  |  | ||||||
| 			response, err := cl.ServiceUpdate(ctx, service.ID, service.Version, serviceSpec, updateOpts) | 			response, err := cl.ServiceUpdate(ctx, service.ID, service.Version, serviceSpec, updateOpts) | ||||||
| @ -410,7 +399,7 @@ func deployServices( | |||||||
| 				return errors.Wrapf(err, "failed to update service %s", name) | 				return errors.Wrapf(err, "failed to update service %s", name) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			serviceIDs = append(serviceIDs, service.ID) | 			serviceIDs[service.ID] = name | ||||||
|  |  | ||||||
| 			for _, warning := range response.Warnings { | 			for _, warning := range response.Warnings { | ||||||
| 				logrus.Warn(warning) | 				logrus.Warn(warning) | ||||||
| @ -430,18 +419,27 @@ func deployServices( | |||||||
| 				return errors.Wrapf(err, "failed to create service %s", name) | 				return errors.Wrapf(err, "failed to create service %s", name) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			serviceIDs = append(serviceIDs, serviceCreateResponse.ID) | 			serviceIDs[serviceCreateResponse.ID] = name | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	logrus.Infof("waiting for services to converge: %s", strings.Join(serviceIDs, ", ")) | 	var serviceNames []string | ||||||
|  | 	for _, serviceName := range serviceIDs { | ||||||
|  | 		serviceNames = append(serviceNames, serviceName) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if dontWait { | ||||||
|  | 		logrus.Warn("skipping converge logic checks") | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Infof("waiting for services to converge: %s", strings.Join(serviceNames, ", ")) | ||||||
| 	ch := make(chan error, len(serviceIDs)) | 	ch := make(chan error, len(serviceIDs)) | ||||||
| 	for _, serviceID := range serviceIDs { | 	for serviceID, serviceName := range serviceIDs { | ||||||
| 		logrus.Debugf("waiting on %s to converge", serviceID) | 		logrus.Debugf("waiting on %s to converge", serviceName) | ||||||
| 		go func(s string) { | 		go func(sID, sName, aName string) { | ||||||
| 			ch <- waitOnService(ctx, cl, s) | 			ch <- WaitOnService(ctx, cl, sID, aName) | ||||||
| 		}(serviceID) | 		}(serviceID, serviceName, appName) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, serviceID := range serviceIDs { | 	for _, serviceID := range serviceIDs { | ||||||
| @ -471,7 +469,7 @@ func getStackConfigs(ctx context.Context, dockerclient client.APIClient, namespa | |||||||
|  |  | ||||||
| // https://github.com/docker/cli/blob/master/cli/command/service/helpers.go | // https://github.com/docker/cli/blob/master/cli/command/service/helpers.go | ||||||
| // https://github.com/docker/cli/blob/master/cli/command/service/progress/progress.go | // https://github.com/docker/cli/blob/master/cli/command/service/progress/progress.go | ||||||
| func waitOnService(ctx context.Context, cl *dockerclient.Client, serviceID string) error { | func WaitOnService(ctx context.Context, cl *dockerclient.Client, serviceID, appName string) error { | ||||||
| 	errChan := make(chan error, 1) | 	errChan := make(chan error, 1) | ||||||
| 	pipeReader, pipeWriter := io.Pipe() | 	pipeReader, pipeWriter := io.Pipe() | ||||||
|  |  | ||||||
| @ -481,12 +479,27 @@ func waitOnService(ctx context.Context, cl *dockerclient.Client, serviceID strin | |||||||
|  |  | ||||||
| 	go io.Copy(ioutil.Discard, pipeReader) | 	go io.Copy(ioutil.Discard, pipeReader) | ||||||
|  |  | ||||||
| 	timeout := 60 * time.Second | 	timeout := 30 * time.Second | ||||||
|  |  | ||||||
| 	select { | 	select { | ||||||
| 	case err := <-errChan: | 	case err := <-errChan: | ||||||
| 		return err | 		return err | ||||||
| 	case <-time.After(timeout): | 	case <-time.After(timeout): | ||||||
| 		return fmt.Errorf("%s has still not converged (%s second timeout)?", serviceID, timeout) | 		return fmt.Errorf(fmt.Sprintf(` | ||||||
|  | %s has still not converged (%s second timeout reached) | ||||||
|  |  | ||||||
|  | This does not necessarily mean your deployment has failed, it may just be that | ||||||
|  | the app is taking longer to deploy based on your server resources or network | ||||||
|  | latency. Please run the following the inspect the logs of your deployed app: | ||||||
|  |  | ||||||
|  |     abra app logs %s | ||||||
|  |  | ||||||
|  | If a service is failing to even start (run "abra app ps %s" to see what | ||||||
|  | services are running) there could be a few things. The follow command will | ||||||
|  | try to smoke those out for you: | ||||||
|  |  | ||||||
|  |     abra app errors --watch %s | ||||||
|  |  | ||||||
|  | `, appName, timeout, appName, appName, appName)) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										25
									
								
								pkg/web/client.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								pkg/web/client.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | package web | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"github.com/hashicorp/go-retryablehttp" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // customLeveledLogger is custom logger with logrus baked in | ||||||
|  | type customLeveledLogger struct { | ||||||
|  | 	retryablehttp.Logger | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Printf wires up logrus into the custom retryablehttp logger | ||||||
|  | func (l customLeveledLogger) Printf(msg string, args ...interface{}) { | ||||||
|  | 	logrus.Debugf(fmt.Sprintf(msg, args...)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewHTTPRetryClient instantiates a new http client with retries baked in | ||||||
|  | func NewHTTPRetryClient() *retryablehttp.Client { | ||||||
|  | 	retryClient := retryablehttp.NewClient() | ||||||
|  | 	retryClient.Logger = customLeveledLogger{} | ||||||
|  | 	return retryClient | ||||||
|  | } | ||||||
| @ -3,7 +3,10 @@ package web | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"os" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @ -13,7 +16,7 @@ const Timeout = 10 * time.Second | |||||||
|  |  | ||||||
| // ReadJSON reads JSON and parses it into your chosen interface pointer | // ReadJSON reads JSON and parses it into your chosen interface pointer | ||||||
| func ReadJSON(url string, target interface{}) error { | func ReadJSON(url string, target interface{}) error { | ||||||
| 	httpClient := &http.Client{Timeout: Timeout} | 	httpClient := NewHTTPRetryClient() | ||||||
| 	res, err := httpClient.Get(url) | 	res, err := httpClient.Get(url) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -21,3 +24,29 @@ func ReadJSON(url string, target interface{}) error { | |||||||
| 	defer res.Body.Close() | 	defer res.Body.Close() | ||||||
| 	return json.NewDecoder(res.Body).Decode(target) | 	return json.NewDecoder(res.Body).Decode(target) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetFile downloads a file and saves it to a filepath | ||||||
|  | func GetFile(filepath string, url string) (err error) { | ||||||
|  | 	out, err := os.Create(filepath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer out.Close() | ||||||
|  |  | ||||||
|  | 	resp, err := http.Get(url) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer resp.Body.Close() | ||||||
|  |  | ||||||
|  | 	if resp.StatusCode != http.StatusOK { | ||||||
|  | 		return fmt.Errorf("bad status: %s", resp.Status) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	_, err = io.Copy(out, resp.Body) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| ABRA_VERSION="0.3.0-alpha" | ABRA_VERSION="0.3.0-alpha" | ||||||
| ABRA_RELEASE_URL="https://git.coopcloud.tech/api/v1/repos/coop-cloud/abra/releases/tags/$ABRA_VERSION" | ABRA_RELEASE_URL="https://git.coopcloud.tech/api/v1/repos/coop-cloud/abra/releases/tags/$ABRA_VERSION" | ||||||
| RC_VERSION="0.3.1-alpha-rc2" | RC_VERSION="0.4.0-alpha-rc1" | ||||||
| RC_VERSION_URL="https://git.coopcloud.tech/api/v1/repos/coop-cloud/abra/releases/tags/$RC_VERSION" | RC_VERSION_URL="https://git.coopcloud.tech/api/v1/repos/coop-cloud/abra/releases/tags/$RC_VERSION" | ||||||
|  |  | ||||||
| for arg in "$@"; do | for arg in "$@"; do | ||||||
| @ -28,7 +28,7 @@ function show_banner { | |||||||
| } | } | ||||||
|  |  | ||||||
| function print_checksum_error { | function print_checksum_error { | ||||||
|       echo "$(tput setaf 1)ERROR: the checksum of downloaded file doesn't match the checksum in release!!! Either the file was corrupted during download or the file has been changed during transport!$(tput sgr0)" |       echo "$(tput setaf 1)ERROR: the checksum of downloaded file doesn't match the checksum in release! Either the file was corrupted during download or the file has been changed during transport$(tput sgr0)" | ||||||
|       echo "expected checksum: $checksum" |       echo "expected checksum: $checksum" | ||||||
|       echo "checksum of downloaded file: $localsum" |       echo "checksum of downloaded file: $localsum" | ||||||
|       echo "abra was NOT installed/upgraded" |       echo "abra was NOT installed/upgraded" | ||||||
| @ -37,29 +37,28 @@ function print_checksum_error { | |||||||
| function install_abra_release { | function install_abra_release { | ||||||
|   mkdir -p "$HOME/.local/bin" |   mkdir -p "$HOME/.local/bin" | ||||||
|  |  | ||||||
|   if ! type "curl" > /dev/null 2>&1; then |   if ! type "wget" > /dev/null 2>&1; then | ||||||
|     echo "'curl' is not installed, cannot proceed..." |     echo "'wget' is not installed, cannot proceed..." | ||||||
|     echo "perhaps try installing manually via the releases URL?" |     echo "perhaps try installing manually via the releases URL?" | ||||||
|     echo "https://git.coopcloud.tech/coop-cloud/abra/releases" |     echo "https://git.coopcloud.tech/coop-cloud/abra/releases" | ||||||
|     exit 1 |     exit 1 | ||||||
|   fi |   fi | ||||||
|  |  | ||||||
|  |  | ||||||
|   # FIXME: support different architectures |  | ||||||
|   PLATFORM=$(uname -s | tr '[:upper:]' '[:lower:]')_$(uname -m) |   PLATFORM=$(uname -s | tr '[:upper:]' '[:lower:]')_$(uname -m) | ||||||
|   FILENAME="abra_"$ABRA_VERSION"_"$PLATFORM"" |   FILENAME="abra_"$ABRA_VERSION"_"$PLATFORM"" | ||||||
|   sed_command_rel='s/.*"assets":\[\{[^]]*"name":"'$FILENAME'"[^}]*"browser_download_url":"([^"]*)".*\].*/\1/p' |   sed_command_rel='s/.*"assets":\[\{[^]]*"name":"'$FILENAME'"[^}]*"browser_download_url":"([^"]*)".*\].*/\1/p' | ||||||
|   sed_command_checksums='s/.*"assets":\[\{[^\]*"name":"checksums.txt"[^}]*"browser_download_url":"([^"]*)".*\].*/\1/p' |   sed_command_checksums='s/.*"assets":\[\{[^\]*"name":"checksums.txt"[^}]*"browser_download_url":"([^"]*)".*\].*/\1/p' | ||||||
|  |  | ||||||
|   json=$(curl -s $ABRA_RELEASE_URL) |   json=$(wget -q -O- $ABRA_RELEASE_URL) | ||||||
|   release_url=$(echo $json | sed -En $sed_command_rel) |   release_url=$(echo $json | sed -En $sed_command_rel) | ||||||
|   checksums_url=$(echo $json | sed -En $sed_command_checksums) |   checksums_url=$(echo $json | sed -En $sed_command_checksums) | ||||||
|  |  | ||||||
|   checksums=$(curl -s $checksums_url) |   checksums=$(wget -q -O- $checksums_url) | ||||||
|   checksum=$(echo "$checksums" | grep "$FILENAME" - | sed -En 's/([0-9a-f]{64})\s+'"$FILENAME"'.*/\1/p') |   checksum=$(echo "$checksums" | grep "$FILENAME" - | sed -En 's/([0-9a-f]{64})\s+'"$FILENAME"'.*/\1/p') | ||||||
|  |  | ||||||
|   echo "downloading $ABRA_VERSION $PLATFORM binary release for abra..." |   echo "downloading $ABRA_VERSION $PLATFORM binary release for abra..." | ||||||
|   curl --progress-bar "$release_url" --output "$HOME/.local/bin/.abra-download" |   wget -q --show-progress "$release_url" -O "$HOME/.local/bin/.abra-download" | ||||||
|   localsum=$(sha256sum $HOME/.local/bin/.abra-download | sed -En 's/([0-9a-f]{64})\s+.*/\1/p') |   localsum=$(sha256sum $HOME/.local/bin/.abra-download | sed -En 's/([0-9a-f]{64})\s+.*/\1/p') | ||||||
|   echo "checking if checksums match..." |   echo "checking if checksums match..." | ||||||
|   if [[ "$localsum" != "$checksum" ]]; then |   if [[ "$localsum" != "$checksum" ]]; then | ||||||
| @ -76,17 +75,19 @@ function install_abra_release { | |||||||
|       p=$HOME/.local/bin |       p=$HOME/.local/bin | ||||||
|       com="echo PATH=\$PATH:$p" |       com="echo PATH=\$PATH:$p" | ||||||
|       if [[ $SHELL =~ "bash" ]]; then |       if [[ $SHELL =~ "bash" ]]; then | ||||||
|           echo "echo $com >> $HOME/.bashrc" |           echo "$com >> $HOME/.bashrc" | ||||||
|       elif [[ $SHELL =~ "fizsh" ]]; then |       elif [[ $SHELL =~ "fizsh" ]]; then | ||||||
|           echo "echo $com >> $HOME/.fizsh/.fizshrc" |           echo "$com >> $HOME/.fizsh/.fizshrc" | ||||||
|       elif [[ $SHELL =~ "zsh" ]]; then |       elif [[ $SHELL =~ "zsh" ]]; then | ||||||
|           echo "echo $com >> $HOME/.zshrc" |           echo "$com >> $HOME/.zshrc" | ||||||
|       else |       else | ||||||
|           echo "echo $com >> $HOME/.profile" |           echo "$com >> $HOME/.profile" | ||||||
|       fi |       fi | ||||||
|   fi |   fi | ||||||
|  |  | ||||||
|   echo "abra installed to $HOME/.local/bin/abra" |   echo "abra installed to $HOME/.local/bin/abra" | ||||||
|  |   echo "test your installation is working by running \"abra\" on your command-line" | ||||||
|  |   echo "run \"abra autocomplete -h\" to see how to set up command-line autocompletion" | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										101
									
								
								tests/integration/core.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										101
									
								
								tests/integration/core.sh
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,101 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | # the goal of this script is to ensure basic core functionality is working | ||||||
|  | # before we make new releases. we try to make a balance between manual testing | ||||||
|  | # and automated testing, i.e. we don't invest too much time in a fragile | ||||||
|  | # automation that costs us more time to maintain and instead just do the test | ||||||
|  | # manually (see `../manual/manual.md` for more). it is a balance which we | ||||||
|  | # figure out together. | ||||||
|  |  | ||||||
|  | set -ex | ||||||
|  |  | ||||||
|  | ABRA="$HOME/.local/bin/abra -d" | ||||||
|  | INSTALLER_URL="https://install.abra.coopcloud.tech" | ||||||
|  |  | ||||||
|  | for arg in "$@"; do | ||||||
|  |   if [ "$arg" == "--dev" ]; then | ||||||
|  |     ABRA="/src/abra -d" | ||||||
|  |     INSTALLER_URL="https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/installer/installer" | ||||||
|  |   fi | ||||||
|  | done | ||||||
|  |  | ||||||
|  | export PATH=$PATH:$HOME/.local/bin | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # choosing abra executable for test run | ||||||
|  | # ======================================================================== | ||||||
|  | echo "choosing $ABRA as abra executable" | ||||||
|  | echo "choosing $INSTALLER_URL as abra installer url" | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # latest stable release | ||||||
|  | # ======================================================================== | ||||||
|  | wget -O- https://install.abra.autonomic.zone | bash | ||||||
|  | ~/.local/bin/abra -v | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # latest rc release | ||||||
|  | # ======================================================================== | ||||||
|  | wget -O- https://install.abra.autonomic.zone | bash -s -- --rc | ||||||
|  | ~/.local/bin/abra -v | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # upgrade to stable in-place | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA upgrade | ||||||
|  | ~/.local/bin/abra -v | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # upgrade to rc in-place | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA upgrade --rc | ||||||
|  | ~/.local/bin/abra -v | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # autocomplete | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA autocomplete bash | ||||||
|  | $ABRA autocomplete fizsh | ||||||
|  | $ABRA autocomplete zsh | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # record command | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA record new -p gandi -t A -n e2e -v 192.157.2.21 coopcloud.tech | ||||||
|  | $ABRA record list -p gandi coopcloud.tech | grep -q e2e | ||||||
|  | $ABRA -n record rm -p gandi -t A -n e2e coopcloud.tech | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # catalogue command | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA catalogue generate | ||||||
|  | $ABRA catalogue generate -s gitea | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # recipe command | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA recipe new testrecipe | ||||||
|  |  | ||||||
|  | $ABRA recipe list | ||||||
|  | $ABRA recipe list -p cloud | ||||||
|  |  | ||||||
|  | $ABRA recipe versions peertube | ||||||
|  |  | ||||||
|  | $ABRA recipe lint gitea | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # server command | ||||||
|  | # ======================================================================== | ||||||
|  | $ABRA -n server new -p hetzner-cloud --hn e2e | ||||||
|  |  | ||||||
|  | $ABRA server ls | grep -q e2e | ||||||
|  |  | ||||||
|  | $ABRA -n server rm -s -p hetzner-cloud --hn e2e | ||||||
|  |  | ||||||
|  | # ======================================================================== | ||||||
|  | # app command | ||||||
|  | # ======================================================================== | ||||||
|  |  | ||||||
|  | $ABRA app ls | ||||||
|  |  | ||||||
|  | $ABRA app ls -S | ||||||
							
								
								
									
										1
									
								
								tests/manual/apps/works/.env.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/manual/apps/works/.env.sample
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | TYPE=works | ||||||
							
								
								
									
										84
									
								
								tests/manual/apps/works/compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								tests/manual/apps/works/compose.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,84 @@ | |||||||
|  | --- | ||||||
|  |  | ||||||
|  | # The goal of this compose file is to have a testing ground for understanding | ||||||
|  | # what cases we need to handle to get stable deployments. For that, we need to | ||||||
|  | # work with healthchecks and deploy configurations quite closely. If you run | ||||||
|  | # the `make symlink` target then this will be loaded into a "fake" app on your | ||||||
|  | # local machine which you can deploy with `abra`. | ||||||
|  |  | ||||||
|  | version: "3.8" | ||||||
|  | services: | ||||||
|  |   r1_should_work: | ||||||
|  |     image: redis:alpine | ||||||
|  |     deploy: | ||||||
|  |       update_config: | ||||||
|  |         failure_action: rollback | ||||||
|  |         order: start-first | ||||||
|  |       rollback_config: | ||||||
|  |         order: start-first | ||||||
|  |       restart_policy: | ||||||
|  |         max_attempts: 1 | ||||||
|  |     healthcheck: | ||||||
|  |       test: redis-cli ping | ||||||
|  |       interval: 2s | ||||||
|  |       retries: 3 | ||||||
|  |       start_period: 1s | ||||||
|  |       timeout: 3s | ||||||
|  |  | ||||||
|  |   r2_broken_health_check: | ||||||
|  |     image: redis:alpine | ||||||
|  |     deploy: | ||||||
|  |       update_config: | ||||||
|  |         failure_action: rollback | ||||||
|  |         order: start-first | ||||||
|  |       rollback_config: | ||||||
|  |         order: start-first | ||||||
|  |       restart_policy: | ||||||
|  |         max_attempts: 3 | ||||||
|  |     healthcheck: | ||||||
|  |       test: foobar | ||||||
|  |       interval: 2s | ||||||
|  |       retries: 3 | ||||||
|  |       start_period: 1s | ||||||
|  |       timeout: 3s | ||||||
|  |  | ||||||
|  |   r3_no_health_check: | ||||||
|  |     image: redis:alpine | ||||||
|  |     deploy: | ||||||
|  |       update_config: | ||||||
|  |         failure_action: rollback | ||||||
|  |         order: start-first | ||||||
|  |       rollback_config: | ||||||
|  |         order: start-first | ||||||
|  |       restart_policy: | ||||||
|  |         max_attempts: 3 | ||||||
|  |  | ||||||
|  |   r4_disabled_health_check: | ||||||
|  |     image: redis:alpine | ||||||
|  |     deploy: | ||||||
|  |       update_config: | ||||||
|  |         failure_action: rollback | ||||||
|  |         order: start-first | ||||||
|  |       rollback_config: | ||||||
|  |         order: start-first | ||||||
|  |       restart_policy: | ||||||
|  |         max_attempts: 3 | ||||||
|  |     healthcheck: | ||||||
|  |       disable: true | ||||||
|  |  | ||||||
|  |   r5_should_also_work: | ||||||
|  |     image: redis:alpine | ||||||
|  |     deploy: | ||||||
|  |       update_config: | ||||||
|  |         failure_action: rollback | ||||||
|  |         order: start-first | ||||||
|  |       rollback_config: | ||||||
|  |         order: start-first | ||||||
|  |       restart_policy: | ||||||
|  |         max_attempts: 1 | ||||||
|  |     healthcheck: | ||||||
|  |       test: redis-cli ping | ||||||
|  |       interval: 2s | ||||||
|  |       retries: 3 | ||||||
|  |       start_period: 1s | ||||||
|  |       timeout: 3s | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user