-
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
Add support for JSON patch in fake client #69330
Conversation
@p0lyn0mial I'm having a hard time understanding how the modifications on the two files that keep failing the tests should be making their way into the pkg/client/... on the main kubernetes. There's two tests bazel-test and typecheck for example: and I'm at a bit off a loss and feeling like I'm missing something obvious somewhere, got any pointers? Thanks in advance. |
ok, so looks like fake_node_expansion.go and fake_event_expansion.go are not recreated when you run ./hack/update-codegen.sh. it's recreated, but if I remove the fake_*expansion.go they are however not. |
/assign @caesarxuchao |
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 reviewed the first commit. Could you squash the other commits into meaningful ones?
if err = json.Unmarshal(modified, obj); err != nil { | ||
return true, nil, err | ||
} | ||
default: |
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 we return error by default? Does it break tests?
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.
perhaps I misunderstand what you mean here, but by default is what the code did before that assumed that everything was a strategic patch (by ignoring the incoming patch type). So, I assume this code was added for a reason (with not explicit tests that I can see) so if I start returning error, that seems bad to me. Do you mean, that if the patch type is not strategicmergepatch that I return error and see if tests break?
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.
In general the patch strategy is specified by the value of the patchStrategy
key in a field tag.(in a struct) Additionally kubectl
defaults to strategic
, I'm not sure if there is server side defaulting ?
I think that the idea here is to return an error if the strategy was not specified because it would be better if the tests were more explicit. Hopefully returning an error by default will not fail other tests :)
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 real server returns error is the patch type isn't specified:
if !sets.NewString(patchTypes...).Has(contentType) { | |
scope.err(negotiation.NewUnsupportedMediaTypeError(patchTypes), w, req) | |
return |
Do you mean, that if the patch type is not strategicmergepatch that I return error and see if tests break?
Yes. If tests break, and it requires significant changes to fix, please let us know. We can fix it in another PR.
// Before adding JSONPatch support, PatchType was ignored and everything | ||
// was treated as strategic merge patch. JSONPatch flat out failed, | ||
// so support for it has been added, not sure about the merge patch | ||
// behaviour, so leaving it as is. |
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 history isn't useful. I would remove the comments.
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.
done
// so support for it has been added, not sure about the merge patch | ||
// behaviour, so leaving it as is. | ||
|
||
// obj gets set to the new unmarshalled patched object |
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.
Isn't this as expected? The comment doesn't seem to add additional information.
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.
done
// Name is a descriptive naem for this test suitable as a first argument to t.Run() | ||
Name string | ||
|
||
// Object contain the object to start the test 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.
contains
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 we make this struct private and remove the comment? The fields are easy to understand.
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.
done
PatchedObject: newUnstructuredWithSpec(map[string]interface{}{"foo": "bar"}), | ||
}, { | ||
Name: "strategic merge change", | ||
Object: newUnstructured(testAPIVersion, testKind, testNamespace, testName), |
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 use a v1.Pod in the strategic merge patch test cases? Strategic merge patch is not supported for Unstructured
. The test passed because the patch modified a field that's not a map or list.
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.
Perhaps I'm confused here on what you are asking for, but since this is the fake for dynamic/simple, all it deals with is unstructured objects. I'm not modifying any of the existing code wrt. to strategic merge patch, just adding a JSON patch semantics, which were impossible to pass on before this change and everything failed at runtime. Bsically, I just want to add a test here that simply keeps the existing behavior
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.
Yes, the dynamic client deals with unstructured objects. But the patch is applied to an object at the server side, the server doesn't care if it's sent via a dynamic client or not.
Concretely, if the dynamic client sends a strategic merge patch for a custom resource (which is implemented as Unstructured
), it will receive an error. (see #58414 (comment) for background).
I don't want to have a test case that demonstrates wrong use case.
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've removed those test cases and just added a test that returns a failure for the patchtype when not specified. I feel like adding those tests is something that should have been done when the code was initially checked in, so adding those tests could be done in a follow on PR. Fair?
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.
ok, so looks like fake_node_expansion.go and fake_event_expansion.go are not recreated when you run ./hack/update-codegen.sh.
you are right, the fake_*expansion.go are not generated. They were created manually. The others files were generated by the client-gen
https://github.com/kubernetes/kubernetes/tree/master/staging/src/k8s.io/code-generator/cmd/client-gen
PatchBytes []byte | ||
|
||
// If test should fail or not | ||
WantErr bool |
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.
how about removing this field and using only WantErrMsg
? Where a non empty vale implies that an error was wanted.
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.
done
@@ -40,6 +52,20 @@ func newUnstructured(apiVersion, kind, namespace, name string) *unstructured.Uns | |||
} | |||
} | |||
|
|||
func newUnstructuredWithSpec(spec map[string]interface{}) *unstructured.Unstructured { | |||
return &unstructured.Unstructured{ |
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 could call newUnstructured
method and then just add spec
key.
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.
done
@@ -64,3 +90,144 @@ func TestList(t *testing.T) { | |||
t.Fatal(diff.ObjectGoPrintDiff(expected, listFirst.Items)) | |||
} | |||
} | |||
|
|||
type PatchTestCase 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.
I would either make it private or inline (TestPatch), so that it doesn't leak when someone imports fake
pkg.
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.
done
/test pull-kubernetes-e2e-gce-100-performance |
Thanks for the reviews, rebased and addressed the points I understood. PTAL. |
expectedPatchedObject runtime.Object | ||
} | ||
|
||
func (tc *patchTestCase) Runner(t *testing.T) { |
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 make the methods private as well?
Thanks! PTAL. |
/lgtm |
yay! had to rebase, @caesarxuchao could you re-apply lgtm please. |
/approve |
Updated the release notes. Leaving lgtm to @caesarxuchao. |
/lgtm |
@lavalamp can you help approve? Thanks. |
is there somebody else besides @lavalamp that could take a look. My worry is that something else gets merged, need to rebase and need to start re-hunting for lgtm/approve from scratch again. |
/approve |
[APPROVALNOTIFIER] This PR is APPROVED This pull-request has been approved by: lavalamp, sttts, vaikas-google The full list of commands accepted by this bot can be found here. The pull request process is described here
Needs approval from an approver in each of these files:
Approvers can indicate their approval by writing |
if err != nil { | ||
return true, nil, err | ||
} | ||
if err = json.Unmarshal(modified, obj); err != nil { |
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 unfortunately only works with Unstructured objects. Normal kube objects will fail on this because Unmarshal doesn't reset maps and a few other behaviors. I'll take fixing this.
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.
Fixed in #78743
What this PR does / why we need it:
Adds support for JSON Patch in client-go/dynamic/fake
Which issue(s) this PR fixes (optional, in
fixes #<issue number>(, fixes #<issue_number>, ...)
format, will close the issue(s) when PR gets merged):Fixes kubernetes/client-go#478
Special notes for your reviewer:
Release note: