Merge pull request #688 from davidnemec/allow-plugins-with-confirm-dialog

Optionally allow plugin commands to have confirm dialog
mine
Fernand Galiana 2020-05-01 14:34:55 -06:00 committed by GitHub
commit c56616527d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 30 additions and 8 deletions

View File

@ -316,6 +316,7 @@ Entering the command mode and typing a resource name or alias, could be cumberso
K9s allows you to extend your command line and tooling by defining your very own cluster commands via plugins. K9s will look at `$HOME/.k9s/plugin.yml` to locate all available plugins. A plugin is defined as follows:
* Shortcut option represents the key combination a user would type to activate the plugin
* Confirm option (when enabled) lets you see the command that is going to be executed and gives you an option to confirm or prevent execution
* Description will be printed next to the shortcut in the k9s menu
* Scopes defines a collection of resources names/shortnames for the views associated with the plugin. You can specify `all` to provide this shortcut for all views.
* Command represents adhoc commands the plugin runs upon activation
@ -346,6 +347,7 @@ plugin:
# Defines a plugin to provide a `ctrl-l` shorcut to tail the logs while in pod view.
fred:
shortCut: Ctrl-L
confirm: false
description: Pod logs
scopes:
- pods

View File

@ -18,6 +18,7 @@ type Plugins struct {
// Plugin describes a K9s plugin
type Plugin struct {
ShortCut string `yaml:"shortCut"`
Confirm bool `yaml:"confirm"`
Scopes []string `yaml:"scopes"`
Description string `yaml:"description"`
Command string `yaml:"command"`

View File

@ -15,8 +15,10 @@ func TestPluginLoad(t *testing.T) {
k, ok := p.Plugin["blah"]
assert.True(t, ok)
assert.Equal(t, "shift-s", k.ShortCut)
assert.True(t, k.Confirm)
assert.Equal(t, "blee", k.Description)
assert.Equal(t, []string{"po", "dp"}, k.Scopes)
assert.Equal(t, "duh", k.Command)
assert.False(t, k.Background)
assert.Equal(t, []string{"-n", "$NAMESPACE", "-boolean"}, k.Args)
}

View File

@ -1,11 +1,13 @@
plugin:
blah:
shortCut: shift-s
confirm: true
description: blee
scopes:
- po
- dp
command: duh
background: false
args:
- -n
- $NAMESPACE

View File

@ -2,9 +2,11 @@ package view
import (
"fmt"
"strings"
"github.com/derailed/k9s/internal/config"
"github.com/derailed/k9s/internal/ui"
"github.com/derailed/k9s/internal/ui/dialog"
"github.com/gdamore/tcell"
"github.com/rs/zerolog/log"
)
@ -107,12 +109,12 @@ func pluginActions(r Runner, aa ui.KeyActions) {
}
aa[key] = ui.NewKeyAction(
plugin.Description,
execCmd(r, plugin.Command, plugin.Background, plugin.Args...),
pluginAction(r, plugin),
true)
}
}
func execCmd(r Runner, bin string, bg bool, args ...string) ui.ActionHandler {
func pluginAction(r Runner, p config.Plugin) ui.ActionHandler {
return func(evt *tcell.EventKey) *tcell.EventKey {
path := r.GetSelectedItem()
if path == "" {
@ -123,19 +125,28 @@ func execCmd(r Runner, bin string, bg bool, args ...string) ui.ActionHandler {
return nil
}
aa := make([]string, len(args))
for i, a := range args {
args := make([]string, len(p.Args))
for i, a := range p.Args {
arg, err := r.EnvFn()().Substitute(a)
if err != nil {
log.Error().Err(err).Msg("Plugin Args match failed")
return nil
}
aa[i] = arg
args[i] = arg
}
if run(r.App(), shellOpts{clear: true, binary: bin, background: bg, args: aa}) {
r.App().Flash().Info("Plugin command launched successfully!")
fn := func() {
if run(r.App(), shellOpts{clear: true, binary: p.Command, background: p.Background, args: args}) {
r.App().Flash().Info("Plugin command launched successfully!")
} else {
r.App().Flash().Info("Plugin command failed!")
}
}
if p.Confirm {
dialog.ShowConfirm(r.App().Content.Pages, "Confirm plugin action", fmt.Sprintf("Will execute:\n%s %s", p.Command, strings.Join(args, " ")), func() {
fn()
}, func() {})
} else {
r.App().Flash().Info("Plugin command failed!")
fn()
}
return nil

View File

@ -1,6 +1,7 @@
plugin:
dive:
shortCut: d
confirm: false
description: "Dive image"
scopes:
- containers

View File

@ -2,6 +2,7 @@ plugin:
# Suspends/Resumes a cronjob
toggleCronjob:
shortCut: Ctrl-S
confirm: true
scopes:
- cj
description: Toggle to suspend or resume a running cronjob

View File

@ -2,6 +2,7 @@ plugin:
# Sends logs over to jq for processing. This leverages kubectl plugin kubectl-jq.
jqlogs:
shortCut: Ctrl-J
confirm: false
description: "Logs (jq)"
scopes:
- po

View File

@ -2,6 +2,7 @@ plugin:
# Leverage stern (https://github.com/wercker/stern) to output logs.
stern:
shortCut: Ctrl-L
confirm: false
description: "Logs <Stern>"
scopes:
- pods