labels.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. /*
  2. Copyright 2018 The Knative 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 kmeta
  14. import (
  15. "fmt"
  16. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  17. "k8s.io/apimachinery/pkg/labels"
  18. "k8s.io/apimachinery/pkg/selection"
  19. )
  20. // The methods in this file are used for managing subresources in cases where
  21. // a controller instantiates different resources for each version of itself.
  22. // There are two sets of methods available here:
  23. // * `*VersionLabel*`: these methods act on `metadata.resourceVersion` and
  24. // create new labels for EVERY change to the resource (incl. `/status`).
  25. // * `*GenerationLabel*`: these methods act on `metadata.generation` and
  26. // create new labels for changes to the resource's "spec" (typically, but
  27. // some K8s resources change `metadata.generation` for annotations as well
  28. // e.g. Deployment).
  29. //
  30. // For example, if an A might instantiate N B's at version 1 and M B's at
  31. // version 2 then it can use MakeVersionLabels to decorate each subresource
  32. // with the appropriate labels for the version at which it was instantiated.
  33. //
  34. // During reconciliation, MakeVersionLabelSelector can be used with the
  35. // informer listers to access the appropriate subresources for the current
  36. // version of the parent resource.
  37. //
  38. // Likewise during reconciliation, MakeOldVersionLabelSelector can be used
  39. // with the API client's DeleteCollection method to clean up subresources
  40. // for older versions of the resource.
  41. // MakeVersionLabels constructs a set of labels to apply to subresources
  42. // instantiated at this version of the parent resource, so that we can
  43. // efficiently select them.
  44. func MakeVersionLabels(om metav1.ObjectMetaAccessor) labels.Set {
  45. return map[string]string{
  46. "controller": string(om.GetObjectMeta().GetUID()),
  47. "version": om.GetObjectMeta().GetResourceVersion(),
  48. }
  49. }
  50. // MakeVersionLabelSelector constructs a selector for subresources
  51. // instantiated at this version of the parent resource. This keys
  52. // off of the labels populated by MakeVersionLabels.
  53. func MakeVersionLabelSelector(om metav1.ObjectMetaAccessor) labels.Selector {
  54. return labels.SelectorFromSet(MakeVersionLabels(om))
  55. }
  56. // MakeOldVersionLabelSelector constructs a selector for subresources
  57. // instantiated at an older version of the parent resource. This keys
  58. // off of the labels populated by MakeVersionLabels.
  59. func MakeOldVersionLabelSelector(om metav1.ObjectMetaAccessor) labels.Selector {
  60. return labels.NewSelector().Add(
  61. mustNewRequirement("controller", selection.Equals, []string{string(om.GetObjectMeta().GetUID())}),
  62. mustNewRequirement("version", selection.NotEquals, []string{om.GetObjectMeta().GetResourceVersion()}),
  63. )
  64. }
  65. // MakeGenerationLabels constructs a set of labels to apply to subresources
  66. // instantiated at this version of the parent resource, so that we can
  67. // efficiently select them.
  68. func MakeGenerationLabels(om metav1.ObjectMetaAccessor) labels.Set {
  69. return map[string]string{
  70. "controller": string(om.GetObjectMeta().GetUID()),
  71. "generation": genStr(om),
  72. }
  73. }
  74. // MakeGenerationLabelSelector constructs a selector for subresources
  75. // instantiated at this version of the parent resource. This keys
  76. // off of the labels populated by MakeGenerationLabels.
  77. func MakeGenerationLabelSelector(om metav1.ObjectMetaAccessor) labels.Selector {
  78. return labels.SelectorFromSet(MakeGenerationLabels(om))
  79. }
  80. // MakeOldGenerationLabelSelector constructs a selector for subresources
  81. // instantiated at an older version of the parent resource. This keys
  82. // off of the labels populated by MakeGenerationLabels.
  83. func MakeOldGenerationLabelSelector(om metav1.ObjectMetaAccessor) labels.Selector {
  84. return labels.NewSelector().Add(
  85. mustNewRequirement("controller", selection.Equals, []string{string(om.GetObjectMeta().GetUID())}),
  86. mustNewRequirement("generation", selection.NotEquals, []string{genStr(om)}),
  87. )
  88. }
  89. func genStr(om metav1.ObjectMetaAccessor) string {
  90. return fmt.Sprintf("%05d", om.GetObjectMeta().GetGeneration())
  91. }
  92. // mustNewRequirement panics if there are any errors constructing our selectors.
  93. func mustNewRequirement(key string, op selection.Operator, vals []string) labels.Requirement {
  94. r, err := labels.NewRequirement(key, op, vals)
  95. if err != nil {
  96. panic(fmt.Sprintf("mustNewRequirement(%v, %v, %v) = %v", key, op, vals, err))
  97. }
  98. return *r
  99. }