-
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
Make UnstructuredContent return contents without mutating the source #62063
Make UnstructuredContent return contents without mutating the source #62063
Conversation
/retest |
out := u.Object | ||
if out == nil { | ||
out = make(map[string]interface{}) | ||
out := make(map[string]interface{}, len(u.Object)+1) |
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 +1?
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 contents may not have items
.
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 the case in line 62 should not happen?
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.
u.Object
may or may not contain items
. We can remove if
in the loop because the key will be overwritten anyway. It made sense to have it before (in the previous PR) because a deep copy was made here and skipping items
reduced the unnecessary work. 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.
I've removed the if
.
Could we make |
It is quite convenient to be able to construct one of these as a literal, especially as part of a bigger literal. For example https://github.com/atlassian/smith/blob/ba1ff1e867c74ee29f1172257b5b555134ca2e77/pkg/controller/spec_processor_test.go#L332-L338 However I can also see value in hiding it and making everyone access the contents via the methods. Maybe we can have a constructor |
A constructor sounds good. |
@sttts can we merge this one first and then I'll submit another one with the constructor change? It is going to be a big PR and I think it makes sense to keep the changes separate. Also this is a bugfix and a refactoring - should be separate. |
5d14bf4
to
d5fdac3
Compare
@@ -2133,14 +2133,14 @@ URL: http://localhost | |||
"name": "MyName", | |||
"namespace": "MyNamespace", | |||
"creationTimestamp": "2017-04-01T00:00:00Z", | |||
"resourceVersion": 123, | |||
"resourceVersion": int64(123), |
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 is unrelated. Can you move it to another PR? (I cannot approve 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.
Most likely it was causing a failure because 123
literal has an int
type by default but deep copy only likes float64
and int64
. But this fix is no longer needed because we are not making a deep copy.
var u Unstructured | ||
content := u.UnstructuredContent() | ||
expContent := make(map[string]interface{}) | ||
assert.EqualValues(t, expContent, content) |
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 check that the actual object stays the same? Make a deep copy before and then compare afterwards.
@@ -0,0 +1,30 @@ | |||
/* | |||
Copyright 2017 The Kubernetes Authors. |
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.
2018
yes, makes sense. |
switch x := x.(type) { | ||
case map[string]interface{}: | ||
if x == 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.
does this really exist?
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, without this change (and below for completeness) the newly added check in the test (deep copy and compare) was failing. https://play.golang.org/p/KASoa2DeTnp
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 will change the output - this code used to produce non-nil maps and slices for typed nils but now will produce typed nils. JSON encoding will change - typed nils as null
in JSON (https://play.golang.org/p/Uz2ajLEQx9A).
I think this change is a good one - it makes the deep copy more precise.
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 should never see typed nils somewhere deep inside IMO. On the toplevel maybe.
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, we should not see them, but they exist. I'm fine with removing this fix from this PR if that's what you think is best. However I think that way deep copy is more correct. It just mirrors the input precisely. If there are no typed nils, we don't introduce them here. If they create a problem somewhere then the place that "spawns" them should be fixed. Or, even better, a place that does not handle them well could be made more robust.
Also, I'm not sure what exactly you wanted me to compare? typed and untyped nils are both marshaled into null
, yes :)
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.
Have we ever seen this "cannot deep copy" panic? You say that they show up. Do you know where?
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.
Panic? Current code does not cause any panics AFAIK. Are you talking about failing test without these changes? It does not panic, it just fails because objects are not equal. They are not equal because the deep copied Object
field ("expected" in the test) is an empty map and the original is `nil.
func TestNilUnstructuredContent(t *testing.T) {
var u Unstructured
uCopy := u.DeepCopy()
content := u.UnstructuredContent()
expContent := make(map[string]interface{})
assert.EqualValues(t, expContent, content)
assert.Equal(t, uCopy, &u)
}
--- FAIL: TestNilUnstructuredContent (0.00s)
Error Trace: unstructured_test.go:31
Error: Not equal:
expected: &unstructured.Unstructured{Object:map[string]interface {}{}}
actual: &unstructured.Unstructured{Object:map[string]interface {}(nil)}
Diff:
--- Expected
+++ Actual
@@ -1,4 +1,3 @@
(*unstructured.Unstructured)({
- Object: (map[string]interface {}) {
- }
+ Object: (map[string]interface {}) <nil>
})
/retest |
clone := make([]interface{}, len(x)) | ||
for i, v := range x { | ||
clone[i] = DeepCopyJSONValue(v) | ||
} | ||
return clone | ||
case string, int64, bool, float64, nil, encodingjson.Number: | ||
case string, int64, bool, float64, encodingjson.Number: |
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 not keep the nil and remove the if clause above?
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.
Good question. My IntelliJ is complaining that nil
is not a valid type but Go compiler is happy 🤷♂️ It actually works fine https://play.golang.org/p/PBIyhFgWn8H
I'll revert this change.
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.
Don't feel strong about it. Was just curious.
/retest |
the verify job is broken: #62162 /lgtm |
[APPROVALNOTIFIER] This PR is APPROVED This pull-request has been approved by: ash2k, sttts 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 |
/retest Review the full test history for this PR. Silence the bot with an |
2 similar comments
/retest Review the full test history for this PR. Silence the bot with an |
/retest Review the full test history for this PR. Silence the bot with an |
/test all [submit-queue is verifying that this PR is safe to merge] |
Automatic merge from submit-queue. If you want to cherry-pick this change to another branch, please follow the instructions here. |
cc @mbohlool |
What this PR does / why we need it:
This PR solves the issues described in #56316
Before this change:
UnstructuredContent()
potentially modifiedObject
UnstructuredContent()
could be manipulated to modify the value inObject
. Going through the history it looks like this behavior was added before the addition ofSetUnstructuredContent()
. IMO it makes more sense now to useSetUnstructuredContent()
or make changes to the exposedObject
propertyUnstructuredList
did not implement the behavior described in the godoc. The godoc stated that the value returned should be mutable, but if u.Object == nil the map returned had no effect on ObjectWith this PR I'm proposing
UnstructuredContent()
returns the data without providing the contract of a mutable map. It also ensures all implementations of theUnstructured
interface abide by the docWhich issue(s) this PR fixes:
Fixes #56316
Special notes for your reviewer:
This PR continues work started in #57713.
Release note:
/kind bug
/sig api-machinery
/cc sttts deads2k