k9s/internal/view/helpers.go

217 lines
4.9 KiB
Go

package view
import (
"context"
"errors"
"fmt"
"strconv"
"strings"
"github.com/derailed/k9s/internal"
"github.com/derailed/k9s/internal/client"
"github.com/derailed/k9s/internal/config"
"github.com/derailed/k9s/internal/render"
"github.com/derailed/k9s/internal/ui"
"github.com/gdamore/tcell"
"github.com/rs/zerolog/log"
)
func k8sEnv(c *client.Config) Env {
ctx, err := c.CurrentContextName()
if err != nil {
ctx = render.NAValue
}
cluster, err := c.CurrentClusterName()
if err != nil {
cluster = render.NAValue
}
user, err := c.CurrentUserName()
if err != nil {
user = render.NAValue
}
groups, err := c.CurrentGroupNames()
if err != nil {
groups = []string{render.NAValue}
}
var cfg string
kcfg := c.Flags().KubeConfig
if kcfg != nil && *kcfg != "" {
cfg = *kcfg
}
return Env{
"CONTEXT": ctx,
"CLUSTER": cluster,
"USER": user,
"GROUPS": strings.Join(groups, ","),
"KUBECONFIG": cfg,
}
}
func defaultEnv(c *client.Config, path string, header render.Header, row render.Row) Env {
env := k8sEnv(c)
env["NAMESPACE"], env["NAME"] = client.Namespaced(path)
for _, col := range header.Columns(true) {
env["COL-"+col] = row.Fields[header.IndexOf(col, true)]
}
return env
}
func describeResource(app *App, model ui.Tabular, gvr, path string) {
ctx := context.Background()
ctx = context.WithValue(ctx, internal.KeyFactory, app.factory)
yaml, err := model.Describe(ctx, path)
if err != nil {
app.Flash().Errf("Describe command failed: %s", err)
return
}
details := NewDetails(app, "Describe", path, true).Update(yaml)
if err := app.inject(details); err != nil {
app.Flash().Err(err)
}
}
func showPodsWithLabels(app *App, path string, sel map[string]string) {
labels := make([]string, 0, len(sel))
for k, v := range sel {
labels = append(labels, fmt.Sprintf("%s=%s", k, v))
}
showPods(app, path, strings.Join(labels, ","), "")
}
func showPods(app *App, path, labelSel, fieldSel string) {
if err := app.switchNS(client.AllNamespaces); err != nil {
app.Flash().Err(err)
return
}
v := NewPod(client.NewGVR("v1/pods"))
v.SetContextFn(podCtx(app, path, labelSel, fieldSel))
v.GetTable().SetColorerFn(render.Pod{}.ColorerFunc())
ns, _ := client.Namespaced(path)
if err := app.Config.SetActiveNamespace(ns); err != nil {
log.Error().Err(err).Msg("Config NS set failed!")
}
if err := app.inject(v); err != nil {
app.Flash().Err(err)
}
}
func podCtx(app *App, path, labelSel, fieldSel string) ContextFunc {
return func(ctx context.Context) context.Context {
ctx = context.WithValue(ctx, internal.KeyPath, path)
ctx = context.WithValue(ctx, internal.KeyLabels, labelSel)
ns, _ := client.Namespaced(path)
mx := client.NewMetricsServer(app.factory.Client())
nmx, err := mx.FetchPodsMetrics(ctx, ns)
if err != nil {
log.Debug().Err(err).Msgf("No pods metrics")
}
ctx = context.WithValue(ctx, internal.KeyMetrics, nmx)
return context.WithValue(ctx, internal.KeyFields, fieldSel)
}
}
func extractApp(ctx context.Context) (*App, error) {
app, ok := ctx.Value(internal.KeyApp).(*App)
if !ok {
return nil, errors.New("No application found in context")
}
return app, nil
}
// AsKey maps a string representation of a key to a tcell key.
func asKey(key string) (tcell.Key, error) {
for k, v := range tcell.KeyNames {
if v == key {
return k, nil
}
}
return 0, fmt.Errorf("No matching key found %s", key)
}
// FwFQN returns a fully qualified ns/name:container id.
func fwFQN(po, co string) string {
return po + ":" + co
}
func isTCPPort(p string) bool {
return !strings.Contains(p, "UDP")
}
// ContainerID computes container ID based on ns/po/co.
func containerID(path, co string) string {
ns, n := client.Namespaced(path)
po := strings.Split(n, "-")[0]
return ns + "/" + po + ":" + co
}
// UrlFor computes fq url for a given benchmark configuration.
func urlFor(cfg config.BenchConfig, port string) string {
host := "localhost"
if cfg.HTTP.Host != "" {
host = cfg.HTTP.Host
}
path := "/"
if cfg.HTTP.Path != "" {
path = cfg.HTTP.Path
}
return "http://" + host + ":" + port + path
}
func fqn(ns, n string) string {
if ns == "" {
return n
}
return ns + "/" + n
}
func decorateCpuMemHeaderRows(app *App, data render.TableData) render.TableData {
for colIndex, header := range data.Header {
check := ""
if header.Name == "%CPU/L" {
check = "cpu"
}
if header.Name == "%MEM/L" {
check = "memory"
}
if len(check) == 0 {
continue
}
for _, re := range data.RowEvents {
if re.Row.Fields[colIndex] == render.NAValue {
continue
}
n, err := strconv.Atoi(re.Row.Fields[colIndex])
if err != nil {
continue
}
if n > 100 {
n = 100
}
severity := app.Config.K9s.Thresholds.LevelFor(check, n)
if severity == config.SeverityLow {
continue
}
color := app.Config.K9s.Thresholds.SeverityColor(check, n)
if len(color) > 0 {
re.Row.Fields[colIndex] = "[" + color + "::b]" + re.Row.Fields[colIndex]
}
}
}
return data
}