Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

OpenAPI support for kubectl #44531

Merged
merged 4 commits into from
Apr 26, 2017

Conversation

pwittrock
Copy link
Member

Support for openapi spec in kubectl.

Includes:

  • downloading and caching openapi spec to a local file
  • parsing openapi spec into binary serializable datastructures (10x faster load times 600ms -> 40ms)
  • caching parsed openapi spec in memory for each command
NONE

@pwittrock pwittrock self-assigned this Apr 15, 2017
@k8s-ci-robot k8s-ci-robot added the cncf-cla: yes Indicates the PR's author has signed the CNCF CLA. label Apr 15, 2017
@k8s-github-robot k8s-github-robot added the size/XXL Denotes a PR that changes 1000+ lines, ignoring generated files. label Apr 15, 2017
@k8s-github-robot k8s-github-robot added the release-note-none Denotes a PR that doesn't merit a release note. label Apr 15, 2017
@k8s-reviewable
Copy link

This change is Reviewable

@k8s-github-robot k8s-github-robot added the needs-rebase Indicates a PR cannot be merged because it has merge conflicts with HEAD. label Apr 16, 2017
@pwittrock pwittrock force-pushed the kubectl-openapi branch 2 times, most recently from 1c4ef67 to bf281e5 Compare April 16, 2017 18:55
@k8s-github-robot k8s-github-robot removed the needs-rebase Indicates a PR cannot be merged because it has merge conflicts with HEAD. label Apr 16, 2017
@pwittrock pwittrock force-pushed the kubectl-openapi branch 4 times, most recently from f12943c to 20e7af1 Compare April 17, 2017 00:12
@pwittrock
Copy link
Member Author

@k8s-bot verify test this

W0416 18:58:18.522] fatal: unable to access 'https://bitbucket.org/bertimus9/systemstat.git/': The requested URL returned error: 502
W0416 18:58:18.522] godep: error downloading dep (bitbucket.org/bertimus9/systemstat): exit status 128
W0416 18:58:24.761] godep: Error downloading some deps. Aborting restore and check.
W0416 18:58:24.765] !!! Error in ./hack/godep-restore.sh:28

@pwittrock pwittrock force-pushed the kubectl-openapi branch 2 times, most recently from e758ec4 to 18d9b0b Compare April 17, 2017 18:03
@pwittrock
Copy link
Member Author

@caesarxuchao - plz review the client-go changes
@mbohlool - plz review the openapi parsing
@fabianofranz / @monopole - plz review for kubectl

@@ -401,6 +401,11 @@ func AddValidateFlags(cmd *cobra.Command) {
cmd.MarkFlagFilename("schema-cache-dir")
}

func AddOpenAPIFlags(cmd *cobra.Command) {
cmd.Flags().String("schema-cache-dir", fmt.Sprintf("~/%s/%s", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName), fmt.Sprintf("If non-empty, load/store cached API schemas in this directory, default is '$HOME/%s/%s'", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've seen three reference to schema, openapi, and swagger in code/comments/messages. should we just use one? I suggest openapi for all of them.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This doesn't introduce a new flag, it is just allows registering the cache dir without registering the validation flags. See 5 lines above this one.


// isPrimitive returns true if s is a primitive type
func (o *Resources) isPrimitive(s spec.Schema) bool {
return !o.isArray(s) && !o.isMap(s) && !o.isDefinitionReference(s) && len(s.Type) > 0
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The assumption of a schema is primitive if it does not have a ref may not be true all the time. I would say a primitive schema is one with no "properties". does that make sense?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated the checks to be more explicit.

There is a unit check that test that every definition encountered has exactly 1 of (isPrimitive, isArray, isMap, isDefinitionReference) which should give some confidence that these are working as intended.

}

// encodeSpec binary encodes the openapi spec
func (c *cachingOpenAPIClient) encodeSpec(parsed *Resources) ([]byte, error) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can't we just encode/decode spec.swagger?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No. :( As discussed offline, the Ref type which contains the json reference has private fields that are dropped by binary serialization.

@monopole
Copy link
Contributor

Reviewed 3 of 3 files at r1, 19 of 19 files at r2, 6 of 6 files at r3.
Review status: all files reviewed at latest revision, 20 unresolved discussions.


pkg/kubectl/cmd/testing/fake.go, line 406 at r3 (raw file):

func (f *FakeFactory) SwaggerSchema(schema.GroupVersionKind) (*swagger.ApiDeclaration, error) {
	return nil, nil
}

should the swagger stuff be marked deprecated and/or be given a TODO(#issue) to track its removal?


pkg/kubectl/cmd/util/factory.go, line 445 at r3 (raw file):

	cacheFile := path.Join(fullDir, prefix, groupVersion, schemaFileName)

	abs, _ := filepath.Abs(cacheDir)

this can be scoped to the 'if' body


pkg/kubectl/cmd/util/factory_object_mapping.go, line 442 at r3 (raw file):

// openAPIData returns an openAPIData containing metadata and structural information about Kubernetes
// models and operations.  Will try to cache the spec locally if possible.

pls add note about what (if anything) will make the cache stale, and how it is freshened

test coverage for that?


pkg/kubectl/cmd/util/helpers.go, line 405 at r3 (raw file):

func AddOpenAPIFlags(cmd *cobra.Command) {
	cmd.Flags().String("schema-cache-dir", fmt.Sprintf("~/%s/%s", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName), fmt.Sprintf("If non-empty, load/store cached API schemas in this directory, default is '$HOME/%s/%s'", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName))

BTW , go fmt will retain newlines added for readability.


pkg/kubectl/cmd/util/openapi/openapi.go, line 30 at r3 (raw file):

)

const gvkOpenAPIKey = "x-kubernetes-group-version-kind"

Please comment the const. Does the x mean something? could this string be arbitrary, e.g. peach-banana-sandwich?


pkg/kubectl/cmd/util/openapi/openapi.go, line 40 at r3 (raw file):

}

// KindDefinition defines a Kubernetes object Kind

Seems odd to have PeachDefinition instead of just Peach. The doc doesn't help.
Maybe for doc inspiration see https://docs.oracle.com/javase/7/docs/api/java/lang/Class.html

Went digging in apimachinery/blob/master/pkg/runtime/schema/group_version.go and am even more confused about Kind


pkg/kubectl/cmd/util/openapi/openapi.go, line 42 at r3 (raw file):

// KindDefinition defines a Kubernetes object Kind
type KindDefinition struct {
	// Name is the name of the type

BTW, this looks like L65, hence earlier request for commentary on Kind.
"TypeDefinition" below raises the same questions


pkg/kubectl/cmd/util/openapi/openapi.go, line 48 at r3 (raw file):

	IsResource bool

	// GroupVersionKind is the group version kind of a resource.

// GroupVersionKind (does this useful thing in the context) of a resource.


pkg/kubectl/cmd/util/openapi/openapi.go, line 63 at r3 (raw file):

}

// TypeDefinition defines a field definition

"defines a Type"

and again wondering why not just "Type" and how this differs from "Kind".


pkg/kubectl/cmd/util/openapi/openapi.go, line 193 at r3 (raw file):

// isArray returns true if s is an array type.
func (o *Resources) isArray(s spec.Schema) bool {
	return len(s.Type) > 0 && s.Type[0] == "array"

BTW: Could "ARRAY" show up?


pkg/kubectl/cmd/util/openapi/openapi.go, line 196 at r3 (raw file):

}

// isMap

isObject? (given s.Type[0] == "object")


pkg/kubectl/cmd/util/openapi/openapi.go, line 232 at r3 (raw file):

}

// getElementType returns the type of an element for arrays and maps

Comment looks odd given !o.isArray returns error


pkg/kubectl/cmd/util/openapi/openapi.go, line 259 at r3 (raw file):

// getGroupVersionKind implements openAPIData
// getGVK parses the gropuversionkind for a resource definition from the x-kubernetes

sic (gropuversionkind)

More and more want a primer in this doc about Group, Kind, Version, Type, Field
(or a ptr to a primer).

Doc under https://github.com/kubernetes/apimachinery/blob/master/pkg/runtime/schema/group_version.go doesn't help
as much as i'd hoped.


pkg/kubectl/cmd/util/openapi/openapi_cache.go, line 48 at r3 (raw file):

// newCachingOpenAPIClient returns a new discovery.OpenAPISchemaInterface
// that will read the openapi spec from a local cache if it exists, and
// if not fetch an openapi spec using client

using client.

People are not picky about sentence formation?


pkg/kubectl/cmd/util/openapi/openapi_cache_test.go, line 60 at r3 (raw file):

		result, err := instance.openAPIData()
		Expect(err).To(BeNil())
		expectEqual(result, expectedData)

FYI: make an Equals method for Resources, test it in openapi_test.go, and use it directly here? instead of the expectEqual utility below?


pkg/kubectl/cmd/util/openapi/openapi_test.go, line 67 at r3 (raw file):

	})

	It("should find the definition fields", func() {

BTW: nifty framework 👍 tests easy to read


staging/src/k8s.io/client-go/discovery/discovery_client.go, line 380 at r3 (raw file):

	// WARNING: We don't use the loads.Analyze because this is very expensive
	// and loads.Analyze takes 2x as long (it Unmarshal's 2x)
	// Warning: this is very expensive (~600ms)
  1. comment above function?
  2. loads.Analyzed called below, seems to contradict comment (though technically it differs by one letter)
  3. Warning: twice?

Comments from Reviewable

package openapi

// The openapi package is a collection of libraries for fetching the openapi spec
// from a Kubernetes server and then indexing the type definitions.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if these are meant to be Godoc-compatible comments, they need to go above the package keyword

@pwittrock pwittrock requested a review from mengqiy April 17, 2017 22:16
@pwittrock
Copy link
Member Author

Review status: 20 of 28 files reviewed at latest revision, 21 unresolved discussions.


pkg/kubectl/cmd/testing/fake.go, line 406 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

should the swagger stuff be marked deprecated and/or be given a TODO(#issue) to track its removal?

Added


pkg/kubectl/cmd/util/factory.go, line 445 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

this can be scoped to the 'if' body

Done


pkg/kubectl/cmd/util/factory_object_mapping.go, line 442 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

pls add note about what (if anything) will make the cache stale, and how it is freshened

test coverage for that?

Updated with a comment.

Test coverage for the caching logic is in the openapi_cache_test.go file


pkg/kubectl/cmd/util/helpers.go, line 405 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

BTW , go fmt will retain newlines added for readability.

Added some newlines for readability


pkg/kubectl/cmd/util/openapi/doc.go, line 20 at r3 (raw file):

Previously, arschles (Aaron Schlesinger) wrote…

if these are meant to be Godoc-compatible comments, they need to go above the package keyword

Done thx.


pkg/kubectl/cmd/util/openapi/openapi.go, line 30 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

Please comment the const. Does the x mean something? could this string be arbitrary, e.g. peach-banana-sandwich?

Done


pkg/kubectl/cmd/util/openapi/openapi.go, line 40 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

Seems odd to have PeachDefinition instead of just Peach. The doc doesn't help.
Maybe for doc inspiration see https://docs.oracle.com/javase/7/docs/api/java/lang/Class.html

Went digging in apimachinery/blob/master/pkg/runtime/schema/group_version.go and am even more confused about Kind

Updated to drop the Definition piece. GVK deserves a good doc.


pkg/kubectl/cmd/util/openapi/openapi.go, line 42 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

BTW, this looks like L65, hence earlier request for commentary on Kind.
"TypeDefinition" below raises the same questions

Yes it is a bit confusing. Added a comment to clarify


pkg/kubectl/cmd/util/openapi/openapi.go, line 48 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

// GroupVersionKind (does this useful thing in the context) of a resource.

Updated the comment


pkg/kubectl/cmd/util/openapi/openapi.go, line 63 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

"defines a Type"

and again wondering why not just "Type" and how this differs from "Kind".

Updated the comment. Let me know if this makes more sense.


pkg/kubectl/cmd/util/openapi/openapi.go, line 193 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

BTW: Could "ARRAY" show up?

Changed to compare against lowercase. I don't think that this should actually show up, but provides a bit of future proofing.

I've changed the matching logic a bit so it should match one of the primitive types "string", "boolean", "integer" or "array", "object", or be empty for definition references. The unit test parses the k8s swagger.json and validates that each definition matches exactly one of primitive, array, map, definitionReference, so we should catch if there is anything that isn't looking correct.


pkg/kubectl/cmd/util/openapi/openapi.go, line 196 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

isObject? (given s.Type[0] == "object")

Updated with a comment to explain this. Also moved the strings into constants with descriptions.


pkg/kubectl/cmd/util/openapi/openapi.go, line 232 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

Comment looks odd given !o.isArray returns error

Done


pkg/kubectl/cmd/util/openapi/openapi.go, line 259 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

sic (gropuversionkind)

More and more want a primer in this doc about Group, Kind, Version, Type, Field
(or a ptr to a primer).

Doc under https://github.com/kubernetes/apimachinery/blob/master/pkg/runtime/schema/group_version.go doesn't help
as much as i'd hoped.

GroupVersionKind is a more general API concept. There is a little bit here. Could probably deserve something more in depth.


pkg/kubectl/cmd/util/openapi/openapi_cache.go, line 48 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

using client.

People are not picky about sentence formation?

Sometimes they are :)


pkg/kubectl/cmd/util/openapi/openapi_cache_test.go, line 60 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…

FYI: make an Equals method for Resources, test it in openapi_test.go, and use it directly here? instead of the expectEqual utility below?

Thx


staging/src/k8s.io/client-go/discovery/discovery_client.go, line 380 at r3 (raw file):

Previously, monopole (Jeff Regan) wrote…
  1. comment above function?
  2. loads.Analyzed called below, seems to contradict comment (though technically it differs by one letter)
  3. Warning: twice?

:P. Changed the logic after I started caching. Updated the comments to be correct.


Comments from Reviewable

Copy link
Member

@mengqiy mengqiy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Other LGTM

@@ -33,6 +33,7 @@ import (
"github.com/spf13/cobra"
"github.com/spf13/pflag"

"github.com/golang/glog"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: This line should be moved into github import group

@@ -28,6 +28,8 @@ import (

"github.com/emicklei/go-restful/swagger"

"sync"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: should move this up.

@@ -91,6 +94,11 @@ type SwaggerSchemaInterface interface {
SwaggerSchema(version schema.GroupVersion) (*swagger.ApiDeclaration, error)
}

type OpenAPISchemaInterface interface {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: for consistency, could you put a comment for the interface?

// OpenAPISchema fetches the open api schema using a rest client and parses the json.
// Warning: this is very expensive (~1.2s)
func (d *DiscoveryClient) OpenAPISchema() (*spec.Swagger, error) {
data, err := d.restClient.Get().AbsPath("/swagger.json").Do().Raw()
Copy link
Member

@caesarxuchao caesarxuchao Apr 21, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This comment is off-topic. /swagger.json is not shown as an path when i visit the root of the server. It only shows "/swaggerapi/" which leads me to swagger-1.2. @mbohlool do you think we should fix that?

}

func TestGetOpenAPISchema(t *testing.T) {
expect := spec.Swagger{}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we add some contents to the spec?

Copy link
Member

@caesarxuchao caesarxuchao left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A couple of nits for the changes in staging/client-go.

openAPIGetter openAPIGetter
}

type openAPIGetter struct {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please avoid the anonymous inclusion in the struct. It makes the code below harder to read.

})

// Delegate to the OpenAPIGetter
return f.openAPIGetter.OpenAPIData()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This method is on the Getter I guess?

}

// Create the caching OpenAPIGetter
f.openAPIGetter.Getter = openapi.NewOpenAPIGetter(cacheDir, versionString, discovery)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why does it care about the server version? Why doesn't it use the discovery method and rely on the caching of the discovery client?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does the discovery client cache the result to disk?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does the discovery client expose a method to that will give the client stable a hash / key for for exposed API?

@monopole
Copy link
Contributor

Review status: 20 of 28 files reviewed at latest revision, 24 unresolved discussions.


pkg/kubectl/cmd/util/openapi/openapi_cache.go, line 48 at r3 (raw file):

Previously, pwittrock (Phillip Wittrock) wrote…

Sometimes they are :)

OK :) it all /lgtm


Comments from Reviewable

@monopole
Copy link
Contributor

Reviewed 4 of 8 files at r4, 4 of 4 files at r5.
Review status: all files reviewed at latest revision, 23 unresolved discussions.


Comments from Reviewable

@pwittrock
Copy link
Member Author

Review status: 22 of 28 files reviewed at latest revision, 23 unresolved discussions.


pkg/kubectl/cmd/util/factory.go, line 36 at r5 (raw file):

Previously, mengqiy (Mengqi Yu) wrote…

nit: This line should be moved into github import group

Done.


pkg/kubectl/cmd/util/factory_object_mapping.go, line 64 at r4 (raw file):

Previously, deads2k (David Eads) wrote…

please avoid the anonymous inclusion in the struct. It makes the code below harder to read.

Done.


pkg/kubectl/cmd/util/factory_object_mapping.go, line 477 at r4 (raw file):

Previously, deads2k (David Eads) wrote…

This method is on the Getter I guess?

Done.


pkg/kubectl/cmd/util/factory_object_mapping.go, line 31 at r5 (raw file):

Previously, mengqiy (Mengqi Yu) wrote…

nit: should move this up.

Done.


pkg/kubectl/cmd/util/openapi/openapi_cache.go, line 82 at r4 (raw file):

Previously, fabianofranz (Fabiano Franz) wrote…

Same here.

Done.


staging/src/k8s.io/client-go/discovery/discovery_client.go, line 97 at r5 (raw file):

Previously, caesarxuchao (Chao Xu) wrote…

nit: for consistency, could you put a comment for the interface?

Done.


staging/src/k8s.io/client-go/discovery/discovery_client.go, line 380 at r5 (raw file):

Previously, caesarxuchao (Chao Xu) wrote…

This comment is off-topic. /swagger.json is not shown as an path when i visit the root of the server. It only shows "/swaggerapi/" which leads me to swagger-1.2. @mbohlool do you think we should fix that?

ACK


staging/src/k8s.io/client-go/discovery/discovery_client_test.go, line 354 at r5 (raw file):

Previously, caesarxuchao (Chao Xu) wrote…

Can we add some contents to the spec?

Done.


Comments from Reviewable

@pwittrock
Copy link
Member Author

PTAL

@pwittrock
Copy link
Member Author

@k8s-bot gce etcd3 e2e test this

@caesarxuchao
Copy link
Member

client-go changes lgtm.

@pwittrock
Copy link
Member Author

@caesarxuchao @deads2k Can I get an approval for client-go?

@k8s-github-robot k8s-github-robot added the needs-rebase Indicates a PR cannot be merged because it has merge conflicts with HEAD. label Apr 25, 2017
@deads2k
Copy link
Contributor

deads2k commented Apr 25, 2017

/approve

@k8s-github-robot k8s-github-robot added the approved Indicates a PR has been approved by an approver from all required OWNERS files. label Apr 25, 2017
@k8s-github-robot k8s-github-robot removed the needs-rebase Indicates a PR cannot be merged because it has merge conflicts with HEAD. label Apr 25, 2017
@monopole
Copy link
Contributor

/lgtm

@k8s-ci-robot k8s-ci-robot added the lgtm "Looks good to me", indicates that a PR is ready to be merged. label Apr 26, 2017
@k8s-github-robot
Copy link

[APPROVALNOTIFIER] This PR is APPROVED

This pull-request has been approved by: deads2k, monopole, pwittrock

Needs approval from an approver in each of these OWNERS Files:

You can indicate your approval by writing /approve in a comment
You can cancel your approval by writing /approve cancel in a comment

@k8s-github-robot
Copy link

Automatic merge from submit-queue

@k8s-github-robot k8s-github-robot merged commit 433aec1 into kubernetes:master Apr 26, 2017
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
approved Indicates a PR has been approved by an approver from all required OWNERS files. cncf-cla: yes Indicates the PR's author has signed the CNCF CLA. lgtm "Looks good to me", indicates that a PR is ready to be merged. release-note-none Denotes a PR that doesn't merit a release note. size/XXL Denotes a PR that changes 1000+ lines, ignoring generated files.
Projects
None yet
Development

Successfully merging this pull request may close these issues.