managedfields.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. /*
  2. Copyright 2018 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. "encoding/json"
  16. "fmt"
  17. "sort"
  18. "k8s.io/apimachinery/pkg/api/meta"
  19. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  20. "k8s.io/apimachinery/pkg/runtime"
  21. "sigs.k8s.io/structured-merge-diff/v4/fieldpath"
  22. )
  23. // ManagedInterface groups a fieldpath.ManagedFields together with the timestamps associated with each operation.
  24. type ManagedInterface interface {
  25. // Fields gets the fieldpath.ManagedFields.
  26. Fields() fieldpath.ManagedFields
  27. // Times gets the timestamps associated with each operation.
  28. Times() map[string]*metav1.Time
  29. }
  30. type managedStruct struct {
  31. fields fieldpath.ManagedFields
  32. times map[string]*metav1.Time
  33. }
  34. var _ ManagedInterface = &managedStruct{}
  35. // Fields implements ManagedInterface.
  36. func (m *managedStruct) Fields() fieldpath.ManagedFields {
  37. return m.fields
  38. }
  39. // Times implements ManagedInterface.
  40. func (m *managedStruct) Times() map[string]*metav1.Time {
  41. return m.times
  42. }
  43. // NewEmptyManaged creates an empty ManagedInterface.
  44. func NewEmptyManaged() ManagedInterface {
  45. return NewManaged(fieldpath.ManagedFields{}, map[string]*metav1.Time{})
  46. }
  47. // NewManaged creates a ManagedInterface from a fieldpath.ManagedFields and the timestamps associated with each operation.
  48. func NewManaged(f fieldpath.ManagedFields, t map[string]*metav1.Time) ManagedInterface {
  49. return &managedStruct{
  50. fields: f,
  51. times: t,
  52. }
  53. }
  54. // RemoveObjectManagedFields removes the ManagedFields from the object
  55. // before we merge so that it doesn't appear in the ManagedFields
  56. // recursively.
  57. func RemoveObjectManagedFields(obj runtime.Object) {
  58. accessor, err := meta.Accessor(obj)
  59. if err != nil {
  60. panic(fmt.Sprintf("couldn't get accessor: %v", err))
  61. }
  62. accessor.SetManagedFields(nil)
  63. }
  64. // EncodeObjectManagedFields converts and stores the fieldpathManagedFields into the objects ManagedFields
  65. func EncodeObjectManagedFields(obj runtime.Object, managed ManagedInterface) error {
  66. accessor, err := meta.Accessor(obj)
  67. if err != nil {
  68. panic(fmt.Sprintf("couldn't get accessor: %v", err))
  69. }
  70. encodedManagedFields, err := encodeManagedFields(managed)
  71. if err != nil {
  72. return fmt.Errorf("failed to convert back managed fields to API: %v", err)
  73. }
  74. accessor.SetManagedFields(encodedManagedFields)
  75. return nil
  76. }
  77. // DecodeManagedFields converts ManagedFields from the wire format (api format)
  78. // to the format used by sigs.k8s.io/structured-merge-diff
  79. func DecodeManagedFields(encodedManagedFields []metav1.ManagedFieldsEntry) (ManagedInterface, error) {
  80. managed := managedStruct{}
  81. managed.fields = make(fieldpath.ManagedFields, len(encodedManagedFields))
  82. managed.times = make(map[string]*metav1.Time, len(encodedManagedFields))
  83. for i, encodedVersionedSet := range encodedManagedFields {
  84. switch encodedVersionedSet.Operation {
  85. case metav1.ManagedFieldsOperationApply, metav1.ManagedFieldsOperationUpdate:
  86. default:
  87. return nil, fmt.Errorf("operation must be `Apply` or `Update`")
  88. }
  89. if len(encodedVersionedSet.APIVersion) < 1 {
  90. return nil, fmt.Errorf("apiVersion must not be empty")
  91. }
  92. switch encodedVersionedSet.FieldsType {
  93. case "FieldsV1":
  94. // Valid case.
  95. case "":
  96. return nil, fmt.Errorf("missing fieldsType in managed fields entry %d", i)
  97. default:
  98. return nil, fmt.Errorf("invalid fieldsType %q in managed fields entry %d", encodedVersionedSet.FieldsType, i)
  99. }
  100. manager, err := BuildManagerIdentifier(&encodedVersionedSet)
  101. if err != nil {
  102. return nil, fmt.Errorf("error decoding manager from %v: %v", encodedVersionedSet, err)
  103. }
  104. managed.fields[manager], err = decodeVersionedSet(&encodedVersionedSet)
  105. if err != nil {
  106. return nil, fmt.Errorf("error decoding versioned set from %v: %v", encodedVersionedSet, err)
  107. }
  108. managed.times[manager] = encodedVersionedSet.Time
  109. }
  110. return &managed, nil
  111. }
  112. // BuildManagerIdentifier creates a manager identifier string from a ManagedFieldsEntry
  113. func BuildManagerIdentifier(encodedManager *metav1.ManagedFieldsEntry) (manager string, err error) {
  114. encodedManagerCopy := *encodedManager
  115. // Never include fields type in the manager identifier
  116. encodedManagerCopy.FieldsType = ""
  117. // Never include the fields in the manager identifier
  118. encodedManagerCopy.FieldsV1 = nil
  119. // Never include the time in the manager identifier
  120. encodedManagerCopy.Time = nil
  121. // For appliers, don't include the APIVersion in the manager identifier,
  122. // so it will always have the same manager identifier each time it applied.
  123. if encodedManager.Operation == metav1.ManagedFieldsOperationApply {
  124. encodedManagerCopy.APIVersion = ""
  125. }
  126. // Use the remaining fields to build the manager identifier
  127. b, err := json.Marshal(&encodedManagerCopy)
  128. if err != nil {
  129. return "", fmt.Errorf("error marshalling manager identifier: %v", err)
  130. }
  131. return string(b), nil
  132. }
  133. func decodeVersionedSet(encodedVersionedSet *metav1.ManagedFieldsEntry) (versionedSet fieldpath.VersionedSet, err error) {
  134. fields := EmptyFields
  135. if encodedVersionedSet.FieldsV1 != nil {
  136. fields = *encodedVersionedSet.FieldsV1
  137. }
  138. set, err := FieldsToSet(fields)
  139. if err != nil {
  140. return nil, fmt.Errorf("error decoding set: %v", err)
  141. }
  142. return fieldpath.NewVersionedSet(&set, fieldpath.APIVersion(encodedVersionedSet.APIVersion), encodedVersionedSet.Operation == metav1.ManagedFieldsOperationApply), nil
  143. }
  144. // encodeManagedFields converts ManagedFields from the format used by
  145. // sigs.k8s.io/structured-merge-diff to the wire format (api format)
  146. func encodeManagedFields(managed ManagedInterface) (encodedManagedFields []metav1.ManagedFieldsEntry, err error) {
  147. if len(managed.Fields()) == 0 {
  148. return nil, nil
  149. }
  150. encodedManagedFields = []metav1.ManagedFieldsEntry{}
  151. for manager := range managed.Fields() {
  152. versionedSet := managed.Fields()[manager]
  153. v, err := encodeManagerVersionedSet(manager, versionedSet)
  154. if err != nil {
  155. return nil, fmt.Errorf("error encoding versioned set for %v: %v", manager, err)
  156. }
  157. if t, ok := managed.Times()[manager]; ok {
  158. v.Time = t
  159. }
  160. encodedManagedFields = append(encodedManagedFields, *v)
  161. }
  162. return sortEncodedManagedFields(encodedManagedFields)
  163. }
  164. func sortEncodedManagedFields(encodedManagedFields []metav1.ManagedFieldsEntry) (sortedManagedFields []metav1.ManagedFieldsEntry, err error) {
  165. sort.Slice(encodedManagedFields, func(i, j int) bool {
  166. p, q := encodedManagedFields[i], encodedManagedFields[j]
  167. if p.Operation != q.Operation {
  168. return p.Operation < q.Operation
  169. }
  170. pSeconds, qSeconds := int64(0), int64(0)
  171. if p.Time != nil {
  172. pSeconds = p.Time.Unix()
  173. }
  174. if q.Time != nil {
  175. qSeconds = q.Time.Unix()
  176. }
  177. if pSeconds != qSeconds {
  178. return pSeconds < qSeconds
  179. }
  180. if p.Manager != q.Manager {
  181. return p.Manager < q.Manager
  182. }
  183. if p.APIVersion != q.APIVersion {
  184. return p.APIVersion < q.APIVersion
  185. }
  186. return p.Subresource < q.Subresource
  187. })
  188. return encodedManagedFields, nil
  189. }
  190. func encodeManagerVersionedSet(manager string, versionedSet fieldpath.VersionedSet) (encodedVersionedSet *metav1.ManagedFieldsEntry, err error) {
  191. encodedVersionedSet = &metav1.ManagedFieldsEntry{}
  192. // Get as many fields as we can from the manager identifier
  193. err = json.Unmarshal([]byte(manager), encodedVersionedSet)
  194. if err != nil {
  195. return nil, fmt.Errorf("error unmarshalling manager identifier %v: %v", manager, err)
  196. }
  197. // Get the APIVersion, Operation, and Fields from the VersionedSet
  198. encodedVersionedSet.APIVersion = string(versionedSet.APIVersion())
  199. if versionedSet.Applied() {
  200. encodedVersionedSet.Operation = metav1.ManagedFieldsOperationApply
  201. }
  202. encodedVersionedSet.FieldsType = "FieldsV1"
  203. fields, err := SetToFields(*versionedSet.Set())
  204. if err != nil {
  205. return nil, fmt.Errorf("error encoding set: %v", err)
  206. }
  207. encodedVersionedSet.FieldsV1 = &fields
  208. return encodedVersionedSet, nil
  209. }