324 lines
7.1 KiB
Go
324 lines
7.1 KiB
Go
package resource
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"github.com/derailed/k9s/internal/k8s"
|
|
"github.com/rs/zerolog/log"
|
|
v1 "k8s.io/api/core/v1"
|
|
"k8s.io/apimachinery/pkg/util/sets"
|
|
)
|
|
|
|
const (
|
|
labelNodeRolePrefix = "node-role.kubernetes.io/"
|
|
nodeLabelRole = "kubernetes.io/role"
|
|
)
|
|
|
|
// Node tracks a kubernetes resource.
|
|
type Node struct {
|
|
*Base
|
|
instance *v1.Node
|
|
MetricsServer MetricsServer
|
|
metrics k8s.NodeMetrics
|
|
}
|
|
|
|
// NewNodeList returns a new resource list.
|
|
func NewNodeList(c Connection, mx MetricsServer, ns string) List {
|
|
return NewList(
|
|
NotNamespaced,
|
|
"no",
|
|
NewNode(c, mx),
|
|
ViewAccess|DescribeAccess,
|
|
)
|
|
}
|
|
|
|
// NewNode instantiates a new Node.
|
|
func NewNode(c Connection, mx MetricsServer) *Node {
|
|
n := &Node{&Base{Connection: c, Resource: k8s.NewNode(c)}, nil, mx, k8s.NodeMetrics{}}
|
|
n.Factory = n
|
|
|
|
return n
|
|
}
|
|
|
|
// New builds a new Node instance from a k8s resource.
|
|
func (r *Node) New(i interface{}) Columnar {
|
|
c := NewNode(r.Connection, r.MetricsServer)
|
|
switch instance := i.(type) {
|
|
case *v1.Node:
|
|
c.instance = instance
|
|
case v1.Node:
|
|
c.instance = &instance
|
|
default:
|
|
log.Fatal().Msgf("unknown Node type %#v", i)
|
|
}
|
|
c.path = c.namespacedName(c.instance.ObjectMeta)
|
|
|
|
return c
|
|
}
|
|
|
|
// List all resources for a given namespace.
|
|
func (r *Node) List(ns string) (Columnars, error) {
|
|
nn, err := r.Resource.List(ns)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
mx := make(k8s.NodesMetrics, len(nn))
|
|
if r.MetricsServer.HasMetrics() {
|
|
nmx, _ := r.MetricsServer.FetchNodesMetrics()
|
|
r.MetricsServer.NodesMetrics(nn, nmx, mx)
|
|
}
|
|
|
|
cc := make(Columnars, 0, len(nn))
|
|
for i := range nn {
|
|
node := nn[i].(v1.Node)
|
|
no := r.New(&node).(*Node)
|
|
no.metrics = mx[node.Name]
|
|
cc = append(cc, no)
|
|
}
|
|
|
|
return cc, nil
|
|
}
|
|
|
|
// Marshal a resource to yaml.
|
|
func (r *Node) Marshal(path string) (string, error) {
|
|
ns, n := namespaced(path)
|
|
i, err := r.Resource.Get(ns, n)
|
|
if err != nil {
|
|
log.Error().Err(err)
|
|
return "", err
|
|
}
|
|
|
|
no := i.(*v1.Node)
|
|
no.TypeMeta.APIVersion = "v1"
|
|
no.TypeMeta.Kind = "Node"
|
|
|
|
return r.marshalObject(no)
|
|
}
|
|
|
|
// Header returns resource header.
|
|
func (*Node) Header(ns string) Row {
|
|
return Row{
|
|
"NAME",
|
|
"STATUS",
|
|
"ROLE",
|
|
"VERSION",
|
|
"KERNEL",
|
|
"INTERNAL-IP",
|
|
"EXTERNAL-IP",
|
|
"CPU",
|
|
"MEM",
|
|
// "RCPU",
|
|
// "RMEM",
|
|
"ACPU",
|
|
"AMEM",
|
|
"AGE",
|
|
}
|
|
}
|
|
|
|
// Fields returns displayable fields.
|
|
func (r *Node) Fields(ns string) Row {
|
|
ff := make(Row, 0, len(r.Header(ns)))
|
|
|
|
i := r.instance
|
|
iIP, eIP := r.getIPs(i.Status.Addresses)
|
|
iIP, eIP = missing(iIP), missing(eIP)
|
|
|
|
// reqs, _, err := r.podsResources(i.Name)
|
|
// if err != nil {
|
|
// if !errors.IsForbidden(err) {
|
|
// log.Warn().Msgf("User is not authorized to list pods on nodes: %v", err)
|
|
// }
|
|
// log.Error().Msgf("%#v", err)
|
|
// }
|
|
|
|
// rcpu, rmem := reqs["cpu"], reqs["memory"]
|
|
// pcpur := toPerc(float64(rcpu.MilliValue()), float64(r.metrics.AvailCPU))
|
|
// pmemr := toPerc(k8s.ToMB(rmem.Value()), float64(r.metrics.AvailMEM))
|
|
|
|
return append(ff,
|
|
i.Name,
|
|
r.status(i.Status, i.Spec.Unschedulable),
|
|
r.nodeRoles(i),
|
|
i.Status.NodeInfo.KubeletVersion,
|
|
i.Status.NodeInfo.KernelVersion,
|
|
iIP,
|
|
eIP,
|
|
withPerc(
|
|
ToMillicore(r.metrics.CurrentCPU),
|
|
AsPerc(toPerc(float64(r.metrics.CurrentCPU), float64(r.metrics.AvailCPU))),
|
|
),
|
|
withPerc(
|
|
ToMi(r.metrics.CurrentMEM),
|
|
AsPerc(toPerc(r.metrics.CurrentMEM, r.metrics.AvailMEM)),
|
|
),
|
|
// withPerc(rcpu.String(), AsPerc(pcpur)),
|
|
// withPerc(rmem.String(), AsPerc(pmemr)),
|
|
ToMillicore(r.metrics.AvailCPU),
|
|
ToMi(r.metrics.AvailMEM),
|
|
toAge(i.ObjectMeta.CreationTimestamp),
|
|
)
|
|
}
|
|
|
|
func withPerc(v, p string) string {
|
|
return v + " (" + p + ")"
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// Helpers...
|
|
|
|
func (*Node) nodeRoles(node *v1.Node) string {
|
|
const (
|
|
labelNodeRolePrefix = "node-role.kubernetes.io/"
|
|
nodeLabelRole = "kubernetes.io/role"
|
|
)
|
|
|
|
roles := sets.NewString()
|
|
for k, v := range node.Labels {
|
|
switch {
|
|
case strings.HasPrefix(k, labelNodeRolePrefix):
|
|
if role := strings.TrimPrefix(k, labelNodeRolePrefix); len(role) > 0 {
|
|
roles.Insert(role)
|
|
}
|
|
|
|
case k == nodeLabelRole && v != "":
|
|
roles.Insert(v)
|
|
}
|
|
}
|
|
|
|
if len(roles) == 0 {
|
|
return MissingValue
|
|
}
|
|
|
|
return strings.Join(roles.List(), ",")
|
|
}
|
|
|
|
func (*Node) getIPs(addrs []v1.NodeAddress) (iIP, eIP string) {
|
|
for _, a := range addrs {
|
|
switch a.Type {
|
|
case v1.NodeExternalIP:
|
|
eIP = a.Address
|
|
case v1.NodeInternalIP:
|
|
iIP = a.Address
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (*Node) status(status v1.NodeStatus, exempt bool) string {
|
|
conditions := make(map[v1.NodeConditionType]*v1.NodeCondition)
|
|
for n := range status.Conditions {
|
|
cond := status.Conditions[n]
|
|
conditions[cond.Type] = &cond
|
|
}
|
|
|
|
var conds []string
|
|
validConditions := []v1.NodeConditionType{v1.NodeReady}
|
|
for _, validCondition := range validConditions {
|
|
condition, ok := conditions[validCondition]
|
|
if !ok {
|
|
continue
|
|
}
|
|
neg := ""
|
|
if condition.Status != v1.ConditionTrue {
|
|
neg = "Not"
|
|
}
|
|
conds = append(conds, neg+string(condition.Type))
|
|
|
|
}
|
|
if len(conds) == 0 {
|
|
conds = append(conds, "Unknown")
|
|
}
|
|
if exempt {
|
|
conds = append(conds, "SchedulingDisabled")
|
|
}
|
|
|
|
return strings.Join(conds, ",")
|
|
}
|
|
|
|
func findNodeRoles(i *v1.Node) []string {
|
|
roles := sets.NewString()
|
|
for k, v := range i.Labels {
|
|
switch {
|
|
case strings.HasPrefix(k, labelNodeRolePrefix):
|
|
if role := strings.TrimPrefix(k, labelNodeRolePrefix); len(role) > 0 {
|
|
roles.Insert(role)
|
|
}
|
|
case k == nodeLabelRole && v != "":
|
|
roles.Insert(v)
|
|
}
|
|
}
|
|
|
|
return roles.List()
|
|
}
|
|
|
|
func (r *Node) podsResources(name string) (v1.ResourceList, v1.ResourceList, error) {
|
|
reqs, limits := v1.ResourceList{}, v1.ResourceList{}
|
|
pods, err := r.Connection.NodePods(name)
|
|
if err != nil {
|
|
return reqs, limits, err
|
|
}
|
|
for _, p := range pods.Items {
|
|
preq, plim := podResources(&p)
|
|
for k, v := range preq {
|
|
if value, ok := reqs[k]; !ok {
|
|
reqs[k] = *v.Copy()
|
|
} else {
|
|
value.Add(v)
|
|
reqs[k] = value
|
|
}
|
|
}
|
|
for k, v := range plim {
|
|
if value, ok := limits[k]; !ok {
|
|
limits[k] = *v.Copy()
|
|
} else {
|
|
value.Add(v)
|
|
limits[k] = value
|
|
}
|
|
}
|
|
}
|
|
|
|
return reqs, limits, nil
|
|
}
|
|
|
|
func podResources(pod *v1.Pod) (v1.ResourceList, v1.ResourceList) {
|
|
reqs, limits := v1.ResourceList{}, v1.ResourceList{}
|
|
for _, container := range pod.Spec.Containers {
|
|
addResources(reqs, container.Resources.Requests)
|
|
addResources(limits, container.Resources.Limits)
|
|
}
|
|
// init containers define the minimum of any resource
|
|
for _, container := range pod.Spec.InitContainers {
|
|
maxResources(reqs, container.Resources.Requests)
|
|
maxResources(limits, container.Resources.Limits)
|
|
}
|
|
|
|
return reqs, limits
|
|
}
|
|
|
|
// AddResources adds the resources from l2 to l1.
|
|
func addResources(l1, l2 v1.ResourceList) {
|
|
for name, quantity := range l2 {
|
|
if value, ok := l1[name]; ok {
|
|
value.Add(quantity)
|
|
l1[name] = value
|
|
} else {
|
|
l1[name] = *quantity.Copy()
|
|
}
|
|
}
|
|
}
|
|
|
|
// MaxResourceList sets list to the greater of l1/l2 for every resource.
|
|
func maxResources(l1, l2 v1.ResourceList) {
|
|
for name, quantity := range l2 {
|
|
if value, ok := l1[name]; ok {
|
|
if quantity.Cmp(value) > 0 {
|
|
l1[name] = *quantity.Copy()
|
|
}
|
|
} else {
|
|
l1[name] = *quantity.Copy()
|
|
}
|
|
}
|
|
}
|