forked from toolshed/abra
		
	Compare commits
	
		
			22 Commits
		
	
	
		
			cp-enhance
			...
			upgrade-cl
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b81f5651d3 | |||
| 7d80f4d56b | |||
| 114bdc5ce9 | |||
| 40c0fb4bac | |||
| 8a7d17f37b | |||
| deb4293fba | |||
| ac39d6ab97 | |||
| 428426b6b7 | |||
| 0643df6d73 | |||
| e9b99fe921 | |||
| 4920dfedb3 | |||
| 0a3624c15b | |||
| c5687dfbd7 | |||
| ca91abbed9 | |||
| d4727db8f9 | |||
| af8cd1f67a | |||
| cdd7516e54 | |||
| 99e3ed416f | |||
| 02b726db02 | |||
| 2de6934322 | |||
| cb49cf06d1 | |||
| 9affda8a70 | 
| @ -1,7 +1,7 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var AppCommand = cli.Command{ | var AppCommand = cli.Command{ | ||||||
| @ -10,28 +10,28 @@ var AppCommand = cli.Command{ | |||||||
| 	Usage:       "Manage apps", | 	Usage:       "Manage apps", | ||||||
| 	ArgsUsage:   "<domain>", | 	ArgsUsage:   "<domain>", | ||||||
| 	Description: "Functionality for managing the life cycle of your apps", | 	Description: "Functionality for managing the life cycle of your apps", | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		appBackupCommand, | 		&appBackupCommand, | ||||||
| 		appCheckCommand, | 		&appCheckCommand, | ||||||
| 		appCmdCommand, | 		&appCmdCommand, | ||||||
| 		appConfigCommand, | 		&appConfigCommand, | ||||||
| 		appCpCommand, | 		&appCpCommand, | ||||||
| 		appDeployCommand, | 		&appDeployCommand, | ||||||
| 		appErrorsCommand, | 		&appErrorsCommand, | ||||||
| 		appListCommand, | 		&appListCommand, | ||||||
| 		appLogsCommand, | 		&appLogsCommand, | ||||||
| 		appNewCommand, | 		&appNewCommand, | ||||||
| 		appPsCommand, | 		&appPsCommand, | ||||||
| 		appRemoveCommand, | 		&appRemoveCommand, | ||||||
| 		appRestartCommand, | 		&appRestartCommand, | ||||||
| 		appRestoreCommand, | 		&appRestoreCommand, | ||||||
| 		appRollbackCommand, | 		&appRollbackCommand, | ||||||
| 		appRunCommand, | 		&appRunCommand, | ||||||
| 		appSecretCommand, | 		&appSecretCommand, | ||||||
| 		appServicesCommand, | 		&appServicesCommand, | ||||||
| 		appUndeployCommand, | 		&appUndeployCommand, | ||||||
| 		appUpgradeCommand, | 		&appUpgradeCommand, | ||||||
| 		appVersionCommand, | 		&appVersionCommand, | ||||||
| 		appVolumeCommand, | 		&appVolumeCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,414 +1,296 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"archive/tar" |  | ||||||
| 	"context" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	containerPkg "coopcloud.tech/abra/pkg/container" |  | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" |  | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/container" |  | ||||||
| 	"github.com/docker/cli/cli/command" |  | ||||||
| 	"github.com/docker/docker/api/types" |  | ||||||
| 	"github.com/docker/docker/api/types/filters" |  | ||||||
| 	dockerClient "github.com/docker/docker/client" |  | ||||||
| 	"github.com/docker/docker/pkg/archive" |  | ||||||
| 	"github.com/klauspost/pgzip" |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type backupConfig struct { | var snapshot string | ||||||
| 	preHookCmd  string | var snapshotFlag = &cli.StringFlag{ | ||||||
| 	postHookCmd string | 	Name:        "snapshot, s", | ||||||
| 	backupPaths []string | 	Usage:       "Lists specific snapshot", | ||||||
|  | 	Destination: &snapshot, | ||||||
| } | } | ||||||
|  |  | ||||||
| var appBackupCommand = cli.Command{ | var includePath string | ||||||
| 	Name:      "backup", | var includePathFlag = &cli.StringFlag{ | ||||||
| 	Aliases:   []string{"bk"}, | 	Name:        "path, p", | ||||||
| 	Usage:     "Run app backup", | 	Usage:       "Include path", | ||||||
| 	ArgsUsage: "<domain> [<service>]", | 	Destination: &includePath, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var resticRepo string | ||||||
|  | var resticRepoFlag = &cli.StringFlag{ | ||||||
|  | 	Name:        "repo, r", | ||||||
|  | 	Usage:       "Restic repository", | ||||||
|  | 	Destination: &resticRepo, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var appBackupListCommand = cli.Command{ | ||||||
|  | 	Name:    "list", | ||||||
|  | 	Aliases: []string{"ls"}, | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.DebugFlag, | 		internal.DebugFlag, | ||||||
| 		internal.OfflineFlag, | 		internal.OfflineFlag, | ||||||
| 		internal.ChaosFlag, | 		snapshotFlag, | ||||||
|  | 		includePathFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Before:       internal.SubCommandBefore, | 	Before:       internal.SubCommandBefore, | ||||||
|  | 	Usage:        "List all backups", | ||||||
| 	BashComplete: autocomplete.AppNameComplete, | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Description: ` |  | ||||||
| Run an app backup. |  | ||||||
|  |  | ||||||
| A backup command and pre/post hook commands are defined in the recipe |  | ||||||
| configuration. Abra reads this configuration and run the comands in the context |  | ||||||
| of the deployed services. Pass <service> if you only want to back up a single |  | ||||||
| service. All backups are placed in the ~/.abra/backups directory. |  | ||||||
|  |  | ||||||
| A single backup file is produced for all backup paths specified for a service. |  | ||||||
| If we have the following backup configuration: |  | ||||||
|  |  | ||||||
|     - "backupbot.backup.path=/var/lib/foo,/var/lib/bar" |  | ||||||
|  |  | ||||||
| And we run "abra app backup example.com app", Abra will produce a file that |  | ||||||
| looks like: |  | ||||||
|  |  | ||||||
|     ~/.abra/backups/example_com_app_609341138.tar.gz |  | ||||||
|  |  | ||||||
| This file is a compressed archive which contains all backup paths. To see paths, run: |  | ||||||
|  |  | ||||||
|     tar -tf ~/.abra/backups/example_com_app_609341138.tar.gz |  | ||||||
|  |  | ||||||
| (Make sure to change the name of the backup file) |  | ||||||
|  |  | ||||||
| This single file can be used to restore your app. See "abra app restore" for more. |  | ||||||
| `, |  | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		recipe, err := recipePkg.Get(app.Recipe, internal.Offline) | 		if err := recipe.EnsureExists(app.Recipe); err != nil { | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !internal.Chaos { | 		if !internal.Chaos { | ||||||
| 			if err := recipePkg.EnsureIsClean(app.Recipe); err != nil { | 			if err := recipe.EnsureIsClean(app.Recipe); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if !internal.Offline { | 			if !internal.Offline { | ||||||
| 				if err := recipePkg.EnsureUpToDate(app.Recipe); err != nil { | 				if err := recipe.EnsureUpToDate(app.Recipe); err != nil { | ||||||
| 					logrus.Fatal(err) | 					logrus.Fatal(err) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if err := recipePkg.EnsureLatest(app.Recipe); err != nil { | 			if err := recipe.EnsureLatest(app.Recipe); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		backupConfigs := make(map[string]backupConfig) |  | ||||||
| 		for _, service := range recipe.Config.Services { |  | ||||||
| 			if backupsEnabled, ok := service.Deploy.Labels["backupbot.backup"]; ok { |  | ||||||
| 				if backupsEnabled == "true" { |  | ||||||
| 					fullServiceName := fmt.Sprintf("%s_%s", app.StackName(), service.Name) |  | ||||||
| 					bkConfig := backupConfig{} |  | ||||||
|  |  | ||||||
| 					logrus.Debugf("backup config detected for %s", fullServiceName) |  | ||||||
|  |  | ||||||
| 					if paths, ok := service.Deploy.Labels["backupbot.backup.path"]; ok { |  | ||||||
| 						logrus.Debugf("detected backup paths for %s: %s", fullServiceName, paths) |  | ||||||
| 						bkConfig.backupPaths = strings.Split(paths, ",") |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					if preHookCmd, ok := service.Deploy.Labels["backupbot.backup.pre-hook"]; ok { |  | ||||||
| 						logrus.Debugf("detected pre-hook command for %s: %s", fullServiceName, preHookCmd) |  | ||||||
| 						bkConfig.preHookCmd = preHookCmd |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					if postHookCmd, ok := service.Deploy.Labels["backupbot.backup.post-hook"]; ok { |  | ||||||
| 						logrus.Debugf("detected post-hook command for %s: %s", fullServiceName, postHookCmd) |  | ||||||
| 						bkConfig.postHookCmd = postHookCmd |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					backupConfigs[service.Name] = bkConfig |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		cl, err := client.New(app.Server) | 		cl, err := client.New(app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		serviceName := c.Args().Get(1) | 		targetContainer, err := internal.RetrieveBackupBotContainer(cl) | ||||||
| 		if serviceName != "" { | 		if err != nil { | ||||||
| 			backupConfig, ok := backupConfigs[serviceName] | 			logrus.Fatal(err) | ||||||
| 			if !ok { | 		} | ||||||
| 				logrus.Fatalf("no backup config for %s? does %s exist?", serviceName, serviceName) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			logrus.Infof("running backup for the %s service", serviceName) | 		execEnv := []string{fmt.Sprintf("SERVICE=%s", app.Domain)} | ||||||
|  | 		if snapshot != "" { | ||||||
|  | 			logrus.Debugf("including SNAPSHOT=%s in backupbot exec invocation", snapshot) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot)) | ||||||
|  | 		} | ||||||
|  | 		if includePath != "" { | ||||||
|  | 			logrus.Debugf("including INCLUDE_PATH=%s in backupbot exec invocation", includePath) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("INCLUDE_PATH=%s", includePath)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 			if err := runBackup(cl, app, serviceName, backupConfig); err != nil { | 		if err := internal.RunBackupCmdRemote(cl, "ls", targetContainer.ID, execEnv); err != nil { | ||||||
| 				logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if len(backupConfigs) == 0 { |  | ||||||
| 				logrus.Fatalf("no backup configs discovered for %s?", app.Name) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			for serviceName, backupConfig := range backupConfigs { |  | ||||||
| 				logrus.Infof("running backup for the %s service", serviceName) |  | ||||||
|  |  | ||||||
| 				if err := runBackup(cl, app, serviceName, backupConfig); err != nil { |  | ||||||
| 					logrus.Fatal(err) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| // TimeStamp generates a file name friendly timestamp. | var appBackupDownloadCommand = cli.Command{ | ||||||
| func TimeStamp() string { | 	Name:    "download", | ||||||
| 	ts := time.Now().UTC().Format(time.RFC3339) | 	Aliases: []string{"d"}, | ||||||
| 	return strings.Replace(ts, ":", "-", -1) | 	Flags: []cli.Flag{ | ||||||
| } | 		internal.DebugFlag, | ||||||
|  | 		internal.OfflineFlag, | ||||||
|  | 		snapshotFlag, | ||||||
|  | 		includePathFlag, | ||||||
|  | 	}, | ||||||
|  | 	Before:       internal.SubCommandBefore, | ||||||
|  | 	Usage:        "Download a backup", | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| // runBackup does the actual backup logic. | 		if err := recipe.EnsureExists(app.Recipe); err != nil { | ||||||
| func runBackup(cl *dockerClient.Client, app config.App, serviceName string, bkConfig backupConfig) error { | 			logrus.Fatal(err) | ||||||
| 	if len(bkConfig.backupPaths) == 0 { |  | ||||||
| 		return fmt.Errorf("backup paths are empty for %s?", serviceName) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// FIXME: avoid instantiating a new CLI |  | ||||||
| 	dcli, err := command.NewDockerCli() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	filters := filters.NewArgs() |  | ||||||
| 	filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(), serviceName)) |  | ||||||
|  |  | ||||||
| 	targetContainer, err := containerPkg.GetContainer(context.Background(), cl, filters, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	fullServiceName := fmt.Sprintf("%s_%s", app.StackName(), serviceName) |  | ||||||
| 	if bkConfig.preHookCmd != "" { |  | ||||||
| 		splitCmd := internal.SafeSplit(bkConfig.preHookCmd) |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("split pre-hook command for %s into %s", fullServiceName, splitCmd) |  | ||||||
|  |  | ||||||
| 		preHookExecOpts := types.ExecConfig{ |  | ||||||
| 			AttachStderr: true, |  | ||||||
| 			AttachStdin:  true, |  | ||||||
| 			AttachStdout: true, |  | ||||||
| 			Cmd:          splitCmd, |  | ||||||
| 			Detach:       false, |  | ||||||
| 			Tty:          true, |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, targetContainer.ID, &preHookExecOpts); err != nil { | 		if !internal.Chaos { | ||||||
| 			return fmt.Errorf("failed to run %s on %s: %s", bkConfig.preHookCmd, targetContainer.ID, err.Error()) | 			if err := recipe.EnsureIsClean(app.Recipe); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if !internal.Offline { | ||||||
|  | 				if err := recipe.EnsureUpToDate(app.Recipe); err != nil { | ||||||
|  | 					logrus.Fatal(err) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if err := recipe.EnsureLatest(app.Recipe); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logrus.Infof("succesfully ran %s pre-hook command: %s", fullServiceName, bkConfig.preHookCmd) | 		cl, err := client.New(app.Server) | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var tempBackupPaths []string |  | ||||||
| 	for _, remoteBackupPath := range bkConfig.backupPaths { |  | ||||||
| 		sanitisedPath := strings.ReplaceAll(remoteBackupPath, "/", "_") |  | ||||||
| 		localBackupPath := filepath.Join(config.BACKUP_DIR, fmt.Sprintf("%s%s_%s.tar.gz", fullServiceName, sanitisedPath, TimeStamp())) |  | ||||||
| 		logrus.Debugf("temporarily backing up %s:%s to %s", fullServiceName, remoteBackupPath, localBackupPath) |  | ||||||
|  |  | ||||||
| 		logrus.Infof("backing up %s:%s", fullServiceName, remoteBackupPath) |  | ||||||
|  |  | ||||||
| 		content, _, err := cl.CopyFromContainer(context.Background(), targetContainer.ID, remoteBackupPath) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Debugf("failed to copy %s from container: %s", remoteBackupPath, err.Error()) | 			logrus.Fatal(err) | ||||||
| 			if err := cleanupTempArchives(tempBackupPaths); err != nil { | 		} | ||||||
| 				return fmt.Errorf("failed to clean up temporary archives: %s", err.Error()) |  | ||||||
|  | 		targetContainer, err := internal.RetrieveBackupBotContainer(cl) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		execEnv := []string{fmt.Sprintf("SERVICE=%s", app.Domain)} | ||||||
|  | 		if snapshot != "" { | ||||||
|  | 			logrus.Debugf("including SNAPSHOT=%s in backupbot exec invocation", snapshot) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot)) | ||||||
|  | 		} | ||||||
|  | 		if includePath != "" { | ||||||
|  | 			logrus.Debugf("including INCLUDE_PATH=%s in backupbot exec invocation", includePath) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("INCLUDE_PATH=%s", includePath)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := internal.RunBackupCmdRemote(cl, "download", targetContainer.ID, execEnv); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		remoteBackupDir := "/tmp/backup.tar.gz" | ||||||
|  | 		currentWorkingDir := "." | ||||||
|  | 		if err = CopyFromContainer(cl, targetContainer.ID, remoteBackupDir, currentWorkingDir); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		fmt.Println("backup successfully downloaded to current working directory") | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var appBackupCreateCommand = cli.Command{ | ||||||
|  | 	Name:    "create", | ||||||
|  | 	Aliases: []string{"c"}, | ||||||
|  | 	Flags: []cli.Flag{ | ||||||
|  | 		internal.DebugFlag, | ||||||
|  | 		internal.OfflineFlag, | ||||||
|  | 		resticRepoFlag, | ||||||
|  | 	}, | ||||||
|  | 	Before:       internal.SubCommandBefore, | ||||||
|  | 	Usage:        "Create a new backup", | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
|  | 		if err := recipe.EnsureExists(app.Recipe); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if !internal.Chaos { | ||||||
|  | 			if err := recipe.EnsureIsClean(app.Recipe); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 			return fmt.Errorf("failed to copy %s from container: %s", remoteBackupPath, err.Error()) |  | ||||||
| 		} |  | ||||||
| 		defer content.Close() |  | ||||||
|  |  | ||||||
| 		_, srcBase := archive.SplitPathDirEntry(remoteBackupPath) | 			if !internal.Offline { | ||||||
| 		preArchive := archive.RebaseArchiveEntries(content, srcBase, remoteBackupPath) | 				if err := recipe.EnsureUpToDate(app.Recipe); err != nil { | ||||||
| 		if err := copyToFile(localBackupPath, preArchive); err != nil { | 					logrus.Fatal(err) | ||||||
| 			logrus.Debugf("failed to create tar archive (%s): %s", localBackupPath, err.Error()) | 				} | ||||||
| 			if err := cleanupTempArchives(tempBackupPaths); err != nil { |  | ||||||
| 				return fmt.Errorf("failed to clean up temporary archives: %s", err.Error()) |  | ||||||
| 			} | 			} | ||||||
| 			return fmt.Errorf("failed to create tar archive (%s): %s", localBackupPath, err.Error()) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		tempBackupPaths = append(tempBackupPaths, localBackupPath) | 			if err := recipe.EnsureLatest(app.Recipe); err != nil { | ||||||
| 	} | 				logrus.Fatal(err) | ||||||
|  |  | ||||||
| 	logrus.Infof("compressing and merging archives...") |  | ||||||
|  |  | ||||||
| 	if err := mergeArchives(tempBackupPaths, fullServiceName); err != nil { |  | ||||||
| 		logrus.Debugf("failed to merge archive files: %s", err.Error()) |  | ||||||
| 		if err := cleanupTempArchives(tempBackupPaths); err != nil { |  | ||||||
| 			return fmt.Errorf("failed to clean up temporary archives: %s", err.Error()) |  | ||||||
| 		} |  | ||||||
| 		return fmt.Errorf("failed to merge archive files: %s", err.Error()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := cleanupTempArchives(tempBackupPaths); err != nil { |  | ||||||
| 		return fmt.Errorf("failed to clean up temporary archives: %s", err.Error()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if bkConfig.postHookCmd != "" { |  | ||||||
| 		splitCmd := internal.SafeSplit(bkConfig.postHookCmd) |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("split post-hook command for %s into %s", fullServiceName, splitCmd) |  | ||||||
|  |  | ||||||
| 		postHookExecOpts := types.ExecConfig{ |  | ||||||
| 			AttachStderr: true, |  | ||||||
| 			AttachStdin:  true, |  | ||||||
| 			AttachStdout: true, |  | ||||||
| 			Cmd:          splitCmd, |  | ||||||
| 			Detach:       false, |  | ||||||
| 			Tty:          true, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, targetContainer.ID, &postHookExecOpts); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Infof("succesfully ran %s post-hook command: %s", fullServiceName, bkConfig.postHookCmd) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func copyToFile(outfile string, r io.Reader) error { |  | ||||||
| 	tmpFile, err := os.CreateTemp(filepath.Dir(outfile), ".tar_temp") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	tmpPath := tmpFile.Name() |  | ||||||
|  |  | ||||||
| 	_, err = io.Copy(tmpFile, r) |  | ||||||
| 	tmpFile.Close() |  | ||||||
|  |  | ||||||
| 	if err != nil { |  | ||||||
| 		os.Remove(tmpPath) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = os.Rename(tmpPath, outfile); err != nil { |  | ||||||
| 		os.Remove(tmpPath) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func cleanupTempArchives(tarPaths []string) error { |  | ||||||
| 	for _, tarPath := range tarPaths { |  | ||||||
| 		if err := os.RemoveAll(tarPath); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("remove temporary archive file %s", tarPath) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func mergeArchives(tarPaths []string, serviceName string) error { |  | ||||||
| 	var out io.Writer |  | ||||||
| 	var cout *pgzip.Writer |  | ||||||
|  |  | ||||||
| 	localBackupPath := filepath.Join(config.BACKUP_DIR, fmt.Sprintf("%s_%s.tar.gz", serviceName, TimeStamp())) |  | ||||||
|  |  | ||||||
| 	fout, err := os.Create(localBackupPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("Failed to open %s: %s", localBackupPath, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	defer fout.Close() |  | ||||||
| 	out = fout |  | ||||||
|  |  | ||||||
| 	cout = pgzip.NewWriter(out) |  | ||||||
| 	out = cout |  | ||||||
|  |  | ||||||
| 	tw := tar.NewWriter(out) |  | ||||||
|  |  | ||||||
| 	for _, tarPath := range tarPaths { |  | ||||||
| 		if err := addTar(tw, tarPath); err != nil { |  | ||||||
| 			return fmt.Errorf("failed to merge %s: %v", tarPath, err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := tw.Close(); err != nil { |  | ||||||
| 		return fmt.Errorf("failed to close tar writer %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if cout != nil { |  | ||||||
| 		if err := cout.Flush(); err != nil { |  | ||||||
| 			return fmt.Errorf("failed to flush: %s", err) |  | ||||||
| 		} else if err = cout.Close(); err != nil { |  | ||||||
| 			return fmt.Errorf("failed to close compressed writer: %s", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Infof("backed up %s to %s", serviceName, localBackupPath) |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func addTar(tw *tar.Writer, pth string) (err error) { |  | ||||||
| 	var tr *tar.Reader |  | ||||||
| 	var rc io.ReadCloser |  | ||||||
| 	var hdr *tar.Header |  | ||||||
|  |  | ||||||
| 	if tr, rc, err = openTarFile(pth); err != nil { |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for { |  | ||||||
| 		if hdr, err = tr.Next(); err != nil { |  | ||||||
| 			if err == io.EOF { |  | ||||||
| 				err = nil |  | ||||||
| 			} | 			} | ||||||
| 			break |  | ||||||
| 		} | 		} | ||||||
| 		if err = tw.WriteHeader(hdr); err != nil { |  | ||||||
| 			break | 		cl, err := client.New(app.Server) | ||||||
| 		} else if _, err = io.Copy(tw, tr); err != nil { | 		if err != nil { | ||||||
| 			break | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| 	} |  | ||||||
| 	if err == nil { | 		targetContainer, err := internal.RetrieveBackupBotContainer(cl) | ||||||
| 		err = rc.Close() | 		if err != nil { | ||||||
| 	} else { | 			logrus.Fatal(err) | ||||||
| 		rc.Close() | 		} | ||||||
| 	} |  | ||||||
| 	return | 		execEnv := []string{fmt.Sprintf("SERVICE=%s", app.Domain)} | ||||||
|  | 		if resticRepo != "" { | ||||||
|  | 			logrus.Debugf("including RESTIC_REPO=%s in backupbot exec invocation", resticRepo) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("RESTIC_REPO=%s", resticRepo)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := internal.RunBackupCmdRemote(cl, "create", targetContainer.ID, execEnv); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func openTarFile(pth string) (tr *tar.Reader, rc io.ReadCloser, err error) { | var appBackupSnapshotsCommand = cli.Command{ | ||||||
| 	var fin *os.File | 	Name:    "snapshots", | ||||||
| 	var n int | 	Aliases: []string{"s"}, | ||||||
| 	buff := make([]byte, 1024) | 	Flags: []cli.Flag{ | ||||||
|  | 		internal.DebugFlag, | ||||||
|  | 		internal.OfflineFlag, | ||||||
|  | 		snapshotFlag, | ||||||
|  | 	}, | ||||||
|  | 	Before:       internal.SubCommandBefore, | ||||||
|  | 	Usage:        "List backup snapshots", | ||||||
|  | 	BashComplete: autocomplete.AppNameComplete, | ||||||
|  | 	Action: func(c *cli.Context) error { | ||||||
|  | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 	if fin, err = os.Open(pth); err != nil { | 		if err := recipe.EnsureExists(app.Recipe); err != nil { | ||||||
| 		return | 			logrus.Fatal(err) | ||||||
| 	} | 		} | ||||||
|  |  | ||||||
| 	if n, err = fin.Read(buff); err != nil { | 		if !internal.Chaos { | ||||||
| 		fin.Close() | 			if err := recipe.EnsureIsClean(app.Recipe); err != nil { | ||||||
| 		return | 				logrus.Fatal(err) | ||||||
| 	} else if n == 0 { | 			} | ||||||
| 		fin.Close() |  | ||||||
| 		err = fmt.Errorf("%s is empty", pth) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err = fin.Seek(0, 0); err != nil { | 			if !internal.Offline { | ||||||
| 		fin.Close() | 				if err := recipe.EnsureUpToDate(app.Recipe); err != nil { | ||||||
| 		return | 					logrus.Fatal(err) | ||||||
| 	} | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 	rc = fin | 			if err := recipe.EnsureLatest(app.Recipe); err != nil { | ||||||
| 	tr = tar.NewReader(rc) | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 	return tr, rc, nil | 		cl, err := client.New(app.Server) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		targetContainer, err := internal.RetrieveBackupBotContainer(cl) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		execEnv := []string{fmt.Sprintf("SERVICE=%s", app.Domain)} | ||||||
|  | 		if snapshot != "" { | ||||||
|  | 			logrus.Debugf("including SNAPSHOT=%s in backupbot exec invocation", snapshot) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := internal.RunBackupCmdRemote(cl, "snapshots", targetContainer.ID, execEnv); err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return nil | ||||||
|  | 	}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var appBackupCommand = cli.Command{ | ||||||
|  | 	Name:      "backup", | ||||||
|  | 	Aliases:   []string{"b"}, | ||||||
|  | 	Usage:     "Manage app backups", | ||||||
|  | 	ArgsUsage: "<domain>", | ||||||
|  | 	Subcommands: []*cli.Command{ | ||||||
|  | 		&appBackupListCommand, | ||||||
|  | 		&appBackupSnapshotsCommand, | ||||||
|  | 		&appBackupDownloadCommand, | ||||||
|  | 		&appBackupCreateCommand, | ||||||
|  | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appCheckCommand = cli.Command{ | var appCheckCommand = cli.Command{ | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appCmdCommand = cli.Command{ | var appCmdCommand = cli.Command{ | ||||||
| @ -45,10 +45,10 @@ Example: | |||||||
| 		internal.ChaosFlag, | 		internal.ChaosFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Before:      internal.SubCommandBefore, | 	Before:      internal.SubCommandBefore, | ||||||
| 	Subcommands: []cli.Command{appCmdListCommand}, | 	Subcommands: []*cli.Command{&appCmdListCommand}, | ||||||
| 	BashComplete: func(ctx *cli.Context) { | 	BashComplete: func(ctx *cli.Context) { | ||||||
| 		args := ctx.Args() | 		args := ctx.Args() | ||||||
| 		switch len(args) { | 		switch args.Len() { | ||||||
| 		case 0: | 		case 0: | ||||||
| 			autocomplete.AppNameComplete(ctx) | 			autocomplete.AppNameComplete(ctx) | ||||||
| 		case 1: | 		case 1: | ||||||
| @ -84,7 +84,7 @@ Example: | |||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use --local & --user together")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("cannot use --local & --user together")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		hasCmdArgs, parsedCmdArgs := parseCmdArgs(c.Args(), internal.LocalCmd) | 		hasCmdArgs, parsedCmdArgs := parseCmdArgs(c.Args().Slice()) | ||||||
|  |  | ||||||
| 		abraSh := path.Join(config.RECIPES_DIR, app.Recipe, "abra.sh") | 		abraSh := path.Join(config.RECIPES_DIR, app.Recipe, "abra.sh") | ||||||
| 		if _, err := os.Stat(abraSh); err != nil { | 		if _, err := os.Stat(abraSh); err != nil { | ||||||
| @ -95,7 +95,7 @@ Example: | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if internal.LocalCmd { | 		if internal.LocalCmd { | ||||||
| 			if !(len(c.Args()) >= 2) { | 			if !(c.Args().Len() >= 2) { | ||||||
| 				internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments")) | 				internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments")) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @ -131,7 +131,7 @@ Example: | |||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			if !(len(c.Args()) >= 3) { | 			if !(c.Args().Len() >= 3) { | ||||||
| 				internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments")) | 				internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments")) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @ -180,23 +180,16 @@ Example: | |||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func parseCmdArgs(args []string, isLocal bool) (bool, string) { | // Parse the command arguments from the cli args. | ||||||
| 	var ( | // Arguments should look like this: | ||||||
| 		parsedCmdArgs string | // | ||||||
| 		hasCmdArgs    bool | //	DOMAIN COMMAND -- ARGUMENT1 ARGUMENT2 ... | ||||||
| 	) | func parseCmdArgs(args []string) (bool, string) { | ||||||
|  | 	if len(args) < 4 { | ||||||
| 	if isLocal { | 		return false, "" | ||||||
| 		if len(args) > 2 { |  | ||||||
| 			return true, fmt.Sprintf("%s ", strings.Join(args[2:], " ")) |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		if len(args) > 3 { |  | ||||||
| 			return true, fmt.Sprintf("%s ", strings.Join(args[3:], " ")) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return hasCmdArgs, parsedCmdArgs | 	return true, fmt.Sprintf("%s ", strings.Join(args[3:], " ")) | ||||||
| } | } | ||||||
|  |  | ||||||
| func cmdNameComplete(appName string) { | func cmdNameComplete(appName string) { | ||||||
|  | |||||||
| @ -20,7 +20,7 @@ func TestParseCmdArgs(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, test := range tests { | 	for _, test := range tests { | ||||||
| 		ok, parsed := parseCmdArgs(test.input, false) | 		ok, parsed := parseCmdArgs(test.input) | ||||||
| 		if ok != test.shouldParse { | 		if ok != test.shouldParse { | ||||||
| 			t.Fatalf("[%s] should not parse", strings.Join(test.input, " ")) | 			t.Fatalf("[%s] should not parse", strings.Join(test.input, " ")) | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ import ( | |||||||
| 	"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" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appConfigCommand = cli.Command{ | var appConfigCommand = cli.Command{ | ||||||
|  | |||||||
| @ -22,7 +22,7 @@ import ( | |||||||
| 	"github.com/docker/docker/errdefs" | 	"github.com/docker/docker/errdefs" | ||||||
| 	"github.com/docker/docker/pkg/archive" | 	"github.com/docker/docker/pkg/archive" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appCpCommand = cli.Command{ | var appCpCommand = cli.Command{ | ||||||
| @ -76,9 +76,9 @@ And if you want to copy that file back to your current working directory locally | |||||||
| 		logrus.Debugf("retrieved %s as target container on %s", formatter.ShortenID(container.ID), app.Server) | 		logrus.Debugf("retrieved %s as target container on %s", formatter.ShortenID(container.ID), app.Server) | ||||||
|  |  | ||||||
| 		if toContainer { | 		if toContainer { | ||||||
| 			err = copyToContainer(cl, container.ID, srcPath, dstPath) | 			err = CopyToContainer(cl, container.ID, srcPath, dstPath) | ||||||
| 		} else { | 		} else { | ||||||
| 			err = copyFromContainer(cl, container.ID, srcPath, dstPath) | 			err = CopyFromContainer(cl, container.ID, srcPath, dstPath) | ||||||
| 		} | 		} | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| @ -106,9 +106,9 @@ func parseSrcAndDst(src, dst string) (srcPath string, dstPath string, service st | |||||||
| 	return "", "", "", false, errServiceMissing | 	return "", "", "", false, errServiceMissing | ||||||
| } | } | ||||||
|  |  | ||||||
| // copyToContainer copies a file or directory from the local file system to the container. | // CopyToContainer copies a file or directory from the local file system to the container. | ||||||
| // See the possible copy modes and their documentation. | // See the possible copy modes and their documentation. | ||||||
| func copyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error { | func CopyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error { | ||||||
| 	srcStat, err := os.Stat(srcPath) | 	srcStat, err := os.Stat(srcPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("local %s ", err) | 		return fmt.Errorf("local %s ", err) | ||||||
| @ -140,7 +140,7 @@ func copyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath stri | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		if err := container.RunExec(dcli, cl, containerID, &types.ExecConfig{ | 		if _, err := container.RunExec(dcli, cl, containerID, &types.ExecConfig{ | ||||||
| 			AttachStderr: true, | 			AttachStderr: true, | ||||||
| 			AttachStdin:  true, | 			AttachStdin:  true, | ||||||
| 			AttachStdout: true, | 			AttachStdout: true, | ||||||
| @ -179,7 +179,7 @@ func copyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath stri | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		if err := container.RunExec(dcli, cl, containerID, &types.ExecConfig{ | 		if _, err := container.RunExec(dcli, cl, containerID, &types.ExecConfig{ | ||||||
| 			AttachStderr: true, | 			AttachStderr: true, | ||||||
| 			AttachStdin:  true, | 			AttachStdin:  true, | ||||||
| 			AttachStdout: true, | 			AttachStdout: true, | ||||||
| @ -194,9 +194,9 @@ func copyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath stri | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // copyFromContainer copies a file or directory from the given container to the local file system. | // CopyFromContainer copies a file or directory from the given container to the local file system. | ||||||
| // See the possible copy modes and their documentation. | // See the possible copy modes and their documentation. | ||||||
| func copyFromContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error { | func CopyFromContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error { | ||||||
| 	srcStat, err := cl.ContainerStatPath(context.Background(), containerID, srcPath) | 	srcStat, err := cl.ContainerStatPath(context.Background(), containerID, srcPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if errdefs.IsNotFound(err) { | 		if errdefs.IsNotFound(err) { | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" | 	"coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appDeployCommand = cli.Command{ | var appDeployCommand = cli.Command{ | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ import ( | |||||||
| 	"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" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appErrorsCommand = cli.Command{ | var appErrorsCommand = cli.Command{ | ||||||
|  | |||||||
| @ -13,19 +13,21 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var status bool | var status bool | ||||||
| var statusFlag = &cli.BoolFlag{ | var statusFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "status, S", | 	Name:        "status", | ||||||
|  | 	Aliases:     []string{"S"}, | ||||||
| 	Usage:       "Show app deployment status", | 	Usage:       "Show app deployment status", | ||||||
| 	Destination: &status, | 	Destination: &status, | ||||||
| } | } | ||||||
|  |  | ||||||
| var recipeFilter string | var recipeFilter string | ||||||
| var recipeFlag = &cli.StringFlag{ | var recipeFlag = &cli.StringFlag{ | ||||||
| 	Name:        "recipe, r", | 	Name:        "recipe", | ||||||
|  | 	Aliases:     []string{"r"}, | ||||||
| 	Value:       "", | 	Value:       "", | ||||||
| 	Usage:       "Show apps of a specific recipe", | 	Usage:       "Show apps of a specific recipe", | ||||||
| 	Destination: &recipeFilter, | 	Destination: &recipeFilter, | ||||||
| @ -33,7 +35,8 @@ var recipeFlag = &cli.StringFlag{ | |||||||
|  |  | ||||||
| var listAppServer string | var listAppServer string | ||||||
| var listAppServerFlag = &cli.StringFlag{ | var listAppServerFlag = &cli.StringFlag{ | ||||||
| 	Name:        "server, s", | 	Name:        "server", | ||||||
|  | 	Aliases:     []string{"s"}, | ||||||
| 	Value:       "", | 	Value:       "", | ||||||
| 	Usage:       "Show apps of a specific server", | 	Usage:       "Show apps of a specific server", | ||||||
| 	Destination: &listAppServer, | 	Destination: &listAppServer, | ||||||
|  | |||||||
							
								
								
									
										147
									
								
								cli/app/logs.go
									
									
									
									
									
								
							
							
						
						
									
										147
									
								
								cli/app/logs.go
									
									
									
									
									
								
							| @ -2,75 +2,26 @@ package app | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" |  | ||||||
| 	"io" | 	"io" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"slices" | ||||||
| 	"sync" | 	"sync" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"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/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"coopcloud.tech/abra/pkg/service" |  | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" | 	"coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"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" | ||||||
|  | 	"github.com/docker/docker/api/types/swarm" | ||||||
| 	dockerClient "github.com/docker/docker/client" | 	dockerClient "github.com/docker/docker/client" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var logOpts = types.ContainerLogsOptions{ |  | ||||||
| 	ShowStderr: true, |  | ||||||
| 	ShowStdout: true, |  | ||||||
| 	Since:      "", |  | ||||||
| 	Until:      "", |  | ||||||
| 	Timestamps: true, |  | ||||||
| 	Follow:     true, |  | ||||||
| 	Tail:       "20", |  | ||||||
| 	Details:    false, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // stackLogs lists logs for all stack services |  | ||||||
| func stackLogs(c *cli.Context, app config.App, client *dockerClient.Client) { |  | ||||||
| 	filters, err := app.Filters(true, false) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	serviceOpts := types.ServiceListOptions{Filters: filters} |  | ||||||
| 	services, err := client.ServiceList(context.Background(), serviceOpts) |  | ||||||
| 	if err != nil { |  | ||||||
| 		logrus.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var wg sync.WaitGroup |  | ||||||
| 	for _, service := range services { |  | ||||||
| 		wg.Add(1) |  | ||||||
| 		go func(s string) { |  | ||||||
| 			if internal.StdErrOnly { |  | ||||||
| 				logOpts.ShowStdout = false |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			logs, err := client.ServiceLogs(context.Background(), s, 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) |  | ||||||
| 			} |  | ||||||
| 		}(service.ID) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	wg.Wait() |  | ||||||
|  |  | ||||||
| 	os.Exit(0) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var appLogsCommand = cli.Command{ | var appLogsCommand = cli.Command{ | ||||||
| 	Name:      "logs", | 	Name:      "logs", | ||||||
| 	Aliases:   []string{"l"}, | 	Aliases:   []string{"l"}, | ||||||
| @ -105,46 +56,84 @@ var appLogsCommand = cli.Command{ | |||||||
| 			logrus.Fatalf("%s is not deployed?", app.Name) | 			logrus.Fatalf("%s is not deployed?", app.Name) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		logOpts.Since = internal.SinceLogs |  | ||||||
|  |  | ||||||
| 		serviceName := c.Args().Get(1) | 		serviceName := c.Args().Get(1) | ||||||
| 		if serviceName == "" { | 		serviceNames := []string{} | ||||||
| 			logrus.Debugf("tailing logs for all %s services", app.Recipe) | 		if serviceName != "" { | ||||||
| 			stackLogs(c, app, cl) | 			serviceNames = []string{serviceName} | ||||||
| 		} else { | 		} | ||||||
| 			logrus.Debugf("tailing logs for %s", serviceName) | 		err = tailLogs(cl, app, serviceNames) | ||||||
| 			if err := tailServiceLogs(c, cl, app, serviceName); err != nil { | 		if err != nil { | ||||||
| 				logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func tailServiceLogs(c *cli.Context, cl *dockerClient.Client, app config.App, serviceName string) error { | // tailLogs prints logs for the given app with optional service names to be | ||||||
| 	filters := filters.NewArgs() | // filtered on. It also checks if the latest task is not runnning and then | ||||||
| 	filters.Add("name", fmt.Sprintf("%s_%s", app.StackName(), serviceName)) | // prints the past tasks. | ||||||
|  | func tailLogs(cl *dockerClient.Client, app config.App, serviceNames []string) error { | ||||||
| 	chosenService, err := service.GetService(context.Background(), cl, filters, internal.NoInput) | 	f, err := app.Filters(true, false, serviceNames...) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if internal.StdErrOnly { | 	services, err := cl.ServiceList(context.Background(), types.ServiceListOptions{Filters: f}) | ||||||
| 		logOpts.ShowStdout = false |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logs, err := cl.ServiceLogs(context.Background(), chosenService.ID, logOpts) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		logrus.Fatal(err) | 		return err | ||||||
| 	} | 	} | ||||||
| 	defer logs.Close() |  | ||||||
|  |  | ||||||
| 	_, err = io.Copy(os.Stdout, logs) | 	var wg sync.WaitGroup | ||||||
| 	if err != nil && err != io.EOF { | 	for _, service := range services { | ||||||
| 		logrus.Fatal(err) | 		filters := filters.NewArgs() | ||||||
|  | 		filters.Add("name", service.Spec.Name) | ||||||
|  | 		tasks, err := cl.TaskList(context.Background(), types.TaskListOptions{Filters: f}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if len(tasks) > 0 { | ||||||
|  | 			// Need to sort the tasks by the CreatedAt field in the inverse order. | ||||||
|  | 			// Otherwise they are in the reversed order and not sorted properly. | ||||||
|  | 			slices.SortFunc[[]swarm.Task](tasks, func(t1, t2 swarm.Task) int { | ||||||
|  | 				return int(t2.Meta.CreatedAt.Unix() - t1.Meta.CreatedAt.Unix()) | ||||||
|  | 			}) | ||||||
|  | 			lastTask := tasks[0].Status | ||||||
|  | 			if lastTask.State != swarm.TaskStateRunning { | ||||||
|  | 				for _, task := range tasks { | ||||||
|  | 					logrus.Errorf("[%s] %s State %s: %s", service.Spec.Name, task.Meta.CreatedAt.Format(time.RFC3339), task.Status.State, task.Status.Err) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Collect the logs in a go routine, so the logs from all services are | ||||||
|  | 		// collected in parallel. | ||||||
|  | 		wg.Add(1) | ||||||
|  | 		go func(serviceID string) { | ||||||
|  | 			logs, err := cl.ServiceLogs(context.Background(), serviceID, types.ContainerLogsOptions{ | ||||||
|  | 				ShowStderr: true, | ||||||
|  | 				ShowStdout: !internal.StdErrOnly, | ||||||
|  | 				Since:      internal.SinceLogs, | ||||||
|  | 				Until:      "", | ||||||
|  | 				Timestamps: true, | ||||||
|  | 				Follow:     true, | ||||||
|  | 				Tail:       "20", | ||||||
|  | 				Details:    false, | ||||||
|  | 			}) | ||||||
|  | 			if err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			defer logs.Close() | ||||||
|  |  | ||||||
|  | 			_, err = io.Copy(os.Stdout, logs) | ||||||
|  | 			if err != nil && err != io.EOF { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 		}(service.ID) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// Wait for all log streams to be closed. | ||||||
|  | 	wg.Wait() | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,7 +16,7 @@ import ( | |||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	dockerClient "github.com/docker/docker/client" | 	dockerClient "github.com/docker/docker/client" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appNewDescription = ` | var appNewDescription = ` | ||||||
| @ -54,9 +54,17 @@ var appNewCommand = cli.Command{ | |||||||
| 		internal.OfflineFlag, | 		internal.OfflineFlag, | ||||||
| 		internal.ChaosFlag, | 		internal.ChaosFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Before:       internal.SubCommandBefore, | 	Before:    internal.SubCommandBefore, | ||||||
| 	ArgsUsage:    "[<recipe>]", | 	ArgsUsage: "[<recipe>] [<version>]", | ||||||
| 	BashComplete: autocomplete.RecipeNameComplete, | 	BashComplete: func(ctx *cli.Context) { | ||||||
|  | 		args := ctx.Args() | ||||||
|  | 		switch args.Len() { | ||||||
|  | 		case 0: | ||||||
|  | 			autocomplete.RecipeNameComplete(ctx) | ||||||
|  | 		case 1: | ||||||
|  | 			autocomplete.RecipeVersionComplete(ctx.Args().Get(0)) | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		recipe := internal.ValidateRecipe(c) | 		recipe := internal.ValidateRecipe(c) | ||||||
|  |  | ||||||
| @ -69,8 +77,14 @@ var appNewCommand = cli.Command{ | |||||||
| 					logrus.Fatal(err) | 					logrus.Fatal(err) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			if err := recipePkg.EnsureLatest(recipe.Name); err != nil { | 			if c.Args().Get(1) == "" { | ||||||
| 				logrus.Fatal(err) | 				if err := recipePkg.EnsureLatest(recipe.Name); err != nil { | ||||||
|  | 					logrus.Fatal(err) | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				if err := recipePkg.EnsureVersion(recipe.Name, c.Args().Get(1)); err != nil { | ||||||
|  | 					logrus.Fatal(err) | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -108,7 +122,7 @@ var appNewCommand = cli.Command{ | |||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			envSamplePath := path.Join(config.RECIPES_DIR, recipe.Name, ".env.sample") | 			envSamplePath := path.Join(config.RECIPES_DIR, recipe.Name, ".env.sample") | ||||||
| 			secretsConfig, err := secret.ReadSecretsConfig(envSamplePath, composeFiles, recipe.Name) | 			secretsConfig, err := secret.ReadSecretsConfig(envSamplePath, composeFiles, config.StackName(internal.Domain)) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| @ -168,14 +182,8 @@ var appNewCommand = cli.Command{ | |||||||
| type AppSecrets map[string]string | type AppSecrets map[string]string | ||||||
|  |  | ||||||
| // createSecrets creates all secrets for a new app. | // createSecrets creates all secrets for a new app. | ||||||
| func createSecrets(cl *dockerClient.Client, secretsConfig map[string]secret.SecretValue, sanitisedAppName string) (AppSecrets, error) { | func createSecrets(cl *dockerClient.Client, secretsConfig map[string]secret.Secret, sanitisedAppName string) (AppSecrets, error) { | ||||||
| 	// NOTE(d1): trim to match app.StackName() implementation | 	secrets, err := secret.GenerateSecrets(cl, secretsConfig, internal.NewAppServer) | ||||||
| 	if len(sanitisedAppName) > 45 { |  | ||||||
| 		logrus.Debugf("trimming %s to %s to avoid runtime limits", sanitisedAppName, sanitisedAppName[:45]) |  | ||||||
| 		sanitisedAppName = sanitisedAppName[:45] |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	secrets, err := secret.GenerateSecrets(cl, secretsConfig, sanitisedAppName, internal.NewAppServer) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -217,7 +225,7 @@ func ensureDomainFlag(recipe recipe.Recipe, server string) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // promptForSecrets asks if we should generate secrets for a new app. | // promptForSecrets asks if we should generate secrets for a new app. | ||||||
| func promptForSecrets(recipeName string, secretsConfig map[string]secret.SecretValue) error { | func promptForSecrets(recipeName string, secretsConfig map[string]secret.Secret) error { | ||||||
| 	if len(secretsConfig) == 0 { | 	if len(secretsConfig) == 0 { | ||||||
| 		logrus.Debugf("%s has no secrets to generate, skipping...", recipeName) | 		logrus.Debugf("%s has no secrets to generate, skipping...", recipeName) | ||||||
| 		return nil | 		return nil | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ import ( | |||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	dockerClient "github.com/docker/docker/client" | 	dockerClient "github.com/docker/docker/client" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appPsCommand = cli.Command{ | var appPsCommand = cli.Command{ | ||||||
|  | |||||||
| @ -3,7 +3,9 @@ package app | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"log" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
| @ -13,7 +15,7 @@ import ( | |||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/docker/docker/api/types/volume" | 	"github.com/docker/docker/api/types/volume" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appRemoveCommand = cli.Command{ | var appRemoveCommand = cli.Command{ | ||||||
| @ -124,9 +126,11 @@ flag. | |||||||
|  |  | ||||||
| 		if len(vols) > 0 { | 		if len(vols) > 0 { | ||||||
| 			for _, vol := range vols { | 			for _, vol := range vols { | ||||||
| 				err := cl.VolumeRemove(context.Background(), vol, internal.Force) // last argument is for force removing | 				err = retryFunc(5, func() error { | ||||||
|  | 					return cl.VolumeRemove(context.Background(), vol, internal.Force) // last argument is for force removing | ||||||
|  | 				}) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					logrus.Fatal(err) | 					log.Fatalf("removing volumes failed: %s", err) | ||||||
| 				} | 				} | ||||||
| 				logrus.Info(fmt.Sprintf("volume %s removed", vol)) | 				logrus.Info(fmt.Sprintf("volume %s removed", vol)) | ||||||
| 			} | 			} | ||||||
| @ -143,3 +147,21 @@ flag. | |||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // retryFunc retries the given function for the given retries. After the nth | ||||||
|  | // retry it waits (n + 1)^2 seconds before the next retry (starting with n=0). | ||||||
|  | // It returns an error if the function still failed after the last retry. | ||||||
|  | func retryFunc(retries int, fn func() error) error { | ||||||
|  | 	for i := 0; i < retries; i++ { | ||||||
|  | 		err := fn() | ||||||
|  | 		if err == nil { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		if i+1 < retries { | ||||||
|  | 			sleep := time.Duration(i+1) * time.Duration(i+1) | ||||||
|  | 			logrus.Infof("%s: waiting %d seconds before next retry", err, sleep) | ||||||
|  | 			time.Sleep(sleep * time.Second) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return fmt.Errorf("%d retries failed", retries) | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								cli/app/remove_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								cli/app/remove_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | package app | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestRetryFunc(t *testing.T) { | ||||||
|  | 	err := retryFunc(1, func() error { return nil }) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Errorf("should not return an error: %s", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	i := 0 | ||||||
|  | 	fn := func() error { | ||||||
|  | 		i++ | ||||||
|  | 		return fmt.Errorf("oh no, something went wrong!") | ||||||
|  | 	} | ||||||
|  | 	err = retryFunc(2, fn) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Error("should return an error") | ||||||
|  | 	} | ||||||
|  | 	if i != 2 { | ||||||
|  | 		t.Errorf("The function should have been called 1 times, got %d", i) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -11,7 +11,7 @@ import ( | |||||||
| 	upstream "coopcloud.tech/abra/pkg/upstream/service" | 	upstream "coopcloud.tech/abra/pkg/upstream/service" | ||||||
| 	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" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appRestartCommand = cli.Command{ | var appRestartCommand = cli.Command{ | ||||||
|  | |||||||
| @ -1,223 +1,82 @@ | |||||||
| package app | package app | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"os" |  | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"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/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" |  | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/container" |  | ||||||
| 	"github.com/docker/cli/cli/command" |  | ||||||
| 	"github.com/docker/docker/api/types" |  | ||||||
| 	"github.com/docker/docker/api/types/filters" |  | ||||||
| 	dockerClient "github.com/docker/docker/client" |  | ||||||
| 	"github.com/docker/docker/pkg/archive" |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type restoreConfig struct { | var targetPath string | ||||||
| 	preHookCmd  string | var targetPathFlag = &cli.StringFlag{ | ||||||
| 	postHookCmd string | 	Name:        "target, t", | ||||||
|  | 	Usage:       "Target path", | ||||||
|  | 	Destination: &targetPath, | ||||||
| } | } | ||||||
|  |  | ||||||
| var appRestoreCommand = cli.Command{ | var appRestoreCommand = cli.Command{ | ||||||
| 	Name:      "restore", | 	Name:      "restore", | ||||||
| 	Aliases:   []string{"rs"}, | 	Aliases:   []string{"rs"}, | ||||||
| 	Usage:     "Run app restore", | 	Usage:     "Restore an app backup", | ||||||
| 	ArgsUsage: "<domain> <service> <file>", | 	ArgsUsage: "<domain> <service>", | ||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.DebugFlag, | 		internal.DebugFlag, | ||||||
| 		internal.OfflineFlag, | 		internal.OfflineFlag, | ||||||
| 		internal.ChaosFlag, | 		targetPathFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Before:       internal.SubCommandBefore, | 	Before:       internal.SubCommandBefore, | ||||||
| 	BashComplete: autocomplete.AppNameComplete, | 	BashComplete: autocomplete.AppNameComplete, | ||||||
| 	Description: ` |  | ||||||
| Run an app restore. |  | ||||||
|  |  | ||||||
| Pre/post hook commands are defined in the recipe configuration. Abra reads this |  | ||||||
| configuration and run the comands in the context of the service before |  | ||||||
| restoring the backup. |  | ||||||
|  |  | ||||||
| Unlike "abra app backup", restore must be run on a per-service basis. You can |  | ||||||
| not restore all services in one go. Backup files produced by Abra are |  | ||||||
| compressed archives which use absolute paths. This allows Abra to restore |  | ||||||
| according to standard tar command logic, i.e. the backup will be restored to |  | ||||||
| the path it was originally backed up from. |  | ||||||
|  |  | ||||||
| Example: |  | ||||||
|  |  | ||||||
|     abra app restore example.com app ~/.abra/backups/example_com_app_609341138.tar.gz |  | ||||||
| `, |  | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		recipe, err := recipe.Get(app.Recipe, internal.Offline) | 		if err := recipe.EnsureExists(app.Recipe); err != nil { | ||||||
| 		if err != nil { |  | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if !internal.Chaos { | 		if !internal.Chaos { | ||||||
| 			if err := recipePkg.EnsureIsClean(app.Recipe); err != nil { | 			if err := recipe.EnsureIsClean(app.Recipe); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if !internal.Offline { | 			if !internal.Offline { | ||||||
| 				if err := recipePkg.EnsureUpToDate(app.Recipe); err != nil { | 				if err := recipe.EnsureUpToDate(app.Recipe); err != nil { | ||||||
| 					logrus.Fatal(err) | 					logrus.Fatal(err) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if err := recipePkg.EnsureLatest(app.Recipe); err != nil { | 			if err := recipe.EnsureLatest(app.Recipe); err != nil { | ||||||
| 				logrus.Fatal(err) | 				logrus.Fatal(err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		serviceName := c.Args().Get(1) |  | ||||||
| 		if serviceName == "" { |  | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("missing <service>?")) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		backupPath := c.Args().Get(2) |  | ||||||
| 		if backupPath == "" { |  | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("missing <file>?")) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if _, err := os.Stat(backupPath); err != nil { |  | ||||||
| 			if os.IsNotExist(err) { |  | ||||||
| 				logrus.Fatalf("%s doesn't exist?", backupPath) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		restoreConfigs := make(map[string]restoreConfig) |  | ||||||
| 		for _, service := range recipe.Config.Services { |  | ||||||
| 			if restoreEnabled, ok := service.Deploy.Labels["backupbot.restore"]; ok { |  | ||||||
| 				if restoreEnabled == "true" { |  | ||||||
| 					fullServiceName := fmt.Sprintf("%s_%s", app.StackName(), service.Name) |  | ||||||
| 					rsConfig := restoreConfig{} |  | ||||||
|  |  | ||||||
| 					logrus.Debugf("restore config detected for %s", fullServiceName) |  | ||||||
|  |  | ||||||
| 					if preHookCmd, ok := service.Deploy.Labels["backupbot.restore.pre-hook"]; ok { |  | ||||||
| 						logrus.Debugf("detected pre-hook command for %s: %s", fullServiceName, preHookCmd) |  | ||||||
| 						rsConfig.preHookCmd = preHookCmd |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					if postHookCmd, ok := service.Deploy.Labels["backupbot.restore.post-hook"]; ok { |  | ||||||
| 						logrus.Debugf("detected post-hook command for %s: %s", fullServiceName, postHookCmd) |  | ||||||
| 						rsConfig.postHookCmd = postHookCmd |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					restoreConfigs[service.Name] = rsConfig |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		rsConfig, ok := restoreConfigs[serviceName] |  | ||||||
| 		if !ok { |  | ||||||
| 			rsConfig = restoreConfig{} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		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 err := runRestore(cl, app, backupPath, serviceName, rsConfig); err != nil { | 		targetContainer, err := internal.RetrieveBackupBotContainer(cl) | ||||||
|  | 		if err != nil { | ||||||
|  | 			logrus.Fatal(err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		execEnv := []string{fmt.Sprintf("SERVICE=%s", app.Domain)} | ||||||
|  | 		if snapshot != "" { | ||||||
|  | 			logrus.Debugf("including SNAPSHOT=%s in backupbot exec invocation", snapshot) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot)) | ||||||
|  | 		} | ||||||
|  | 		if targetPath != "" { | ||||||
|  | 			logrus.Debugf("including TARGET=%s in backupbot exec invocation", targetPath) | ||||||
|  | 			execEnv = append(execEnv, fmt.Sprintf("TARGET=%s", targetPath)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err := internal.RunBackupCmdRemote(cl, "restore", targetContainer.ID, execEnv); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| // runRestore does the actual restore logic. |  | ||||||
| func runRestore(cl *dockerClient.Client, app config.App, backupPath, serviceName string, rsConfig restoreConfig) error { |  | ||||||
| 	// FIXME: avoid instantiating a new CLI |  | ||||||
| 	dcli, err := command.NewDockerCli() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	filters := filters.NewArgs() |  | ||||||
| 	filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(), serviceName)) |  | ||||||
|  |  | ||||||
| 	targetContainer, err := containerPkg.GetContainer(context.Background(), cl, filters, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	fullServiceName := fmt.Sprintf("%s_%s", app.StackName(), serviceName) |  | ||||||
| 	if rsConfig.preHookCmd != "" { |  | ||||||
| 		splitCmd := internal.SafeSplit(rsConfig.preHookCmd) |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("split pre-hook command for %s into %s", fullServiceName, splitCmd) |  | ||||||
|  |  | ||||||
| 		preHookExecOpts := types.ExecConfig{ |  | ||||||
| 			AttachStderr: true, |  | ||||||
| 			AttachStdin:  true, |  | ||||||
| 			AttachStdout: true, |  | ||||||
| 			Cmd:          splitCmd, |  | ||||||
| 			Detach:       false, |  | ||||||
| 			Tty:          true, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, targetContainer.ID, &preHookExecOpts); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Infof("succesfully ran %s pre-hook command: %s", fullServiceName, rsConfig.preHookCmd) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	backupReader, err := os.Open(backupPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	content, err := archive.DecompressStream(backupReader) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// NOTE(d1): we use absolute paths so tar knows what to do. it will restore |  | ||||||
| 	// files according to the paths set in the compressed archive |  | ||||||
| 	restorePath := "/" |  | ||||||
|  |  | ||||||
| 	copyOpts := types.CopyToContainerOptions{AllowOverwriteDirWithFile: false, CopyUIDGID: false} |  | ||||||
| 	if err := cl.CopyToContainer(context.Background(), targetContainer.ID, restorePath, content, copyOpts); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logrus.Infof("restored %s to %s", backupPath, fullServiceName) |  | ||||||
|  |  | ||||||
| 	if rsConfig.postHookCmd != "" { |  | ||||||
| 		splitCmd := internal.SafeSplit(rsConfig.postHookCmd) |  | ||||||
|  |  | ||||||
| 		logrus.Debugf("split post-hook command for %s into %s", fullServiceName, splitCmd) |  | ||||||
|  |  | ||||||
| 		postHookExecOpts := types.ExecConfig{ |  | ||||||
| 			AttachStderr: true, |  | ||||||
| 			AttachStdin:  true, |  | ||||||
| 			AttachStdout: true, |  | ||||||
| 			Cmd:          splitCmd, |  | ||||||
| 			Detach:       false, |  | ||||||
| 			Tty:          true, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, targetContainer.ID, &postHookExecOpts); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		logrus.Infof("succesfully ran %s post-hook command: %s", fullServiceName, rsConfig.postHookCmd) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appRollbackCommand = cli.Command{ | var appRollbackCommand = cli.Command{ | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ import ( | |||||||
| 	"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" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var user string | var user string | ||||||
| @ -45,11 +45,11 @@ var appRunCommand = cli.Command{ | |||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		if len(c.Args()) < 2 { | 		if c.Args().Len() < 2 { | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("no <service> provided?")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("no <service> provided?")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(c.Args()) < 3 { | 		if c.Args().Len() < 3 { | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("no <args> provided?")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("no <args> provided?")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -68,7 +68,7 @@ var appRunCommand = cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		cmd := c.Args()[2:] | 		cmd := c.Args().Slice()[2:] | ||||||
| 		execCreateOpts := types.ExecConfig{ | 		execCreateOpts := types.ExecConfig{ | ||||||
| 			AttachStderr: true, | 			AttachStderr: true, | ||||||
| 			AttachStdin:  true, | 			AttachStdin:  true, | ||||||
| @ -91,7 +91,7 @@ var appRunCommand = cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | 		if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ import ( | |||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	dockerClient "github.com/docker/docker/client" | 	dockerClient "github.com/docker/docker/client" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @ -76,7 +76,7 @@ var appSecretGenerateCommand = cli.Command{ | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(c.Args()) == 1 && !allSecrets { | 		if c.Args().Len() == 1 && !allSecrets { | ||||||
| 			err := errors.New("missing arguments <secret>/<version> or '--all'") | 			err := errors.New("missing arguments <secret>/<version> or '--all'") | ||||||
| 			internal.ShowSubcommandHelpAndError(c, err) | 			internal.ShowSubcommandHelpAndError(c, err) | ||||||
| 		} | 		} | ||||||
| @ -91,7 +91,7 @@ var appSecretGenerateCommand = cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.Recipe) | 		secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName()) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -104,7 +104,7 @@ var appSecretGenerateCommand = cli.Command{ | |||||||
| 				logrus.Fatalf("%s doesn't exist in the env config?", secretName) | 				logrus.Fatalf("%s doesn't exist in the env config?", secretName) | ||||||
| 			} | 			} | ||||||
| 			s.Version = secretVersion | 			s.Version = secretVersion | ||||||
| 			secrets = map[string]secret.SecretValue{ | 			secrets = map[string]secret.Secret{ | ||||||
| 				secretName: s, | 				secretName: s, | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @ -114,7 +114,7 @@ var appSecretGenerateCommand = cli.Command{ | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		secretVals, err := secret.GenerateSecrets(cl, secrets, app.StackName(), app.Server) | 		secretVals, err := secret.GenerateSecrets(cl, secrets, app.Server) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -175,7 +175,7 @@ Example: | |||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		app := internal.ValidateApp(c) | 		app := internal.ValidateApp(c) | ||||||
|  |  | ||||||
| 		if len(c.Args()) != 4 { | 		if c.Args().Len() != 4 { | ||||||
| 			internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments?")) | 			internal.ShowSubcommandHelpAndError(c, errors.New("missing arguments?")) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -274,7 +274,7 @@ Example: | |||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.Recipe) | 		secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName()) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
| @ -419,10 +419,10 @@ var appSecretCommand = cli.Command{ | |||||||
| 	Aliases:   []string{"s"}, | 	Aliases:   []string{"s"}, | ||||||
| 	Usage:     "Manage app secrets", | 	Usage:     "Manage app secrets", | ||||||
| 	ArgsUsage: "<domain>", | 	ArgsUsage: "<domain>", | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		appSecretGenerateCommand, | 		&appSecretGenerateCommand, | ||||||
| 		appSecretInsertCommand, | 		&appSecretInsertCommand, | ||||||
| 		appSecretRmCommand, | 		&appSecretRmCommand, | ||||||
| 		appSecretLsCommand, | 		&appSecretLsCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ import ( | |||||||
| 	stack "coopcloud.tech/abra/pkg/upstream/stack" | 	stack "coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/docker/docker/api/types" | 	"github.com/docker/docker/api/types" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appServicesCommand = cli.Command{ | var appServicesCommand = cli.Command{ | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ import ( | |||||||
| 	"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" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var prune bool | var prune bool | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ import ( | |||||||
| 	"coopcloud.tech/tagcmp" | 	"coopcloud.tech/tagcmp" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appUpgradeCommand = cli.Command{ | var appUpgradeCommand = cli.Command{ | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ import ( | |||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
| 	"github.com/olekukonko/tablewriter" | 	"github.com/olekukonko/tablewriter" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func sortServiceByName(versions [][]string) func(i, j int) bool { | func sortServiceByName(versions [][]string) func(i, j int) bool { | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" | 	"coopcloud.tech/abra/pkg/upstream/stack" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var appVolumeListCommand = cli.Command{ | var appVolumeListCommand = cli.Command{ | ||||||
| @ -150,8 +150,8 @@ var appVolumeCommand = cli.Command{ | |||||||
| 	Aliases:   []string{"vl"}, | 	Aliases:   []string{"vl"}, | ||||||
| 	Usage:     "Manage app volumes", | 	Usage:     "Manage app volumes", | ||||||
| 	ArgsUsage: "<domain>", | 	ArgsUsage: "<domain>", | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		appVolumeListCommand, | 		&appVolumeListCommand, | ||||||
| 		appVolumeRemoveCommand, | 		&appVolumeRemoveCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var catalogueGenerateCommand = cli.Command{ | var catalogueGenerateCommand = cli.Command{ | ||||||
| @ -217,7 +217,7 @@ var CatalogueCommand = cli.Command{ | |||||||
| 	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", | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		catalogueGenerateCommand, | 		&catalogueGenerateCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								cli/cli.go
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								cli/cli.go
									
									
									
									
									
								
							| @ -18,7 +18,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/web" | 	"coopcloud.tech/abra/pkg/web" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // AutoCompleteCommand helps people set up auto-complete in their shells | // AutoCompleteCommand helps people set up auto-complete in their shells | ||||||
| @ -153,13 +153,13 @@ func newAbraApp(version, commit string) *cli.App { | |||||||
|                          |_| |                          |_| | ||||||
| `, | `, | ||||||
| 		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, | ||||||
| 			server.ServerCommand, | 			&server.ServerCommand, | ||||||
| 			recipe.RecipeCommand, | 			&recipe.RecipeCommand, | ||||||
| 			catalogue.CatalogueCommand, | 			&catalogue.CatalogueCommand, | ||||||
| 			UpgradeCommand, | 			&UpgradeCommand, | ||||||
| 			AutoCompleteCommand, | 			&AutoCompleteCommand, | ||||||
| 		}, | 		}, | ||||||
| 		BashComplete: autocomplete.SubcommandComplete, | 		BashComplete: autocomplete.SubcommandComplete, | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -1,35 +1,67 @@ | |||||||
| package internal | package internal | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"strings" | 	"context" | ||||||
|  |  | ||||||
|  | 	"coopcloud.tech/abra/pkg/config" | ||||||
|  | 	containerPkg "coopcloud.tech/abra/pkg/container" | ||||||
|  | 	"coopcloud.tech/abra/pkg/service" | ||||||
|  | 	"coopcloud.tech/abra/pkg/upstream/container" | ||||||
|  | 	"github.com/docker/cli/cli/command" | ||||||
|  | 	"github.com/docker/docker/api/types" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	dockerClient "github.com/docker/docker/client" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // SafeSplit splits up a string into a list of commands safely. | // RetrieveBackupBotContainer gets the deployed backupbot container. | ||||||
| func SafeSplit(s string) []string { | func RetrieveBackupBotContainer(cl *dockerClient.Client) (types.Container, error) { | ||||||
| 	split := strings.Split(s, " ") | 	ctx := context.Background() | ||||||
|  | 	chosenService, err := service.GetServiceByLabel(ctx, cl, config.BackupbotLabel, NoInput) | ||||||
| 	var result []string | 	if err != nil { | ||||||
| 	var inquote string | 		return types.Container{}, err | ||||||
| 	var block string |  | ||||||
| 	for _, i := range split { |  | ||||||
| 		if inquote == "" { |  | ||||||
| 			if strings.HasPrefix(i, "'") || strings.HasPrefix(i, "\"") { |  | ||||||
| 				inquote = string(i[0]) |  | ||||||
| 				block = strings.TrimPrefix(i, inquote) + " " |  | ||||||
| 			} else { |  | ||||||
| 				result = append(result, i) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if !strings.HasSuffix(i, inquote) { |  | ||||||
| 				block += i + " " |  | ||||||
| 			} else { |  | ||||||
| 				block += strings.TrimSuffix(i, inquote) |  | ||||||
| 				inquote = "" |  | ||||||
| 				result = append(result, block) |  | ||||||
| 				block = "" |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return result | 	logrus.Debugf("retrieved %s as backup enabled service", chosenService.Spec.Name) | ||||||
|  |  | ||||||
|  | 	filters := filters.NewArgs() | ||||||
|  | 	filters.Add("name", chosenService.Spec.Name) | ||||||
|  | 	targetContainer, err := containerPkg.GetContainer( | ||||||
|  | 		ctx, | ||||||
|  | 		cl, | ||||||
|  | 		filters, | ||||||
|  | 		NoInput, | ||||||
|  | 	) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return types.Container{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return targetContainer, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // RunBackupCmdRemote runs a backup related command on a remote backupbot container. | ||||||
|  | func RunBackupCmdRemote(cl *dockerClient.Client, backupCmd string, containerID string, execEnv []string) error { | ||||||
|  | 	execBackupListOpts := types.ExecConfig{ | ||||||
|  | 		AttachStderr: true, | ||||||
|  | 		AttachStdin:  true, | ||||||
|  | 		AttachStdout: true, | ||||||
|  | 		Cmd:          []string{"/usr/bin/backup", "--", backupCmd}, | ||||||
|  | 		Detach:       false, | ||||||
|  | 		Env:          execEnv, | ||||||
|  | 		Tty:          true, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	logrus.Debugf("running backup %s on %s with exec config %v", backupCmd, containerID, execBackupListOpts) | ||||||
|  |  | ||||||
|  | 	// FIXME: avoid instantiating a new CLI | ||||||
|  | 	dcli, err := command.NewDockerCli() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := container.RunExec(dcli, cl, containerID, &execBackupListOpts); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,7 +5,7 @@ import ( | |||||||
|  |  | ||||||
| 	logrusStack "github.com/Gurpartap/logrus-stack" | 	logrusStack "github.com/Gurpartap/logrus-stack" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Secrets stores the variable from SecretsFlag | // Secrets stores the variable from SecretsFlag | ||||||
| @ -13,7 +13,8 @@ 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, S", | 	Name:        "secrets", | ||||||
|  | 	Aliases:     []string{"S"}, | ||||||
| 	Usage:       "Automatically generate secrets", | 	Usage:       "Automatically generate secrets", | ||||||
| 	Destination: &Secrets, | 	Destination: &Secrets, | ||||||
| } | } | ||||||
| @ -23,7 +24,8 @@ 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, p", | 	Name:        "pass", | ||||||
|  | 	Aliases:     []string{"p"}, | ||||||
| 	Usage:       "Store the generated secrets in a local pass store", | 	Usage:       "Store the generated secrets in a local pass store", | ||||||
| 	Destination: &Pass, | 	Destination: &Pass, | ||||||
| } | } | ||||||
| @ -33,7 +35,8 @@ var PassRemove bool | |||||||
|  |  | ||||||
| // PassRemoveFlag turns on/off removing generated secrets from pass | // PassRemoveFlag turns on/off removing generated secrets from pass | ||||||
| var PassRemoveFlag = &cli.BoolFlag{ | var PassRemoveFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "pass, p", | 	Name:        "pass", | ||||||
|  | 	Aliases:     []string{"p"}, | ||||||
| 	Usage:       "Remove generated secrets from a local pass store", | 	Usage:       "Remove generated secrets from a local pass store", | ||||||
| 	Destination: &PassRemove, | 	Destination: &PassRemove, | ||||||
| } | } | ||||||
| @ -43,7 +46,8 @@ var Force bool | |||||||
|  |  | ||||||
| // ForceFlag turns on/off force functionality. | // ForceFlag turns on/off force functionality. | ||||||
| var ForceFlag = &cli.BoolFlag{ | var ForceFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "force, f", | 	Name:        "force", | ||||||
|  | 	Aliases:     []string{"f"}, | ||||||
| 	Usage:       "Perform action without further prompt. Use with care!", | 	Usage:       "Perform action without further prompt. Use with care!", | ||||||
| 	Destination: &Force, | 	Destination: &Force, | ||||||
| } | } | ||||||
| @ -53,7 +57,8 @@ var Chaos bool | |||||||
|  |  | ||||||
| // ChaosFlag turns on/off chaos functionality. | // ChaosFlag turns on/off chaos functionality. | ||||||
| var ChaosFlag = &cli.BoolFlag{ | var ChaosFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "chaos, C", | 	Name:        "chaos", | ||||||
|  | 	Aliases:     []string{"C"}, | ||||||
| 	Usage:       "Proceed with uncommitted recipes changes. Use with care!", | 	Usage:       "Proceed with uncommitted recipes changes. Use with care!", | ||||||
| 	Destination: &Chaos, | 	Destination: &Chaos, | ||||||
| } | } | ||||||
| @ -63,24 +68,29 @@ var Tty bool | |||||||
|  |  | ||||||
| // TtyFlag turns on/off tty mode. | // TtyFlag turns on/off tty mode. | ||||||
| var TtyFlag = &cli.BoolFlag{ | var TtyFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "tty, T", | 	Name:        "tty", | ||||||
|  | 	Aliases:     []string{"T"}, | ||||||
| 	Usage:       "Disables TTY mode to run this command from a script.", | 	Usage:       "Disables TTY mode to run this command from a script.", | ||||||
| 	Destination: &Tty, | 	Destination: &Tty, | ||||||
| } | } | ||||||
|  |  | ||||||
| var NoInput bool | var ( | ||||||
| var NoInputFlag = &cli.BoolFlag{ | 	NoInput     bool | ||||||
| 	Name:        "no-input, n", | 	NoInputFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Toggle non-interactive mode", | 		Name:        "no-input", | ||||||
| 	Destination: &NoInput, | 		Aliases:     []string{"n"}, | ||||||
| } | 		Usage:       "Toggle non-interactive mode", | ||||||
|  | 		Destination: &NoInput, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| // Debug stores the variable from DebugFlag. | // Debug stores the variable from DebugFlag. | ||||||
| var Debug bool | var Debug bool | ||||||
|  |  | ||||||
| // DebugFlag turns on/off verbose logging down to the DEBUG level. | // DebugFlag turns on/off verbose logging down to the DEBUG level. | ||||||
| var DebugFlag = &cli.BoolFlag{ | var DebugFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "debug, d", | 	Name:        "debug", | ||||||
|  | 	Aliases:     []string{"d"}, | ||||||
| 	Destination: &Debug, | 	Destination: &Debug, | ||||||
| 	Usage:       "Show DEBUG messages", | 	Usage:       "Show DEBUG messages", | ||||||
| } | } | ||||||
| @ -90,7 +100,8 @@ var Offline bool | |||||||
|  |  | ||||||
| // DebugFlag turns on/off offline mode. | // DebugFlag turns on/off offline mode. | ||||||
| var OfflineFlag = &cli.BoolFlag{ | var OfflineFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "offline, o", | 	Name:        "offline", | ||||||
|  | 	Aliases:     []string{"o"}, | ||||||
| 	Destination: &Offline, | 	Destination: &Offline, | ||||||
| 	Usage:       "Prefer offline & filesystem access when possible", | 	Usage:       "Prefer offline & filesystem access when possible", | ||||||
| } | } | ||||||
| @ -100,7 +111,8 @@ var MachineReadable bool | |||||||
|  |  | ||||||
| // MachineReadableFlag turns on/off machine readable output where supported | // MachineReadableFlag turns on/off machine readable output where supported | ||||||
| var MachineReadableFlag = &cli.BoolFlag{ | var MachineReadableFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "machine, m", | 	Name:        "machine", | ||||||
|  | 	Aliases:     []string{"m"}, | ||||||
| 	Destination: &MachineReadable, | 	Destination: &MachineReadable, | ||||||
| 	Usage:       "Output in a machine-readable format (where supported)", | 	Usage:       "Output in a machine-readable format (where supported)", | ||||||
| } | } | ||||||
| @ -110,133 +122,185 @@ var RC bool | |||||||
|  |  | ||||||
| // RCFlag chooses the latest release candidate for install | // RCFlag chooses the latest release candidate for install | ||||||
| var RCFlag = &cli.BoolFlag{ | var RCFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "rc, r", | 	Name:        "rc", | ||||||
|  | 	Aliases:     []string{"c"}, | ||||||
| 	Destination: &RC, | 	Destination: &RC, | ||||||
| 	Usage:       "Install the latest release candidate", | 	Usage:       "Install the latest release candidate", | ||||||
| } | } | ||||||
|  |  | ||||||
| var Major bool | var ( | ||||||
| var MajorFlag = &cli.BoolFlag{ | 	Major     bool | ||||||
| 	Name:        "major, x", | 	MajorFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Increase the major part of the version", | 		Name:        "major", | ||||||
| 	Destination: &Major, | 		Aliases:     []string{"x"}, | ||||||
| } | 		Usage:       "Increase the major part of the version", | ||||||
|  | 		Destination: &Major, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Minor bool | var ( | ||||||
| var MinorFlag = &cli.BoolFlag{ | 	Minor     bool | ||||||
| 	Name:        "minor, y", | 	MinorFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Increase the minor part of the version", | 		Name:        "minor", | ||||||
| 	Destination: &Minor, | 		Aliases:     []string{"y"}, | ||||||
| } | 		Usage:       "Increase the minor part of the version", | ||||||
|  | 		Destination: &Minor, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Patch bool | var ( | ||||||
| var PatchFlag = &cli.BoolFlag{ | 	Patch     bool | ||||||
| 	Name:        "patch, z", | 	PatchFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Increase the patch part of the version", | 		Name:        "patch", | ||||||
| 	Destination: &Patch, | 		Aliases:     []string{"z"}, | ||||||
| } | 		Usage:       "Increase the patch part of the version", | ||||||
|  | 		Destination: &Patch, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Dry bool | var ( | ||||||
| var DryFlag = &cli.BoolFlag{ | 	Dry     bool | ||||||
| 	Name:        "dry-run, r", | 	DryFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Only reports changes that would be made", | 		Name:        "dry-run", | ||||||
| 	Destination: &Dry, | 		Aliases:     []string{"r"}, | ||||||
| } | 		Usage:       "Only reports changes that would be made", | ||||||
|  | 		Destination: &Dry, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Publish bool | var ( | ||||||
| var PublishFlag = &cli.BoolFlag{ | 	Publish     bool | ||||||
| 	Name:        "publish, p", | 	PublishFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Publish changes to git.coopcloud.tech", | 		Name:        "publish", | ||||||
| 	Destination: &Publish, | 		Aliases:     []string{"p"}, | ||||||
| } | 		Usage:       "Publish changes to git.coopcloud.tech", | ||||||
|  | 		Destination: &Publish, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Domain string | var ( | ||||||
| var DomainFlag = &cli.StringFlag{ | 	Domain     string | ||||||
| 	Name:        "domain, D", | 	DomainFlag = &cli.StringFlag{ | ||||||
| 	Value:       "", | 		Name:        "domain", | ||||||
| 	Usage:       "Choose a domain name", | 		Aliases:     []string{"D"}, | ||||||
| 	Destination: &Domain, | 		Value:       "", | ||||||
| } | 		Usage:       "Choose a domain name", | ||||||
|  | 		Destination: &Domain, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var NewAppServer string | var ( | ||||||
| var NewAppServerFlag = &cli.StringFlag{ | 	NewAppServer     string | ||||||
| 	Name:        "server, s", | 	NewAppServerFlag = &cli.StringFlag{ | ||||||
| 	Value:       "", | 		Name:        "server", | ||||||
| 	Usage:       "Show apps of a specific server", | 		Aliases:     []string{"s"}, | ||||||
| 	Destination: &NewAppServer, | 		Value:       "", | ||||||
| } | 		Usage:       "Show apps of a specific server", | ||||||
|  | 		Destination: &NewAppServer, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var NoDomainChecks bool | var ( | ||||||
| var NoDomainChecksFlag = &cli.BoolFlag{ | 	NoDomainChecks     bool | ||||||
| 	Name:        "no-domain-checks, D", | 	NoDomainChecksFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Disable app domain sanity checks", | 		Name:        "no-domain-checks", | ||||||
| 	Destination: &NoDomainChecks, | 		Aliases:     []string{"D"}, | ||||||
| } | 		Usage:       "Disable app domain sanity checks", | ||||||
|  | 		Destination: &NoDomainChecks, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var StdErrOnly bool | var ( | ||||||
| var StdErrOnlyFlag = &cli.BoolFlag{ | 	StdErrOnly     bool | ||||||
| 	Name:        "stderr, s", | 	StdErrOnlyFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Only tail stderr", | 		Name:        "stderr", | ||||||
| 	Destination: &StdErrOnly, | 		Aliases:     []string{"s"}, | ||||||
| } | 		Usage:       "Only tail stderr", | ||||||
|  | 		Destination: &StdErrOnly, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var SinceLogs string | var ( | ||||||
| var SinceLogsFlag = &cli.StringFlag{ | 	SinceLogs     string | ||||||
| 	Name:        "since, S", | 	SinceLogsFlag = &cli.StringFlag{ | ||||||
| 	Value:       "", | 		Name:        "since", | ||||||
| 	Usage:       "tail logs since YYYY-MM-DDTHH:MM:SSZ", | 		Aliases:     []string{"S"}, | ||||||
| 	Destination: &SinceLogs, | 		Value:       "", | ||||||
| } | 		Usage:       "tail logs since YYYY-MM-DDTHH:MM:SSZ", | ||||||
|  | 		Destination: &SinceLogs, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var DontWaitConverge bool | var ( | ||||||
| var DontWaitConvergeFlag = &cli.BoolFlag{ | 	DontWaitConverge     bool | ||||||
| 	Name:        "no-converge-checks, c", | 	DontWaitConvergeFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Don't wait for converge logic checks", | 		Name:        "no-converge-checks", | ||||||
| 	Destination: &DontWaitConverge, | 		Aliases:     []string{"c"}, | ||||||
| } | 		Usage:       "Don't wait for converge logic checks", | ||||||
|  | 		Destination: &DontWaitConverge, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var Watch bool | var ( | ||||||
| var WatchFlag = &cli.BoolFlag{ | 	Watch     bool | ||||||
| 	Name:        "watch, w", | 	WatchFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Watch status by polling repeatedly", | 		Name:        "watch", | ||||||
| 	Destination: &Watch, | 		Aliases:     []string{"w"}, | ||||||
| } | 		Usage:       "Watch status by polling repeatedly", | ||||||
|  | 		Destination: &Watch, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var OnlyErrors bool | var ( | ||||||
| var OnlyErrorFlag = &cli.BoolFlag{ | 	OnlyErrors    bool | ||||||
| 	Name:        "errors, e", | 	OnlyErrorFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Only show errors", | 		Name:        "errors", | ||||||
| 	Destination: &OnlyErrors, | 		Aliases:     []string{"e"}, | ||||||
| } | 		Usage:       "Only show errors", | ||||||
|  | 		Destination: &OnlyErrors, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var SkipUpdates bool | var ( | ||||||
| var SkipUpdatesFlag = &cli.BoolFlag{ | 	SkipUpdates     bool | ||||||
| 	Name:        "skip-updates, s", | 	SkipUpdatesFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Skip updating recipe repositories", | 		Name:        "skip-updates", | ||||||
| 	Destination: &SkipUpdates, | 		Aliases:     []string{"s"}, | ||||||
| } | 		Usage:       "Skip updating recipe repositories", | ||||||
|  | 		Destination: &SkipUpdates, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var AllTags bool | var ( | ||||||
| var AllTagsFlag = &cli.BoolFlag{ | 	AllTags     bool | ||||||
| 	Name:        "all-tags, a", | 	AllTagsFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "List all tags, not just upgrades", | 		Name:        "all-tags", | ||||||
| 	Destination: &AllTags, | 		Aliases:     []string{"a"}, | ||||||
| } | 		Usage:       "List all tags, not just upgrades", | ||||||
|  | 		Destination: &AllTags, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var LocalCmd bool | var ( | ||||||
| var LocalCmdFlag = &cli.BoolFlag{ | 	LocalCmd     bool | ||||||
| 	Name:        "local, l", | 	LocalCmdFlag = &cli.BoolFlag{ | ||||||
| 	Usage:       "Run command locally", | 		Name:        "local", | ||||||
| 	Destination: &LocalCmd, | 		Aliases:     []string{"l"}, | ||||||
| } | 		Usage:       "Run command locally", | ||||||
|  | 		Destination: &LocalCmd, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| var RemoteUser string | var ( | ||||||
| var RemoteUserFlag = &cli.StringFlag{ | 	RemoteUser     string | ||||||
| 	Name:        "user, u", | 	RemoteUserFlag = &cli.StringFlag{ | ||||||
| 	Value:       "", | 		Name:        "user", | ||||||
| 	Usage:       "User to run command within a service context", | 		Aliases:     []string{"u"}, | ||||||
| 	Destination: &RemoteUser, | 		Value:       "", | ||||||
| } | 		Usage:       "User to run command within a service context", | ||||||
|  | 		Destination: &RemoteUser, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| // SubCommandBefore wires up pre-action machinery (e.g. --debug handling). | // SubCommandBefore wires up pre-action machinery (e.g. --debug handling). | ||||||
| func SubCommandBefore(c *cli.Context) error { | func SubCommandBefore(c *cli.Context) error { | ||||||
|  | |||||||
| @ -60,7 +60,7 @@ func RunCmdRemote(cl *dockerClient.Client, app config.App, abraSh, serviceName, | |||||||
| 		Tty:          false, | 		Tty:          false, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | 	if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | ||||||
| 		logrus.Infof("%s does not exist for %s, use /bin/sh as fallback", shell, app.Name) | 		logrus.Infof("%s does not exist for %s, use /bin/sh as fallback", shell, app.Name) | ||||||
| 		shell = "/bin/sh" | 		shell = "/bin/sh" | ||||||
| 	} | 	} | ||||||
| @ -85,7 +85,7 @@ func RunCmdRemote(cl *dockerClient.Client, app config.App, abraSh, serviceName, | |||||||
| 		execCreateOpts.Tty = false | 		execCreateOpts.Tty = false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | 	if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
|  |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // ShowSubcommandHelpAndError exits the program on error, logs the error to the | // ShowSubcommandHelpAndError exits the program on error, logs the error to the | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // ValidateRecipe ensures the recipe arg is valid. | // ValidateRecipe ensures the recipe arg is valid. | ||||||
| @ -120,9 +120,9 @@ func ValidateDomain(c *cli.Context) string { | |||||||
|  |  | ||||||
| // ValidateSubCmdFlags ensures flag order conforms to correct order | // ValidateSubCmdFlags ensures flag order conforms to correct order | ||||||
| func ValidateSubCmdFlags(c *cli.Context) bool { | func ValidateSubCmdFlags(c *cli.Context) bool { | ||||||
| 	for argIdx, arg := range c.Args() { | 	for argIdx, arg := range c.Args().Slice() { | ||||||
| 		if !strings.HasPrefix(arg, "--") { | 		if !strings.HasPrefix(arg, "--") { | ||||||
| 			for _, flag := range c.Args()[argIdx:] { | 			for _, flag := range c.Args().Slice()[argIdx:] { | ||||||
| 				if strings.HasPrefix(flag, "--") { | 				if strings.HasPrefix(flag, "--") { | ||||||
| 					return false | 					return false | ||||||
| 				} | 				} | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	gitPkg "coopcloud.tech/abra/pkg/git" | 	gitPkg "coopcloud.tech/abra/pkg/git" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeDiffCommand = cli.Command{ | var recipeDiffCommand = cli.Command{ | ||||||
|  | |||||||
| @ -3,9 +3,10 @@ package recipe | |||||||
| import ( | import ( | ||||||
| 	"coopcloud.tech/abra/cli/internal" | 	"coopcloud.tech/abra/cli/internal" | ||||||
| 	"coopcloud.tech/abra/pkg/autocomplete" | 	"coopcloud.tech/abra/pkg/autocomplete" | ||||||
|  | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeFetchCommand = cli.Command{ | var recipeFetchCommand = cli.Command{ | ||||||
| @ -17,26 +18,31 @@ var recipeFetchCommand = cli.Command{ | |||||||
| 	Flags: []cli.Flag{ | 	Flags: []cli.Flag{ | ||||||
| 		internal.DebugFlag, | 		internal.DebugFlag, | ||||||
| 		internal.NoInputFlag, | 		internal.NoInputFlag, | ||||||
|  | 		internal.OfflineFlag, | ||||||
| 	}, | 	}, | ||||||
| 	Before:       internal.SubCommandBefore, | 	Before:       internal.SubCommandBefore, | ||||||
| 	BashComplete: autocomplete.RecipeNameComplete, | 	BashComplete: autocomplete.RecipeNameComplete, | ||||||
| 	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.ValidateRecipe(c) | 			internal.ValidateRecipe(c) | ||||||
|  | 			if err := recipe.Ensure(recipeName); err != nil { | ||||||
|  | 				logrus.Fatal(err) | ||||||
|  | 			} | ||||||
|  | 			return nil | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := recipe.EnsureExists(recipeName); err != nil { | 		catalogue, err := recipe.ReadRecipeCatalogue(internal.Offline) | ||||||
|  | 		if err != nil { | ||||||
| 			logrus.Fatal(err) | 			logrus.Fatal(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := recipe.EnsureUpToDate(recipeName); err != nil { | 		catlBar := formatter.CreateProgressbar(len(catalogue), "fetching latest recipes...") | ||||||
| 			logrus.Fatal(err) | 		for recipeName := range catalogue { | ||||||
| 		} | 			if err := recipe.Ensure(recipeName); err != nil { | ||||||
|  | 				logrus.Error(err) | ||||||
| 		if err := recipe.EnsureLatest(recipeName); err != nil { | 			} | ||||||
| 			logrus.Fatal(err) | 			catlBar.Add(1) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return nil | 		return nil | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/lint" | 	"coopcloud.tech/abra/pkg/lint" | ||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeLintCommand = cli.Command{ | var recipeLintCommand = cli.Command{ | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/formatter" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var pattern string | var pattern string | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/git" | 	"coopcloud.tech/abra/pkg/git" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // recipeMetadata is the recipe metadata for the README.md | // recipeMetadata is the recipe metadata for the README.md | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| package recipe | package recipe | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // RecipeCommand defines all recipe related sub-commands. | // RecipeCommand defines all recipe related sub-commands. | ||||||
| @ -21,16 +21,16 @@ 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 | manner. Abra supports convenient automation for recipe maintainenace, see the | ||||||
| "abra recipe upgrade", "abra recipe sync" and "abra recipe release" commands. | "abra recipe upgrade", "abra recipe sync" and "abra recipe release" commands. | ||||||
| `, | `, | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		recipeFetchCommand, | 		&recipeFetchCommand, | ||||||
| 		recipeLintCommand, | 		&recipeLintCommand, | ||||||
| 		recipeListCommand, | 		&recipeListCommand, | ||||||
| 		recipeNewCommand, | 		&recipeNewCommand, | ||||||
| 		recipeReleaseCommand, | 		&recipeReleaseCommand, | ||||||
| 		recipeSyncCommand, | 		&recipeSyncCommand, | ||||||
| 		recipeUpgradeCommand, | 		&recipeUpgradeCommand, | ||||||
| 		recipeVersionCommand, | 		&recipeVersionCommand, | ||||||
| 		recipeResetCommand, | 		&recipeResetCommand, | ||||||
| 		recipeDiffCommand, | 		&recipeDiffCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,7 +1,9 @@ | |||||||
| package recipe | package recipe | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"os" | ||||||
| 	"path" | 	"path" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| @ -18,7 +20,7 @@ import ( | |||||||
| 	"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/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeReleaseCommand = cli.Command{ | var recipeReleaseCommand = cli.Command{ | ||||||
| @ -140,7 +142,7 @@ your SSH keys configured on your account. | |||||||
|  |  | ||||||
| // getImageVersions retrieves image versions for a recipe | // getImageVersions retrieves image versions for a recipe | ||||||
| func getImageVersions(recipe recipe.Recipe) (map[string]string, error) { | func getImageVersions(recipe recipe.Recipe) (map[string]string, error) { | ||||||
| 	var services = make(map[string]string) | 	services := make(map[string]string) | ||||||
|  |  | ||||||
| 	missingTag := false | 	missingTag := false | ||||||
| 	for _, service := range recipe.Config.Services { | 	for _, service := range recipe.Config.Services { | ||||||
| @ -207,6 +209,10 @@ func createReleaseFromTag(recipe recipe.Recipe, tagString, mainAppVersion string | |||||||
| 		tagString = fmt.Sprintf("%s+%s", tag.String(), mainAppVersion) | 		tagString = fmt.Sprintf("%s+%s", tag.String(), mainAppVersion) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if err := addReleaseNotes(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if err := commitRelease(recipe, tagString); err != nil { | 	if err := commitRelease(recipe, tagString); err != nil { | ||||||
| 		logrus.Fatal(err) | 		logrus.Fatal(err) | ||||||
| 	} | 	} | ||||||
| @ -237,6 +243,82 @@ func getTagCreateOptions(tag string) (git.CreateTagOptions, error) { | |||||||
| 	return git.CreateTagOptions{Message: msg}, nil | 	return git.CreateTagOptions{Message: msg}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // addReleaseNotes checks if the release/next release note exists and moves the | ||||||
|  | // file to release/<tag>. | ||||||
|  | func addReleaseNotes(recipe recipe.Recipe, tag string) error { | ||||||
|  | 	repoPath := path.Join(config.RECIPES_DIR, recipe.Name) | ||||||
|  | 	tagReleaseNotePath := path.Join(repoPath, "release", tag) | ||||||
|  | 	if _, err := os.Stat(tagReleaseNotePath); err == nil { | ||||||
|  | 		// Release note for current tag already exist exists. | ||||||
|  | 		return nil | ||||||
|  | 	} else if !errors.Is(err, os.ErrNotExist) { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	nextReleaseNotePath := path.Join(repoPath, "release", "next") | ||||||
|  | 	if _, err := os.Stat(nextReleaseNotePath); err == nil { | ||||||
|  | 		// release/next note exists. Move it to release/<tag> | ||||||
|  | 		if internal.Dry { | ||||||
|  | 			logrus.Debugf("dry run: move release note from 'next' to %s", tag) | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		if !internal.NoInput { | ||||||
|  | 			prompt := &survey.Input{ | ||||||
|  | 				Message: "Use release note in release/next?", | ||||||
|  | 			} | ||||||
|  | 			var addReleaseNote bool | ||||||
|  | 			if err := survey.AskOne(prompt, &addReleaseNote); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if !addReleaseNote { | ||||||
|  | 				return nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		err := os.Rename(nextReleaseNotePath, tagReleaseNotePath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		err = gitPkg.Add(repoPath, path.Join("release", "next"), internal.Dry) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		err = gitPkg.Add(repoPath, path.Join("release", tag), internal.Dry) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} else if !errors.Is(err, os.ErrNotExist) { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// No release note exists for the current release. | ||||||
|  | 	if internal.NoInput { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	prompt := &survey.Input{ | ||||||
|  | 		Message: "Release Note (leave empty for no release note)", | ||||||
|  | 	} | ||||||
|  | 	var releaseNote string | ||||||
|  | 	if err := survey.AskOne(prompt, &releaseNote); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if releaseNote == "" { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err := os.WriteFile(tagReleaseNotePath, []byte(releaseNote), 0o644) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	err = gitPkg.Add(repoPath, path.Join("release", tag), internal.Dry) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| func commitRelease(recipe recipe.Recipe, tag string) error { | func commitRelease(recipe recipe.Recipe, tag string) error { | ||||||
| 	if internal.Dry { | 	if internal.Dry { | ||||||
| 		logrus.Debugf("dry run: no changes committed") | 		logrus.Debugf("dry run: no changes committed") | ||||||
| @ -404,6 +486,10 @@ func createReleaseFromPreviousTag(tagString, mainAppVersion string, recipe recip | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if err := addReleaseNotes(recipe, tagString); err != nil { | ||||||
|  | 		logrus.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if err := commitRelease(recipe, tagString); err != nil { | 	if err := commitRelease(recipe, tagString); err != nil { | ||||||
| 		logrus.Fatalf("failed to commit changes: %s", err.Error()) | 		logrus.Fatalf("failed to commit changes: %s", err.Error()) | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeResetCommand = cli.Command{ | var recipeResetCommand = cli.Command{ | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ import ( | |||||||
| 	"github.com/go-git/go-git/v5" | 	"github.com/go-git/go-git/v5" | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" | 	"github.com/go-git/go-git/v5/plumbing" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var recipeSyncCommand = cli.Command{ | var recipeSyncCommand = cli.Command{ | ||||||
|  | |||||||
| @ -20,7 +20,7 @@ import ( | |||||||
| 	"github.com/AlecAivazis/survey/v2" | 	"github.com/AlecAivazis/survey/v2" | ||||||
| 	"github.com/docker/distribution/reference" | 	"github.com/docker/distribution/reference" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type imgPin struct { | type imgPin struct { | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ import ( | |||||||
| 	recipePkg "coopcloud.tech/abra/pkg/recipe" | 	recipePkg "coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/olekukonko/tablewriter" | 	"github.com/olekukonko/tablewriter" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func sortServiceByName(versions [][]string) func(i, j int) bool { | func sortServiceByName(versions [][]string) func(i, j int) bool { | ||||||
|  | |||||||
| @ -13,12 +13,13 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/server" | 	"coopcloud.tech/abra/pkg/server" | ||||||
| 	sshPkg "coopcloud.tech/abra/pkg/ssh" | 	sshPkg "coopcloud.tech/abra/pkg/ssh" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var local bool | var local bool | ||||||
| var localFlag = &cli.BoolFlag{ | var localFlag = &cli.BoolFlag{ | ||||||
| 	Name:        "local, l", | 	Name:        "local", | ||||||
|  | 	Aliases:     []string{"l"}, | ||||||
| 	Usage:       "Use local server", | 	Usage:       "Use local server", | ||||||
| 	Destination: &local, | 	Destination: &local, | ||||||
| } | } | ||||||
| @ -122,7 +123,7 @@ developer machine. | |||||||
| 	Before:    internal.SubCommandBefore, | 	Before:    internal.SubCommandBefore, | ||||||
| 	ArgsUsage: "<domain>", | 	ArgsUsage: "<domain>", | ||||||
| 	Action: func(c *cli.Context) error { | 	Action: func(c *cli.Context) error { | ||||||
| 		if len(c.Args()) > 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) | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/formatter" | 	"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" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var problemsFilter bool | var problemsFilter bool | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/formatter" | 	"coopcloud.tech/abra/pkg/formatter" | ||||||
| 	"github.com/docker/docker/api/types/filters" | 	"github.com/docker/docker/api/types/filters" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var allFilter bool | var allFilter bool | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/client" | 	"coopcloud.tech/abra/pkg/client" | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var serverRemoveCommand = cli.Command{ | var serverRemoveCommand = cli.Command{ | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| package server | package server | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // ServerCommand defines the `abra server` command and its subcommands | // ServerCommand defines the `abra server` command and its subcommands | ||||||
| @ -9,10 +9,10 @@ var ServerCommand = cli.Command{ | |||||||
| 	Name:    "server", | 	Name:    "server", | ||||||
| 	Aliases: []string{"s"}, | 	Aliases: []string{"s"}, | ||||||
| 	Usage:   "Manage servers", | 	Usage:   "Manage servers", | ||||||
| 	Subcommands: []cli.Command{ | 	Subcommands: []*cli.Command{ | ||||||
| 		serverAddCommand, | 		&serverAddCommand, | ||||||
| 		serverListCommand, | 		&serverListCommand, | ||||||
| 		serverRemoveCommand, | 		&serverRemoveCommand, | ||||||
| 		serverPruneCommand, | 		&serverPruneCommand, | ||||||
| 	}, | 	}, | ||||||
| } | } | ||||||
|  | |||||||
| @ -21,24 +21,28 @@ import ( | |||||||
| 	dockerclient "github.com/docker/docker/client" | 	dockerclient "github.com/docker/docker/client" | ||||||
|  |  | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const SERVER = "localhost" | const SERVER = "localhost" | ||||||
|  |  | ||||||
| var majorUpdate bool | var ( | ||||||
| var majorFlag = &cli.BoolFlag{ | 	majorUpdate bool | ||||||
| 	Name:        "major, m", | 	majorFlag   = &cli.BoolFlag{ | ||||||
| 	Usage:       "Also check for major updates", | 		Name:        "major", | ||||||
| 	Destination: &majorUpdate, | 		Aliases:     []string{"m"}, | ||||||
| } | 		Usage:       "Also check for major updates", | ||||||
|  | 		Destination: &majorUpdate, | ||||||
|  | 	} | ||||||
|  |  | ||||||
| var updateAll bool | 	updateAll bool | ||||||
| var allFlag = &cli.BoolFlag{ | 	allFlag   = &cli.BoolFlag{ | ||||||
| 	Name:        "all, a", | 		Name:        "all", | ||||||
| 	Usage:       "Update all deployed apps", | 		Aliases:     []string{"a"}, | ||||||
| 	Destination: &updateAll, | 		Usage:       "Update all deployed apps", | ||||||
| } | 		Destination: &updateAll, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
| // Notify checks for available upgrades | // Notify checks for available upgrades | ||||||
| var Notify = cli.Command{ | var Notify = cli.Command{ | ||||||
| @ -271,7 +275,8 @@ func getDeployedVersion(cl *dockerclient.Client, stackName string, recipeName st | |||||||
| // than the deployed version. It only includes major upgrades if the "--major" | // than the deployed version. It only includes major upgrades if the "--major" | ||||||
| // flag is set. | // flag is set. | ||||||
| func getAvailableUpgrades(cl *dockerclient.Client, stackName string, recipeName string, | func getAvailableUpgrades(cl *dockerclient.Client, stackName string, recipeName string, | ||||||
| 	deployedVersion string) ([]string, error) { | 	deployedVersion string, | ||||||
|  | ) ([]string, error) { | ||||||
| 	catl, err := recipe.ReadRecipeCatalogue(internal.Offline) | 	catl, err := recipe.ReadRecipeCatalogue(internal.Offline) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -429,7 +434,8 @@ func tryUpgrade(cl *dockerclient.Client, stackName, recipeName string) error { | |||||||
|  |  | ||||||
| // upgrade performs all necessary steps to upgrade an app. | // upgrade performs all necessary steps to upgrade an app. | ||||||
| func upgrade(cl *dockerclient.Client, stackName, recipeName, | func upgrade(cl *dockerclient.Client, stackName, recipeName, | ||||||
| 	upgradeVersion string) error { | 	upgradeVersion string, | ||||||
|  | ) error { | ||||||
| 	env, err := getEnv(cl, stackName) | 	env, err := getEnv(cl, stackName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -474,9 +480,9 @@ func newAbraApp(version, commit string) *cli.App { | |||||||
|                          |_| |                          |_| | ||||||
| `, | `, | ||||||
| 		Version: fmt.Sprintf("%s-%s", version, commit[:7]), | 		Version: fmt.Sprintf("%s-%s", version, commit[:7]), | ||||||
| 		Commands: []cli.Command{ | 		Commands: []*cli.Command{ | ||||||
| 			Notify, | 			&Notify, | ||||||
| 			UpgradeApp, | 			&UpgradeApp, | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								go.mod
									
									
									
									
									
								
							| @ -4,7 +4,7 @@ go 1.21 | |||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	coopcloud.tech/tagcmp v0.0.0-20211103052201-885b22f77d52 | 	coopcloud.tech/tagcmp v0.0.0-20211103052201-885b22f77d52 | ||||||
| 	git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100106-7462d91acefd | 	git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100509-01bff8284355 | ||||||
| 	github.com/AlecAivazis/survey/v2 v2.3.7 | 	github.com/AlecAivazis/survey/v2 v2.3.7 | ||||||
| 	github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 | 	github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 | ||||||
| 	github.com/docker/cli v24.0.7+incompatible | 	github.com/docker/cli v24.0.7+incompatible | ||||||
| @ -12,6 +12,7 @@ require ( | |||||||
| 	github.com/docker/docker v24.0.7+incompatible | 	github.com/docker/docker v24.0.7+incompatible | ||||||
| 	github.com/docker/go-units v0.5.0 | 	github.com/docker/go-units v0.5.0 | ||||||
| 	github.com/go-git/go-git/v5 v5.10.0 | 	github.com/go-git/go-git/v5 v5.10.0 | ||||||
|  | 	github.com/google/go-cmp v0.5.9 | ||||||
| 	github.com/moby/sys/signal v0.7.0 | 	github.com/moby/sys/signal v0.7.0 | ||||||
| 	github.com/moby/term v0.5.0 | 	github.com/moby/term v0.5.0 | ||||||
| 	github.com/olekukonko/tablewriter v0.0.5 | 	github.com/olekukonko/tablewriter v0.0.5 | ||||||
| @ -24,7 +25,7 @@ require ( | |||||||
| require ( | require ( | ||||||
| 	dario.cat/mergo v1.0.0 // indirect | 	dario.cat/mergo v1.0.0 // indirect | ||||||
| 	github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 // indirect | 	github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 // indirect | ||||||
| 	github.com/BurntSushi/toml v1.0.0 // indirect | 	github.com/BurntSushi/toml v1.3.2 // indirect | ||||||
| 	github.com/Microsoft/go-winio v0.6.1 // indirect | 	github.com/Microsoft/go-winio v0.6.1 // indirect | ||||||
| 	github.com/Microsoft/hcsshim v0.9.2 // indirect | 	github.com/Microsoft/hcsshim v0.9.2 // indirect | ||||||
| 	github.com/ProtonMail/go-crypto v0.0.0-20230828082145-3c4c8a2d2371 // indirect | 	github.com/ProtonMail/go-crypto v0.0.0-20230828082145-3c4c8a2d2371 // indirect | ||||||
| @ -32,7 +33,7 @@ require ( | |||||||
| 	github.com/beorn7/perks v1.0.1 // indirect | 	github.com/beorn7/perks v1.0.1 // indirect | ||||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||||
| 	github.com/cloudflare/circl v1.3.3 // indirect | 	github.com/cloudflare/circl v1.3.3 // indirect | ||||||
| 	github.com/cpuguy83/go-md2man/v2 v2.0.1 // indirect | 	github.com/cpuguy83/go-md2man/v2 v2.0.3 // indirect | ||||||
| 	github.com/cyphar/filepath-securejoin v0.2.4 // indirect | 	github.com/cyphar/filepath-securejoin v0.2.4 // indirect | ||||||
| 	github.com/davecgh/go-spew v1.1.1 // indirect | 	github.com/davecgh/go-spew v1.1.1 // indirect | ||||||
| 	github.com/distribution/reference v0.5.0 // indirect | 	github.com/distribution/reference v0.5.0 // indirect | ||||||
| @ -47,7 +48,6 @@ require ( | |||||||
| 	github.com/gogo/protobuf v1.3.2 // indirect | 	github.com/gogo/protobuf v1.3.2 // indirect | ||||||
| 	github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect | 	github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect | ||||||
| 	github.com/golang/protobuf v1.5.3 // indirect | 	github.com/golang/protobuf v1.5.3 // indirect | ||||||
| 	github.com/google/go-cmp v0.5.9 // indirect |  | ||||||
| 	github.com/hashicorp/go-cleanhttp v0.5.2 // indirect | 	github.com/hashicorp/go-cleanhttp v0.5.2 // indirect | ||||||
| 	github.com/imdario/mergo v0.3.12 // indirect | 	github.com/imdario/mergo v0.3.12 // indirect | ||||||
| 	github.com/inconshreveable/mousetrap v1.0.0 // indirect | 	github.com/inconshreveable/mousetrap v1.0.0 // indirect | ||||||
| @ -75,9 +75,11 @@ require ( | |||||||
| 	github.com/russross/blackfriday/v2 v2.1.0 // indirect | 	github.com/russross/blackfriday/v2 v2.1.0 // indirect | ||||||
| 	github.com/skeema/knownhosts v1.2.0 // indirect | 	github.com/skeema/knownhosts v1.2.0 // indirect | ||||||
| 	github.com/spf13/pflag v1.0.5 // indirect | 	github.com/spf13/pflag v1.0.5 // indirect | ||||||
|  | 	github.com/urfave/cli/v2 v2.27.1 // indirect | ||||||
| 	github.com/xanzy/ssh-agent v0.3.3 // indirect | 	github.com/xanzy/ssh-agent v0.3.3 // indirect | ||||||
| 	github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect | 	github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect | ||||||
| 	github.com/xeipuuv/gojsonschema v1.2.0 // indirect | 	github.com/xeipuuv/gojsonschema v1.2.0 // indirect | ||||||
|  | 	github.com/xrash/smetrics v0.0.0-20231213231151-1d8dd44e695e // indirect | ||||||
| 	golang.org/x/crypto v0.14.0 // indirect | 	golang.org/x/crypto v0.14.0 // indirect | ||||||
| 	golang.org/x/mod v0.12.0 // indirect | 	golang.org/x/mod v0.12.0 // indirect | ||||||
| 	golang.org/x/net v0.17.0 // indirect | 	golang.org/x/net v0.17.0 // indirect | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								go.sum
									
									
									
									
									
								
							| @ -51,8 +51,8 @@ coopcloud.tech/tagcmp v0.0.0-20211103052201-885b22f77d52/go.mod h1:ESVm0wQKcbcFi | |||||||
| dario.cat/mergo v1.0.0 h1:AGCNq9Evsj31mOgNPcLyXc+4PNABt905YmuqPYYpBWk= | dario.cat/mergo v1.0.0 h1:AGCNq9Evsj31mOgNPcLyXc+4PNABt905YmuqPYYpBWk= | ||||||
| dario.cat/mergo v1.0.0/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk= | dario.cat/mergo v1.0.0/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk= | ||||||
| dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | ||||||
| git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100106-7462d91acefd h1:dctCkMhcsgIWMrkB1Br8S0RJF17eG+LKiqcXXVr3mdU= | git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100509-01bff8284355 h1:tCv2B4qoN6RMheKDnCzIafOkWS5BB1h7hwhmo+9bVeE= | ||||||
| git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100106-7462d91acefd/go.mod h1:Q8V1zbtPAlzYSr/Dvky3wS6x58IQAl3rot2me1oSO2Q= | git.coopcloud.tech/coop-cloud/godotenv v1.5.2-0.20231130100509-01bff8284355/go.mod h1:Q8V1zbtPAlzYSr/Dvky3wS6x58IQAl3rot2me1oSO2Q= | ||||||
| github.com/AdaLogics/go-fuzz-headers v0.0.0-20230106234847-43070de90fa1 h1:EKPd1INOIyr5hWOWhvpmQpY6tKjeG0hT1s3AMC/9fic= | github.com/AdaLogics/go-fuzz-headers v0.0.0-20230106234847-43070de90fa1 h1:EKPd1INOIyr5hWOWhvpmQpY6tKjeG0hT1s3AMC/9fic= | ||||||
| github.com/AdaLogics/go-fuzz-headers v0.0.0-20230106234847-43070de90fa1/go.mod h1:VzwV+t+dZ9j/H867F1M2ziD+yLHtB46oM35FxxMJ4d0= | github.com/AdaLogics/go-fuzz-headers v0.0.0-20230106234847-43070de90fa1/go.mod h1:VzwV+t+dZ9j/H867F1M2ziD+yLHtB46oM35FxxMJ4d0= | ||||||
| github.com/AlecAivazis/survey/v2 v2.3.7 h1:6I/u8FvytdGsgonrYsVn2t8t4QiRnh6QSTqkkhIiSjQ= | github.com/AlecAivazis/survey/v2 v2.3.7 h1:6I/u8FvytdGsgonrYsVn2t8t4QiRnh6QSTqkkhIiSjQ= | ||||||
| @ -74,6 +74,8 @@ github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBp | |||||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||||
| github.com/BurntSushi/toml v1.0.0 h1:dtDWrepsVPfW9H/4y7dDgFc2MBUSeJhlaDtK13CxFlU= | github.com/BurntSushi/toml v1.0.0 h1:dtDWrepsVPfW9H/4y7dDgFc2MBUSeJhlaDtK13CxFlU= | ||||||
| github.com/BurntSushi/toml v1.0.0/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= | github.com/BurntSushi/toml v1.0.0/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= | ||||||
|  | github.com/BurntSushi/toml v1.3.2 h1:o7IhLm0Msx3BaB+n3Ag7L8EVlByGnpq14C4YWiu/gL8= | ||||||
|  | github.com/BurntSushi/toml v1.3.2/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= | ||||||
| github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | ||||||
| github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= | github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= | ||||||
| github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 h1:vdT7QwBhJJEVNFMBNhRSFDRCB6O16T28VhvqRgqFyn8= | github.com/Gurpartap/logrus-stack v0.0.0-20170710170904-89c00d8a28f4 h1:vdT7QwBhJJEVNFMBNhRSFDRCB6O16T28VhvqRgqFyn8= | ||||||
| @ -311,6 +313,8 @@ github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:ma | |||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= | github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= | ||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU= | github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU= | ||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= | github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= | ||||||
|  | github.com/cpuguy83/go-md2man/v2 v2.0.3 h1:qMCsGGgs+MAzDFyp9LpAe1Lqy/fY/qCovCm0qnXZOBM= | ||||||
|  | github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= | ||||||
| github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= | github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= | ||||||
| github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | ||||||
| github.com/creack/pty v1.1.17/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= | github.com/creack/pty v1.1.17/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= | ||||||
| @ -893,6 +897,7 @@ github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6L | |||||||
| github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | ||||||
| github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= | github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= | ||||||
| github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= | github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= | ||||||
|  | github.com/russross/blackfriday v1.6.0 h1:KqfZb0pUVN2lYqZUYRddxF4OR8ZMURnJIG5Y3VRLtww= | ||||||
| github.com/russross/blackfriday v1.6.0/go.mod h1:ti0ldHuxg49ri4ksnFxlkCfN+hvslNlmVHqNRXXJNAY= | github.com/russross/blackfriday v1.6.0/go.mod h1:ti0ldHuxg49ri4ksnFxlkCfN+hvslNlmVHqNRXXJNAY= | ||||||
| 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/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk= | github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk= | ||||||
| @ -992,6 +997,8 @@ github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtX | |||||||
| github.com/urfave/cli v1.22.4/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | github.com/urfave/cli v1.22.4/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | ||||||
| github.com/urfave/cli v1.22.9 h1:cv3/KhXGBGjEXLC4bH0sLuJ9BewaAbpk5oyMOveu4pw= | github.com/urfave/cli v1.22.9 h1:cv3/KhXGBGjEXLC4bH0sLuJ9BewaAbpk5oyMOveu4pw= | ||||||
| github.com/urfave/cli v1.22.9/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | github.com/urfave/cli v1.22.9/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | ||||||
|  | github.com/urfave/cli/v2 v2.27.1 h1:8xSQ6szndafKVRmfyeUMxkNUJQMjL1F2zmsZ+qHpfho= | ||||||
|  | github.com/urfave/cli/v2 v2.27.1/go.mod h1:8qnjx1vcq5s2/wpsqoZFndg2CE5tNFyrTvS6SinrnYQ= | ||||||
| github.com/vbatts/tar-split v0.11.2/go.mod h1:vV3ZuO2yWSVsz+pfFzDG/upWH1JhjOiEaWq6kXyQ3VI= | github.com/vbatts/tar-split v0.11.2/go.mod h1:vV3ZuO2yWSVsz+pfFzDG/upWH1JhjOiEaWq6kXyQ3VI= | ||||||
| github.com/vishvananda/netlink v0.0.0-20181108222139-023a6dafdcdf/go.mod h1:+SR5DhBJrl6ZM7CoCKvpw5BKroDKQ+PJqOg65H/2ktk= | github.com/vishvananda/netlink v0.0.0-20181108222139-023a6dafdcdf/go.mod h1:+SR5DhBJrl6ZM7CoCKvpw5BKroDKQ+PJqOg65H/2ktk= | ||||||
| github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= | github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= | ||||||
| @ -1013,6 +1020,8 @@ github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17 | |||||||
| github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y= | github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y= | ||||||
| github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= | github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= | ||||||
| github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= | github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= | ||||||
|  | github.com/xrash/smetrics v0.0.0-20231213231151-1d8dd44e695e h1:+SOyEddqYF09QP7vr7CgJ1eti3pY9Fn3LHO1M1r/0sI= | ||||||
|  | github.com/xrash/smetrics v0.0.0-20231213231151-1d8dd44e695e/go.mod h1:N3UwUGtsrSj3ccvlPHLoLsHnpR27oXr4ZE984MbSER8= | ||||||
| github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||||
| github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||||
| github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||||
| @ -1315,7 +1324,6 @@ golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | |||||||
| golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.14.0 h1:Vz7Qs629MkJkGyHxUlRHizWJRG2j8fbQKjELVSNhy7Q= | golang.org/x/sys v0.14.0 h1:Vz7Qs629MkJkGyHxUlRHizWJRG2j8fbQKjELVSNhy7Q= | ||||||
| golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||||
| 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= | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ import ( | |||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| 	"github.com/urfave/cli" | 	"github.com/urfave/cli/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // AppNameComplete copletes app names. | // AppNameComplete copletes app names. | ||||||
| @ -51,6 +51,20 @@ func RecipeNameComplete(c *cli.Context) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // RecipeVersionComplete completes versions for the recipe. | ||||||
|  | func RecipeVersionComplete(recipeName string) { | ||||||
|  | 	catl, err := recipe.ReadRecipeCatalogue(false) | ||||||
|  | 	if err != nil { | ||||||
|  | 		logrus.Warn(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, v := range catl[recipeName].Versions { | ||||||
|  | 		for v2 := range v { | ||||||
|  | 			fmt.Println(v2) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| // ServerNameComplete completes server names. | // ServerNameComplete completes server names. | ||||||
| func ServerNameComplete(c *cli.Context) { | func ServerNameComplete(c *cli.Context) { | ||||||
| 	files, err := config.LoadAppFiles("") | 	files, err := config.LoadAppFiles("") | ||||||
|  | |||||||
| @ -50,34 +50,61 @@ type App struct { | |||||||
| 	Path   string | 	Path   string | ||||||
| } | } | ||||||
|  |  | ||||||
| // StackName gets whatever the docker safe (uses the right delimiting | // See documentation of config.StackName | ||||||
| // character, e.g. "_") stack name is for the app. In general, you don't want |  | ||||||
| // to use this to show anything to end-users, you want use a.Name instead. |  | ||||||
| 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 := StackName(a.Name) | ||||||
|  |  | ||||||
| 	if len(stackName) > 45 { |  | ||||||
| 		logrus.Debugf("trimming %s to %s to avoid runtime limits", stackName, stackName[:45]) |  | ||||||
| 		stackName = stackName[:45] |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	a.Env["STACK_NAME"] = stackName | 	a.Env["STACK_NAME"] = stackName | ||||||
|  |  | ||||||
| 	return stackName | 	return stackName | ||||||
| } | } | ||||||
|  |  | ||||||
| // Filters retrieves exact app filters for querying the container runtime. Due | // StackName gets whatever the docker safe (uses the right delimiting | ||||||
| // to upstream issues, filtering works different depending on what you're | // character, e.g. "_") stack name is for the app. In general, you don't want | ||||||
|  | // to use this to show anything to end-users, you want use a.Name instead. | ||||||
|  | func StackName(appName string) string { | ||||||
|  | 	stackName := SanitiseAppName(appName) | ||||||
|  |  | ||||||
|  | 	if len(stackName) > 45 { | ||||||
|  | 		logrus.Debugf("trimming %s to %s to avoid runtime limits", stackName, stackName[:45]) | ||||||
|  | 		stackName = stackName[:45] | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return stackName | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Filters retrieves app filters for querying the container runtime. By default | ||||||
|  | // it filters on all services in the app. It is also possible to pass an | ||||||
|  | // otional list of service names, which get filtered instead. | ||||||
|  | // | ||||||
|  | // Due to upstream issues, filtering works different depending on what you're | ||||||
| // querying. So, for example, secrets don't work with regex! The caller needs | // querying. So, for example, secrets don't work with regex! The caller needs | ||||||
| // to implement their own validation that the right secrets are matched. In | // to implement their own validation that the right secrets are matched. In | ||||||
| // order to handle these cases, we provide the `appendServiceNames` / | // order to handle these cases, we provide the `appendServiceNames` / | ||||||
| // `exactMatch` modifiers. | // `exactMatch` modifiers. | ||||||
| func (a App) Filters(appendServiceNames, exactMatch bool) (filters.Args, error) { | func (a App) Filters(appendServiceNames, exactMatch bool, services ...string) (filters.Args, error) { | ||||||
| 	filters := filters.NewArgs() | 	filters := filters.NewArgs() | ||||||
|  | 	if len(services) > 0 { | ||||||
|  | 		for _, serviceName := range services { | ||||||
|  | 			filters.Add("name", ServiceFilter(a.StackName(), serviceName, exactMatch)) | ||||||
|  | 		} | ||||||
|  | 		return filters, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// When not appending the service name, just add one filter for the whole | ||||||
|  | 	// stack. | ||||||
|  | 	if !appendServiceNames { | ||||||
|  | 		f := fmt.Sprintf("%s", a.StackName()) | ||||||
|  | 		if exactMatch { | ||||||
|  | 			f = fmt.Sprintf("^%s", f) | ||||||
|  | 		} | ||||||
|  | 		filters.Add("name", f) | ||||||
|  | 		return filters, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	composeFiles, err := GetComposeFiles(a.Recipe, a.Env) | 	composeFiles, err := GetComposeFiles(a.Recipe, a.Env) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -91,28 +118,23 @@ func (a App) Filters(appendServiceNames, exactMatch bool) (filters.Args, error) | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, service := range compose.Services { | 	for _, service := range compose.Services { | ||||||
| 		var filter string | 		f := ServiceFilter(a.StackName(), service.Name, exactMatch) | ||||||
|  | 		filters.Add("name", f) | ||||||
| 		if appendServiceNames { |  | ||||||
| 			if exactMatch { |  | ||||||
| 				filter = fmt.Sprintf("^%s_%s", a.StackName(), service.Name) |  | ||||||
| 			} else { |  | ||||||
| 				filter = fmt.Sprintf("%s_%s", a.StackName(), service.Name) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if exactMatch { |  | ||||||
| 				filter = fmt.Sprintf("^%s", a.StackName()) |  | ||||||
| 			} else { |  | ||||||
| 				filter = fmt.Sprintf("%s", a.StackName()) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		filters.Add("name", filter) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return filters, nil | 	return filters, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ServiceFilter creates a filter string for filtering a service in the docker | ||||||
|  | // container runtime. When exact match is true, it uses regex to match the | ||||||
|  | // string exactly. | ||||||
|  | func ServiceFilter(stack, service string, exact bool) string { | ||||||
|  | 	if exact { | ||||||
|  | 		return fmt.Sprintf("^%s_%s", stack, service) | ||||||
|  | 	} | ||||||
|  | 	return fmt.Sprintf("%s_%s", stack, service) | ||||||
|  | } | ||||||
|  |  | ||||||
| // ByServer sort a slice of Apps | // ByServer sort a slice of Apps | ||||||
| type ByServer []App | type ByServer []App | ||||||
|  |  | ||||||
| @ -333,7 +355,7 @@ func TemplateAppEnvSample(recipeName, appName, server, domain string) error { | |||||||
| 		return fmt.Errorf("%s already exists?", appEnvPath) | 		return fmt.Errorf("%s already exists?", appEnvPath) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = ioutil.WriteFile(appEnvPath, envSample, 0664) | 	err = ioutil.WriteFile(appEnvPath, envSample, 0o664) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| @ -595,7 +617,7 @@ func GetLabel(compose *composetypes.Config, stackName string, label string) stri | |||||||
|  |  | ||||||
| // GetTimeoutFromLabel reads the timeout value from docker label "coop-cloud.${STACK_NAME}.TIMEOUT" and returns 50 as default value | // GetTimeoutFromLabel reads the timeout value from docker label "coop-cloud.${STACK_NAME}.TIMEOUT" and returns 50 as default value | ||||||
| func GetTimeoutFromLabel(compose *composetypes.Config, stackName string) (int, error) { | func GetTimeoutFromLabel(compose *composetypes.Config, stackName string) (int, error) { | ||||||
| 	var timeout = 50 // Default Timeout | 	timeout := 50 // Default Timeout | ||||||
| 	var err error = nil | 	var err error = nil | ||||||
| 	if timeoutLabel := GetLabel(compose, stackName, "timeout"); timeoutLabel != "" { | 	if timeoutLabel := GetLabel(compose, stackName, "timeout"); timeoutLabel != "" { | ||||||
| 		logrus.Debugf("timeout label: %s", timeoutLabel) | 		logrus.Debugf("timeout label: %s", timeoutLabel) | ||||||
|  | |||||||
| @ -1,12 +1,15 @@ | |||||||
| package config_test | package config_test | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" | 	"coopcloud.tech/abra/pkg/config" | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" | 	"coopcloud.tech/abra/pkg/recipe" | ||||||
|  | 	"github.com/docker/docker/api/types/filters" | ||||||
|  | 	"github.com/google/go-cmp/cmp" | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @ -106,3 +109,89 @@ func TestGetComposeFilesError(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestFilters(t *testing.T) { | ||||||
|  | 	oldDir := config.RECIPES_DIR | ||||||
|  | 	config.RECIPES_DIR = "./testdir" | ||||||
|  | 	defer func() { | ||||||
|  | 		config.RECIPES_DIR = oldDir | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	app, err := config.NewApp(config.AppEnv{ | ||||||
|  | 		"DOMAIN": "test.example.com", | ||||||
|  | 		"RECIPE": "test-recipe", | ||||||
|  | 	}, "test_example_com", config.AppFile{ | ||||||
|  | 		Path:   "./testdir/filtertest.end", | ||||||
|  | 		Server: "local", | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	f, err := app.Filters(false, false) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	compareFilter(t, f, map[string]map[string]bool{ | ||||||
|  | 		"name": { | ||||||
|  | 			"test_example_com": true, | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	f2, err := app.Filters(false, true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	compareFilter(t, f2, map[string]map[string]bool{ | ||||||
|  | 		"name": { | ||||||
|  | 			"^test_example_com": true, | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	f3, err := app.Filters(true, false) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	compareFilter(t, f3, map[string]map[string]bool{ | ||||||
|  | 		"name": { | ||||||
|  | 			"test_example_com_bar": true, | ||||||
|  | 			"test_example_com_foo": true, | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	f4, err := app.Filters(true, true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	compareFilter(t, f4, map[string]map[string]bool{ | ||||||
|  | 		"name": { | ||||||
|  | 			"^test_example_com_bar": true, | ||||||
|  | 			"^test_example_com_foo": true, | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	f5, err := app.Filters(false, false, "foo") | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	compareFilter(t, f5, map[string]map[string]bool{ | ||||||
|  | 		"name": { | ||||||
|  | 			"test_example_com_foo": true, | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func compareFilter(t *testing.T, f1 filters.Args, f2 map[string]map[string]bool) { | ||||||
|  | 	t.Helper() | ||||||
|  | 	j1, err := f1.MarshalJSON() | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	j2, err := json.Marshal(f2) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	if diff := cmp.Diff(string(j2), string(j1)); diff != "" { | ||||||
|  | 		t.Errorf("filters mismatch (-want +got):\n%s", diff) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | |||||||
| @ -36,6 +36,8 @@ var REPOS_BASE_URL = "https://git.coopcloud.tech/coop-cloud" | |||||||
| var CATALOGUE_JSON_REPO_NAME = "recipes-catalogue-json" | var CATALOGUE_JSON_REPO_NAME = "recipes-catalogue-json" | ||||||
| var SSH_URL_TEMPLATE = "ssh://git@git.coopcloud.tech:2222/coop-cloud/%s.git" | var SSH_URL_TEMPLATE = "ssh://git@git.coopcloud.tech:2222/coop-cloud/%s.git" | ||||||
|  |  | ||||||
|  | var BackupbotLabel = "coop-cloud.backupbot.enabled" | ||||||
|  |  | ||||||
| // envVarModifiers is a list of env var modifier strings. These are added to | // envVarModifiers is a list of env var modifier strings. These are added to | ||||||
| // env vars as comments and modify their processing by Abra, e.g. determining | // env vars as comments and modify their processing by Abra, e.g. determining | ||||||
| // how long secrets should be. | // how long secrets should be. | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								pkg/config/testdir/filtertest.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								pkg/config/testdir/filtertest.env
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | RECIPE=test-recipe | ||||||
|  | DOMAIN=test.example.com | ||||||
							
								
								
									
										6
									
								
								pkg/config/testdir/test-recipe/compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								pkg/config/testdir/test-recipe/compose.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | version: "3.8" | ||||||
|  | services: | ||||||
|  |   foo: | ||||||
|  |     image: debian | ||||||
|  |   bar: | ||||||
|  |     image: debian | ||||||
| @ -28,7 +28,7 @@ func GetContainer(c context.Context, cl *client.Client, filters filters.Args, no | |||||||
| 		return types.Container{}, fmt.Errorf("no containers matching the %v filter found?", filter) | 		return types.Container{}, fmt.Errorf("no containers matching the %v filter found?", filter) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if len(containers) != 1 { | 	if len(containers) > 1 { | ||||||
| 		var containersRaw []string | 		var containersRaw []string | ||||||
| 		for _, container := range containers { | 		for _, container := range containers { | ||||||
| 			containerName := strings.Join(container.Names, " ") | 			containerName := strings.Join(container.Names, " ") | ||||||
|  | |||||||
							
								
								
									
										27
									
								
								pkg/git/add.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								pkg/git/add.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | package git | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/go-git/go-git/v5" | ||||||
|  | 	"github.com/sirupsen/logrus" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Add adds a file to the git index. | ||||||
|  | func Add(repoPath, path string, dryRun bool) error { | ||||||
|  | 	repo, err := git.PlainOpen(repoPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	worktree, err := repo.Worktree() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if dryRun { | ||||||
|  | 		logrus.Debugf("dry run: adding %s", path) | ||||||
|  | 	} else { | ||||||
|  | 		worktree.Add(path) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @ -264,6 +264,20 @@ func (r Recipe) SampleEnv() (map[string]string, error) { | |||||||
| 	return sampleEnv, nil | 	return sampleEnv, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Ensure makes sure the recipe exists, is up to date and has the latest version checked out. | ||||||
|  | func Ensure(recipeName string) error { | ||||||
|  | 	if err := EnsureExists(recipeName); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if err := EnsureUpToDate(recipeName); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if err := EnsureLatest(recipeName); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| // EnsureExists ensures that a recipe is locally cloned | // EnsureExists ensures that a recipe is locally cloned | ||||||
| func EnsureExists(recipeName string) error { | func EnsureExists(recipeName string) error { | ||||||
| 	recipeDir := path.Join(config.RECIPES_DIR, recipeName) | 	recipeDir := path.Join(config.RECIPES_DIR, recipeName) | ||||||
|  | |||||||
| @ -21,11 +21,24 @@ import ( | |||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // SecretValue represents a parsed `SECRET_FOO=v1 # length=bar` env var config | // Secret represents a secret. | ||||||
| // secret definition. | type Secret struct { | ||||||
| type SecretValue struct { | 	// Version comes from the secret version environment variable. | ||||||
|  | 	// For example: | ||||||
|  | 	//  SECRET_FOO=v1 | ||||||
| 	Version string | 	Version string | ||||||
| 	Length  int | 	// Length comes from the length modifier at the secret version environment | ||||||
|  | 	// variable. For Example: | ||||||
|  | 	//   SECRET_FOO=v1 # length=12 | ||||||
|  | 	Length int | ||||||
|  | 	// RemoteName is the name of the secret on the server. For example: | ||||||
|  | 	//   name: ${STACK_NAME}_test_pass_two_${SECRET_TEST_PASS_TWO_VERSION} | ||||||
|  | 	// With the following: | ||||||
|  | 	//   STACK_NAME=test_example_com | ||||||
|  | 	//   SECRET_TEST_PASS_TWO_VERSION=v2 | ||||||
|  | 	// Will have this remote name: | ||||||
|  | 	//   test_example_com_test_pass_two_v2 | ||||||
|  | 	RemoteName string | ||||||
| } | } | ||||||
|  |  | ||||||
| // GeneratePasswords generates passwords. | // GeneratePasswords generates passwords. | ||||||
| @ -67,17 +80,20 @@ func GeneratePassphrases(count uint) ([]string, error) { | |||||||
| // and some times you don't (as the caller). We need to be able to handle the | // and some times you don't (as the caller). We need to be able to handle the | ||||||
| // "app new" case where we pass in the .env.sample and the "secret generate" | // "app new" case where we pass in the .env.sample and the "secret generate" | ||||||
| // case where the app is created. | // case where the app is created. | ||||||
| func ReadSecretsConfig(appEnvPath string, composeFiles []string, recipeName string) (map[string]SecretValue, error) { | func ReadSecretsConfig(appEnvPath string, composeFiles []string, stackName string) (map[string]Secret, error) { | ||||||
| 	appEnv, appModifiers, err := config.ReadEnvWithModifiers(appEnvPath) | 	appEnv, appModifiers, err := config.ReadEnvWithModifiers(appEnvPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	// Set the STACK_NAME to be able to generate the remote name correctly. | ||||||
|  | 	appEnv["STACK_NAME"] = stackName | ||||||
|  |  | ||||||
| 	opts := stack.Deploy{Composefiles: composeFiles} | 	opts := stack.Deploy{Composefiles: composeFiles} | ||||||
| 	config, err := loader.LoadComposefile(opts, appEnv) | 	config, err := loader.LoadComposefile(opts, appEnv) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	// Read the compose files without injecting environment variables. | ||||||
| 	configWithoutEnv, err := loader.LoadComposefile(opts, map[string]string{}, loader.SkipInterpolation) | 	configWithoutEnv, err := loader.LoadComposefile(opts, map[string]string{}, loader.SkipInterpolation) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -95,7 +111,7 @@ func ReadSecretsConfig(appEnvPath string, composeFiles []string, recipeName stri | |||||||
| 		return nil, nil | 		return nil, nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	secretValues := map[string]SecretValue{} | 	secretValues := map[string]Secret{} | ||||||
| 	for secretId, secretConfig := range config.Secrets { | 	for secretId, secretConfig := range config.Secrets { | ||||||
| 		if string(secretConfig.Name[len(secretConfig.Name)-1]) == "_" { | 		if string(secretConfig.Name[len(secretConfig.Name)-1]) == "_" { | ||||||
| 			return nil, fmt.Errorf("missing version for secret? (%s)", secretId) | 			return nil, fmt.Errorf("missing version for secret? (%s)", secretId) | ||||||
| @ -108,15 +124,19 @@ func ReadSecretsConfig(appEnvPath string, composeFiles []string, recipeName stri | |||||||
|  |  | ||||||
| 		lastIdx := strings.LastIndex(secretConfig.Name, "_") | 		lastIdx := strings.LastIndex(secretConfig.Name, "_") | ||||||
| 		secretVersion := secretConfig.Name[lastIdx+1:] | 		secretVersion := secretConfig.Name[lastIdx+1:] | ||||||
| 		value := SecretValue{Version: secretVersion} | 		value := Secret{Version: secretVersion, RemoteName: secretConfig.Name} | ||||||
|  |  | ||||||
| 		// Check if the length modifier is set for this secret. | 		// Check if the length modifier is set for this secret. | ||||||
| 		for k, v := range appModifiers { | 		for envName, modifierValues := range appModifiers { | ||||||
| 			// configWithoutEnv contains the raw name as defined in the compose.yaml | 			// configWithoutEnv contains the raw name as defined in the compose.yaml | ||||||
| 			if !strings.Contains(configWithoutEnv.Secrets[secretId].Name, k) { | 			// The name will look something like this: | ||||||
|  | 			//   name: ${STACK_NAME}_test_pass_two_${SECRET_TEST_PASS_TWO_VERSION} | ||||||
|  | 			// To check if the current modifier is for the current secret we check | ||||||
|  | 			// if the raw name contains the env name (e.g. SECRET_TEST_PASS_TWO_VERSION). | ||||||
|  | 			if !strings.Contains(configWithoutEnv.Secrets[secretId].Name, envName) { | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 			lengthRaw, ok := v["length"] | 			lengthRaw, ok := modifierValues["length"] | ||||||
| 			if ok { | 			if ok { | ||||||
| 				length, err := strconv.Atoi(lengthRaw) | 				length, err := strconv.Atoi(lengthRaw) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| @ -133,7 +153,7 @@ func ReadSecretsConfig(appEnvPath string, composeFiles []string, recipeName stri | |||||||
| } | } | ||||||
|  |  | ||||||
| // GenerateSecrets generates secrets locally and sends them to a remote server for storage. | // GenerateSecrets generates secrets locally and sends them to a remote server for storage. | ||||||
| func GenerateSecrets(cl *dockerClient.Client, secrets map[string]SecretValue, appName, server string) (map[string]string, error) { | func GenerateSecrets(cl *dockerClient.Client, secrets map[string]Secret, server string) (map[string]string, error) { | ||||||
| 	secretsGenerated := map[string]string{} | 	secretsGenerated := map[string]string{} | ||||||
| 	var mutex sync.Mutex | 	var mutex sync.Mutex | ||||||
| 	var wg sync.WaitGroup | 	var wg sync.WaitGroup | ||||||
| @ -141,11 +161,10 @@ func GenerateSecrets(cl *dockerClient.Client, secrets map[string]SecretValue, ap | |||||||
| 	for n, v := range secrets { | 	for n, v := range secrets { | ||||||
| 		wg.Add(1) | 		wg.Add(1) | ||||||
|  |  | ||||||
| 		go func(secretName string, secret SecretValue) { | 		go func(secretName string, secret Secret) { | ||||||
| 			defer wg.Done() | 			defer wg.Done() | ||||||
|  |  | ||||||
| 			secretRemoteName := fmt.Sprintf("%s_%s_%s", appName, secretName, secret.Version) | 			logrus.Debugf("attempting to generate and store %s on %s", secret.RemoteName, server) | ||||||
| 			logrus.Debugf("attempting to generate and store %s on %s", secretRemoteName, server) |  | ||||||
|  |  | ||||||
| 			if secret.Length > 0 { | 			if secret.Length > 0 { | ||||||
| 				passwords, err := GeneratePasswords(1, uint(secret.Length)) | 				passwords, err := GeneratePasswords(1, uint(secret.Length)) | ||||||
| @ -154,9 +173,9 @@ func GenerateSecrets(cl *dockerClient.Client, secrets map[string]SecretValue, ap | |||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				if err := client.StoreSecret(cl, secretRemoteName, passwords[0], server); err != nil { | 				if err := client.StoreSecret(cl, secret.RemoteName, passwords[0], server); err != nil { | ||||||
| 					if strings.Contains(err.Error(), "AlreadyExists") { | 					if strings.Contains(err.Error(), "AlreadyExists") { | ||||||
| 						logrus.Warnf("%s already exists, moving on...", secretRemoteName) | 						logrus.Warnf("%s already exists, moving on...", secret.RemoteName) | ||||||
| 						ch <- nil | 						ch <- nil | ||||||
| 					} else { | 					} else { | ||||||
| 						ch <- err | 						ch <- err | ||||||
| @ -174,9 +193,9 @@ func GenerateSecrets(cl *dockerClient.Client, secrets map[string]SecretValue, ap | |||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				if err := client.StoreSecret(cl, secretRemoteName, passphrases[0], server); err != nil { | 				if err := client.StoreSecret(cl, secret.RemoteName, passphrases[0], server); err != nil { | ||||||
| 					if strings.Contains(err.Error(), "AlreadyExists") { | 					if strings.Contains(err.Error(), "AlreadyExists") { | ||||||
| 						logrus.Warnf("%s already exists, moving on...", secretRemoteName) | 						logrus.Warnf("%s already exists, moving on...", secret.RemoteName) | ||||||
| 						ch <- nil | 						ch <- nil | ||||||
| 					} else { | 					} else { | ||||||
| 						ch <- err | 						ch <- err | ||||||
| @ -225,7 +244,7 @@ func PollSecretsStatus(cl *dockerClient.Client, app config.App) (secretStatuses, | |||||||
| 		return secStats, err | 		return secStats, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	secretsConfig, err := ReadSecretsConfig(app.Path, composeFiles, app.Recipe) | 	secretsConfig, err := ReadSecretsConfig(app.Path, composeFiles, app.StackName()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return secStats, err | 		return secStats, err | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -1,42 +1,30 @@ | |||||||
| package secret | package secret | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"path" |  | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"coopcloud.tech/abra/pkg/config" |  | ||||||
| 	"coopcloud.tech/abra/pkg/recipe" |  | ||||||
| 	"coopcloud.tech/abra/pkg/upstream/stack" |  | ||||||
| 	loader "coopcloud.tech/abra/pkg/upstream/stack" |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestReadSecretsConfig(t *testing.T) { | func TestReadSecretsConfig(t *testing.T) { | ||||||
| 	offline := true | 	composeFiles := []string{"./testdir/compose.yaml"} | ||||||
| 	recipe, err := recipe.Get("matrix-synapse", offline) | 	secretsFromConfig, err := ReadSecretsConfig("./testdir/.env.sample", composeFiles, "test_example_com") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	sampleEnv, err := recipe.SampleEnv() | 	// Simple secret | ||||||
| 	if err != nil { | 	assert.Equal(t, "test_example_com_test_pass_one_v2", secretsFromConfig["test_pass_one"].RemoteName) | ||||||
| 		t.Fatal(err) | 	assert.Equal(t, "v2", secretsFromConfig["test_pass_one"].Version) | ||||||
| 	} | 	assert.Equal(t, 0, secretsFromConfig["test_pass_one"].Length) | ||||||
|  |  | ||||||
| 	composeFiles := []string{path.Join(config.RECIPES_DIR, recipe.Name, "compose.yml")} | 	// Has a length modifier | ||||||
| 	envSamplePath := path.Join(config.RECIPES_DIR, recipe.Name, ".env.sample") | 	assert.Equal(t, "test_example_com_test_pass_two_v1", secretsFromConfig["test_pass_two"].RemoteName) | ||||||
| 	secretsFromConfig, err := ReadSecretsConfig(envSamplePath, composeFiles, recipe.Name) | 	assert.Equal(t, "v1", secretsFromConfig["test_pass_two"].Version) | ||||||
| 	if err != nil { | 	assert.Equal(t, 10, secretsFromConfig["test_pass_two"].Length) | ||||||
| 		t.Fatal(err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	opts := stack.Deploy{Composefiles: composeFiles} | 	// Secret name does not include the secret id | ||||||
| 	config, err := loader.LoadComposefile(opts, sampleEnv) | 	assert.Equal(t, "test_example_com_pass_three_v2", secretsFromConfig["test_pass_three"].RemoteName) | ||||||
| 	if err != nil { | 	assert.Equal(t, "v2", secretsFromConfig["test_pass_three"].Version) | ||||||
| 		t.Fatal(err) | 	assert.Equal(t, 0, secretsFromConfig["test_pass_three"].Length) | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for secretId := range config.Secrets { |  | ||||||
| 		assert.Contains(t, secretsFromConfig, secretId) |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								pkg/secret/testdir/.env.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								pkg/secret/testdir/.env.sample
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | SECRET_TEST_PASS_ONE_VERSION=v2 | ||||||
|  | SECRET_TEST_PASS_TWO_VERSION=v1 # length=10 | ||||||
|  | SECRET_TEST_PASS_THREE_VERSION=v2 | ||||||
							
								
								
									
										21
									
								
								pkg/secret/testdir/compose.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								pkg/secret/testdir/compose.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | --- | ||||||
|  | version: "3.8" | ||||||
|  |  | ||||||
|  | services: | ||||||
|  |   app: | ||||||
|  |     image: nginx:1.21.0 | ||||||
|  |     secrets: | ||||||
|  |       - test_pass_one | ||||||
|  |       - test_pass_two | ||||||
|  |       - test_pass_three | ||||||
|  |  | ||||||
|  | secrets: | ||||||
|  |   test_pass_one: | ||||||
|  |     external: true | ||||||
|  |     name: ${STACK_NAME}_test_pass_one_${SECRET_TEST_PASS_ONE_VERSION}  # should be removed | ||||||
|  |   test_pass_two: | ||||||
|  |     external: true | ||||||
|  |     name: ${STACK_NAME}_test_pass_two_${SECRET_TEST_PASS_TWO_VERSION} | ||||||
|  |   test_pass_three: | ||||||
|  |     external: true | ||||||
|  |     name: ${STACK_NAME}_pass_three_${SECRET_TEST_PASS_THREE_VERSION} # secretId and name don't match | ||||||
| @ -14,6 +14,70 @@ import ( | |||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | // GetService retrieves a service container based on a label. 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. An error is returned when no service is | ||||||
|  | // found. | ||||||
|  | func GetServiceByLabel(c context.Context, cl *client.Client, label string, prompt bool) (swarm.Service, error) { | ||||||
|  | 	services, err := cl.ServiceList(c, types.ServiceListOptions{}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return swarm.Service{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(services) == 0 { | ||||||
|  | 		return swarm.Service{}, fmt.Errorf("no services deployed?") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var matchingServices []swarm.Service | ||||||
|  | 	for _, service := range services { | ||||||
|  | 		if enabled, exists := service.Spec.Labels[label]; exists && enabled == "true" { | ||||||
|  | 			matchingServices = append(matchingServices, service) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(matchingServices) == 0 { | ||||||
|  | 		return swarm.Service{}, fmt.Errorf("no services deployed matching label '%s'?", label) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(matchingServices) > 1 { | ||||||
|  | 		var servicesRaw []string | ||||||
|  | 		for _, service := range matchingServices { | ||||||
|  | 			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(matchingServices), 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 matchingServices { | ||||||
|  | 			serviceName := strings.ToLower(service.Spec.Name) | ||||||
|  | 			if serviceName == chosenService { | ||||||
|  | 				return service, nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		logrus.Panic("failed to match chosen service") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return matchingServices[0], nil | ||||||
|  | } | ||||||
|  |  | ||||||
| // GetService retrieves a service container. If prompt is true and the retrievd | // 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 | // 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. | // let the user choose. A count of 0 is handled gracefully. | ||||||
|  | |||||||
| @ -13,7 +13,10 @@ import ( | |||||||
| 	"github.com/sirupsen/logrus" | 	"github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func RunExec(dockerCli command.Cli, client *apiclient.Client, containerID string, execConfig *types.ExecConfig) error { | // RunExec runs a command on a remote container. io.Writer corresponds to the | ||||||
|  | // command output. | ||||||
|  | func RunExec(dockerCli command.Cli, client *apiclient.Client, containerID string, | ||||||
|  | 	execConfig *types.ExecConfig) (io.Writer, error) { | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
|  |  | ||||||
| 	// We need to check the tty _before_ we do the ContainerExecCreate, because | 	// We need to check the tty _before_ we do the ContainerExecCreate, because | ||||||
| @ -21,22 +24,22 @@ func RunExec(dockerCli command.Cli, client *apiclient.Client, containerID string | |||||||
| 	// there's no easy way to clean those up). But also in order to make "not | 	// there's no easy way to clean those up). But also in order to make "not | ||||||
| 	// exist" errors take precedence we do a dummy inspect first. | 	// exist" errors take precedence we do a dummy inspect first. | ||||||
| 	if _, err := client.ContainerInspect(ctx, containerID); err != nil { | 	if _, err := client.ContainerInspect(ctx, containerID); err != nil { | ||||||
| 		return err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if !execConfig.Detach { | 	if !execConfig.Detach { | ||||||
| 		if err := dockerCli.In().CheckTty(execConfig.AttachStdin, execConfig.Tty); err != nil { | 		if err := dockerCli.In().CheckTty(execConfig.AttachStdin, execConfig.Tty); err != nil { | ||||||
| 			return err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	response, err := client.ContainerExecCreate(ctx, containerID, *execConfig) | 	response, err := client.ContainerExecCreate(ctx, containerID, *execConfig) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	execID := response.ID | 	execID := response.ID | ||||||
| 	if execID == "" { | 	if execID == "" { | ||||||
| 		return errors.New("exec ID empty") | 		return nil, errors.New("exec ID empty") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if execConfig.Detach { | 	if execConfig.Detach { | ||||||
| @ -44,13 +47,13 @@ func RunExec(dockerCli command.Cli, client *apiclient.Client, containerID string | |||||||
| 			Detach: execConfig.Detach, | 			Detach: execConfig.Detach, | ||||||
| 			Tty:    execConfig.Tty, | 			Tty:    execConfig.Tty, | ||||||
| 		} | 		} | ||||||
| 		return client.ContainerExecStart(ctx, execID, execStartCheck) | 		return nil, client.ContainerExecStart(ctx, execID, execStartCheck) | ||||||
| 	} | 	} | ||||||
| 	return interactiveExec(ctx, dockerCli, client, execConfig, execID) | 	return interactiveExec(ctx, dockerCli, client, execConfig, execID) | ||||||
| } | } | ||||||
|  |  | ||||||
| func interactiveExec(ctx context.Context, dockerCli command.Cli, client *apiclient.Client, | func interactiveExec(ctx context.Context, dockerCli command.Cli, client *apiclient.Client, | ||||||
| 	execConfig *types.ExecConfig, execID string) error { | 	execConfig *types.ExecConfig, execID string) (io.Writer, error) { | ||||||
| 	// Interactive exec requested. | 	// Interactive exec requested. | ||||||
| 	var ( | 	var ( | ||||||
| 		out, stderr io.Writer | 		out, stderr io.Writer | ||||||
| @ -76,7 +79,7 @@ func interactiveExec(ctx context.Context, dockerCli command.Cli, client *apiclie | |||||||
| 	} | 	} | ||||||
| 	resp, err := client.ContainerExecAttach(ctx, execID, execStartCheck) | 	resp, err := client.ContainerExecAttach(ctx, execID, execStartCheck) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return out, err | ||||||
| 	} | 	} | ||||||
| 	defer resp.Close() | 	defer resp.Close() | ||||||
|  |  | ||||||
| @ -107,10 +110,10 @@ func interactiveExec(ctx context.Context, dockerCli command.Cli, client *apiclie | |||||||
|  |  | ||||||
| 	if err := <-errCh; err != nil { | 	if err := <-errCh; err != nil { | ||||||
| 		logrus.Debugf("Error hijack: %s", err) | 		logrus.Debugf("Error hijack: %s", err) | ||||||
| 		return err | 		return out, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return getExecExitStatus(ctx, client, execID) | 	return out, getExecExitStatus(ctx, client, execID) | ||||||
| } | } | ||||||
|  |  | ||||||
| func getExecExitStatus(ctx context.Context, client apiclient.ContainerAPIClient, execID string) error { | func getExecExitStatus(ctx context.Context, client apiclient.ContainerAPIClient, execID string) error { | ||||||
|  | |||||||
| @ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| ABRA_VERSION="0.8.1-beta" | ABRA_VERSION="0.8.1-beta" | ||||||
| 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.8.1-beta" | RC_VERSION="0.8.0-rc1-beta" | ||||||
| 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 | ||||||
| @ -65,17 +65,19 @@ function install_abra_release { | |||||||
|  |  | ||||||
|   checksums=$(wget -q -O- $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') | ||||||
|  |   abra_download="/tmp/abra-download" | ||||||
|  |  | ||||||
|   echo "downloading $ABRA_VERSION $PLATFORM binary release for abra..." |   echo "downloading $ABRA_VERSION $PLATFORM binary release for abra..." | ||||||
|   wget -q "$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') |   wget -q "$release_url" -O $abra_download  | ||||||
|  |   localsum=$(sha256sum $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 | ||||||
|       print_checksum_error |       print_checksum_error | ||||||
|       exit 1 |       exit 1 | ||||||
|   fi |   fi | ||||||
|   echo "$(tput setaf 2)check successful!$(tput sgr0)" |   echo "$(tput setaf 2)check successful!$(tput sgr0)" | ||||||
|   mv "$HOME/.local/bin/.abra-download" "$HOME/.local/bin/abra" |   mv "$abra_download" "$HOME/.local/bin/abra" | ||||||
|   chmod +x "$HOME/.local/bin/abra" |   chmod +x "$HOME/.local/bin/abra" | ||||||
|  |  | ||||||
|   x=$(echo $PATH | grep $HOME/.local/bin) |   x=$(echo $PATH | grep $HOME/.local/bin) | ||||||
|  | |||||||
| @ -70,13 +70,13 @@ setup(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app check "$TEST_APP_DOMAIN" |   run $ABRA app check "$TEST_APP_DOMAIN" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
| @ -86,7 +86,7 @@ setup(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 1' |   assert_output --partial "Your branch is behind 'origin/main' by 1 commit" | ||||||
|  |  | ||||||
|   # NOTE(d1): we can't quite tell if this will fail or not in the future, so, |   # NOTE(d1): we can't quite tell if this will fail or not in the future, so, | ||||||
|   # since it isn't an important part of what we're testing here, we don't check |   # since it isn't an important part of what we're testing here, we don't check | ||||||
| @ -94,7 +94,7 @@ setup(){ | |||||||
|   run $ABRA app check "$TEST_APP_DOMAIN" --offline |   run $ABRA app check "$TEST_APP_DOMAIN" --offline | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 1' |   assert_output --partial "Your branch is behind 'origin/main' by 1 commit" | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
|  | |||||||
| @ -58,7 +58,7 @@ test_cmd_export" | |||||||
|   assert_success |   assert_success | ||||||
|   assert_not_exists "$ABRA_DIR/recipes/$TEST_RECIPE" |   assert_not_exists "$ABRA_DIR/recipes/$TEST_RECIPE" | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'baz' |   assert_output --partial 'baz' | ||||||
|  |  | ||||||
| @ -70,7 +70,7 @@ test_cmd_export" | |||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/recipes/$TEST_RECIPE/foo" |   assert_exists "$ABRA_DIR/recipes/$TEST_RECIPE/foo" | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial 'locally unstaged changes' |   assert_output --partial 'locally unstaged changes' | ||||||
|  |  | ||||||
| @ -83,7 +83,7 @@ test_cmd_export" | |||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/recipes/$TEST_RECIPE/foo" |   assert_exists "$ABRA_DIR/recipes/$TEST_RECIPE/foo" | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local --chaos |   run $ABRA app cmd --local --chaos "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'baz' |   assert_output --partial 'baz' | ||||||
|  |  | ||||||
| @ -96,14 +96,14 @@ test_cmd_export" | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'baz' |   assert_output --partial 'baz' | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --partial "up to date" | ||||||
|  |  | ||||||
|   _reset_recipe "$TEST_RECIPE" |   _reset_recipe "$TEST_RECIPE" | ||||||
| } | } | ||||||
| @ -113,14 +113,14 @@ test_cmd_export" | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local --offline |   run $ABRA app cmd --local --offline "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'baz' |   assert_output --partial 'baz' | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe "$TEST_RECIPE" |   _reset_recipe "$TEST_RECIPE" | ||||||
| } | } | ||||||
| @ -132,13 +132,13 @@ test_cmd_export" | |||||||
| } | } | ||||||
|  |  | ||||||
| @test "error if missing arguments when passing --local" { | @test "error if missing arguments when passing --local" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial 'missing arguments' |   assert_output --partial 'missing arguments' | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "cannot use --local and --user at same time" { | @test "cannot use --local and --user at same time" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local --user root |   run $ABRA app cmd --local --user root "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial 'cannot use --local & --user together' |   assert_output --partial 'cannot use --local & --user together' | ||||||
| } | } | ||||||
| @ -147,7 +147,7 @@ test_cmd_export" | |||||||
|   run rm -rf "$ABRA_DIR/recipes/$TEST_RECIPE/abra.sh" |   run rm -rf "$ABRA_DIR/recipes/$TEST_RECIPE/abra.sh" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local --chaos |   run $ABRA app cmd --local --chaos "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial "$ABRA_DIR/recipes/$TEST_RECIPE/abra.sh does not exist" |   assert_output --partial "$ABRA_DIR/recipes/$TEST_RECIPE/abra.sh does not exist" | ||||||
|  |  | ||||||
| @ -155,25 +155,25 @@ test_cmd_export" | |||||||
| } | } | ||||||
|  |  | ||||||
| @test "error if missing command" { | @test "error if missing command" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" doesnt_exist --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" doesnt_exist | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial "doesn't have a doesnt_exist function" |   assert_output --partial "doesn't have a doesnt_exist function" | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "run --local command" { | @test "run --local command" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd --local |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'baz' |   assert_output --partial 'baz' | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "run command with single arg" { | @test "run command with single arg" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd_arg --local -- bing |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd_arg -- bing | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'bing' |   assert_output --partial 'bing' | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "run command with several args" { | @test "run command with several args" { | ||||||
|   run $ABRA app cmd "$TEST_APP_DOMAIN" test_cmd_args --local -- bong bang |   run $ABRA app cmd --local "$TEST_APP_DOMAIN" test_cmd_args -- bong bang | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'bong bang' |   assert_output --partial 'bong bang' | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,31 +19,6 @@ setup(){ | |||||||
|   _common_setup |   _common_setup | ||||||
| } | } | ||||||
|  |  | ||||||
| _mkfile() { |  | ||||||
|   run bash -c "echo $2 > $1" |  | ||||||
|   assert_success |  | ||||||
| } |  | ||||||
|  |  | ||||||
| _mkfile_remote() { |  | ||||||
|   run $ABRA app run "$TEST_APP_DOMAIN" app "bash -c \"echo $2 > $1\"" |  | ||||||
|   assert_success |  | ||||||
| } |  | ||||||
|  |  | ||||||
| _mkdir() { |  | ||||||
|   run bash -c "mkdir -p $1" |  | ||||||
|   assert_success |  | ||||||
| } |  | ||||||
|  |  | ||||||
| _rm() { |  | ||||||
|   run rm -rf "$1" |  | ||||||
|   assert_success |  | ||||||
| } |  | ||||||
|  |  | ||||||
| _rm_remote() { |  | ||||||
|   run "$ABRA" app run "$TEST_APP_DOMAIN" app rm -rf "$1" |  | ||||||
|   assert_success |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @test "validate app argument" { | @test "validate app argument" { | ||||||
|   run $ABRA app cp |   run $ABRA app cp | ||||||
|   assert_failure |   assert_failure | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ teardown_file(){ | |||||||
| setup(){ | setup(){ | ||||||
|   load "$PWD/tests/integration/helpers/common" |   load "$PWD/tests/integration/helpers/common" | ||||||
|   _common_setup |   _common_setup | ||||||
|  |   _reset_recipe | ||||||
| } | } | ||||||
|  |  | ||||||
| teardown(){ | teardown(){ | ||||||
| @ -82,13 +83,13 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app deploy "$TEST_APP_DOMAIN" --no-input --no-converge-checks |   run $ABRA app deploy "$TEST_APP_DOMAIN" --no-input --no-converge-checks | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   refute_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
| @ -100,7 +101,7 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   # NOTE(d1): need to use --chaos to force same commit |   # NOTE(d1): need to use --chaos to force same commit | ||||||
|   run $ABRA app deploy "$TEST_APP_DOMAIN" \ |   run $ABRA app deploy "$TEST_APP_DOMAIN" \ | ||||||
| @ -108,7 +109,7 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| @ -116,6 +117,9 @@ teardown(){ | |||||||
|  |  | ||||||
| # bats test_tags=slow | # bats test_tags=slow | ||||||
| @test "deploy latest commit if no published versions and no --chaos" { | @test "deploy latest commit if no published versions and no --chaos" { | ||||||
|  |   # TODO(d1): fix with a new test recipe which has no published versions? | ||||||
|  |   skip "known issue, abra-test-recipe has published versions now" | ||||||
|  |  | ||||||
|   latestCommit="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" |   latestCommit="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" | ||||||
|  |  | ||||||
|   _remove_tags |   _remove_tags | ||||||
| @ -140,7 +144,7 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   threeCommitsBack="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" |   threeCommitsBack="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" | ||||||
|  |  | ||||||
| @ -273,6 +277,10 @@ teardown(){ | |||||||
| } | } | ||||||
|  |  | ||||||
| @test "ensure domain is checked" { | @test "ensure domain is checked" { | ||||||
|  |   if [[ "$TEST_SERVER" == "default" ]]; then | ||||||
|  |       skip "domain checks are disabled for local server" | ||||||
|  |   fi | ||||||
|  |  | ||||||
|   appDomain="custom-html.DOESNTEXIST" |   appDomain="custom-html.DOESNTEXIST" | ||||||
|  |  | ||||||
|   run $ABRA app new custom-html \ |   run $ABRA app new custom-html \ | ||||||
|  | |||||||
| @ -18,9 +18,24 @@ setup(){ | |||||||
| } | } | ||||||
|  |  | ||||||
| teardown(){ | teardown(){ | ||||||
|  |   load "$PWD/tests/integration/helpers/common" | ||||||
|   _rm_app |   _rm_app | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @test "autocomplete" { | ||||||
|  |   run $ABRA app new --generate-bash-completion | ||||||
|  |   assert_success | ||||||
|  |   assert_output --partial "traefik" | ||||||
|  |   assert_output --partial "abra-test-recipe" | ||||||
|  |  | ||||||
|  |   # Note: this test needs to be updated when a new version of the test recipe is published. | ||||||
|  |   run $ABRA app new abra-test-recipe --generate-bash-completion | ||||||
|  |   assert_success | ||||||
|  |   assert_output "0.1.0+1.20.0 | ||||||
|  | 0.1.1+1.20.2 | ||||||
|  | 0.2.0+1.21.0" | ||||||
|  | } | ||||||
|  |  | ||||||
| @test "create new app" { | @test "create new app" { | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
| @ -28,10 +43,29 @@ teardown(){ | |||||||
|     --domain "$TEST_APP_DOMAIN" |     --domain "$TEST_APP_DOMAIN" | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|  |   assert_output --partial "up to date" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @test "create new app with version" { | ||||||
|  |   run $ABRA app new "$TEST_RECIPE" 0.1.1+1.20.2 \ | ||||||
|  |     --no-input \ | ||||||
|  |     --server "$TEST_SERVER" \ | ||||||
|  |     --domain "$TEST_APP_DOMAIN" | ||||||
|  |   assert_success | ||||||
|  |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" log -1 | ||||||
|  |   assert_output --partial "453db7121c0a56a7a8f15378f18fe3bf21ccfdef" | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "does not overwrite existing env files" { | @test "does not overwrite existing env files" { | ||||||
|   _new_app |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
|  |     --no-input \ | ||||||
|  |     --server "$TEST_SERVER" \ | ||||||
|  |     --domain "$TEST_APP_DOMAIN" | ||||||
|  |   assert_success | ||||||
|  |  | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
| @ -74,8 +108,7 @@ teardown(){ | |||||||
|     --no-input \ |     --no-input \ | ||||||
|     --chaos \ |     --chaos \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" | ||||||
|     --secrets |  | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
| @ -88,18 +121,17 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" | ||||||
|     --secrets |  | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --partial "up to date" | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
| @ -109,7 +141,7 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   # NOTE(d1): need to use --chaos to force same commit |   # NOTE(d1): need to use --chaos to force same commit | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
| @ -117,13 +149,12 @@ teardown(){ | |||||||
|     --offline \ |     --offline \ | ||||||
|     --chaos \ |     --chaos \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" | ||||||
|     --secrets |  | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
|  | |||||||
| @ -104,10 +104,10 @@ teardown(){ | |||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |   # TODO: should wait as long as volume is no longer in use | ||||||
|   sleep 5 |   sleep 10 | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --no-input | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run $ABRA app volume ls "$TEST_APP_DOMAIN" |   run $ABRA app volume ls "$TEST_APP_DOMAIN" | ||||||
| @ -132,9 +132,6 @@ teardown(){ | |||||||
|  |  | ||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |  | ||||||
|   sleep 5 |  | ||||||
|  |  | ||||||
|   run $ABRA app rm "$TEST_APP_DOMAIN" --no-input |   run $ABRA app rm "$TEST_APP_DOMAIN" --no-input | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'test-volume' |   assert_output --partial 'test-volume' | ||||||
|  | |||||||
| @ -109,13 +109,13 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app restore "$TEST_APP_DOMAIN" app DOESNTEXIST |   run $ABRA app restore "$TEST_APP_DOMAIN" app | ||||||
|   assert_failure |   assert_failure | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --partial "up to date" | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "ensure recipe not up to date if --offline" { | @test "ensure recipe not up to date if --offline" { | ||||||
| @ -126,19 +126,19 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app restore "$TEST_APP_DOMAIN" app DOESNTEXIST --offline |   run $ABRA app restore "$TEST_APP_DOMAIN" app --offline | ||||||
|   assert_failure |   assert_failure | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" checkout "$latestCommit" |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" checkout "$latestCommit" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --partial "HEAD detached at $latestCommit" | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "error if missing service" { | @test "error if missing service" { | ||||||
|  | |||||||
| @ -50,13 +50,13 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app rollback "$TEST_APP_DOMAIN" --no-input --no-converge-checks |   run $ABRA app rollback "$TEST_APP_DOMAIN" --no-input --no-converge-checks | ||||||
|   assert_failure |   assert_failure | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --partial "up to date" | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "ensure recipe not up to date if --offline" { | @test "ensure recipe not up to date if --offline" { | ||||||
| @ -67,14 +67,14 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA app rollback "$TEST_APP_DOMAIN" \ |   run $ABRA app rollback "$TEST_APP_DOMAIN" \ | ||||||
|     --no-input --no-converge-checks --offline |     --no-input --no-converge-checks --offline | ||||||
|   assert_failure |   assert_failure | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" checkout "$latestCommit" |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" checkout "$latestCommit" | ||||||
|   assert_success |   assert_success | ||||||
| @ -131,7 +131,7 @@ teardown(){ | |||||||
|   latestCommit="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" |   latestCommit="$(git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rev-parse --short HEAD)" | ||||||
|  |  | ||||||
|   run $ABRA app deploy "$TEST_APP_DOMAIN" \ |   run $ABRA app deploy "$TEST_APP_DOMAIN" \ | ||||||
|     --no-input --no-converge-checks --chaos |     --no-input --chaos | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial "$latestCommit" |   assert_output --partial "$latestCommit" | ||||||
|   assert_output --partial 'chaos' |   assert_output --partial 'chaos' | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ setup_file(){ | |||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new "$TEST_RECIPE" \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
| } | } | ||||||
| @ -19,13 +19,6 @@ teardown_file(){ | |||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
|  |  | ||||||
| teardown(){ |  | ||||||
|   # https://github.com/bats-core/bats-core/issues/383#issuecomment-738628888 |  | ||||||
|   if [[ -z "${BATS_TEST_COMPLETED}" ]]; then |  | ||||||
|     _undeploy_app |  | ||||||
|   fi |  | ||||||
| } |  | ||||||
|  |  | ||||||
| setup(){ | setup(){ | ||||||
|   load "$PWD/tests/integration/helpers/common" |   load "$PWD/tests/integration/helpers/common" | ||||||
|   _common_setup |   _common_setup | ||||||
|  | |||||||
| @ -59,6 +59,8 @@ teardown(){ | |||||||
|  |  | ||||||
| # bats test_tags=slow | # bats test_tags=slow | ||||||
| @test "error if not in catalogue" { | @test "error if not in catalogue" { | ||||||
|  |   skip "known issue, see https://git.coopcloud.tech/coop-cloud/recipes-catalogue-json/issues/6" | ||||||
|  |  | ||||||
|   _deploy_app |   _deploy_app | ||||||
|  |  | ||||||
|   run $ABRA app version "$TEST_APP_DOMAIN" |   run $ABRA app version "$TEST_APP_DOMAIN" | ||||||
| @ -92,7 +94,7 @@ teardown(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|  # NOTE(d1): to let the stack come down before nuking volumes |  # NOTE(d1): to let the stack come down before nuking volumes | ||||||
|   sleep 3 |   sleep 5 | ||||||
|  |  | ||||||
|   run $ABRA app volume remove "$appDomain" --no-input |   run $ABRA app volume remove "$appDomain" --no-input | ||||||
|   assert_success |   assert_success | ||||||
|  | |||||||
| @ -79,7 +79,7 @@ teardown(){ | |||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |   # NOTE(d1): to let the stack come down before nuking volumes | ||||||
|   sleep 5 |   sleep 10 | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force | ||||||
|   assert_success |   assert_success | ||||||
| @ -93,7 +93,7 @@ teardown(){ | |||||||
|   _undeploy_app |   _undeploy_app | ||||||
|  |  | ||||||
|   # NOTE(d1): to let the stack come down before nuking volumes |   # NOTE(d1): to let the stack come down before nuking volumes | ||||||
|   sleep 5 |   sleep 10 | ||||||
|  |  | ||||||
|   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force |   run $ABRA app volume rm "$TEST_APP_DOMAIN" --force | ||||||
|   assert_success |   assert_success | ||||||
|  | |||||||
| @ -1,17 +1,18 @@ | |||||||
| #!/usr/bin/env bash | #!/usr/bin/env bash | ||||||
|  |  | ||||||
| _new_app() { | _new_app() { | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" \ | ||||||
|     --secrets |     --secrets \ | ||||||
|  |     "$TEST_RECIPE" | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
| } | } | ||||||
|  |  | ||||||
| _deploy_app() { | _deploy_app() { | ||||||
|   run $ABRA app deploy "$TEST_APP_DOMAIN" --no-input |   run $ABRA app deploy --no-input "$TEST_APP_DOMAIN" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run $ABRA app ls --server "$TEST_SERVER" --status |   run $ABRA app ls --server "$TEST_SERVER" --status | ||||||
| @ -21,7 +22,7 @@ _deploy_app() { | |||||||
| } | } | ||||||
|  |  | ||||||
| _undeploy_app() { | _undeploy_app() { | ||||||
|   run $ABRA app undeploy "$TEST_APP_DOMAIN" --no-input |   run $ABRA app undeploy --no-input "$TEST_APP_DOMAIN" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run $ABRA app ls --server "$TEST_SERVER" --status |   run $ABRA app ls --server "$TEST_SERVER" --status | ||||||
| @ -34,10 +35,10 @@ _rm_app() { | |||||||
|   # NOTE(d1): not asserting outcomes on teardown here since some might fail |   # NOTE(d1): not asserting outcomes on teardown here since some might fail | ||||||
|   # depending on what the test created. all commands run through anyway |   # depending on what the test created. all commands run through anyway | ||||||
|   if [[ -f "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" ]]; then |   if [[ -f "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" ]]; then | ||||||
|     run $ABRA app undeploy "$TEST_APP_DOMAIN" --no-input |     run $ABRA app undeploy --no-input "$TEST_APP_DOMAIN" | ||||||
|     run $ABRA app secret remove "$TEST_APP_DOMAIN" --all --no-input |     run $ABRA app secret remove --all --no-input "$TEST_APP_DOMAIN" | ||||||
|     run $ABRA app volume remove "$TEST_APP_DOMAIN" --no-input |     run $ABRA app volume remove --no-input "$TEST_APP_DOMAIN" | ||||||
|     run $ABRA app remove "$TEST_APP_DOMAIN" --no-input |     run $ABRA app remove --no-input "$TEST_APP_DOMAIN" | ||||||
|   fi |   fi | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -46,10 +47,11 @@ _reset_app(){ | |||||||
|   assert_success |   assert_success | ||||||
|   assert_not_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_not_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
|  |  | ||||||
|   run $ABRA app new "$TEST_RECIPE" \ |   run $ABRA app new \ | ||||||
|     --no-input \ |     --no-input \ | ||||||
|     --server "$TEST_SERVER" \ |     --server "$TEST_SERVER" \ | ||||||
|     --domain "$TEST_APP_DOMAIN" \ |     --domain "$TEST_APP_DOMAIN" \ | ||||||
|  |     "$TEST_RECIPE" | ||||||
|   assert_success |   assert_success | ||||||
|   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" |   assert_exists "$ABRA_DIR/servers/$TEST_SERVER/$TEST_APP_DOMAIN.env" | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,6 +5,7 @@ _common_setup() { | |||||||
|   bats_load_library bats-assert |   bats_load_library bats-assert | ||||||
|   bats_load_library bats-file |   bats_load_library bats-file | ||||||
|  |  | ||||||
|  |   load "$PWD/tests/integration/helpers/file" | ||||||
|   load "$PWD/tests/integration/helpers/app" |   load "$PWD/tests/integration/helpers/app" | ||||||
|   load "$PWD/tests/integration/helpers/git" |   load "$PWD/tests/integration/helpers/git" | ||||||
|   load "$PWD/tests/integration/helpers/recipe" |   load "$PWD/tests/integration/helpers/recipe" | ||||||
|  | |||||||
							
								
								
									
										24
									
								
								tests/integration/helpers/file.bash
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								tests/integration/helpers/file.bash
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,24 @@ | |||||||
|  | _mkfile() { | ||||||
|  |   run bash -c "echo $2 > $1" | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
|  |  | ||||||
|  | _mkfile_remote() { | ||||||
|  |   run $ABRA app run "$TEST_APP_DOMAIN" app "bash -c \"echo $2 > $1\"" | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
|  |  | ||||||
|  | _mkdir() { | ||||||
|  |   run bash -c "mkdir -p $1" | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
|  |  | ||||||
|  | _rm() { | ||||||
|  |   run rm -rf "$1" | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
|  |  | ||||||
|  | _rm_remote() { | ||||||
|  |   run "$ABRA" app run "$TEST_APP_DOMAIN" app rm -rf "$1" | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
| @ -28,3 +28,10 @@ _reset_tags() { | |||||||
|   assert_success |   assert_success | ||||||
|   refute_output '0' |   refute_output '0' | ||||||
| } | } | ||||||
|  |  | ||||||
|  | _set_git_author() { | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" config --local user.email test@example.com | ||||||
|  |   assert_success | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" config --local user.name test | ||||||
|  |   assert_success | ||||||
|  | } | ||||||
|  | |||||||
| @ -11,7 +11,11 @@ _add_server() { | |||||||
| } | } | ||||||
|  |  | ||||||
| _rm_server() { | _rm_server() { | ||||||
|   run $ABRA server remove --no-input "$TEST_SERVER" |   if [[ "$TEST_SERVER" == "default" ]]; then | ||||||
|  |       run rm -rf "$ABRA_DIR/servers/default" | ||||||
|  |   else | ||||||
|  |       run $ABRA server remove --no-input "$TEST_SERVER" | ||||||
|  |   fi | ||||||
|   assert_success |   assert_success | ||||||
|   assert_not_exists "$ABRA_DIR/servers/$TEST_SERVER" |   assert_not_exists "$ABRA_DIR/servers/$TEST_SERVER" | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,7 +5,17 @@ setup() { | |||||||
|   _common_setup |   _common_setup | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "recipe fetch" { | @test "recipe fetch all" { | ||||||
|  |   run rm -rf "$ABRA_DIR/recipes/matrix-synapse" | ||||||
|  |   assert_success | ||||||
|  |   assert_not_exists "$ABRA_DIR/recipes/matrix-synapse" | ||||||
|  |  | ||||||
|  |   run $ABRA recipe fetch | ||||||
|  |   assert_success | ||||||
|  |   assert_exists "$ABRA_DIR/recipes/matrix-synapse" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @test "recipe fetch single recipe" { | ||||||
|   run rm -rf "$ABRA_DIR/recipes/matrix-synapse" |   run rm -rf "$ABRA_DIR/recipes/matrix-synapse" | ||||||
|   assert_success |   assert_success | ||||||
|   assert_not_exists "$ABRA_DIR/recipes/matrix-synapse" |   assert_not_exists "$ABRA_DIR/recipes/matrix-synapse" | ||||||
|  | |||||||
| @ -66,13 +66,13 @@ setup() { | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA recipe lint "$TEST_RECIPE" |   run $ABRA recipe lint "$TEST_RECIPE" | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
| @ -82,13 +82,13 @@ setup() { | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA recipe lint "$TEST_RECIPE" --offline |   run $ABRA recipe lint "$TEST_RECIPE" --offline | ||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,6 +15,11 @@ teardown_file(){ | |||||||
| setup(){ | setup(){ | ||||||
|   load "$PWD/tests/integration/helpers/common" |   load "$PWD/tests/integration/helpers/common" | ||||||
|   _common_setup |   _common_setup | ||||||
|  |   _set_git_author | ||||||
|  | } | ||||||
|  |  | ||||||
|  | teardown() { | ||||||
|  |   _reset_recipe | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "validate recipe argument" { | @test "validate recipe argument" { | ||||||
| @ -51,8 +56,6 @@ setup(){ | |||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" tag --list |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" tag --list | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial '0.2.1+1.21.6' |   assert_output --partial '0.2.1+1.21.6' | ||||||
|  |  | ||||||
|   _reset_recipe |  | ||||||
| } | } | ||||||
|  |  | ||||||
| # NOTE(d1): this test can't assert hardcoded versions since we upgrade a minor | # NOTE(d1): this test can't assert hardcoded versions since we upgrade a minor | ||||||
| @ -81,8 +84,6 @@ setup(){ | |||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" tag --list |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" tag --list | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --regexp '0\.3\.0\+1\.2.*' |   assert_output --regexp '0\.3\.0\+1\.2.*' | ||||||
|  |  | ||||||
|   _reset_recipe "$TEST_RECIPE" |  | ||||||
| } | } | ||||||
|  |  | ||||||
| @test "unknown files not committed" { | @test "unknown files not committed" { | ||||||
| @ -100,6 +101,21 @@ setup(){ | |||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rm foo |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" rm foo | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial "fatal: pathspec 'foo' did not match any files" |   assert_output --partial "fatal: pathspec 'foo' did not match any files" | ||||||
|  | } | ||||||
|   _reset_recipe |  | ||||||
|  | # NOTE: relies on 0.2.x being the last minor version | ||||||
|  | @test "release with next release note" { | ||||||
|  |   _mkfile "$ABRA_DIR/recipes/$TEST_RECIPE/release/next" "those are some release notes for the next release" | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" add release/next | ||||||
|  |   assert_success | ||||||
|  |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" commit -m "added some release notes" | ||||||
|  |   assert_success | ||||||
|  |  | ||||||
|  |   run $ABRA recipe release  "$TEST_RECIPE" --no-input  --minor | ||||||
|  |   assert_success | ||||||
|  |   assert_output --partial 'no -p/--publish passed, not publishing' | ||||||
|  |  | ||||||
|  |   assert_not_exists "$ABRA_DIR/recipes/$TEST_RECIPE/release/next" | ||||||
|  |   assert_exists "$ABRA_DIR/recipes/$TEST_RECIPE/release/0.3.0+1.21.0" | ||||||
|  |   assert_file_contains "$ABRA_DIR/recipes/$TEST_RECIPE/release/0.3.0+1.21.0" "those are some release notes for the next release" | ||||||
| } | } | ||||||
|  | |||||||
| @ -61,14 +61,14 @@ setup(){ | |||||||
|   assert_success |   assert_success | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   assert_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   run $ABRA recipe upgrade "$TEST_RECIPE" --no-input |   run $ABRA recipe upgrade "$TEST_RECIPE" --no-input | ||||||
|   assert_success |   assert_success | ||||||
|   assert_output --partial 'can upgrade service: app' |   assert_output --partial 'can upgrade service: app' | ||||||
|  |  | ||||||
|   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status |   run git -C "$ABRA_DIR/recipes/$TEST_RECIPE" status | ||||||
|   refute_output --partial 'behind 3' |   assert_output --regexp 'behind .* 3 commits' | ||||||
|  |  | ||||||
|   _reset_recipe |   _reset_recipe | ||||||
| } | } | ||||||
|  | |||||||
| @ -12,6 +12,8 @@ setup() { | |||||||
| } | } | ||||||
|  |  | ||||||
| @test "error if not present in catalogue" { | @test "error if not present in catalogue" { | ||||||
|  |   skip "known issue, see https://git.coopcloud.tech/coop-cloud/recipes-catalogue-json/issues/6" | ||||||
|  |  | ||||||
|   run $ABRA recipe versions "$TEST_RECIPE" |   run $ABRA recipe versions "$TEST_RECIPE" | ||||||
|   assert_failure |   assert_failure | ||||||
|   assert_output --partial "is not published on the catalogue" |   assert_output --partial "is not published on the catalogue" | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	