Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cli: uninstall command #725

Merged
merged 13 commits into from
Sep 23, 2021
46 changes: 45 additions & 1 deletion cli/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,11 @@ This repository contains a CLI tool for installing and operating [Consul](https:

## Installation & Setup
Currently the tool is not available on any releases page. Instead clone the repository and run `go build -o bin/consul-k8s`
and proceed to run the binary.
from this directory and proceed to run the binary.

## Commands
* [consul-k8s install](#consul-k8s-install)
* [consul-k8s uninstall](#consul-k8s-uninstall)

### consul-k8s install
This command installs Consul on a Kubernetes cluster. It allows `demo` and `secure` installations via preset configurations
Expand Down Expand Up @@ -81,3 +82,46 @@ Global Options:
Path to kubeconfig file. This is aliased as "-c".

```

### consul-k8s uninstall
This command uninstalls Consul on Kubernetes, while prompting whether to uninstall the release and whether to delete all
related resources such as PVCs, Secrets, and ServiceAccounts.

Get started with:
```bash
consul-k8s uninstall
```

```
Usage: consul-k8s uninstall [flags]
Uninstall Consul with options to delete data and resources associated with Consul installation.

Command Options:

-auto-approve
Skip approval prompt for uninstalling Consul. The default is false.

-name=<string>
Name of the installation. This can be used to uninstall and/or delete
the resources of a specific Helm release.

-namespace=<string>
Namespace for the Consul installation.

-timeout=<string>
Timeout to wait for uninstall. The default is 10m.

-wipe-data
When used in combination with -auto-approve, all persisted data (PVCs
and Secrets) from previous installations will be deleted. Only set this
to true when data from previous installations is no longer necessary.
The default is false.

Global Options:

-context=<string>
Kubernetes context to use.

-kubeconfig=<string>
Path to kubeconfig file. This is aliased as "-c".
```
34 changes: 34 additions & 0 deletions cli/cmd/common/utils.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
package common

import (
"fmt"
"os"
"strings"

"helm.sh/helm/v3/pkg/action"
helmCLI "helm.sh/helm/v3/pkg/cli"
"k8s.io/cli-runtime/pkg/genericclioptions"
)

// Abort returns true if the raw input string is not equal to "y" or "yes".
func Abort(raw string) bool {
confirmation := strings.TrimSuffix(raw, "\n")
if !(strings.ToLower(confirmation) == "y" || strings.ToLower(confirmation) == "yes") {
return true
}
return false
}

// InitActionConfig initializes a Helm Go SDK action configuration. This function currently uses a hack to override the
// namespace field that gets set in the K8s client set up by the SDK.
func InitActionConfig(actionConfig *action.Configuration, namespace string, settings *helmCLI.EnvSettings, logger action.DebugLog) (*action.Configuration, error) {
getter := settings.RESTClientGetter()
configFlags := getter.(*genericclioptions.ConfigFlags)
configFlags.Namespace = &namespace
err := actionConfig.Init(settings.RESTClientGetter(), namespace,
os.Getenv("HELM_DRIVER"), logger)
if err != nil {
return nil, fmt.Errorf("error setting up helm action configuration to find existing installations: %s", err)
}
return actionConfig, nil
}
168 changes: 83 additions & 85 deletions cli/cmd/install/install.go
Original file line number Diff line number Diff line change
Expand Up @@ -86,82 +86,80 @@ func (c *Command) init() {
}

c.set = flag.NewSets()
{
ndhanushkodi marked this conversation as resolved.
Show resolved Hide resolved
f := c.set.NewSet("Command Options")
f.BoolVar(&flag.BoolVar{
Name: flagNameAutoApprove,
Target: &c.flagAutoApprove,
Default: defaultAutoApprove,
Usage: "Skip confirmation prompt.",
})
f.BoolVar(&flag.BoolVar{
Name: flagNameDryRun,
Target: &c.flagDryRun,
Default: defaultDryRun,
Usage: "Run pre-install checks and display summary of installation.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameConfigFile,
Aliases: []string{"f"},
Target: &c.flagValueFiles,
Usage: "Path to a file to customize the installation, such as Consul Helm chart values file. Can be specified multiple times.",
})
f.StringVar(&flag.StringVar{
Name: flagNameNamespace,
Target: &c.flagNamespace,
Default: defaultNamespace,
Usage: "Namespace for the Consul installation.",
})
f.StringVar(&flag.StringVar{
Name: flagNamePreset,
Target: &c.flagPreset,
Default: defaultPreset,
Usage: fmt.Sprintf("Use an installation preset, one of %s. Defaults to none", strings.Join(presetList, ", ")),
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameSetValues,
Target: &c.flagSetValues,
Usage: "Set a value to customize. Can be specified multiple times. Supports Consul Helm chart values.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameFileValues,
Target: &c.flagFileValues,
Usage: "Set a value to customize via a file. The contents of the file will be set as the value. Can be " +
"specified multiple times. Supports Consul Helm chart values.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameSetStringValues,
Target: &c.flagSetStringValues,
Usage: "Set a string value to customize. Can be specified multiple times. Supports Consul Helm chart values.",
})
f.StringVar(&flag.StringVar{
Name: flagNameTimeout,
Target: &c.flagTimeout,
Default: defaultTimeout,
Usage: "Timeout to wait for installation to be ready.",
})
f.BoolVar(&flag.BoolVar{
Name: flagNameWait,
Target: &c.flagWait,
Default: defaultWait,
Usage: "Determines whether to wait for resources in installation to be ready before exiting command.",
})

f = c.set.NewSet("Global Options")
f.StringVar(&flag.StringVar{
Name: "kubeconfig",
Aliases: []string{"c"},
Target: &c.flagKubeConfig,
Default: "",
Usage: "Path to kubeconfig file.",
})
f.StringVar(&flag.StringVar{
Name: "context",
Target: &c.flagKubeContext,
Default: "",
Usage: "Kubernetes context to use.",
})
}
f := c.set.NewSet("Command Options")
f.BoolVar(&flag.BoolVar{
Name: flagNameAutoApprove,
Target: &c.flagAutoApprove,
Default: defaultAutoApprove,
Usage: "Skip confirmation prompt.",
})
f.BoolVar(&flag.BoolVar{
Name: flagNameDryRun,
Target: &c.flagDryRun,
Default: defaultDryRun,
Usage: "Run pre-install checks and display summary of installation.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameConfigFile,
Aliases: []string{"f"},
Target: &c.flagValueFiles,
Usage: "Path to a file to customize the installation, such as Consul Helm chart values file. Can be specified multiple times.",
})
f.StringVar(&flag.StringVar{
Name: flagNameNamespace,
Target: &c.flagNamespace,
Default: defaultNamespace,
Usage: "Namespace for the Consul installation.",
})
f.StringVar(&flag.StringVar{
Name: flagNamePreset,
Target: &c.flagPreset,
Default: defaultPreset,
Usage: fmt.Sprintf("Use an installation preset, one of %s. Defaults to none", strings.Join(presetList, ", ")),
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameSetValues,
Target: &c.flagSetValues,
Usage: "Set a value to customize. Can be specified multiple times. Supports Consul Helm chart values.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameFileValues,
Target: &c.flagFileValues,
Usage: "Set a value to customize via a file. The contents of the file will be set as the value. Can be " +
"specified multiple times. Supports Consul Helm chart values.",
})
f.StringSliceVar(&flag.StringSliceVar{
Name: flagNameSetStringValues,
Target: &c.flagSetStringValues,
Usage: "Set a string value to customize. Can be specified multiple times. Supports Consul Helm chart values.",
})
f.StringVar(&flag.StringVar{
Name: flagNameTimeout,
Target: &c.flagTimeout,
Default: defaultTimeout,
Usage: "Timeout to wait for installation to be ready.",
})
f.BoolVar(&flag.BoolVar{
Name: flagNameWait,
Target: &c.flagWait,
Default: defaultWait,
Usage: "Determines whether to wait for resources in installation to be ready before exiting command.",
})

f = c.set.NewSet("Global Options")
f.StringVar(&flag.StringVar{
Name: "kubeconfig",
Aliases: []string{"c"},
Target: &c.flagKubeConfig,
Default: "",
Usage: "Path to kubeconfig file.",
})
f.StringVar(&flag.StringVar{
Name: "context",
Target: &c.flagKubeContext,
Default: "",
Usage: "Kubernetes context to use.",
})

c.help = c.set.Help()

Expand All @@ -172,15 +170,16 @@ func (c *Command) init() {
func (c *Command) Run(args []string) int {
c.once.Do(c.init)

// The logger is initialized in main with the name cli. Here, we reset the name to install so log lines would be prefixed with install.
c.Log.ResetNamed("install")

defer func() {
if err := c.Close(); err != nil {
c.UI.Output(err.Error())
c.Log.Error(err.Error())
os.Exit(1)
}
}()

// The logger is initialized in main with the name cli. Here, we reset the name to install so log lines would be prefixed with install.
c.Log.ResetNamed("install")

if err := c.validateFlags(args); err != nil {
c.UI.Output(err.Error())
return 1
Expand Down Expand Up @@ -292,8 +291,7 @@ func (c *Command) Run(args []string) int {
c.UI.Output(err.Error(), terminal.WithErrorStyle())
return 1
}
confirmation = strings.TrimSuffix(confirmation, "\n")
if !(strings.ToLower(confirmation) == "y" || strings.ToLower(confirmation) == "yes") {
if common.Abort(confirmation) {
c.UI.Output("Install aborted. To learn how to customize your installation, run:\nconsul-k8s install --help", terminal.WithInfoStyle())
return 1
}
Expand All @@ -303,9 +301,9 @@ func (c *Command) Run(args []string) int {

// Setup action configuration for Helm Go SDK function calls.
actionConfig := new(action.Configuration)
if err := actionConfig.Init(settings.RESTClientGetter(), c.flagNamespace,
os.Getenv("HELM_DRIVER"), uiLogger); err != nil {
c.UI.Output(err.Error())
actionConfig, err = common.InitActionConfig(actionConfig, c.flagNamespace, settings, uiLogger)
if err != nil {
c.UI.Output(err.Error(), terminal.WithErrorStyle())
return 1
}

Expand Down
Loading