Merge pull request #51829 from fabianofranz/dynamic_categories
Automatic merge from submit-queue (batch tested with PRs 51921, 51829, 51968, 51988, 51986) Category expansion fully based on discovery **What this PR does / why we need it**: Makes the expansion of resource names in `kubectl` (e.g. "all" in "kubectl get all") respect the "categories" field in the API, and fallback to the legacy expander. **Which issue this PR fixes** *(optional, in `fixes #<issue number>(, fixes #<issue_number>, ...)` format, will close that issue when PR gets merged)*: fixes https://github.com/kubernetes/kubernetes/issues/41353 **Release note**: ```release-note NONE ```
This commit is contained in:
		| @@ -119,17 +119,22 @@ func (f *ring1Factory) UnstructuredObject() (meta.RESTMapper, runtime.ObjectType | |||||||
| } | } | ||||||
|  |  | ||||||
| func (f *ring1Factory) CategoryExpander() resource.CategoryExpander { | func (f *ring1Factory) CategoryExpander() resource.CategoryExpander { | ||||||
| 	var categoryExpander resource.CategoryExpander | 	legacyExpander := resource.LegacyCategoryExpander | ||||||
| 	categoryExpander = resource.LegacyCategoryExpander |  | ||||||
| 	discoveryClient, err := f.clientAccessFactory.DiscoveryClient() | 	discoveryClient, err := f.clientAccessFactory.DiscoveryClient() | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		// wrap with discovery based filtering | 		// fallback is the legacy expander wrapped with discovery based filtering | ||||||
| 		categoryExpander, err = resource.NewDiscoveryFilteredExpander(categoryExpander, discoveryClient) | 		fallbackExpander, err := resource.NewDiscoveryFilteredExpander(legacyExpander, discoveryClient) | ||||||
| 		// you only have an error on missing discoveryClient, so this shouldn't fail.  Check anyway. |  | ||||||
| 		CheckErr(err) | 		CheckErr(err) | ||||||
|  |  | ||||||
|  | 		// by default use the expander that discovers based on "categories" field from the API | ||||||
|  | 		discoveryCategoryExpander, err := resource.NewDiscoveryCategoryExpander(fallbackExpander, discoveryClient) | ||||||
|  | 		CheckErr(err) | ||||||
|  |  | ||||||
|  | 		return discoveryCategoryExpander | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return categoryExpander | 	return legacyExpander | ||||||
| } | } | ||||||
|  |  | ||||||
| func (f *ring1Factory) ClientForMapping(mapping *meta.RESTMapping) (resource.RESTClient, error) { | func (f *ring1Factory) ClientForMapping(mapping *meta.RESTMapping) (resource.RESTClient, error) { | ||||||
|   | |||||||
| @@ -60,7 +60,9 @@ go_test( | |||||||
|         "//pkg/api:go_default_library", |         "//pkg/api:go_default_library", | ||||||
|         "//pkg/api/testapi:go_default_library", |         "//pkg/api/testapi:go_default_library", | ||||||
|         "//pkg/api/testing:go_default_library", |         "//pkg/api/testing:go_default_library", | ||||||
|  |         "//vendor/github.com/emicklei/go-restful-swagger12:go_default_library", | ||||||
|         "//vendor/github.com/ghodss/yaml:go_default_library", |         "//vendor/github.com/ghodss/yaml:go_default_library", | ||||||
|  |         "//vendor/github.com/googleapis/gnostic/OpenAPIv2:go_default_library", | ||||||
|         "//vendor/github.com/stretchr/testify/assert:go_default_library", |         "//vendor/github.com/stretchr/testify/assert:go_default_library", | ||||||
|         "//vendor/k8s.io/api/core/v1:go_default_library", |         "//vendor/k8s.io/api/core/v1:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/api/equality:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/api/equality:go_default_library", | ||||||
| @@ -72,7 +74,10 @@ go_test( | |||||||
|         "//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/runtime/serializer/streaming:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/runtime/serializer/streaming:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/util/errors:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/util/errors:go_default_library", | ||||||
|  |         "//vendor/k8s.io/apimachinery/pkg/version:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/watch:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/watch:go_default_library", | ||||||
|  |         "//vendor/k8s.io/client-go/discovery:go_default_library", | ||||||
|  |         "//vendor/k8s.io/client-go/rest:go_default_library", | ||||||
|         "//vendor/k8s.io/client-go/rest/fake:go_default_library", |         "//vendor/k8s.io/client-go/rest/fake:go_default_library", | ||||||
|         "//vendor/k8s.io/client-go/rest/watch:go_default_library", |         "//vendor/k8s.io/client-go/rest/watch:go_default_library", | ||||||
|         "//vendor/k8s.io/client-go/util/testing:go_default_library", |         "//vendor/k8s.io/client-go/util/testing:go_default_library", | ||||||
|   | |||||||
| @@ -17,8 +17,6 @@ limitations under the License. | |||||||
| package resource | package resource | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" |  | ||||||
|  |  | ||||||
| 	"k8s.io/apimachinery/pkg/runtime/schema" | 	"k8s.io/apimachinery/pkg/runtime/schema" | ||||||
| 	"k8s.io/client-go/discovery" | 	"k8s.io/client-go/discovery" | ||||||
| ) | ) | ||||||
| @@ -36,6 +34,58 @@ func (e SimpleCategoryExpander) Expand(category string) ([]schema.GroupResource, | |||||||
| 	return ret, ok | 	return ret, ok | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type discoveryCategoryExpander struct { | ||||||
|  | 	fallbackExpander CategoryExpander | ||||||
|  | 	discoveryClient  discovery.DiscoveryInterface | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewDiscoveryCategoryExpander returns a category expander that makes use of the "categories" fields from | ||||||
|  | // the API, found through the discovery client. In case of any error or no category found (which likely | ||||||
|  | // means we're at a cluster prior to categories support, fallback to the expander provided. | ||||||
|  | func NewDiscoveryCategoryExpander(fallbackExpander CategoryExpander, client discovery.DiscoveryInterface) (discoveryCategoryExpander, error) { | ||||||
|  | 	if client == nil { | ||||||
|  | 		panic("Please provide discovery client to shortcut expander") | ||||||
|  | 	} | ||||||
|  | 	return discoveryCategoryExpander{fallbackExpander: fallbackExpander, discoveryClient: client}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e discoveryCategoryExpander) Expand(category string) ([]schema.GroupResource, bool) { | ||||||
|  | 	apiResourceLists, _ := e.discoveryClient.ServerResources() | ||||||
|  | 	if len(apiResourceLists) == 0 { | ||||||
|  | 		return e.fallbackExpander.Expand(category) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	discoveredExpansions := map[string][]schema.GroupResource{} | ||||||
|  |  | ||||||
|  | 	for _, apiResourceList := range apiResourceLists { | ||||||
|  | 		gv, err := schema.ParseGroupVersion(apiResourceList.GroupVersion) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return e.fallbackExpander.Expand(category) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for _, apiResource := range apiResourceList.APIResources { | ||||||
|  | 			if categories := apiResource.Categories; len(categories) > 0 { | ||||||
|  | 				for _, category := range categories { | ||||||
|  | 					groupResource := schema.GroupResource{ | ||||||
|  | 						Group:    gv.Group, | ||||||
|  | 						Resource: apiResource.Name, | ||||||
|  | 					} | ||||||
|  | 					discoveredExpansions[category] = append(discoveredExpansions[category], groupResource) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(discoveredExpansions) == 0 { | ||||||
|  | 		// We don't know if the server really don't have any resource with categories, | ||||||
|  | 		// or we're on a cluster version prior to categories support. Anyways, fallback. | ||||||
|  | 		return e.fallbackExpander.Expand(category) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ret, ok := discoveredExpansions[category] | ||||||
|  | 	return ret, ok | ||||||
|  | } | ||||||
|  |  | ||||||
| type discoveryFilteredExpander struct { | type discoveryFilteredExpander struct { | ||||||
| 	delegate CategoryExpander | 	delegate CategoryExpander | ||||||
|  |  | ||||||
| @@ -46,7 +96,7 @@ type discoveryFilteredExpander struct { | |||||||
| // what the server has available | // what the server has available | ||||||
| func NewDiscoveryFilteredExpander(delegate CategoryExpander, client discovery.DiscoveryInterface) (discoveryFilteredExpander, error) { | func NewDiscoveryFilteredExpander(delegate CategoryExpander, client discovery.DiscoveryInterface) (discoveryFilteredExpander, error) { | ||||||
| 	if client == nil { | 	if client == nil { | ||||||
| 		return discoveryFilteredExpander{}, errors.New("Please provide discovery client to shortcut expander") | 		panic("Please provide discovery client to shortcut expander") | ||||||
| 	} | 	} | ||||||
| 	return discoveryFilteredExpander{delegate: delegate, discoveryClient: client}, nil | 	return discoveryFilteredExpander{delegate: delegate, discoveryClient: client}, nil | ||||||
| } | } | ||||||
|   | |||||||
| @@ -20,7 +20,15 @@ import ( | |||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
|  | 	swagger "github.com/emicklei/go-restful-swagger12" | ||||||
|  |  | ||||||
|  | 	"github.com/googleapis/gnostic/OpenAPIv2" | ||||||
|  | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
| 	"k8s.io/apimachinery/pkg/runtime/schema" | 	"k8s.io/apimachinery/pkg/runtime/schema" | ||||||
|  | 	"k8s.io/apimachinery/pkg/version" | ||||||
|  | 	"k8s.io/client-go/discovery" | ||||||
|  | 	restclient "k8s.io/client-go/rest" | ||||||
|  | 	"k8s.io/client-go/rest/fake" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestCategoryExpansion(t *testing.T) { | func TestCategoryExpansion(t *testing.T) { | ||||||
| @@ -65,3 +73,120 @@ func TestCategoryExpansion(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestDiscoveryCategoryExpander(t *testing.T) { | ||||||
|  | 	tests := []struct { | ||||||
|  | 		category       string | ||||||
|  | 		serverResponse []*metav1.APIResourceList | ||||||
|  | 		expected       []schema.GroupResource | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			category: "all", | ||||||
|  | 			serverResponse: []*metav1.APIResourceList{ | ||||||
|  | 				{ | ||||||
|  | 					GroupVersion: "batch/v1", | ||||||
|  | 					APIResources: []metav1.APIResource{ | ||||||
|  | 						{ | ||||||
|  | 							Name:       "jobs", | ||||||
|  | 							ShortNames: []string{"jz"}, | ||||||
|  | 							Categories: []string{"all"}, | ||||||
|  | 						}, | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expected: []schema.GroupResource{ | ||||||
|  | 				{ | ||||||
|  | 					Group:    "batch", | ||||||
|  | 					Resource: "jobs", | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			category: "all", | ||||||
|  | 			serverResponse: []*metav1.APIResourceList{ | ||||||
|  | 				{ | ||||||
|  | 					GroupVersion: "batch/v1", | ||||||
|  | 					APIResources: []metav1.APIResource{ | ||||||
|  | 						{ | ||||||
|  | 							Name:       "jobs", | ||||||
|  | 							ShortNames: []string{"jz"}, | ||||||
|  | 						}, | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			category: "targaryens", | ||||||
|  | 			serverResponse: []*metav1.APIResourceList{ | ||||||
|  | 				{ | ||||||
|  | 					GroupVersion: "batch/v1", | ||||||
|  | 					APIResources: []metav1.APIResource{ | ||||||
|  | 						{ | ||||||
|  | 							Name:       "jobs", | ||||||
|  | 							ShortNames: []string{"jz"}, | ||||||
|  | 							Categories: []string{"all"}, | ||||||
|  | 						}, | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	dc := &fakeDiscoveryClient{} | ||||||
|  | 	for _, test := range tests { | ||||||
|  | 		dc.serverResourcesHandler = func() ([]*metav1.APIResourceList, error) { | ||||||
|  | 			return test.serverResponse, nil | ||||||
|  | 		} | ||||||
|  | 		expander, err := NewDiscoveryCategoryExpander(SimpleCategoryExpander{}, dc) | ||||||
|  | 		if err != nil { | ||||||
|  | 			t.Fatalf("unexpected error %v", err) | ||||||
|  | 		} | ||||||
|  | 		expanded, _ := expander.Expand(test.category) | ||||||
|  | 		if !reflect.DeepEqual(expanded, test.expected) { | ||||||
|  | 			t.Errorf("expected %v, got %v", test.expected, expanded) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type fakeDiscoveryClient struct { | ||||||
|  | 	serverResourcesHandler func() ([]*metav1.APIResourceList, error) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var _ discovery.DiscoveryInterface = &fakeDiscoveryClient{} | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) RESTClient() restclient.Interface { | ||||||
|  | 	return &fake.RESTClient{} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerGroups() (*metav1.APIGroupList, error) { | ||||||
|  | 	return nil, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerResourcesForGroupVersion(groupVersion string) (*metav1.APIResourceList, error) { | ||||||
|  | 	return &metav1.APIResourceList{}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerResources() ([]*metav1.APIResourceList, error) { | ||||||
|  | 	return c.serverResourcesHandler() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerPreferredResources() ([]*metav1.APIResourceList, error) { | ||||||
|  | 	return nil, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerPreferredNamespacedResources() ([]*metav1.APIResourceList, error) { | ||||||
|  | 	return nil, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) ServerVersion() (*version.Info, error) { | ||||||
|  | 	return &version.Info{}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) SwaggerSchema(version schema.GroupVersion) (*swagger.ApiDeclaration, error) { | ||||||
|  | 	return &swagger.ApiDeclaration{}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *fakeDiscoveryClient) OpenAPISchema() (*openapi_v2.Document, error) { | ||||||
|  | 	return &openapi_v2.Document{}, nil | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Kubernetes Submit Queue
					Kubernetes Submit Queue