package resource import ( "bytes" "context" "errors" "fmt" "path" "github.com/derailed/k9s/internal" "github.com/derailed/k9s/internal/k8s" "github.com/derailed/k9s/internal/watch" "github.com/rs/zerolog/log" v1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" "k8s.io/apimachinery/pkg/runtime" genericprinters "k8s.io/cli-runtime/pkg/printers" "k8s.io/kubectl/pkg/describe" "k8s.io/kubectl/pkg/describe/versioned" mv1beta1 "k8s.io/metrics/pkg/apis/metrics/v1beta1" ) // Base resource. type Base struct { Factory Connection Connection path string Resource Cruder } // NewBase returns a new base func NewBase(c Connection, r Cruder) *Base { return &Base{Connection: c, Resource: r} } // SetPodMetrics attach pod metrics to resource. func (b *Base) SetPodMetrics(*mv1beta1.PodMetrics) {} // SetNodeMetrics attach node metrics to resource. func (b *Base) SetNodeMetrics(*mv1beta1.NodeMetrics) {} // Name returns the resource namespaced name. func (b *Base) Name() string { return b.path } // NumCols designates if column is numerical. func (*Base) NumCols(n string) map[string]bool { return map[string]bool{} } // ExtFields returns extended fields in relation to headers. func (*Base) ExtFields() (TypeMeta, error) { return TypeMeta{}, errors.New("Base does not have extended fields.") } // Get a resource by name func (b *Base) Get(path string) (Columnar, error) { ns, n := Namespaced(path) i, err := b.Resource.Get(ns, n) if err != nil { return nil, err } return b.New(i) } // BOZO!! // List all resources // func (b *Base) List(ctx context.Context, ns string) (Columnars, error) { // ii, err := b.Resource.List(ctx, ns) // if err != nil { // return nil, err // } // cc := make(Columnars, 0, len(ii)) // for i := 0; i < len(ii); i++ { // res, err := b.New(ii[i]) // if err != nil { // return nil, err // } // cc = append(cc) // } // return cc, nil // } // BOZO!! // // List all resources // func (b *Base) List(ns string, opts metav1.ListOptions) (Columnars, error) { // ii, err := b.Resource.List(ns, opts) // if err != nil { // return nil, err // } // cc := make(Columnars, 0, len(ii)) // for i := 0; i < len(ii); i++ { // res, err := b.New(ii[i]) // if err != nil { // return nil, err // } // cc = append(cc, res) // } // return cc, nil // } // Describe a given resource. func (b *Base) Describe(gvr, pa string) (string, error) { mapper := k8s.RestMapper{Connection: b.Connection} m, err := mapper.ToRESTMapper() if err != nil { log.Error().Err(err).Msgf("No REST mapper for resource %s", gvr) return "", err } GVR := k8s.GVR(gvr) gvk, err := m.KindFor(GVR.AsGVR()) if err != nil { log.Error().Err(err).Msgf("No GVK for resource %s", gvr) return "", err } mapping, err := mapper.ResourceFor(GVR.ResName(), gvk.Kind) if err != nil { log.Error().Err(err).Msgf("Unable to find mapper for %s %s", gvr, pa) return "", err } ns, n := Namespaced(pa) d, err := versioned.Describer(b.Connection.Config().Flags(), mapping) if err != nil { log.Error().Err(err).Msgf("Unable to find describer for %#v", mapping) return "", err } return d.Describe(ns, n, describe.DescriberSettings{ShowEvents: true}) } // Delete a resource by name. func (b *Base) Delete(path string, cascade, force bool) error { ns, n := Namespaced(path) return b.Resource.Delete(ns, n, cascade, force) } func (*Base) namespacedName(m metav1.ObjectMeta) string { return path.Join(m.Namespace, m.Name) } func (*Base) marshalObject(o runtime.Object) (string, error) { var ( buff bytes.Buffer p genericprinters.YAMLPrinter ) err := p.PrintObj(o, &buff) if err != nil { log.Error().Msgf("Marshal Error %v", err) return "", err } return buff.String(), nil } func (b *Base) podLogs(ctx context.Context, c chan<- string, sel map[string]string, opts LogOptions) error { f, ok := ctx.Value(internal.KeyFactory).(*watch.Factory) if !ok { return fmt.Errorf("no factory in context for pod logs") } ls, err := metav1.ParseToLabelSelector(toSelector(sel)) if err != nil { return err } lsel, err := metav1.LabelSelectorAsSelector(ls) if err != nil { return err } inf := f.ForResource(opts.Namespace, "v1/pods") pods, err := inf.Lister().List(lsel) if err != nil { return err } if len(pods) > 1 { opts.MultiPods = true } pr := NewPod(b.Connection) for _, p := range pods { var po v1.Pod err := runtime.DefaultUnstructuredConverter.FromUnstructured(p.(*unstructured.Unstructured).Object, &po) if err != nil { // BOZO!! panic(err) } if po.Status.Phase == v1.PodRunning { opts.Namespace, opts.Name = po.Namespace, po.Name if err := pr.PodLogs(ctx, c, opts); err != nil { return err } } } return nil }