Merge pull request #311 from fridokus/master

Implement storageclass view
mine
Fernand Galiana 2019-09-20 06:46:46 -06:00 committed by GitHub
commit b87c6bc0b7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 250 additions and 1 deletions

View File

@ -167,6 +167,11 @@ var resMap = map[string]*meta.RESTMapping{
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "Secret"},
Scope: RestMapping,
},
"StorageClasses": {
Resource: schema.GroupVersionResource{Group: "storage.k8s.io", Version: "v1", Resource: "storageclass"},
GroupVersionKind: schema.GroupVersionKind{Group: "storage.k8s.io", Version: "v1", Kind: "StorageClass"},
Scope: RestMapping,
},
"ServiceAccounts": {
Resource: schema.GroupVersionResource{Group: "", Version: "v1", Resource: "serviceaccount"},
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "ServiceAccount"},

45
internal/k8s/sc.go Normal file
View File

@ -0,0 +1,45 @@
package k8s
import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
// StorageClass represents a Kubernetes StorageClass.
type StorageClass struct {
*base
Connection
}
// NewStorageClass returns a new StorageClass.
func NewStorageClass(c Connection) *StorageClass {
return &StorageClass{&base{}, c}
}
// Get a StorageClass.
func (p *StorageClass) Get(_, n string) (interface{}, error) {
return p.DialOrDie().StorageV1().StorageClasses().Get(n, metav1.GetOptions{})
}
// List all StorageClasses in a given namespace.
func (p *StorageClass) List(_ string) (Collection, error) {
opts := metav1.ListOptions{
LabelSelector: p.labelSelector,
FieldSelector: p.fieldSelector,
}
rr, err := p.DialOrDie().StorageV1().StorageClasses().List(opts)
if err != nil {
return nil, err
}
cc := make(Collection, len(rr.Items))
for i, r := range rr.Items {
cc[i] = r
}
return cc, nil
}
// Delete a StorageClass.
func (p *StorageClass) Delete(_, n string, cascade, force bool) error {
return p.DialOrDie().StorageV1().StorageClasses().Delete(n, nil)
}

87
internal/resource/sc.go Normal file
View File

@ -0,0 +1,87 @@
package resource
import (
"github.com/derailed/k9s/internal/k8s"
"github.com/rs/zerolog/log"
v1 "k8s.io/api/storage/v1"
)
// StorageClass tracks a kubernetes resource.
type StorageClass struct {
*Base
instance *v1.StorageClass
}
// NewStorageClassList returns a new resource list.
func NewStorageClassList(c Connection, ns string) List {
return NewList(
NotNamespaced,
"sc",
NewStorageClass(c),
CRUDAccess|DescribeAccess,
)
}
// NewStorageClass instantiates a new StorageClass.
func NewStorageClass(c Connection) *StorageClass {
p := &StorageClass{&Base{Connection: c, Resource: k8s.NewStorageClass(c)}, nil}
p.Factory = p
return p
}
// New builds a new StorageClass instance from a k8s resource.
func (r *StorageClass) New(i interface{}) Columnar {
c := NewStorageClass(r.Connection)
switch instance := i.(type) {
case *v1.StorageClass:
c.instance = instance
case v1.StorageClass:
c.instance = &instance
default:
log.Fatal().Msgf("unknown StorageClass type %#v", i)
}
c.path = c.namespacedName(c.instance.ObjectMeta)
return c
}
// Marshal resource to yaml.
func (r *StorageClass) Marshal(path string) (string, error) {
ns, n := Namespaced(path)
i, err := r.Resource.Get(ns, n)
if err != nil {
return "", err
}
sc := i.(*v1.StorageClass)
sc.TypeMeta.APIVersion = "storage.k8s.io/v1"
sc.TypeMeta.Kind = "StorageClass"
return r.marshalObject(sc)
}
// Header return resource header.
func (*StorageClass) Header(ns string) Row {
hh := Row{}
if ns == AllNamespaces {
hh = append(hh, "NAMESPACE")
}
return append(hh, "NAME", "PROVISIONER", "AGE")
}
// Fields retrieves displayable fields.
func (r *StorageClass) Fields(ns string) Row {
ff := make(Row, 0, len(r.Header(ns)))
i := r.instance
if ns == AllNamespaces {
ff = append(ff, i.Namespace)
}
return append(ff,
i.Name,
string(i.Provisioner),
toAge(i.ObjectMeta.CreationTimestamp),
)
}

View File

@ -0,0 +1,104 @@
package resource_test
import (
"testing"
"github.com/derailed/k9s/internal/k8s"
"github.com/derailed/k9s/internal/resource"
m "github.com/petergtz/pegomock"
"github.com/stretchr/testify/assert"
v1 "k8s.io/api/storage/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
func NewSCListWithArgs(ns string, r *resource.StorageClass) resource.List {
return resource.NewList(resource.NotNamespaced, "sc", r, resource.CRUDAccess|resource.DescribeAccess)
}
func NewSCWithArgs(conn k8s.Connection, res resource.Cruder) *resource.StorageClass {
r := &resource.StorageClass{Base: resource.NewBase(conn, res)}
r.Factory = r
return r
}
func TestSCListAccess(t *testing.T) {
mc := NewMockConnection()
mr := NewMockCruder()
ns := "blee"
l := NewSCListWithArgs(resource.AllNamespaces, NewSCWithArgs(mc, mr))
l.SetNamespace(ns)
assert.Equal(t, resource.NotNamespaced, l.GetNamespace())
assert.Equal(t, "sc", l.GetName())
for _, a := range []int{resource.GetAccess, resource.ListAccess, resource.DeleteAccess, resource.ViewAccess, resource.EditAccess} {
assert.True(t, l.Access(a))
}
}
func TestSCFields(t *testing.T) {
r := newSC().Fields("blee")
assert.Equal(t, "storage-test", r[0])
}
func TestSCMarshal(t *testing.T) {
mc := NewMockConnection()
mr := NewMockCruder()
m.When(mr.Get("blee", "storage-test")).ThenReturn(k8sSC(), nil)
cm := NewSCWithArgs(mc, mr)
ma, err := cm.Marshal("blee/storage-test")
mr.VerifyWasCalledOnce().Get("blee", "storage-test")
assert.Nil(t, err)
assert.Equal(t, scYaml(), ma)
}
func TestSCListData(t *testing.T) {
mc := NewMockConnection()
mr := NewMockCruder()
m.When(mr.List(resource.NotNamespaced)).ThenReturn(k8s.Collection{*k8sSC()}, nil)
l := NewSCListWithArgs("-", NewSCWithArgs(mc, mr))
// Make sure we mrn get deltas!
for i := 0; i < 2; i++ {
err := l.Reconcile(nil, nil)
assert.Nil(t, err)
}
mr.VerifyWasCalled(m.Times(2)).List(resource.NotNamespaced)
td := l.Data()
assert.Equal(t, 1, len(td.Rows))
assert.Equal(t, resource.NotNamespaced, l.GetNamespace())
row := td.Rows["storage-test"]
assert.Equal(t, 3, len(row.Deltas))
for _, d := range row.Deltas {
assert.Equal(t, "", d)
}
assert.Equal(t, resource.Row{"storage-test"}, row.Fields[:1])
}
// Helpers...
func k8sSC() *v1.StorageClass {
return &v1.StorageClass{
ObjectMeta: metav1.ObjectMeta{
Name: "storage-test",
CreationTimestamp: metav1.Time{Time: testTime()},
},
}
}
func newSC() resource.Columnar {
mc := NewMockConnection()
return resource.NewStorageClass(mc).New(k8sSC())
}
func scYaml() string {
return `apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
creationTimestamp: "2018-12-14T17:36:43Z"
name: storage-test
provisioner: ""
`
}

View File

@ -13,6 +13,6 @@ func TestAliasView(t *testing.T) {
v.Init(nil, "")
assert.Equal(t, 3, len(td.Header))
assert.Equal(t, 32, len(td.Rows))
assert.Equal(t, 33, len(td.Rows))
assert.Equal(t, "Aliases", v.getTitle())
}

View File

@ -174,6 +174,14 @@ func stateRes(m map[string]resCmd) {
viewFn: newSecretView,
listFn: resource.NewSecretList,
}
m["sc"] = resCmd{
title: "StorageClasses",
crdCmd: crdCmd{
api: "storage.k8s.io",
},
viewFn: newResourceView,
listFn: resource.NewStorageClassList,
}
}
func primRes(m map[string]resCmd) {