ingress-nginx-helm/vendor/k8s.io/kubernetes/pkg/kubectl/cmd/util/clientcache.go

198 lines
6.4 KiB
Go
Raw Normal View History

2016-02-22 00:13:08 +00:00
/*
2016-07-12 03:42:47 +00:00
Copyright 2014 The Kubernetes Authors.
2016-02-22 00:13:08 +00:00
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package util
import (
2016-11-10 22:57:28 +00:00
"sync"
2016-06-21 18:58:43 +00:00
fed_clientset "k8s.io/kubernetes/federation/client/clientset_generated/federation_internalclientset"
2016-02-22 00:13:08 +00:00
"k8s.io/kubernetes/pkg/api/unversioned"
"k8s.io/kubernetes/pkg/apimachinery/registered"
2016-11-10 22:57:28 +00:00
"k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
2016-03-19 23:00:11 +00:00
"k8s.io/kubernetes/pkg/client/restclient"
2016-11-10 22:57:28 +00:00
"k8s.io/kubernetes/pkg/client/typed/discovery"
oldclient "k8s.io/kubernetes/pkg/client/unversioned"
2016-02-22 00:13:08 +00:00
"k8s.io/kubernetes/pkg/client/unversioned/clientcmd"
)
func NewClientCache(loader clientcmd.ClientConfig) *ClientCache {
return &ClientCache{
2016-11-10 22:57:28 +00:00
clientsets: make(map[unversioned.GroupVersion]*internalclientset.Clientset),
2016-06-21 18:58:43 +00:00
configs: make(map[unversioned.GroupVersion]*restclient.Config),
fedClientSets: make(map[unversioned.GroupVersion]fed_clientset.Interface),
loader: loader,
2016-02-22 00:13:08 +00:00
}
}
// ClientCache caches previously loaded clients for reuse, and ensures MatchServerVersion
// is invoked only once
type ClientCache struct {
loader clientcmd.ClientConfig
2016-11-10 22:57:28 +00:00
clientsets map[unversioned.GroupVersion]*internalclientset.Clientset
2016-06-21 18:58:43 +00:00
fedClientSets map[unversioned.GroupVersion]fed_clientset.Interface
2016-03-19 23:00:11 +00:00
configs map[unversioned.GroupVersion]*restclient.Config
2016-11-10 22:57:28 +00:00
matchVersion bool
defaultConfigLock sync.Mutex
defaultConfig *restclient.Config
discoveryClient discovery.DiscoveryInterface
2016-02-22 00:13:08 +00:00
}
2016-11-10 22:57:28 +00:00
// also looks up the discovery client. We can't do this during init because the flags won't have been set
// because this is constructed pre-command execution before the command tree is even set up
func (c *ClientCache) getDefaultConfig() (restclient.Config, discovery.DiscoveryInterface, error) {
c.defaultConfigLock.Lock()
defer c.defaultConfigLock.Unlock()
if c.defaultConfig != nil && c.discoveryClient != nil {
return *c.defaultConfig, c.discoveryClient, nil
2016-02-22 00:13:08 +00:00
}
2016-11-10 22:57:28 +00:00
config, err := c.loader.ClientConfig()
if err != nil {
return restclient.Config{}, nil, err
}
discoveryClient, err := discovery.NewDiscoveryClientForConfig(config)
if err != nil {
return restclient.Config{}, nil, err
}
if c.matchVersion {
if err := discovery.MatchesServerVersion(discoveryClient); err != nil {
return restclient.Config{}, nil, err
2016-02-22 00:13:08 +00:00
}
}
2016-11-10 22:57:28 +00:00
c.defaultConfig = config
c.discoveryClient = discoveryClient
return *c.defaultConfig, c.discoveryClient, nil
}
// ClientConfigForVersion returns the correct config for a server
func (c *ClientCache) ClientConfigForVersion(requiredVersion *unversioned.GroupVersion) (*restclient.Config, error) {
2016-02-22 00:13:08 +00:00
// TODO: have a better config copy method
2016-11-10 22:57:28 +00:00
config, discoveryClient, err := c.getDefaultConfig()
if err != nil {
return nil, err
}
if requiredVersion == nil && config.GroupVersion != nil {
// if someone has set the values via flags, our config will have the groupVersion set
// that means it is required.
requiredVersion = config.GroupVersion
}
2016-02-22 00:13:08 +00:00
2016-11-10 22:57:28 +00:00
// required version may still be nil, since config.GroupVersion may have been nil. Do the check
// before looking up from the cache
if requiredVersion != nil {
if config, ok := c.configs[*requiredVersion]; ok {
return config, nil
}
2016-02-22 00:13:08 +00:00
}
2016-11-10 22:57:28 +00:00
negotiatedVersion, err := discovery.NegotiateVersion(discoveryClient, requiredVersion, registered.EnabledVersions())
2016-02-22 00:13:08 +00:00
if err != nil {
return nil, err
}
config.GroupVersion = negotiatedVersion
2016-11-10 22:57:28 +00:00
// TODO this isn't what we want. Each clientset should be setting defaults as it sees fit.
oldclient.SetKubernetesDefaults(&config)
if requiredVersion != nil {
c.configs[*requiredVersion] = &config
2016-02-22 00:13:08 +00:00
}
// `version` does not necessarily equal `config.Version`. However, we know that we call this method again with
2016-08-10 18:53:55 +00:00
// `config.Version`, we should get the config we've just built.
2016-02-22 00:13:08 +00:00
configCopy := config
c.configs[*config.GroupVersion] = &configCopy
return &config, nil
}
2016-11-10 22:57:28 +00:00
// ClientSetForVersion initializes or reuses a clientset for the specified version, or returns an
2016-02-22 00:13:08 +00:00
// error if that is not possible
2016-11-10 22:57:28 +00:00
func (c *ClientCache) ClientSetForVersion(requiredVersion *unversioned.GroupVersion) (*internalclientset.Clientset, error) {
if requiredVersion != nil {
if clientset, ok := c.clientsets[*requiredVersion]; ok {
return clientset, nil
2016-02-22 00:13:08 +00:00
}
}
2016-11-10 22:57:28 +00:00
config, err := c.ClientConfigForVersion(requiredVersion)
2016-02-22 00:13:08 +00:00
if err != nil {
return nil, err
}
2016-11-10 22:57:28 +00:00
clientset, err := internalclientset.NewForConfig(config)
2016-02-22 00:13:08 +00:00
if err != nil {
return nil, err
}
2016-11-10 22:57:28 +00:00
c.clientsets[*config.GroupVersion] = clientset
2016-02-22 00:13:08 +00:00
// `version` does not necessarily equal `config.Version`. However, we know that if we call this method again with
// `version`, we should get a client based on the same config we just found. There's no guarantee that a client
// is copiable, so create a new client and save it in the cache.
2016-11-10 22:57:28 +00:00
if requiredVersion != nil {
2016-02-22 00:13:08 +00:00
configCopy := *config
2016-11-10 22:57:28 +00:00
clientset, err := internalclientset.NewForConfig(&configCopy)
2016-02-22 00:13:08 +00:00
if err != nil {
return nil, err
}
2016-11-10 22:57:28 +00:00
c.clientsets[*requiredVersion] = clientset
2016-02-22 00:13:08 +00:00
}
2016-11-10 22:57:28 +00:00
return clientset, nil
2016-02-22 00:13:08 +00:00
}
2016-06-21 18:58:43 +00:00
func (c *ClientCache) FederationClientSetForVersion(version *unversioned.GroupVersion) (fed_clientset.Interface, error) {
if version != nil {
if clientSet, found := c.fedClientSets[*version]; found {
return clientSet, nil
}
}
config, err := c.ClientConfigForVersion(version)
if err != nil {
return nil, err
}
// TODO: support multi versions of client with clientset
clientSet, err := fed_clientset.NewForConfig(config)
if err != nil {
return nil, err
}
c.fedClientSets[*config.GroupVersion] = clientSet
if version != nil {
configCopy := *config
clientSet, err := fed_clientset.NewForConfig(&configCopy)
if err != nil {
return nil, err
}
c.fedClientSets[*version] = clientSet
}
return clientSet, nil
}
func (c *ClientCache) FederationClientForVersion(version *unversioned.GroupVersion) (*restclient.RESTClient, error) {
fedClientSet, err := c.FederationClientSetForVersion(version)
if err != nil {
return nil, err
}
2016-11-10 22:57:28 +00:00
return fedClientSet.Federation().RESTClient().(*restclient.RESTClient), nil
2016-06-21 18:58:43 +00:00
}