-
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
Useful helper functions for Unstructured #51940
Useful helper functions for Unstructured #51940
Conversation
Hi @ash2k. Thanks for your PR. I'm waiting for a kubernetes member to verify that this patch is reasonable to test. If it is, they should reply with I understand the commands that are listed here. 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. |
/ok-to-test |
nested := getNestedField(obj, fields...) | ||
// NestedInt64Pointer returns the *int64 value of a nested field. | ||
// Returns nil if value is not found or is not an int64/*int64. | ||
func NestedInt64Pointer(obj map[string]interface{}, fields ...string) *int64 { |
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 a pointer and not the value?
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 other funcs return values or clones, never the original object or a reference to it.
// NestedString returns the string value of a nested field. | ||
// Returns empty string if value is not found or is not a string. | ||
func NestedString(obj map[string]interface{}, fields ...string) string { | ||
if str, ok := NestedField(obj, fields...).(string); ok { |
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 prefer to see error
s in the result if the type is wrong.
Same for the other type-casts below.
// NestedStringSlice returns the []string value of a nested field. | ||
// Returns nil if value is not found or is not a []interface{}. | ||
func NestedStringSlice(obj map[string]interface{}, fields ...string) []string { | ||
if m, ok := NestedField(obj, fields...).([]interface{}); ok { | ||
strSlice := make([]string, 0, len(m)) | ||
for _, v := range m { | ||
if str, ok := v.(string); ok { |
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 prefer to return an error if the cast fails.
// NestedStringMap returns the map[string]string value of a nested field. | ||
// Returns nil if value is not found or is not a map[string]interface{}. | ||
func NestedStringMap(obj map[string]interface{}, fields ...string) map[string]string { | ||
if m, ok := NestedField(obj, fields...).(map[string]interface{}); ok { | ||
strMap := make(map[string]string, len(m)) | ||
for k, v := range m { | ||
if str, ok := v.(string); ok { |
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.
same here, I prefer an error in this case
m = m[field].(map[string]interface{}) | ||
for _, field := range fields[:len(fields)-1] { | ||
if _, ok := m[field].(map[string]interface{}); !ok { | ||
m[field] = make(map[string]interface{}) |
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 unexpected that everything of wrong type is overridden.
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.
Also here we need errors.
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.
Well, I agree with you. This is how it is implemented right now, I just made it public. I'm happy to change it.
Do you want all of these functions to return an error
? Maybe bool
for getters?
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.
bool
is also fine, maybe even better.
/assign @krousey |
cbb565a
to
36c5315
Compare
@sttts Would be great to get this merged into 1.9, PTAL. |
return | ||
} | ||
ts, _ := timestamp.MarshalQueryParameter() | ||
u.setNestedField(ts, "metadata", "deletionTimestamp") | ||
} | ||
|
||
func (u *Unstructured) GetDeletionGracePeriodSeconds() *int64 { | ||
return getNestedInt64Pointer(u.Object, "metadata", "deletionGracePeriodSeconds") | ||
val, ok := NestedInt64(u.Object, "metadata", "deletionGracePeriodSeconds") |
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.
related #54803
Can you squash commit 2, 3 and 5? |
func setNestedField(obj map[string]interface{}, value interface{}, fields ...string) { | ||
// SetNestedField sets the value of a nested field. | ||
// Returns false if value cannot be set because one of the nesting levels is not a map[string]interface{}. | ||
func SetNestedField(obj map[string]interface{}, value interface{}, fields ...string) 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.
I wished we had a debugging/strict-test mode where value
is type checked. We use(d) to write arbitrary values into objects (#54803) and the compiler cannot help us with type checks.
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 can send a follow up PR with a environment variable flag a-la
kubernetes/staging/src/k8s.io/apimachinery/pkg/conversion/unstructured/converter.go
Line 81 in 51e653d
DefaultConverter = NewConverter(parseBool(os.Getenv("KUBE_PATCH_CONVERSION_DETECTOR"))) |
mutationDetectionEnabled, _ = strconv.ParseBool(os.Getenv("KUBE_CACHE_MUTATION_DETECTOR")) |
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.
Would be great to run unit and maybe even integration tests with 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.
Follow up in #55297
delete(m, fields[len(fields)-1]) | ||
} | ||
|
||
func getNestedString(obj map[string]interface{}, fields ...string) string { |
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.
nit: nestedStringOrDefault with default string
arg?
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.
getNestedString()
is used in lots of places and default is an empty string everywhere. That is why I kept it as it is. Also I didn't want to touch the name because it would've increased the PR size even further. I can send a follow-up PR with rename if needed.
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.
Makes sense.
func setNestedSlice(obj map[string]interface{}, value []string, fields ...string) { | ||
// SetNestedStringSlice sets the string slice value of a nested field. | ||
// Returns false if value cannot be set because one of the nesting levels is not a map[string]interface{}. | ||
func SetNestedStringSlice(obj map[string]interface{}, value []string, fields ...string) 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.
For the setters the bool return value feels unnatural.
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 an explicit error here look better? Something like that in #55168 (comment)?
No functional changes
55604c9
to
7c10cbc
Compare
Rebased+squashed. I left first commit unchanged - it is just moving code around, no changes. Second one with all the changes. |
/retest |
@deads2k feel free to go through the code. More eyes help in this highly untyped and fragile area. |
} | ||
|
||
gvk := obj.GetObjectKind().GroupVersionKind() | ||
if len(gvk.Kind) == 0 { |
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.
Straight move? I'd expect unstructured objects to always have a gvk.Version
too.
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, I haven't touched this code. See 7aeaedd#diff-31522f2d01b79a28cf7d5adaa03d5b26L685
@@ -478,31 +272,47 @@ func (u *Unstructured) GetDeletionTimestamp() *metav1.Time { | |||
|
|||
func (u *Unstructured) SetDeletionTimestamp(timestamp *metav1.Time) { | |||
if timestamp == nil { | |||
u.setNestedField(nil, "metadata", "deletionTimestamp") | |||
RemoveNestedField(u.Object, "metadata", "deletionTimestamp") |
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'd expect a similar remove on SetCreationTimestamp
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.
Creation timestamp is not optional, at least the argument of the Go func is not. So at worst, we miss a feature here on the Go level.
/lgtm |
[APPROVALNOTIFIER] This PR is APPROVED This pull-request has been approved by: ash2k, deads2k, sttts Associated issue: 40790 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 |
Automatic merge from submit-queue (batch tested with PRs 54787, 51940). If you want to cherry-pick this change to another branch, please follow the instructions here. |
These shouldn't be in this package - I'm moving them into runtime, because meta/v1/unstructured is only for "typed" unstructured (i.e., anything that obeys our metadata rules), not generic to unstructured. |
That makes sense. Thanks for an update. |
Which issue this PR fixes:
Fixes #40790
Release note:
/kind feature
/sig api-machinery
/area client-libraries
/assign @sttts @liggitt