-
Notifications
You must be signed in to change notification settings - Fork 40.3k
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
Polymorphic Scale Client #53743
Polymorphic Scale Client #53743
Conversation
P.S. I think I got most of the open issues around this, but I'm sure there are more. Let me know if I missed one. |
"k8s.io/apimachinery/pkg/runtime/schema" | ||
"k8s.io/client-go/discovery" | ||
discocache "k8s.io/client-go/discovery/cached" // Saturday Night Fever |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
if the rest of the this pull earns it, I'll let you have it :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
whoops, didn't mean to leave that in :-P
r.mu.RLock() | ||
gv, exists := r.subresMap[resource] | ||
if !exists { | ||
// retry on misses in case we have an update |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you build a non-caching version, then build a caching version on top of it? This is a little hard for me to read and the complexity can be built up in another pull.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
sure, can-do
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
hmm... this is a little weirder than I initially thought, since a non-cached version would probably use different code paths entirely (i.e. it wouldn't use the "get-all-at-once" API).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
hmm... this is a little weirder than I initially thought, since a non-cached version would probably use different code paths entirely (i.e. it wouldn't use the "get-all-at-once" API).
Yeah, that's the bit I don't like.
if !exists { | ||
// retry on misses in case we have an update | ||
r.mu.RUnlock() | ||
r.generateKindMap() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ignoring an error doesn't look right.
} | ||
|
||
func NewScaleConverter() *ScaleConverter { | ||
scheme := runtime.NewScheme() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This makes me very happy.
scheme: scheme, | ||
internalVersioner: runtime.NewMultiGroupVersioner( | ||
scalescheme.SchemeGroupVersion, | ||
schema.GroupKind{Group: scaleext.GroupName, Kind: "Scale"}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This ordering controls default preference, right? Make autoscaling come first.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nope, the order shouldn't matter -- this says "if any of the objects passed in match autoscaling/*.Scale
or extensions/*.Scale
, return autoscaling/__scale_internal.Scale
as the target GVK"
) | ||
|
||
// This file contains our own "internal" version of scale that we use for conversions, | ||
// since we can't use the main Kubernetes internal versions. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This makes me very sad, but so be it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIUC with the "alternative API representations", client can request the server to do the conversion. So the internal types and the entire conversion machinery can be removed in the future. Am I too optimistic?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIUC with the "alternative API representations", client can request the server to do the conversion. So the internal types and the entire conversion machinery can be removed in the future. Am I too optimistic?
Too optimistic. This code will still be correct even once we get there. I would guess we're multiple releases away from re-loosening the server to support this.
limitations under the License. | ||
*/ | ||
|
||
package scheme |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@caesarxuchao have a look at the package location. It looks ok to me.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The location lgtm, too.
/cc @nikhita |
@sttts: GitHub didn't allow me to request PR reviews from the following users: nikhita. Note that only kubernetes members can review this PR, and authors cannot review their own PRs. In response to this:
Instructions for interacting with me using PR comments are available here. If you have questions or suggestions related to my behavior, please file an issue against the kubernetes/test-infra repository. |
) | ||
|
||
// GroupName is the group name use in this package | ||
const GroupName = "extensions" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
these have to match exactly and you already have a package dependency. Assign them to the extensionsapiv1beta1
values I think.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ah, good point, will do.
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object | ||
|
||
// Scale represents a scaling request for a resource. | ||
type Scale struct { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
need fuzzing tests on this.
Get(kind schema.GroupKind, name string) (*autoscalingapi.Scale, error) | ||
|
||
// Update updates the scale of the the given scalable resource. | ||
Update(ind schema.GroupKind, scale *autoscalingapi.Scale) (*autoscalingapi.Scale, error) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why aren't you using the hub you defined? If you want to use autoscaling/v1
as your hub, why define a new internal to convert to?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I considered it, but I figured we didn't want to ever expose internal versions in clients. It does make some sense to me to use it, though (although it also make some sense to use internal versions elsewhere, so /me shrugs).
On the flip side, even though we don't expose it, having an internal version makes it a bit easier to reason about the conversions (IMO), especially since it's following the normal Kubernetes hub-and-spoke pattern. AFAIK, there's not a technical reason why we couldn't use external as the hub, though.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
On the flip side, even though we don't expose it, having an internal version makes it a bit easier to reason about the conversions (IMO), especially since it's following the normal Kubernetes hub-and-spoke pattern. AFAIK, there's not a technical reason why we couldn't use external as the hub, though.
I think this demonstrates the inherent problem of not exposing internal types. If you want to keep your own hub, I can live with it. You need fuzzers though.
Resource(mapping.Resource). | ||
Name(name). | ||
SubResource("scale"). | ||
Do(). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If you switched to use a DoRaw
, then used your custom scheme for decoding, wouldn't you be able to use any given rest.Interface
without constructing a new client from config? I like supporting NewForConfig
, I'm just not seeing why I need to have multiple clients and not supporting New
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We still need a rest.Interface
for each group-version, no?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We still need a rest.Interface for each group-version, no?
Is it because you need to point to different endpoints? You can construct a path and call AbsPath()
. We can expose the function the restclient uses to build the path so that we don't have to hard code here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually making defaultServerURLFor
public seems reasonable, but just circumventing the request with AbsPath
seems hacky. I think if we're going to change this, we should have a type that provides clients for a given GroupVersion only, and use it in both the scale client and dynamic client (since the dynamic client currently uses the same approach that I have here).
For example:
type Provider interface {
GroupVersion(gv schema.GroupVersion) rest.Interface
}
provider.GroupVersion(schema.GroupVersion{"apps", "v1beta2"}).Get()
.Resource("deployments")
.Name("foo")
.SubResource("scale")
.Do()
That looks a bit clunky with the GroupVersion
before the Get, but provides a rest.Interface
out of it, so it fits with more existing code. An alternative might be:
type MultiVersionInterface interface {
// the same methods as rest.Interface, but they return a GroupVersionProvider below
}
type GroupVersionProvider interface {
GroupVersion(schema.GroupVersion) rest.Request
}
type client MultiVersionInterface = ...
client.Get().GroupVersion(schema.GroupVersion{"apps", "v1beta2"})
.Resouce("deployments")
.Name("foo")
.Do()
But since that doesn't return a rest.Interface
anywhere, it's mostly useful only here and in the dynamic client.
WDYT?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But since that doesn't return a rest.Interface anywhere, it's mostly useful only here and in the dynamic client.
WDYT?
I think the dynamic client interface needs some work, but I would definitely accept an .AbsPath
here to make the externally facing bits of this API easier to work with.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's use .AbsPath
here and refactor it with the dynamic client in another PR.
Also, with the first approach, it looks like the provider still returns different restclient for different group versions. That sounds unnecessary.
Before taking the second approach, could we try to add a method to the rest.Request to allow us change group/version after its construction?
// the scale subresource. | ||
type ScaleInterface interface { | ||
// Get fetches the scale of the given scalable resource. | ||
Get(kind schema.GroupKind, name string) (*autoscalingapi.Scale, error) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should take a GroupResource
. GroupKind
was a mistake. The responsibility of determining which resource to use should lie with the caller.
pkg/kubectl/cmd/annotate_test.go
Outdated
@@ -451,7 +451,7 @@ func TestAnnotateObject(t *testing.T) { | |||
f, tf, codec, _ := cmdtesting.NewAPIFactory() | |||
tf.Printer = &testPrinter{} | |||
tf.UnstructuredClient = &fake.RESTClient{ | |||
APIRegistry: api.Registry, | |||
GroupVersion: api.Registry.GroupOrDie(api.GroupName).GroupVersion, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this fix in a separate commit. You've got lots of these lines and this update looks good. We could split it out and merge it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah, it's in its own commit (8f9abfd)
a.updateStatusIfNeeded(hpaStatusOriginal, hpa) | ||
return fmt.Errorf("invalid API version in scale target reference: %v", err) | ||
} | ||
targetGK := targetGV.WithKind(hpa.Spec.ScaleTargetRef.Kind).GroupKind() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You need to resolve a GroupResource
here. You client should deal in the correct types. It helps make it clear that the HPA Kind make a mistake in their API.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ack, makes sense to me
Will kubectl use this client? cc @foxish |
@caesarxuchao Yeah, |
@DirectXMan12 could you describe what an ideal world will be, supposing we have the alternative API representation, have deprecated extensions/v1beta1.Autoscale? |
I think you're correct, but I think that is a long way off. |
@@ -184,9 +184,9 @@ func testUnjoinFederationFactory(name, server, secret string) cmdutil.Factory { | |||
tf.ClientConfig = kubefedtesting.DefaultClientConfig() | |||
ns := serializer.NegotiatedSerializerWrapper(runtime.SerializerInfo{Serializer: runtime.NewCodec(f.JSONEncoder(), api.Codecs.UniversalDecoder(fedv1beta1.SchemeGroupVersion))}) | |||
tf.Client = &fake.RESTClient{ | |||
APIRegistry: api.Registry, | |||
GroupVersion: api.Registry.GroupOrDie("federation").GroupVersion, | |||
InternalGroupName: api.Registry.GroupOrDie("federation").GroupVersion.Group, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
do we have cases with distinct internal and external group names? I think we always register the internal types into all relevant internal group versions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think so, but figured I'd let them be explicitly separate here since they were previously controlled separately. I can change that if you want.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It simplifies the interface. So why not.
But kube wouldn't use OpenShift types. It's not a symmetric relation. Also note that we don't have the internal type. So we have to build our own. So in the kube-apiserver binary we will have two autoscaling groups, depending on which scheme you look. |
e016fbd
to
576c1a2
Compare
576c1a2
to
472edea
Compare
@DirectXMan12 failures look real |
yep, I think something broke when I had to rebase. |
Previously, the fake RESTClient in client-go required a Registry. It used the Registry to fetch the GroupVersion for the fake client. However, the way it did so was dubious in some cases (it hard-coded the default API group in places), and not strictly necssary. This updates the fake client to just recieve the GroupVersion and internal group name directly, instead of requiring a Registry, so that it can be consumed in unit tests where a Registry isn't necessarily readily available (e.g. elsewhere in client-go).
The fake discovery client currently returns `nil, nil` for several methods. Among them is the `ServerGroups` method, which is used by the discovery REST mapper implementations. This updates the fake discovery client to actually return server groups so that the discovery REST mapper can be used in tests.
This introduces a polymorphic scale client capable of operating against scale subresources which return different group-versions of Scale. The scale subresources may be in group-versions different than the scale itself, so that we no longer need a copy of every scalable resource in the extensions API group. To discovery which Scale group-versions go to which subresources, discovery is used. The scale client maintains its own internal versions and conversions to several external versions, with a "hub" version that's a copy of the autoscaling internal version. It currently supports the following group-versions for Scale subresources: - extensions/v1beta1.Scale - autoscaling/v1.Scale
Previously, we did not have custom code for fuzzing label selectors. Anything that used a label selector (like Scale) had to manually bypass fuzzing the selector, or write its own fuzzer. This introduces a fuzzer for label selectors which generates random correct selectors with random keys and values.
This removes the override on the extensions fuzzer so that extensions.Scale will use the full label selector fuzzer.
472edea
to
4abac06
Compare
@DirectXMan12: The following tests failed, say
Full PR test history. Your PR dashboard. Please help us cut down on flakes by linking to an open issue when you hit one in your PR. Instructions for interacting with me using PR comments are available here. If you have questions or suggestions related to my behavior, please file an issue against the kubernetes/test-infra repository. I understand the commands that are listed here. |
This updates the HPA controller to use the polymorphic scale client from client-go. This should enable HPAs to work with arbitrary scalable resources, instead of just those in the extensions API group (meaning we can deprecate the copy of ReplicationController in extensions/v1beta1). It also means that the HPA controller now pays attention to the APIVersion field in `scaleTargetRef` (more specifically, the group part of it). Note that currently, discovery information on which resources are available where is only fetched once (the first time that it's requested). In the future, we may want a refreshing discovery REST mapper.
The output of `go test` is passed throug a grep filter that's used when producing junit output. Unfortunately, when testing round-trip conversions with random strings, grep can become convinced that the test output is binary, and will truncate a failed test before any output is displayed, showing "binary file (standard input) matches". This forces grep to consider test output to be text, so that we always see test results.
This adds a new fake scale client (for use in testing) to match the new polymorphic scale client.
4abac06
to
f22bfcd
Compare
ok, rebase issues should be solved |
/lgtm |
[APPROVALNOTIFIER] This PR is APPROVED This pull-request has been approved by: deads2k, DirectXMan12, smarterclayton Associated issue: 29698 The full list of commands accepted by this bot can be found here.
Needs approval from an approver in each of these OWNERS Files:
You can indicate your approval by writing |
/test all [submit-queue is verifying that this PR is safe to merge] |
Automatic merge from submit-queue (batch tested with PRs 53743, 53564). If you want to cherry-pick this change to another branch, please follow the instructions here. |
This PR introduces a polymorphic scale client based on discovery information that's able to scale scalable resources in arbitrary group-versions, as long as they present the scale subresource in their discovery information.
Currently, it supports
extensions/v1beta1.Scale
andautoscaling/v1.Scale
, but supporting other versions of scale if/when we produce them should be fairly trivial.It also updates the HPA to use this client, meaning the HPA will now work on any scalable resource, not just things in the
extensions/v1beta1
API group.Release note:
Unblocks #29698
Unblocks #38756
Unblocks #49504
Fixes #38810