k9s/internal/views/pod.go

245 lines
6.0 KiB
Go

package views
import (
"context"
"fmt"
"github.com/derailed/k9s/internal/k8s"
"github.com/derailed/k9s/internal/resource"
"github.com/derailed/k9s/internal/watch"
"github.com/gdamore/tcell"
"github.com/rs/zerolog/log"
v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
const containerFmt = "[fg:bg:b]%s([hilite:bg:b]%s[fg:bg:-])"
type podView struct {
*resourceView
childCancelFn context.CancelFunc
}
var _ updatable = &podView{}
type loggable interface {
appView() *appView
backFn() actionHandler
getSelection() string
getList() resource.List
switchPage(n string)
}
func newPodView(t string, app *appView, list resource.List) resourceViewer {
v := podView{resourceView: newResourceView(t, app, list).(*resourceView)}
{
v.extraActionsFn = v.extraActions
v.enterFn = v.listContainers
}
picker := newSelectList(&v)
{
picker.setActions(keyActions{
tcell.KeyEscape: {description: "Back", action: v.backCmd, visible: true},
})
}
v.AddPage("picker", picker, true, false)
v.AddPage("logs", newLogsView(list.GetName(), &v), true, false)
return &v
}
func (v *podView) extraActions(aa keyActions) {
aa[KeyL] = newKeyAction("Logs", v.logsCmd, true)
aa[KeyShiftL] = newKeyAction("Logs Previous", v.prevLogsCmd, true)
aa[KeyS] = newKeyAction("Shell", v.shellCmd, true)
aa[KeyShiftR] = newKeyAction("Sort Ready", v.sortColCmd(1, false), true)
aa[KeyShiftS] = newKeyAction("Sort Status", v.sortColCmd(2, true), true)
aa[KeyShiftT] = newKeyAction("Sort Restart", v.sortColCmd(3, false), true)
aa[KeyShiftC] = newKeyAction("Sort CPU", v.sortColCmd(4, false), true)
aa[KeyShiftM] = newKeyAction("Sort MEM", v.sortColCmd(5, false), true)
aa[KeyAltC] = newKeyAction("Sort CPU%", v.sortColCmd(6, false), true)
aa[KeyAltM] = newKeyAction("Sort MEM%", v.sortColCmd(7, false), true)
aa[KeyShiftD] = newKeyAction("Sort IP", v.sortColCmd(8, true), true)
aa[KeyShiftO] = newKeyAction("Sort Node", v.sortColCmd(9, true), true)
}
func (v *podView) listContainers(app *appView, _, res, sel string) {
if !v.rowSelected() {
return
}
po, err := v.app.informer.Get(watch.PodIndex, sel, metav1.GetOptions{})
if err != nil {
log.Error().Err(err).Msgf("Unable to retrieve pod %s", sel)
app.flash().errf("Unable to retrieve pods %s", err)
return
}
pod := po.(*v1.Pod)
mx := k8s.NewMetricsServer(app.conn())
list := resource.NewContainerList(app.conn(), mx, pod)
title := skinTitle(fmt.Sprintf(containerFmt, "Containers", sel), v.app.styles.Style)
// Stop my updater
if v.cancelFn != nil {
v.cancelFn()
}
// Span child view
cv := newContainerView(title, app, list, fqn(pod.Namespace, pod.Name), v.exitFn)
v.AddPage("containers", cv, true, true)
var ctx context.Context
ctx, v.childCancelFn = context.WithCancel(v.parentCtx)
cv.init(ctx, pod.Namespace)
}
func (v *podView) exitFn() {
if v.childCancelFn != nil {
v.childCancelFn()
}
v.RemovePage("containers")
v.switchPage("po")
v.restartUpdates()
}
// Protocol...
func (v *podView) backFn() actionHandler {
return v.backCmd
}
func (v *podView) appView() *appView {
return v.app
}
func (v *podView) getList() resource.List {
return v.list
}
func (v *podView) getSelection() string {
return v.selectedItem
}
// Handlers...
func (v *podView) logsCmd(evt *tcell.EventKey) *tcell.EventKey {
if v.viewLogs(false) {
return nil
}
return evt
}
func (v *podView) prevLogsCmd(evt *tcell.EventKey) *tcell.EventKey {
if v.viewLogs(true) {
return nil
}
return evt
}
func (v *podView) viewLogs(prev bool) bool {
if !v.rowSelected() {
return false
}
cc, err := fetchContainers(v.list, v.selectedItem, true)
if err != nil {
v.app.flash().errf("Unable to retrieve containers %s", err)
log.Error().Err(err)
return false
}
if len(cc) == 1 {
v.showLogs(v.selectedItem, cc[0], v.list.GetName(), v, prev)
return true
}
picker := v.GetPrimitive("picker").(*selectList)
picker.populate(cc)
picker.SetSelectedFunc(func(i int, t, d string, r rune) {
v.showLogs(v.selectedItem, t, "picker", picker, prev)
})
v.switchPage("picker")
return true
}
func (v *podView) showLogs(path, co, view string, parent loggable, prev bool) {
l := v.GetPrimitive("logs").(*logsView)
l.reload(co, parent, view, prev)
v.switchPage("logs")
}
func (v *podView) shellCmd(evt *tcell.EventKey) *tcell.EventKey {
if !v.rowSelected() {
return evt
}
cc, err := fetchContainers(v.list, v.selectedItem, false)
if err != nil {
v.app.flash().errf("Unable to retrieve containers %s", err)
log.Error().Msgf("Error fetching containers %v", err)
return evt
}
if len(cc) == 1 {
v.shellIn(v.selectedItem, "")
return nil
}
p := v.GetPrimitive("picker").(*selectList)
p.populate(cc)
p.SetSelectedFunc(func(i int, t, d string, r rune) {
v.shellIn(v.selectedItem, t)
})
v.switchPage("picker")
return evt
}
func (v *podView) shellIn(path, co string) {
v.stopUpdates()
shellIn(v.app, path, co)
v.restartUpdates()
}
func (v *podView) sortColCmd(col int, asc bool) func(evt *tcell.EventKey) *tcell.EventKey {
return func(evt *tcell.EventKey) *tcell.EventKey {
t := v.getTV()
t.sortCol.index, t.sortCol.asc = t.nameColIndex()+col, asc
t.refresh()
return nil
}
}
// ----------------------------------------------------------------------------
// Helpers...
func fetchContainers(l resource.List, po string, includeInit bool) ([]string, error) {
if len(po) == 0 {
return []string{}, nil
}
return l.Resource().(resource.Containers).Containers(po, includeInit)
}
func shellIn(a *appView, path, co string) {
args := computeShellArgs(path, co, a.config.K9s.CurrentContext, a.conn().Config().Flags().KubeConfig)
log.Debug().Msgf("Shell args %v", args)
runK(true, a, args...)
}
func computeShellArgs(path, co, context string, cfg *string) []string {
a := make([]string, 0, 15)
a = append(a, "exec", "-it")
a = append(a, "--context", context)
ns, po := namespaced(path)
a = append(a, "-n", ns)
a = append(a, po)
if cfg != nil && *cfg != "" {
a = append(a, "--kubeconfig", *cfg)
}
if co != "" {
a = append(a, "-c", co)
}
a = append(a, "--", "sh")
return a
}