k3s/vendor/github.com/rancher/wrangler/pkg/objectset/objectset.go

198 lines
3.7 KiB
Go

package objectset
import (
"fmt"
"reflect"
"sort"
"github.com/pkg/errors"
"github.com/rancher/wrangler/pkg/gvk"
"github.com/rancher/wrangler/pkg/merr"
"k8s.io/apimachinery/pkg/api/meta"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
)
type ObjectKey struct {
Name string
Namespace string
}
func NewObjectKey(obj v1.Object) ObjectKey {
return ObjectKey{
Namespace: obj.GetNamespace(),
Name: obj.GetName(),
}
}
func (o ObjectKey) String() string {
if o.Namespace == "" {
return o.Name
}
return fmt.Sprintf("%s/%s", o.Namespace, o.Name)
}
type ObjectKeyByGVK map[schema.GroupVersionKind][]ObjectKey
type ObjectByGVK map[schema.GroupVersionKind]map[ObjectKey]runtime.Object
func (o ObjectByGVK) Add(obj runtime.Object) (schema.GroupVersionKind, error) {
metadata, err := meta.Accessor(obj)
if err != nil {
return schema.GroupVersionKind{}, err
}
gvk, err := gvk.Get(obj)
if err != nil {
return schema.GroupVersionKind{}, err
}
objs := o[gvk]
if objs == nil {
objs = map[ObjectKey]runtime.Object{}
o[gvk] = objs
}
objs[ObjectKey{
Namespace: metadata.GetNamespace(),
Name: metadata.GetName(),
}] = obj
return gvk, nil
}
type ObjectSet struct {
errs []error
objects ObjectByGVK
objectsByGK ObjectByGK
order []runtime.Object
gvkOrder []schema.GroupVersionKind
gvkSeen map[schema.GroupVersionKind]bool
}
func NewObjectSet(objs ...runtime.Object) *ObjectSet {
os := &ObjectSet{
objects: ObjectByGVK{},
objectsByGK: ObjectByGK{},
gvkSeen: map[schema.GroupVersionKind]bool{},
}
os.Add(objs...)
return os
}
func (o *ObjectSet) ObjectsByGVK() ObjectByGVK {
if o == nil {
return nil
}
return o.objects
}
func (o *ObjectSet) Contains(gk schema.GroupKind, key ObjectKey) bool {
_, ok := o.objectsByGK[gk][key]
return ok
}
func (o *ObjectSet) All() []runtime.Object {
return o.order
}
func (o *ObjectSet) Add(objs ...runtime.Object) *ObjectSet {
for _, obj := range objs {
o.add(obj)
}
return o
}
func (o *ObjectSet) add(obj runtime.Object) {
if obj == nil || reflect.ValueOf(obj).IsNil() {
return
}
gvk, err := o.objects.Add(obj)
if err != nil {
o.err(errors.Wrapf(err, "failed to add %T", obj))
return
}
_, err = o.objectsByGK.Add(obj)
if err != nil {
o.err(errors.Wrapf(err, "failed to add %T", obj))
return
}
o.order = append(o.order, obj)
if !o.gvkSeen[gvk] {
o.gvkSeen[gvk] = true
o.gvkOrder = append(o.gvkOrder, gvk)
}
}
func (o *ObjectSet) err(err error) error {
o.errs = append(o.errs, err)
return o.Err()
}
func (o *ObjectSet) AddErr(err error) {
o.errs = append(o.errs, err)
}
func (o *ObjectSet) Err() error {
return merr.NewErrors(o.errs...)
}
func (o *ObjectSet) Len() int {
return len(o.objects)
}
func (o *ObjectSet) GVKs() []schema.GroupVersionKind {
return o.GVKOrder()
}
func (o *ObjectSet) GVKOrder(known ...schema.GroupVersionKind) []schema.GroupVersionKind {
var rest []schema.GroupVersionKind
for _, gvk := range known {
if o.gvkSeen[gvk] {
continue
}
rest = append(rest, gvk)
}
sort.Slice(rest, func(i, j int) bool {
return rest[i].String() < rest[j].String()
})
return append(o.gvkOrder, rest...)
}
type ObjectByGK map[schema.GroupKind]map[ObjectKey]runtime.Object
func (o ObjectByGK) Add(obj runtime.Object) (schema.GroupKind, error) {
metadata, err := meta.Accessor(obj)
if err != nil {
return schema.GroupKind{}, err
}
gvk, err := gvk.Get(obj)
if err != nil {
return schema.GroupKind{}, err
}
gk := gvk.GroupKind()
objs := o[gk]
if objs == nil {
objs = map[ObjectKey]runtime.Object{}
o[gk] = objs
}
objs[ObjectKey{
Namespace: metadata.GetNamespace(),
Name: metadata.GetName(),
}] = obj
return gk, nil
}