structuredmerge.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. /*
  2. Copyright 2019 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package internal
  14. import (
  15. "fmt"
  16. "sigs.k8s.io/structured-merge-diff/v6/fieldpath"
  17. "sigs.k8s.io/structured-merge-diff/v6/merge"
  18. "sigs.k8s.io/structured-merge-diff/v6/typed"
  19. "k8s.io/apimachinery/pkg/api/errors"
  20. "k8s.io/apimachinery/pkg/api/meta"
  21. "k8s.io/apimachinery/pkg/runtime"
  22. "k8s.io/apimachinery/pkg/runtime/schema"
  23. )
  24. type structuredMergeManager struct {
  25. typeConverter TypeConverter
  26. objectConverter runtime.ObjectConvertor
  27. objectDefaulter runtime.ObjectDefaulter
  28. groupVersion schema.GroupVersion
  29. hubVersion schema.GroupVersion
  30. updater merge.Updater
  31. }
  32. var _ Manager = &structuredMergeManager{}
  33. // NewStructuredMergeManager creates a new Manager that merges apply requests
  34. // and update managed fields for other types of requests.
  35. func NewStructuredMergeManager(typeConverter TypeConverter, objectConverter runtime.ObjectConvertor, objectDefaulter runtime.ObjectDefaulter, gv schema.GroupVersion, hub schema.GroupVersion, resetFields map[fieldpath.APIVersion]fieldpath.Filter) (Manager, error) {
  36. if typeConverter == nil {
  37. return nil, fmt.Errorf("typeconverter must not be nil")
  38. }
  39. return &structuredMergeManager{
  40. typeConverter: typeConverter,
  41. objectConverter: objectConverter,
  42. objectDefaulter: objectDefaulter,
  43. groupVersion: gv,
  44. hubVersion: hub,
  45. updater: merge.Updater{
  46. Converter: newVersionConverter(typeConverter, objectConverter, hub), // This is the converter provided to SMD from k8s
  47. IgnoreFilter: resetFields,
  48. },
  49. }, nil
  50. }
  51. // NewCRDStructuredMergeManager creates a new Manager specifically for
  52. // CRDs. This allows for the possibility of fields which are not defined
  53. // in models, as well as having no models defined at all.
  54. func NewCRDStructuredMergeManager(typeConverter TypeConverter, objectConverter runtime.ObjectConvertor, objectDefaulter runtime.ObjectDefaulter, gv schema.GroupVersion, hub schema.GroupVersion, resetFields map[fieldpath.APIVersion]fieldpath.Filter) (_ Manager, err error) {
  55. return &structuredMergeManager{
  56. typeConverter: typeConverter,
  57. objectConverter: objectConverter,
  58. objectDefaulter: objectDefaulter,
  59. groupVersion: gv,
  60. hubVersion: hub,
  61. updater: merge.Updater{
  62. Converter: newCRDVersionConverter(typeConverter, objectConverter, hub),
  63. IgnoreFilter: resetFields,
  64. },
  65. }, nil
  66. }
  67. func objectGVKNN(obj runtime.Object) string {
  68. name := "<unknown>"
  69. namespace := "<unknown>"
  70. if accessor, err := meta.Accessor(obj); err == nil {
  71. name = accessor.GetName()
  72. namespace = accessor.GetNamespace()
  73. }
  74. return fmt.Sprintf("%v/%v; %v", namespace, name, obj.GetObjectKind().GroupVersionKind())
  75. }
  76. // Update implements Manager.
  77. func (f *structuredMergeManager) Update(liveObj, newObj runtime.Object, managed Managed, manager string) (runtime.Object, Managed, error) {
  78. newObjVersioned, err := f.toVersioned(newObj)
  79. if err != nil {
  80. return nil, nil, fmt.Errorf("failed to convert new object (%v) to proper version (%v): %v", objectGVKNN(newObj), f.groupVersion, err)
  81. }
  82. liveObjVersioned, err := f.toVersioned(liveObj)
  83. if err != nil {
  84. return nil, nil, fmt.Errorf("failed to convert live object (%v) to proper version: %v", objectGVKNN(liveObj), err)
  85. }
  86. newObjTyped, err := f.typeConverter.ObjectToTyped(newObjVersioned, typed.AllowDuplicates)
  87. if err != nil {
  88. return nil, nil, fmt.Errorf("failed to convert new object (%v) to smd typed: %v", objectGVKNN(newObjVersioned), err)
  89. }
  90. liveObjTyped, err := f.typeConverter.ObjectToTyped(liveObjVersioned, typed.AllowDuplicates)
  91. if err != nil {
  92. return nil, nil, fmt.Errorf("failed to convert live object (%v) to smd typed: %v", objectGVKNN(liveObjVersioned), err)
  93. }
  94. apiVersion := fieldpath.APIVersion(f.groupVersion.String())
  95. // TODO(apelisse) use the first return value when unions are implemented
  96. _, managedFields, err := f.updater.Update(liveObjTyped, newObjTyped, apiVersion, managed.Fields(), manager)
  97. if err != nil {
  98. return nil, nil, fmt.Errorf("failed to update ManagedFields (%v): %v", objectGVKNN(newObjVersioned), err)
  99. }
  100. managed = NewManaged(managedFields, managed.Times())
  101. return newObj, managed, nil
  102. }
  103. // Apply implements Manager.
  104. func (f *structuredMergeManager) Apply(liveObj, patchObj runtime.Object, managed Managed, manager string, force bool) (runtime.Object, Managed, error) {
  105. // Check that the patch object has the same version as the live object
  106. if patchVersion := patchObj.GetObjectKind().GroupVersionKind().GroupVersion(); patchVersion != f.groupVersion {
  107. return nil, nil,
  108. errors.NewBadRequest(
  109. fmt.Sprintf("Incorrect version specified in apply patch. "+
  110. "Specified patch version: %s, expected: %s",
  111. patchVersion, f.groupVersion))
  112. }
  113. patchObjMeta, err := meta.Accessor(patchObj)
  114. if err != nil {
  115. return nil, nil, fmt.Errorf("couldn't get accessor: %v", err)
  116. }
  117. if patchObjMeta.GetManagedFields() != nil {
  118. return nil, nil, errors.NewBadRequest("metadata.managedFields must be nil")
  119. }
  120. liveObjVersioned, err := f.toVersioned(liveObj)
  121. if err != nil {
  122. return nil, nil, fmt.Errorf("failed to convert live object (%v) to proper version: %v", objectGVKNN(liveObj), err)
  123. }
  124. // Don't allow duplicates in the applied object.
  125. patchObjTyped, err := f.typeConverter.ObjectToTyped(patchObj)
  126. if err != nil {
  127. return nil, nil, fmt.Errorf("failed to create typed patch object (%v): %v", objectGVKNN(patchObj), err)
  128. }
  129. liveObjTyped, err := f.typeConverter.ObjectToTyped(liveObjVersioned, typed.AllowDuplicates)
  130. if err != nil {
  131. return nil, nil, fmt.Errorf("failed to create typed live object (%v): %v", objectGVKNN(liveObjVersioned), err)
  132. }
  133. apiVersion := fieldpath.APIVersion(f.groupVersion.String())
  134. newObjTyped, managedFields, err := f.updater.Apply(liveObjTyped, patchObjTyped, apiVersion, managed.Fields(), manager, force)
  135. if err != nil {
  136. return nil, nil, err
  137. }
  138. managed = NewManaged(managedFields, managed.Times())
  139. if newObjTyped == nil {
  140. return nil, managed, nil
  141. }
  142. newObj, err := f.typeConverter.TypedToObject(newObjTyped)
  143. if err != nil {
  144. return nil, nil, fmt.Errorf("failed to convert new typed object (%v) to object: %v", objectGVKNN(patchObj), err)
  145. }
  146. newObjVersioned, err := f.toVersioned(newObj)
  147. if err != nil {
  148. return nil, nil, fmt.Errorf("failed to convert new object (%v) to proper version: %v", objectGVKNN(patchObj), err)
  149. }
  150. f.objectDefaulter.Default(newObjVersioned)
  151. newObjUnversioned, err := f.toUnversioned(newObjVersioned)
  152. if err != nil {
  153. return nil, nil, fmt.Errorf("failed to convert to unversioned (%v): %v", objectGVKNN(patchObj), err)
  154. }
  155. return newObjUnversioned, managed, nil
  156. }
  157. func (f *structuredMergeManager) toVersioned(obj runtime.Object) (runtime.Object, error) {
  158. return f.objectConverter.ConvertToVersion(obj, f.groupVersion)
  159. }
  160. func (f *structuredMergeManager) toUnversioned(obj runtime.Object) (runtime.Object, error) {
  161. return f.objectConverter.ConvertToVersion(obj, f.hubVersion)
  162. }