1
0
mirror of https://github.com/rancher/norman.git synced 2025-06-28 16:27:25 +00:00

Allow discovery of API clients

This commit is contained in:
Darren Shepherd 2019-01-11 16:55:37 -07:00
parent ad8537c903
commit 1fd5af67a4
5 changed files with 148 additions and 43 deletions

View File

@ -204,6 +204,8 @@ func (c *Config) defaults(ctx context.Context, r *Runtime, opts Options) (contex
} }
} }
r.LocalConfig = c.Config
if c.ClientGetter == nil { if c.ClientGetter == nil {
cg, err := proxy.NewClientGetterFromConfig(*c.Config) cg, err := proxy.NewClientGetterFromConfig(*c.Config)
if err != nil { if err != nil {

View File

@ -1,21 +1,27 @@
package objectset package objectset
import ( import (
"github.com/rancher/norman/objectclient"
"github.com/rancher/norman/pkg/objectset/injectors" "github.com/rancher/norman/pkg/objectset/injectors"
"github.com/rancher/norman/types" "github.com/rancher/norman/types"
"k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema" "k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/client-go/discovery"
"k8s.io/client-go/rest"
) )
type DesiredSet struct { type DesiredSet struct {
remove bool discoveredClients map[schema.GroupVersionKind]*objectclient.ObjectClient
setID string discovery discovery.DiscoveryInterface
objs *ObjectSet restConfig rest.Config
codeVersion string remove bool
clients map[schema.GroupVersionKind]Client setID string
owner runtime.Object objs *ObjectSet
injectors []injectors.ConfigInjector codeVersion string
errs []error clients map[schema.GroupVersionKind]Client
owner runtime.Object
injectors []injectors.ConfigInjector
errs []error
} }
func (o *DesiredSet) AddInjector(inj injectors.ConfigInjector) { func (o *DesiredSet) AddInjector(inj injectors.ConfigInjector) {

View File

@ -7,15 +7,15 @@ import (
"sort" "sort"
"sync" "sync"
"k8s.io/apimachinery/pkg/selection"
"k8s.io/client-go/util/flowcontrol"
"github.com/pkg/errors" "github.com/pkg/errors"
errors2 "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta" "k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema" "k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/apimachinery/pkg/selection"
"k8s.io/client-go/util/flowcontrol"
) )
const ( const (
@ -27,8 +27,7 @@ const (
) )
var ( var (
ErrObjectSetDelay = errors.New("delaying object set apply") hashOrder = []string{
hashOrder = []string{
LabelID, LabelID,
LabelGVK, LabelGVK,
LabelName, LabelName,
@ -48,7 +47,7 @@ func (o *DesiredSet) getRateLimit(inputID string) flowcontrol.RateLimiter {
} else { } else {
rl = rls[inputID] rl = rls[inputID]
if rl == nil { if rl == nil {
rl = flowcontrol.NewTokenBucketRateLimiter(1.0/60.0, 10) rl = flowcontrol.NewTokenBucketRateLimiter(4.0/60.0, 10)
rls[inputID] = rl rls[inputID] = rl
} }
} }
@ -68,7 +67,7 @@ func (o *DesiredSet) Apply() error {
rl := o.getRateLimit(labelSet[LabelHash]) rl := o.getRateLimit(labelSet[LabelHash])
if rl != nil && !rl.TryAccept() { if rl != nil && !rl.TryAccept() {
return ErrObjectSetDelay return errors2.NewConflict(schema.GroupResource{}, o.setID, errors.New("delaying object set"))
} }
inputID := o.inputID(labelSet[LabelHash]) inputID := o.inputID(labelSet[LabelHash])

View File

@ -4,13 +4,17 @@ import (
"fmt" "fmt"
"sort" "sort"
errors2 "k8s.io/apimachinery/pkg/api/errors"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/rancher/norman/controller"
"github.com/rancher/norman/objectclient"
"github.com/rancher/norman/objectclient/dynamic"
"github.com/rancher/norman/restwatch"
"github.com/rancher/norman/types" "github.com/rancher/norman/types"
"github.com/sirupsen/logrus" "github.com/sirupsen/logrus"
errors2 "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta" "k8s.io/apimachinery/pkg/api/meta"
"k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema" "k8s.io/apimachinery/pkg/runtime/schema"
@ -21,16 +25,60 @@ var (
deletePolicy = v1.DeletePropagationBackground deletePolicy = v1.DeletePropagationBackground
) )
func (o *DesiredSet) process(inputID, debugID string, set labels.Selector, gvk schema.GroupVersionKind, objs map[objectKey]runtime.Object) { func (o *DesiredSet) getControllerAndObjectClient(debugID string, gvk schema.GroupVersionKind) (controller.GenericController, *objectclient.ObjectClient, error) {
client, ok := o.clients[gvk] client, ok := o.clients[gvk]
if !ok { if !ok && o.discovery == nil {
o.err(fmt.Errorf("failed to find client for %s for %s", gvk, debugID)) return nil, nil, fmt.Errorf("failed to find client for %s for %s", gvk, debugID)
}
if client != nil {
return client.Generic(), client.ObjectClient(), nil
}
objectClient := o.discoveredClients[gvk]
if objectClient != nil {
return nil, objectClient, nil
}
resources, err := o.discovery.ServerResourcesForGroupVersion(gvk.GroupVersion().String())
if err != nil {
return nil, nil, err
}
for _, resource := range resources.APIResources {
if resource.Kind != gvk.Kind {
continue
}
restConfig := o.restConfig
if restConfig.NegotiatedSerializer == nil {
restConfig.NegotiatedSerializer = dynamic.NegotiatedSerializer
}
restClient, err := restwatch.UnversionedRESTClientFor(&restConfig)
if err != nil {
return nil, nil, err
}
objectClient := objectclient.NewObjectClient("", restClient, &resource, gvk, &objectclient.UnstructuredObjectFactory{})
if o.discoveredClients == nil {
o.discoveredClients = map[schema.GroupVersionKind]*objectclient.ObjectClient{}
}
o.discoveredClients[gvk] = objectClient
return nil, objectClient, nil
}
return nil, nil, fmt.Errorf("failed to discover client for %s for %s", gvk, debugID)
}
func (o *DesiredSet) process(inputID, debugID string, set labels.Selector, gvk schema.GroupVersionKind, objs map[objectKey]runtime.Object) {
controller, objectClient, err := o.getControllerAndObjectClient(debugID, gvk)
if err != nil {
o.err(err)
return return
} }
indexer := client.Generic().Informer().GetIndexer() existing, err := list(controller, objectClient, set)
existing, err := list(indexer, set)
if err != nil { if err != nil {
o.err(fmt.Errorf("failed to list %s for %s", gvk, debugID)) o.err(fmt.Errorf("failed to list %s for %s", gvk, debugID))
return return
@ -45,10 +93,10 @@ func (o *DesiredSet) process(inputID, debugID string, set labels.Selector, gvk s
continue continue
} }
_, err = client.ObjectClient().Create(obj) _, err = objectClient.Create(obj)
if errors2.IsAlreadyExists(err) { if errors2.IsAlreadyExists(err) {
// Taking over an object that wasn't previously managed by us // Taking over an object that wasn't previously managed by us
existingObj, err := client.ObjectClient().GetNamespaced(k.namespace, k.name, v1.GetOptions{}) existingObj, err := objectClient.GetNamespaced(k.namespace, k.name, v1.GetOptions{})
if err == nil { if err == nil {
toUpdate = append(toUpdate, k) toUpdate = append(toUpdate, k)
existing[k] = existingObj existing[k] = existingObj
@ -63,7 +111,7 @@ func (o *DesiredSet) process(inputID, debugID string, set labels.Selector, gvk s
} }
for _, k := range toUpdate { for _, k := range toUpdate {
err := o.compareObjects(client.ObjectClient(), debugID, inputID, existing[k], objs[k], len(toCreate) > 0 || len(toDelete) > 0) err := o.compareObjects(objectClient, debugID, inputID, existing[k], objs[k], len(toCreate) > 0 || len(toDelete) > 0)
if err != nil { if err != nil {
o.err(errors.Wrapf(err, "failed to update %s %s for %s", k, gvk, debugID)) o.err(errors.Wrapf(err, "failed to update %s %s for %s", k, gvk, debugID))
continue continue
@ -71,7 +119,7 @@ func (o *DesiredSet) process(inputID, debugID string, set labels.Selector, gvk s
} }
for _, k := range toDelete { for _, k := range toDelete {
err := client.ObjectClient().DeleteNamespaced(k.namespace, k.name, &v1.DeleteOptions{ err := objectClient.DeleteNamespaced(k.namespace, k.name, &v1.DeleteOptions{
PropagationPolicy: &deletePolicy, PropagationPolicy: &deletePolicy,
}) })
if err != nil { if err != nil {
@ -110,23 +158,55 @@ func sortObjectKeys(keys []objectKey) {
}) })
} }
func list(indexer cache.Indexer, selector labels.Selector) (map[objectKey]runtime.Object, error) { func addObjectToMap(objs map[objectKey]runtime.Object, obj interface{}) error {
metadata, err := meta.Accessor(obj)
if err != nil {
return err
}
objs[objectKey{
namespace: metadata.GetNamespace(),
name: metadata.GetName(),
}] = obj.(runtime.Object)
return nil
}
func list(controller controller.GenericController, objectClient *objectclient.ObjectClient, selector labels.Selector) (map[objectKey]runtime.Object, error) {
var ( var (
errs []error errs []error
objs = map[objectKey]runtime.Object{} objs = map[objectKey]runtime.Object{}
) )
err := cache.ListAllByNamespace(indexer, "", selector, func(obj interface{}) { if controller == nil {
metadata, err := meta.Accessor(obj) objList, err := objectClient.List(v1.ListOptions{
LabelSelector: selector.String(),
})
if err != nil { if err != nil {
errs = append(errs, err) return nil, err
return
} }
objs[objectKey{ list, ok := objList.(*unstructured.UnstructuredList)
namespace: metadata.GetNamespace(), if !ok {
name: metadata.GetName(), return nil, fmt.Errorf("invalid list type %T", objList)
}] = obj.(runtime.Object) }
if err != nil {
return nil, err
}
for _, obj := range list.Items {
if err := addObjectToMap(objs, obj); err != nil {
errs = append(errs, err)
}
}
return objs, nil
}
err := cache.ListAllByNamespace(controller.Informer().GetIndexer(), "", selector, func(obj interface{}) {
if err := addObjectToMap(objs, obj); err != nil {
errs = append(errs, err)
}
}) })
if err != nil { if err != nil {
errs = append(errs, err) errs = append(errs, err)

View File

@ -5,6 +5,8 @@ import (
"github.com/rancher/norman/objectclient" "github.com/rancher/norman/objectclient"
"k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema" "k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/client-go/discovery"
"k8s.io/client-go/rest"
) )
type Client interface { type Client interface {
@ -13,24 +15,38 @@ type Client interface {
} }
type Processor struct { type Processor struct {
setID string setID string
codeVersion string codeVersion string
clients map[schema.GroupVersionKind]Client discovery discovery.DiscoveryInterface
restConfig rest.Config
allowSlowPath string
slowClient rest.HTTPClient
clients map[schema.GroupVersionKind]Client
} }
func NewProcessor(setID string) Processor { func NewProcessor(setID string) *Processor {
return Processor{ return &Processor{
setID: setID, setID: setID,
clients: map[schema.GroupVersionKind]Client{}, clients: map[schema.GroupVersionKind]Client{},
} }
} }
func (t Processor) CodeVersion(version string) Processor { func (t *Processor) SetID() string {
return t.setID
}
func (t *Processor) CodeVersion(version string) *Processor {
t.codeVersion = version t.codeVersion = version
return t return t
} }
func (t Processor) Client(clients ...Client) Processor { func (t *Processor) AllowDiscovery(discovery discovery.DiscoveryInterface, restConfig rest.Config) *Processor {
t.discovery = discovery
t.restConfig = restConfig
return t
}
func (t *Processor) Client(clients ...Client) *Processor {
// ensure cache is enabled // ensure cache is enabled
for _, client := range clients { for _, client := range clients {
client.Generic() client.Generic()
@ -50,6 +66,8 @@ func (t Processor) NewDesiredSet(owner runtime.Object, objs *ObjectSet) *Desired
objs = &ObjectSet{} objs = &ObjectSet{}
} }
return &DesiredSet{ return &DesiredSet{
discovery: t.discovery,
restConfig: t.restConfig,
remove: remove, remove: remove,
objs: objs, objs: objs,
setID: t.setID, setID: t.setID,