skipnonapplied.go 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  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. "math/rand"
  17. "k8s.io/apimachinery/pkg/api/meta"
  18. "k8s.io/apimachinery/pkg/runtime"
  19. )
  20. type skipNonAppliedManager struct {
  21. fieldManager Manager
  22. objectCreater runtime.ObjectCreater
  23. beforeApplyManagerName string
  24. probability float32
  25. }
  26. var _ Manager = &skipNonAppliedManager{}
  27. // NewSkipNonAppliedManager creates a new wrapped FieldManager that only starts tracking managers after the first apply.
  28. func NewSkipNonAppliedManager(fieldManager Manager, objectCreater runtime.ObjectCreater) Manager {
  29. return NewProbabilisticSkipNonAppliedManager(fieldManager, objectCreater, 0.0)
  30. }
  31. // NewProbabilisticSkipNonAppliedManager creates a new wrapped FieldManager that starts tracking managers after the first apply,
  32. // or starts tracking on create with p probability.
  33. func NewProbabilisticSkipNonAppliedManager(fieldManager Manager, objectCreater runtime.ObjectCreater, p float32) Manager {
  34. return &skipNonAppliedManager{
  35. fieldManager: fieldManager,
  36. objectCreater: objectCreater,
  37. beforeApplyManagerName: "before-first-apply",
  38. probability: p,
  39. }
  40. }
  41. // Update implements Manager.
  42. func (f *skipNonAppliedManager) Update(liveObj, newObj runtime.Object, managed Managed, manager string) (runtime.Object, Managed, error) {
  43. accessor, err := meta.Accessor(liveObj)
  44. if err != nil {
  45. return newObj, managed, nil
  46. }
  47. // If managed fields is empty, we need to determine whether to skip tracking managed fields.
  48. if len(managed.Fields()) == 0 {
  49. // Check if the operation is a create, by checking whether lastObj's UID is empty.
  50. // If the operation is create, P(tracking managed fields) = f.probability
  51. // If the operation is update, skip tracking managed fields, since we already know managed fields is empty.
  52. if len(accessor.GetUID()) == 0 {
  53. if f.probability <= rand.Float32() {
  54. return newObj, managed, nil
  55. }
  56. } else {
  57. return newObj, managed, nil
  58. }
  59. }
  60. return f.fieldManager.Update(liveObj, newObj, managed, manager)
  61. }
  62. // Apply implements Manager.
  63. func (f *skipNonAppliedManager) Apply(liveObj, appliedObj runtime.Object, managed Managed, fieldManager string, force bool) (runtime.Object, Managed, error) {
  64. if len(managed.Fields()) == 0 {
  65. gvk := appliedObj.GetObjectKind().GroupVersionKind()
  66. emptyObj, err := f.objectCreater.New(gvk)
  67. if err != nil {
  68. return nil, nil, fmt.Errorf("failed to create empty object of type %v: %v", gvk, err)
  69. }
  70. liveObj, managed, err = f.fieldManager.Update(emptyObj, liveObj, managed, f.beforeApplyManagerName)
  71. if err != nil {
  72. return nil, nil, fmt.Errorf("failed to create manager for existing fields: %v", err)
  73. }
  74. }
  75. return f.fieldManager.Apply(liveObj, appliedObj, managed, fieldManager, force)
  76. }