Skip to content

Commit dbfc3aa

Browse files
authored
Merge pull request kubernetes#91748 from apelisse/resetting-managed-fields-and-fieldtype
Resetting managed fields and fieldtype
2 parents 11fe6e8 + 3f10709 commit dbfc3aa

File tree

7 files changed

+250
-26
lines changed

7 files changed

+250
-26
lines changed

staging/src/k8s.io/apimachinery/pkg/apis/meta/v1/validation/validation.go

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -178,7 +178,7 @@ func ValidateManagedFields(fieldsList []metav1.ManagedFieldsEntry, fldPath *fiel
178178
default:
179179
allErrs = append(allErrs, field.Invalid(fldPath.Child("operation"), fields.Operation, "must be `Apply` or `Update`"))
180180
}
181-
if fields.FieldsType != "FieldsV1" {
181+
if len(fields.FieldsType) > 0 && fields.FieldsType != "FieldsV1" {
182182
allErrs = append(allErrs, field.Invalid(fldPath.Child("fieldsType"), fields.FieldsType, "must be `FieldsV1`"))
183183
}
184184
}

staging/src/k8s.io/apimachinery/pkg/apis/meta/v1/validation/validation_test.go

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -242,12 +242,8 @@ func TestValidateFieldManagerInvalid(t *testing.T) {
242242
}
243243
}
244244

245-
func TestValidateMangedFieldsInvalid(t *testing.T) {
245+
func TestValidateManagedFieldsInvalid(t *testing.T) {
246246
tests := []metav1.ManagedFieldsEntry{
247-
{
248-
Operation: metav1.ManagedFieldsOperationUpdate,
249-
// FieldsType is missing
250-
},
251247
{
252248
Operation: metav1.ManagedFieldsOperationUpdate,
253249
FieldsType: "RandomVersion",
@@ -274,6 +270,10 @@ func TestValidateMangedFieldsInvalid(t *testing.T) {
274270

275271
func TestValidateMangedFieldsValid(t *testing.T) {
276272
tests := []metav1.ManagedFieldsEntry{
273+
{
274+
Operation: metav1.ManagedFieldsOperationUpdate,
275+
// FieldsType is missing
276+
},
277277
{
278278
Operation: metav1.ManagedFieldsOperationUpdate,
279279
FieldsType: "FieldsV1",

staging/src/k8s.io/apiserver/pkg/endpoints/handlers/fieldmanager/fieldmanager.go

Lines changed: 30 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ package fieldmanager
1818

1919
import (
2020
"fmt"
21+
"reflect"
2122
"time"
2223

2324
"k8s.io/apimachinery/pkg/api/meta"
@@ -111,21 +112,25 @@ func newDefaultFieldManager(f Manager, objectCreater runtime.ObjectCreater, kind
111112
func (f *FieldManager) Update(liveObj, newObj runtime.Object, manager string) (object runtime.Object, err error) {
112113
// If the object doesn't have metadata, we should just return without trying to
113114
// set the managedFields at all, so creates/updates/patches will work normally.
114-
if _, err = meta.Accessor(newObj); err != nil {
115+
newAccessor, err := meta.Accessor(newObj)
116+
if err != nil {
115117
return newObj, nil
116118
}
117119

118120
// First try to decode the managed fields provided in the update,
119121
// This is necessary to allow directly updating managed fields.
120122
var managed Managed
121-
if managed, err = internal.DecodeObjectManagedFields(newObj); err != nil || len(managed.Fields()) == 0 {
123+
if isResetManagedFields(newAccessor.GetManagedFields()) {
124+
managed = internal.NewEmptyManaged()
125+
} else if managed, err = internal.DecodeObjectManagedFields(newAccessor.GetManagedFields()); err != nil || len(managed.Fields()) == 0 {
126+
liveAccessor, err := meta.Accessor(liveObj)
127+
if err != nil {
128+
return newObj, nil
129+
}
122130
// If the managed field is empty or we failed to decode it,
123131
// let's try the live object. This is to prevent clients who
124132
// don't understand managedFields from deleting it accidentally.
125-
managed, err = internal.DecodeObjectManagedFields(liveObj)
126-
if err != nil {
127-
// If we also can't decode the liveObject, then
128-
// just restart managedFields from scratch.
133+
if managed, err = internal.DecodeObjectManagedFields(liveAccessor.GetManagedFields()); err != nil {
129134
managed = internal.NewEmptyManaged()
130135
}
131136
}
@@ -163,17 +168,33 @@ func (f *FieldManager) UpdateNoErrors(liveObj, newObj runtime.Object, manager st
163168
return obj
164169
}
165170

171+
// Returns true if the managedFields indicate that the user is trying to
172+
// reset the managedFields, i.e. if the list is non-nil but empty, or if
173+
// the list has one empty item.
174+
func isResetManagedFields(managedFields []metav1.ManagedFieldsEntry) bool {
175+
if len(managedFields) == 0 {
176+
return managedFields != nil
177+
}
178+
179+
if len(managedFields) == 1 {
180+
return reflect.DeepEqual(managedFields[0], metav1.ManagedFieldsEntry{})
181+
}
182+
183+
return false
184+
}
185+
166186
// Apply is used when server-side apply is called, as it merges the
167187
// object and updates the managed fields.
168188
func (f *FieldManager) Apply(liveObj, appliedObj runtime.Object, manager string, force bool) (object runtime.Object, err error) {
169189
// If the object doesn't have metadata, apply isn't allowed.
170-
if _, err = meta.Accessor(liveObj); err != nil {
190+
accessor, err := meta.Accessor(liveObj)
191+
if err != nil {
171192
return nil, fmt.Errorf("couldn't get accessor: %v", err)
172193
}
173194

174195
// Decode the managed fields in the live object, since it isn't allowed in the patch.
175-
var managed Managed
176-
if managed, err = internal.DecodeObjectManagedFields(liveObj); err != nil {
196+
managed, err := internal.DecodeObjectManagedFields(accessor.GetManagedFields())
197+
if err != nil {
177198
return nil, fmt.Errorf("failed to decode managed fields: %v", err)
178199
}
179200

staging/src/k8s.io/apiserver/pkg/endpoints/handlers/fieldmanager/fieldmanager_test.go

Lines changed: 83 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -783,3 +783,86 @@ func TestNoOpChanges(t *testing.T) {
783783
t.Fatalf("No-op apply has changed the object:\n%v\n---\n%v", before, f.liveObj)
784784
}
785785
}
786+
787+
// Tests that one can reset the managedFields by sending either an empty
788+
// list
789+
func TestResetManagedFieldsEmptyList(t *testing.T) {
790+
f := NewTestFieldManager(schema.FromAPIVersionAndKind("v1", "Pod"))
791+
792+
obj := &unstructured.Unstructured{Object: map[string]interface{}{}}
793+
if err := yaml.Unmarshal([]byte(`{
794+
"apiVersion": "v1",
795+
"kind": "Pod",
796+
"metadata": {
797+
"labels": {
798+
"a": "b"
799+
},
800+
}
801+
}`), &obj.Object); err != nil {
802+
t.Fatalf("error decoding YAML: %v", err)
803+
}
804+
if err := f.Apply(obj, "fieldmanager_test_apply", false); err != nil {
805+
t.Fatalf("failed to apply object: %v", err)
806+
}
807+
808+
if err := yaml.Unmarshal([]byte(`{
809+
"apiVersion": "v1",
810+
"kind": "Pod",
811+
"metadata": {
812+
"managedFields": [],
813+
"labels": {
814+
"a": "b"
815+
},
816+
}
817+
}`), &obj.Object); err != nil {
818+
t.Fatalf("error decoding YAML: %v", err)
819+
}
820+
if err := f.Update(obj, "update_manager"); err != nil {
821+
t.Fatalf("failed to update with empty manager: %v", err)
822+
}
823+
824+
if len(f.ManagedFields()) != 0 {
825+
t.Fatalf("failed to reset managedFields: %v", f.ManagedFields())
826+
}
827+
}
828+
829+
// Tests that one can reset the managedFields by sending either a list with one empty item.
830+
func TestResetManagedFieldsEmptyItem(t *testing.T) {
831+
f := NewTestFieldManager(schema.FromAPIVersionAndKind("v1", "Pod"))
832+
833+
obj := &unstructured.Unstructured{Object: map[string]interface{}{}}
834+
if err := yaml.Unmarshal([]byte(`{
835+
"apiVersion": "v1",
836+
"kind": "Pod",
837+
"metadata": {
838+
"labels": {
839+
"a": "b"
840+
},
841+
}
842+
}`), &obj.Object); err != nil {
843+
t.Fatalf("error decoding YAML: %v", err)
844+
}
845+
if err := f.Apply(obj, "fieldmanager_test_apply", false); err != nil {
846+
t.Fatalf("failed to apply object: %v", err)
847+
}
848+
849+
if err := yaml.Unmarshal([]byte(`{
850+
"apiVersion": "v1",
851+
"kind": "Pod",
852+
"metadata": {
853+
"managedFields": [{}],
854+
"labels": {
855+
"a": "b"
856+
},
857+
}
858+
}`), &obj.Object); err != nil {
859+
t.Fatalf("error decoding YAML: %v", err)
860+
}
861+
if err := f.Update(obj, "update_manager"); err != nil {
862+
t.Fatalf("failed to update with empty manager: %v", err)
863+
}
864+
865+
if len(f.ManagedFields()) != 0 {
866+
t.Fatalf("failed to reset managedFields: %v", f.ManagedFields())
867+
}
868+
}

staging/src/k8s.io/apiserver/pkg/endpoints/handlers/fieldmanager/internal/managedfields.go

Lines changed: 12 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -78,16 +78,8 @@ func RemoveObjectManagedFields(obj runtime.Object) {
7878
}
7979

8080
// DecodeObjectManagedFields extracts and converts the objects ManagedFields into a fieldpath.ManagedFields.
81-
func DecodeObjectManagedFields(from runtime.Object) (ManagedInterface, error) {
82-
if from == nil {
83-
return &managedStruct{}, nil
84-
}
85-
accessor, err := meta.Accessor(from)
86-
if err != nil {
87-
panic(fmt.Sprintf("couldn't get accessor: %v", err))
88-
}
89-
90-
managed, err := decodeManagedFields(accessor.GetManagedFields())
81+
func DecodeObjectManagedFields(from []metav1.ManagedFieldsEntry) (ManagedInterface, error) {
82+
managed, err := decodeManagedFields(from)
9183
if err != nil {
9284
return nil, fmt.Errorf("failed to convert managed fields from API: %v", err)
9385
}
@@ -115,7 +107,16 @@ func EncodeObjectManagedFields(obj runtime.Object, managed ManagedInterface) err
115107
func decodeManagedFields(encodedManagedFields []metav1.ManagedFieldsEntry) (managed managedStruct, err error) {
116108
managed.fields = make(fieldpath.ManagedFields, len(encodedManagedFields))
117109
managed.times = make(map[string]*metav1.Time, len(encodedManagedFields))
118-
for _, encodedVersionedSet := range encodedManagedFields {
110+
111+
for i, encodedVersionedSet := range encodedManagedFields {
112+
switch encodedVersionedSet.FieldsType {
113+
case "FieldsV1":
114+
// Valid case.
115+
case "":
116+
return managedStruct{}, fmt.Errorf("missing fieldsType in managed fields entry %d", i)
117+
default:
118+
return managedStruct{}, fmt.Errorf("invalid fieldsType %q in managed fields entry %d", encodedVersionedSet.FieldsType, i)
119+
}
119120
manager, err := BuildManagerIdentifier(&encodedVersionedSet)
120121
if err != nil {
121122
return managedStruct{}, fmt.Errorf("error decoding manager from %v: %v", encodedVersionedSet, err)

staging/src/k8s.io/apiserver/pkg/endpoints/handlers/fieldmanager/internal/managedfields_test.go

Lines changed: 45 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,51 @@ import (
2727
"sigs.k8s.io/yaml"
2828
)
2929

30+
// TestHasFieldsType makes sure that we fail if we don't have a
31+
// FieldsType set properly.
32+
func TestHasFieldsType(t *testing.T) {
33+
var unmarshaled []metav1.ManagedFieldsEntry
34+
if err := yaml.Unmarshal([]byte(`- apiVersion: v1
35+
fieldsType: FieldsV1
36+
fieldsV1:
37+
f:field: {}
38+
manager: foo
39+
operation: Apply
40+
`), &unmarshaled); err != nil {
41+
t.Fatalf("did not expect yaml unmarshalling error but got: %v", err)
42+
}
43+
if _, err := decodeManagedFields(unmarshaled); err != nil {
44+
t.Fatalf("did not expect decoding error but got: %v", err)
45+
}
46+
47+
// Invalid fieldsType V2.
48+
if err := yaml.Unmarshal([]byte(`- apiVersion: v1
49+
fieldsType: FieldsV2
50+
fieldsV1:
51+
f:field: {}
52+
manager: foo
53+
operation: Apply
54+
`), &unmarshaled); err != nil {
55+
t.Fatalf("did not expect yaml unmarshalling error but got: %v", err)
56+
}
57+
if _, err := decodeManagedFields(unmarshaled); err == nil {
58+
t.Fatal("Expect decoding error but got none")
59+
}
60+
61+
// Missing fieldsType.
62+
if err := yaml.Unmarshal([]byte(`- apiVersion: v1
63+
fieldsV1:
64+
f:field: {}
65+
manager: foo
66+
operation: Apply
67+
`), &unmarshaled); err != nil {
68+
t.Fatalf("did not expect yaml unmarshalling error but got: %v", err)
69+
}
70+
if _, err := decodeManagedFields(unmarshaled); err == nil {
71+
t.Fatal("Expect decoding error but got none")
72+
}
73+
}
74+
3075
// TestRoundTripManagedFields will roundtrip ManagedFields from the wire format
3176
// (api format) to the format used by sigs.k8s.io/structured-merge-diff and back
3277
func TestRoundTripManagedFields(t *testing.T) {

test/integration/apiserver/apply/apply_test.go

Lines changed: 74 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1578,6 +1578,80 @@ func TestErrorsDontFailPatch(t *testing.T) {
15781578

15791579
}
15801580

1581+
// TestClearManagedFieldsWithUpdateEmptyList verifies it's possible to clear the managedFields by sending an empty list.
1582+
func TestClearManagedFieldsWithUpdateEmptyList(t *testing.T) {
1583+
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, genericfeatures.ServerSideApply, true)()
1584+
1585+
_, client, closeFn := setup(t)
1586+
defer closeFn()
1587+
1588+
_, err := client.CoreV1().RESTClient().Patch(types.ApplyPatchType).
1589+
Namespace("default").
1590+
Resource("configmaps").
1591+
Name("test-cm").
1592+
Param("fieldManager", "apply_test").
1593+
Body([]byte(`{
1594+
"apiVersion": "v1",
1595+
"kind": "ConfigMap",
1596+
"metadata": {
1597+
"name": "test-cm",
1598+
"namespace": "default",
1599+
"labels": {
1600+
"test-label": "test"
1601+
}
1602+
},
1603+
"data": {
1604+
"key": "value"
1605+
}
1606+
}`)).
1607+
Do(context.TODO()).
1608+
Get()
1609+
if err != nil {
1610+
t.Fatalf("Failed to create object using Apply patch: %v", err)
1611+
}
1612+
1613+
_, err = client.CoreV1().RESTClient().Put().
1614+
Namespace("default").
1615+
Resource("configmaps").
1616+
Name("test-cm").
1617+
Body([]byte(`{
1618+
"apiVersion": "v1",
1619+
"kind": "ConfigMap",
1620+
"metadata": {
1621+
"name": "test-cm",
1622+
"namespace": "default",
1623+
"managedFields": [],
1624+
"labels": {
1625+
"test-label": "test"
1626+
}
1627+
},
1628+
"data": {
1629+
"key": "value"
1630+
}
1631+
}`)).Do(context.TODO()).Get()
1632+
if err != nil {
1633+
t.Fatalf("Failed to patch object: %v", err)
1634+
}
1635+
1636+
object, err := client.CoreV1().RESTClient().Get().Namespace("default").Resource("configmaps").Name("test-cm").Do(context.TODO()).Get()
1637+
if err != nil {
1638+
t.Fatalf("Failed to retrieve object: %v", err)
1639+
}
1640+
1641+
accessor, err := meta.Accessor(object)
1642+
if err != nil {
1643+
t.Fatalf("Failed to get meta accessor: %v", err)
1644+
}
1645+
1646+
if managedFields := accessor.GetManagedFields(); len(managedFields) != 0 {
1647+
t.Fatalf("Failed to clear managedFields, got: %v", managedFields)
1648+
}
1649+
1650+
if labels := accessor.GetLabels(); len(labels) < 1 {
1651+
t.Fatalf("Expected other fields to stay untouched, got: %v", object)
1652+
}
1653+
}
1654+
15811655
var podBytes = []byte(`
15821656
apiVersion: v1
15831657
kind: Pod

0 commit comments

Comments
 (0)