k9s/internal/view/actions.go

206 lines
4.1 KiB
Go

// SPDX-License-Identifier: Apache-2.0
// Copyright Authors of K9s
package view
import (
"errors"
"fmt"
"strings"
"github.com/derailed/k9s/internal/config"
"github.com/derailed/k9s/internal/ui"
"github.com/derailed/k9s/internal/ui/dialog"
"github.com/derailed/tcell/v2"
"github.com/rs/zerolog/log"
)
// AllScopes represents actions available for all views.
const AllScopes = "all"
// Runner represents a runnable action handler.
type Runner interface {
App() *App
GetSelectedItem() string
Aliases() map[string]struct{}
EnvFn() EnvFunc
}
func hasAll(scopes []string) bool {
for _, s := range scopes {
if s == AllScopes {
return true
}
}
return false
}
func includes(aliases []string, s string) bool {
for _, a := range aliases {
if a == s {
return true
}
}
return false
}
func inScope(scopes []string, aliases map[string]struct{}) bool {
if hasAll(scopes) {
return true
}
for _, s := range scopes {
if _, ok := aliases[s]; ok {
return ok
}
}
return false
}
func hotKeyActions(r Runner, aa ui.KeyActions) error {
hh := config.NewHotKeys()
for k, a := range aa {
if a.Opts.HotKey {
delete(aa, k)
}
}
var errs error
if err := hh.Load(r.App().Config.ContextHotkeysPath()); err != nil {
errs = errors.Join(errs, err)
}
for k, hk := range hh.HotKey {
key, err := asKey(hk.ShortCut)
if err != nil {
errs = errors.Join(errs, err)
continue
}
_, ok := aa[key]
if ok {
errs = errors.Join(errs, fmt.Errorf("duplicated hotkeys found for %q in %q", hk.ShortCut, k))
continue
}
command, err := r.EnvFn()().Substitute(hk.Command)
if err != nil {
log.Warn().Err(err).Msg("Invalid shortcut command")
continue
}
aa[key] = ui.NewKeyActionWithOpts(
hk.Description,
gotoCmd(r, command, "", !hk.KeepHistory),
ui.ActionOpts{
Shared: true,
HotKey: true,
},
)
}
return errs
}
func gotoCmd(r Runner, cmd, path string, clearStack bool) ui.ActionHandler {
return func(evt *tcell.EventKey) *tcell.EventKey {
r.App().gotoResource(cmd, path, clearStack)
return nil
}
}
func pluginActions(r Runner, aa ui.KeyActions) error {
pp := config.NewPlugins()
for k, a := range aa {
if a.Opts.Plugin {
delete(aa, k)
}
}
var errs error
if err := pp.Load(r.App().Config.ContextPluginsPath()); err != nil {
errs = errors.Join(errs, err)
}
aliases := r.Aliases()
for k, plugin := range pp.Plugins {
if !inScope(plugin.Scopes, aliases) {
continue
}
key, err := asKey(plugin.ShortCut)
if err != nil {
errs = errors.Join(errs, err)
continue
}
_, ok := aa[key]
if ok {
errs = errors.Join(errs, fmt.Errorf("duplicated plugin key found for %q in %q", plugin.ShortCut, k))
continue
}
aa[key] = ui.NewKeyActionWithOpts(
plugin.Description,
pluginAction(r, plugin),
ui.ActionOpts{
Visible: true,
Plugin: true,
})
}
return errs
}
func pluginAction(r Runner, p config.Plugin) ui.ActionHandler {
return func(evt *tcell.EventKey) *tcell.EventKey {
path := r.GetSelectedItem()
if path == "" {
return evt
}
if r.EnvFn() == nil {
return nil
}
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
}
args[i] = arg
}
cb := func() {
opts := shellOpts{
binary: p.Command,
background: p.Background,
pipes: p.Pipes,
args: args,
}
suspend, errChan, statusChan := run(r.App(), opts)
if !suspend {
r.App().Flash().Infof("Plugin command failed: %q", p.Description)
return
}
var errs error
for e := range errChan {
errs = errors.Join(errs, e)
}
if errs != nil {
r.App().cowCmd(errs.Error())
return
}
go func() {
for st := range statusChan {
r.App().Flash().Infof("Plugin command launched successfully: %q", st)
}
}()
}
if p.Confirm {
msg := fmt.Sprintf("Run?\n%s %s", p.Command, strings.Join(args, " "))
dialog.ShowConfirm(r.App().Styles.Dialog(), r.App().Content.Pages, "Confirm "+p.Description, msg, cb, func() {})
return nil
}
cb()
return nil
}
}