mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-08 03:33:56 +00:00
use discovery restmapper for kubectl
This commit is contained in:
parent
0a62dab566
commit
500cddc5c3
@ -526,7 +526,7 @@ func (m *DefaultRESTMapper) RESTMapping(gk unversioned.GroupKind, versions ...st
|
|||||||
|
|
||||||
interfaces, err := m.interfacesFunc(gvk.GroupVersion())
|
interfaces, err := m.interfacesFunc(gvk.GroupVersion())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("the provided version %q has no relevant versions", gvk.GroupVersion().String())
|
return nil, fmt.Errorf("the provided version %q has no relevant versions: %v", gvk.GroupVersion().String(), err)
|
||||||
}
|
}
|
||||||
|
|
||||||
retVal := &RESTMapping{
|
retVal := &RESTMapping{
|
||||||
@ -565,7 +565,7 @@ func (m *DefaultRESTMapper) RESTMappings(gk unversioned.GroupKind) ([]*RESTMappi
|
|||||||
|
|
||||||
interfaces, err := m.interfacesFunc(gvk.GroupVersion())
|
interfaces, err := m.interfacesFunc(gvk.GroupVersion())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("the provided version %q has no relevant versions", gvk.GroupVersion().String())
|
return nil, fmt.Errorf("the provided version %q has no relevant versions: %v", gvk.GroupVersion().String(), err)
|
||||||
}
|
}
|
||||||
|
|
||||||
mappings = append(mappings, &RESTMapping{
|
mappings = append(mappings, &RESTMapping{
|
||||||
|
@ -111,6 +111,7 @@ var (
|
|||||||
EnableVersions = DefaultAPIRegistrationManager.EnableVersions
|
EnableVersions = DefaultAPIRegistrationManager.EnableVersions
|
||||||
RegisterGroup = DefaultAPIRegistrationManager.RegisterGroup
|
RegisterGroup = DefaultAPIRegistrationManager.RegisterGroup
|
||||||
RegisterVersions = DefaultAPIRegistrationManager.RegisterVersions
|
RegisterVersions = DefaultAPIRegistrationManager.RegisterVersions
|
||||||
|
InterfacesFor = DefaultAPIRegistrationManager.InterfacesFor
|
||||||
)
|
)
|
||||||
|
|
||||||
// RegisterVersions adds the given group versions to the list of registered group versions.
|
// RegisterVersions adds the given group versions to the list of registered group versions.
|
||||||
@ -265,6 +266,15 @@ func (m *APIRegistrationManager) AddThirdPartyAPIGroupVersions(gvs ...unversione
|
|||||||
return skippedGVs
|
return skippedGVs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InterfacesFor is a union meta.VersionInterfacesFunc func for all registered types
|
||||||
|
func (m *APIRegistrationManager) InterfacesFor(version unversioned.GroupVersion) (*meta.VersionInterfaces, error) {
|
||||||
|
groupMeta, err := m.Group(version.Group)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return groupMeta.InterfacesFor(version)
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: This is an expedient function, because we don't check if a Group is
|
// TODO: This is an expedient function, because we don't check if a Group is
|
||||||
// supported throughout the code base. We will abandon this function and
|
// supported throughout the code base. We will abandon this function and
|
||||||
// checking the error returned by the Group() function.
|
// checking the error returned by the Group() function.
|
||||||
|
@ -32,6 +32,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/blang/semver"
|
||||||
"github.com/emicklei/go-restful/swagger"
|
"github.com/emicklei/go-restful/swagger"
|
||||||
"github.com/imdario/mergo"
|
"github.com/imdario/mergo"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
@ -47,9 +48,9 @@ import (
|
|||||||
"k8s.io/kubernetes/pkg/apimachinery"
|
"k8s.io/kubernetes/pkg/apimachinery"
|
||||||
"k8s.io/kubernetes/pkg/apimachinery/registered"
|
"k8s.io/kubernetes/pkg/apimachinery/registered"
|
||||||
"k8s.io/kubernetes/pkg/apis/apps"
|
"k8s.io/kubernetes/pkg/apis/apps"
|
||||||
"k8s.io/kubernetes/pkg/apis/autoscaling"
|
|
||||||
"k8s.io/kubernetes/pkg/apis/batch"
|
"k8s.io/kubernetes/pkg/apis/batch"
|
||||||
"k8s.io/kubernetes/pkg/apis/extensions"
|
"k8s.io/kubernetes/pkg/apis/extensions"
|
||||||
|
extensionsv1beta1 "k8s.io/kubernetes/pkg/apis/extensions/v1beta1"
|
||||||
"k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
|
"k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
|
||||||
coreclient "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset/typed/core/unversioned"
|
coreclient "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset/typed/core/unversioned"
|
||||||
"k8s.io/kubernetes/pkg/client/restclient"
|
"k8s.io/kubernetes/pkg/client/restclient"
|
||||||
@ -278,8 +279,6 @@ func makeInterfacesFor(versionList []unversioned.GroupVersion) func(version unve
|
|||||||
// if optionalClientConfig is nil, then flags will be bound to a new clientcmd.ClientConfig.
|
// if optionalClientConfig is nil, then flags will be bound to a new clientcmd.ClientConfig.
|
||||||
// if optionalClientConfig is not nil, then this factory will make use of it.
|
// if optionalClientConfig is not nil, then this factory will make use of it.
|
||||||
func NewFactory(optionalClientConfig clientcmd.ClientConfig) *Factory {
|
func NewFactory(optionalClientConfig clientcmd.ClientConfig) *Factory {
|
||||||
mapper := kubectl.ShortcutExpander{RESTMapper: registered.RESTMapper()}
|
|
||||||
|
|
||||||
flags := pflag.NewFlagSet("", pflag.ContinueOnError)
|
flags := pflag.NewFlagSet("", pflag.ContinueOnError)
|
||||||
flags.SetNormalizeFunc(utilflag.WarnWordSepNormalizeFunc) // Warn for "_" flags
|
flags.SetNormalizeFunc(utilflag.WarnWordSepNormalizeFunc) // Warn for "_" flags
|
||||||
|
|
||||||
@ -294,8 +293,6 @@ func NewFactory(optionalClientConfig clientcmd.ClientConfig) *Factory {
|
|||||||
clients: clients,
|
clients: clients,
|
||||||
flags: flags,
|
flags: flags,
|
||||||
|
|
||||||
// If discoverDynamicAPIs is true, make API calls to the discovery service to find APIs that
|
|
||||||
// have been dynamically added to the apiserver
|
|
||||||
Object: func(discoverDynamicAPIs bool) (meta.RESTMapper, runtime.ObjectTyper) {
|
Object: func(discoverDynamicAPIs bool) (meta.RESTMapper, runtime.ObjectTyper) {
|
||||||
cfg, err := clientConfig.ClientConfig()
|
cfg, err := clientConfig.ClientConfig()
|
||||||
checkErrWithPrefix("failed to get client config: ", err)
|
checkErrWithPrefix("failed to get client config: ", err)
|
||||||
@ -303,82 +300,33 @@ func NewFactory(optionalClientConfig clientcmd.ClientConfig) *Factory {
|
|||||||
if cfg.GroupVersion != nil {
|
if cfg.GroupVersion != nil {
|
||||||
cmdApiVersion = *cfg.GroupVersion
|
cmdApiVersion = *cfg.GroupVersion
|
||||||
}
|
}
|
||||||
if discoverDynamicAPIs {
|
|
||||||
clientset, err := clients.ClientSetForVersion(&unversioned.GroupVersion{Version: "v1"})
|
|
||||||
checkErrWithPrefix("failed to find client for version v1: ", err)
|
|
||||||
|
|
||||||
var versions []unversioned.GroupVersion
|
mapper := registered.RESTMapper()
|
||||||
var gvks []unversioned.GroupVersionKind
|
// if we can find the server version and it's current enough to have discovery information, use it. Otherwise,
|
||||||
retries := 3
|
// fallback to our hardcoded list
|
||||||
for i := 0; i < retries; i++ {
|
if discoveryClient, err := discovery.NewDiscoveryClientForConfig(cfg); err == nil {
|
||||||
versions, gvks, err = GetThirdPartyGroupVersions(clientset.Discovery())
|
if serverVersion, err := discoveryClient.ServerVersion(); err == nil && useDiscoveryRESTMapper(serverVersion.GitVersion) {
|
||||||
// Retry if we got a NotFound error, because user may delete
|
// register third party resources with the api machinery groups. This probably should be done, but
|
||||||
// a thirdparty group when the GetThirdPartyGroupVersions is
|
// its consistent with old code, so we'll start with it.
|
||||||
// running.
|
if err := registerThirdPartyResources(discoveryClient); err != nil {
|
||||||
if err == nil || !apierrors.IsNotFound(err) {
|
fmt.Fprintf(os.Stderr, "Unable to register third party resources: %v\n", err)
|
||||||
break
|
}
|
||||||
|
// ThirdPartyResourceData is special. It's not discoverable, but needed for thirdparty resource listing
|
||||||
|
// TODO eliminate this once we're truly generic.
|
||||||
|
thirdPartyResourceDataMapper := meta.NewDefaultRESTMapper([]unversioned.GroupVersion{extensionsv1beta1.SchemeGroupVersion}, registered.InterfacesFor)
|
||||||
|
thirdPartyResourceDataMapper.Add(extensionsv1beta1.SchemeGroupVersion.WithKind("ThirdPartyResourceData"), meta.RESTScopeNamespace)
|
||||||
|
mapper = meta.MultiRESTMapper{
|
||||||
|
discovery.NewDeferredDiscoveryRESTMapper(discoveryClient, registered.InterfacesFor),
|
||||||
|
thirdPartyResourceDataMapper,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
checkErrWithPrefix("failed to get third-party group versions: ", err)
|
}
|
||||||
if len(versions) > 0 {
|
|
||||||
priorityMapper, ok := mapper.RESTMapper.(meta.PriorityRESTMapper)
|
|
||||||
if !ok {
|
|
||||||
CheckErr(fmt.Errorf("expected PriorityMapper, saw: %v", mapper.RESTMapper))
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
multiMapper, ok := priorityMapper.Delegate.(meta.MultiRESTMapper)
|
|
||||||
if !ok {
|
|
||||||
CheckErr(fmt.Errorf("unexpected type: %v", mapper.RESTMapper))
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
groupsMap := map[string][]unversioned.GroupVersion{}
|
|
||||||
for _, version := range versions {
|
|
||||||
groupsMap[version.Group] = append(groupsMap[version.Group], version)
|
|
||||||
}
|
|
||||||
for group, versionList := range groupsMap {
|
|
||||||
preferredExternalVersion := versionList[0]
|
|
||||||
|
|
||||||
thirdPartyMapper, err := kubectl.NewThirdPartyResourceMapper(versionList, getGroupVersionKinds(gvks, group))
|
// wrap with shortcuts
|
||||||
checkErrWithPrefix("failed to create third party resource mapper: ", err)
|
mapper = kubectl.ShortcutExpander{RESTMapper: mapper}
|
||||||
accessor := meta.NewAccessor()
|
// wrap with output preferences
|
||||||
groupMeta := apimachinery.GroupMeta{
|
mapper = kubectl.OutputVersionMapper{RESTMapper: mapper, OutputVersions: []unversioned.GroupVersion{cmdApiVersion}}
|
||||||
GroupVersion: preferredExternalVersion,
|
return mapper, api.Scheme
|
||||||
GroupVersions: versionList,
|
|
||||||
RESTMapper: thirdPartyMapper,
|
|
||||||
SelfLinker: runtime.SelfLinker(accessor),
|
|
||||||
InterfacesFor: makeInterfacesFor(versionList),
|
|
||||||
}
|
|
||||||
|
|
||||||
checkErrWithPrefix("failed to register group: ", registered.RegisterGroup(groupMeta))
|
|
||||||
registered.AddThirdPartyAPIGroupVersions(versionList...)
|
|
||||||
multiMapper = append(meta.MultiRESTMapper{thirdPartyMapper}, multiMapper...)
|
|
||||||
}
|
|
||||||
priorityMapper.Delegate = multiMapper
|
|
||||||
// Reassign to the RESTMapper here because priorityMapper is actually a copy, so if we
|
|
||||||
// don't reassign, the above assignement won't actually update mapper.RESTMapper
|
|
||||||
mapper.RESTMapper = priorityMapper
|
|
||||||
}
|
|
||||||
}
|
|
||||||
outputRESTMapper := kubectl.OutputVersionMapper{RESTMapper: mapper, OutputVersions: []unversioned.GroupVersion{cmdApiVersion}}
|
|
||||||
priorityRESTMapper := meta.PriorityRESTMapper{
|
|
||||||
Delegate: outputRESTMapper,
|
|
||||||
}
|
|
||||||
// TODO: this should come from registered versions
|
|
||||||
groups := []string{api.GroupName, autoscaling.GroupName, extensions.GroupName, federation.GroupName, batch.GroupName}
|
|
||||||
// set a preferred version
|
|
||||||
for _, group := range groups {
|
|
||||||
gvs := registered.EnabledVersionsForGroup(group)
|
|
||||||
if len(gvs) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
priorityRESTMapper.ResourcePriority = append(priorityRESTMapper.ResourcePriority, unversioned.GroupVersionResource{Group: group, Version: gvs[0].Version, Resource: meta.AnyResource})
|
|
||||||
priorityRESTMapper.KindPriority = append(priorityRESTMapper.KindPriority, unversioned.GroupVersionKind{Group: group, Version: gvs[0].Version, Kind: meta.AnyKind})
|
|
||||||
}
|
|
||||||
for _, group := range groups {
|
|
||||||
priorityRESTMapper.ResourcePriority = append(priorityRESTMapper.ResourcePriority, unversioned.GroupVersionResource{Group: group, Version: meta.AnyVersion, Resource: meta.AnyResource})
|
|
||||||
priorityRESTMapper.KindPriority = append(priorityRESTMapper.KindPriority, unversioned.GroupVersionKind{Group: group, Version: meta.AnyVersion, Kind: meta.AnyKind})
|
|
||||||
}
|
|
||||||
return priorityRESTMapper, api.Scheme
|
|
||||||
},
|
},
|
||||||
UnstructuredObject: func() (meta.RESTMapper, runtime.ObjectTyper, error) {
|
UnstructuredObject: func() (meta.RESTMapper, runtime.ObjectTyper, error) {
|
||||||
cfg, err := clients.ClientConfigForVersion(nil)
|
cfg, err := clients.ClientConfigForVersion(nil)
|
||||||
@ -1305,3 +1253,68 @@ func (f *Factory) NewBuilder(thirdPartyDiscovery bool) *resource.Builder {
|
|||||||
|
|
||||||
return resource.NewBuilder(mapper, typer, resource.ClientMapperFunc(f.ClientForMapping), f.Decoder(true))
|
return resource.NewBuilder(mapper, typer, resource.ClientMapperFunc(f.ClientForMapping), f.Decoder(true))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// useDiscoveryRESTMapper checks the server version to see if its recent enough to have
|
||||||
|
// enough discovery information available to reliably build a RESTMapper. If not, use the
|
||||||
|
// hardcoded mapper in this client (legacy behavior)
|
||||||
|
func useDiscoveryRESTMapper(serverVersion string) bool {
|
||||||
|
if len(serverVersion) == 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
serverSemVer, err := semver.Parse(serverVersion[1:])
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return serverSemVer.GE(semver.MustParse("1.3.0"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// registerThirdPartyResources inspects the discovery endpoint to find thirdpartyresources in the discovery doc
|
||||||
|
// and then registers them with the apimachinery code. I think this is done so that scheme/codec stuff works,
|
||||||
|
// but I really don't know. Feels like this code should go away once kubectl is completely generic for generic
|
||||||
|
// CRUD
|
||||||
|
func registerThirdPartyResources(discoveryClient discovery.DiscoveryInterface) error {
|
||||||
|
var versions []unversioned.GroupVersion
|
||||||
|
var gvks []unversioned.GroupVersionKind
|
||||||
|
var err error
|
||||||
|
retries := 3
|
||||||
|
for i := 0; i < retries; i++ {
|
||||||
|
versions, gvks, err = GetThirdPartyGroupVersions(discoveryClient)
|
||||||
|
// Retry if we got a NotFound error, because user may delete
|
||||||
|
// a thirdparty group when the GetThirdPartyGroupVersions is
|
||||||
|
// running.
|
||||||
|
if err == nil || !apierrors.IsNotFound(err) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
groupsMap := map[string][]unversioned.GroupVersion{}
|
||||||
|
for _, version := range versions {
|
||||||
|
groupsMap[version.Group] = append(groupsMap[version.Group], version)
|
||||||
|
}
|
||||||
|
for group, versionList := range groupsMap {
|
||||||
|
preferredExternalVersion := versionList[0]
|
||||||
|
|
||||||
|
thirdPartyMapper, err := kubectl.NewThirdPartyResourceMapper(versionList, getGroupVersionKinds(gvks, group))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
accessor := meta.NewAccessor()
|
||||||
|
groupMeta := apimachinery.GroupMeta{
|
||||||
|
GroupVersion: preferredExternalVersion,
|
||||||
|
GroupVersions: versionList,
|
||||||
|
RESTMapper: thirdPartyMapper,
|
||||||
|
SelfLinker: runtime.SelfLinker(accessor),
|
||||||
|
InterfacesFor: makeInterfacesFor(versionList),
|
||||||
|
}
|
||||||
|
if err := registered.RegisterGroup(groupMeta); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
registered.AddThirdPartyAPIGroupVersions(versionList...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user