commit
b87c6bc0b7
|
|
@ -167,6 +167,11 @@ var resMap = map[string]*meta.RESTMapping{
|
||||||
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "Secret"},
|
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "Secret"},
|
||||||
Scope: RestMapping,
|
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": {
|
"ServiceAccounts": {
|
||||||
Resource: schema.GroupVersionResource{Group: "", Version: "v1", Resource: "serviceaccount"},
|
Resource: schema.GroupVersionResource{Group: "", Version: "v1", Resource: "serviceaccount"},
|
||||||
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "ServiceAccount"},
|
GroupVersionKind: schema.GroupVersionKind{Group: "", Version: "v1", Kind: "ServiceAccount"},
|
||||||
|
|
|
||||||
|
|
@ -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)
|
||||||
|
}
|
||||||
|
|
@ -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),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
@ -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: ""
|
||||||
|
`
|
||||||
|
}
|
||||||
|
|
@ -13,6 +13,6 @@ func TestAliasView(t *testing.T) {
|
||||||
v.Init(nil, "")
|
v.Init(nil, "")
|
||||||
|
|
||||||
assert.Equal(t, 3, len(td.Header))
|
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())
|
assert.Equal(t, "Aliases", v.getTitle())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -174,6 +174,14 @@ func stateRes(m map[string]resCmd) {
|
||||||
viewFn: newSecretView,
|
viewFn: newSecretView,
|
||||||
listFn: resource.NewSecretList,
|
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) {
|
func primRes(m map[string]resCmd) {
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue