externalsecret_controller_template.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. /*
  2. Copyright © 2025 ESO Maintainer Team
  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. https://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 externalsecret
  14. import (
  15. "context"
  16. "fmt"
  17. "maps"
  18. v1 "k8s.io/api/core/v1"
  19. "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
  20. esv1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1"
  21. "github.com/external-secrets/external-secrets/pkg/controllers/templating"
  22. "github.com/external-secrets/external-secrets/pkg/template"
  23. "github.com/external-secrets/external-secrets/pkg/utils"
  24. _ "github.com/external-secrets/external-secrets/pkg/provider/register" // Loading registered providers.
  25. )
  26. // merge template in the following order:
  27. // * template.Data (highest precedence)
  28. // * template.TemplateFrom
  29. // * secret via es.data or es.dataFrom (if template.MergePolicy is Merge, or there is no template)
  30. // * existing secret keys (if CreationPolicy is Merge).
  31. func (r *Reconciler) ApplyTemplate(ctx context.Context, es *esv1.ExternalSecret, secret *v1.Secret, dataMap map[string][]byte) error {
  32. // update metadata (labels, annotations, finalizers) of the secret
  33. if err := setMetadata(secret, es); err != nil {
  34. return err
  35. }
  36. // we only keep existing keys if creation policy is Merge, otherwise we clear the secret
  37. if es.Spec.Target.CreationPolicy != esv1.CreatePolicyMerge {
  38. secret.Data = make(map[string][]byte)
  39. }
  40. // no template: copy data and return
  41. if es.Spec.Target.Template == nil {
  42. maps.Insert(secret.Data, maps.All(dataMap))
  43. return nil
  44. }
  45. // set the secret type if it is defined in the template, otherwise keep the existing type
  46. if es.Spec.Target.Template.Type != "" {
  47. secret.Type = es.Spec.Target.Template.Type
  48. }
  49. // when TemplateMergePolicy is Merge, or there is no data template, we include the keys from `dataMap`
  50. noTemplate := len(es.Spec.Target.Template.Data) == 0 && len(es.Spec.Target.Template.TemplateFrom) == 0
  51. if es.Spec.Target.Template.MergePolicy == esv1.MergePolicyMerge || noTemplate {
  52. maps.Insert(secret.Data, maps.All(dataMap))
  53. }
  54. execute, err := template.EngineForVersion(es.Spec.Target.Template.EngineVersion)
  55. if err != nil {
  56. return err
  57. }
  58. p := templating.Parser{
  59. Client: r.Client,
  60. TargetSecret: secret,
  61. DataMap: dataMap,
  62. Exec: execute,
  63. }
  64. // apply templates defined in template.templateFrom
  65. err = p.MergeTemplateFrom(ctx, es.Namespace, es.Spec.Target.Template)
  66. if err != nil {
  67. return fmt.Errorf(errFetchTplFrom, err)
  68. }
  69. // apply data templates
  70. // NOTE: explicitly defined template.data templates take precedence over templateFrom
  71. err = p.MergeMap(es.Spec.Target.Template.Data, esv1.TemplateTargetData)
  72. if err != nil {
  73. return fmt.Errorf(errExecTpl, err)
  74. }
  75. // apply templates for labels
  76. // NOTE: this only works for v2 templates
  77. err = p.MergeMap(es.Spec.Target.Template.Metadata.Labels, esv1.TemplateTargetLabels)
  78. if err != nil {
  79. return fmt.Errorf(errExecTpl, err)
  80. }
  81. // apply template for annotations
  82. // NOTE: this only works for v2 templates
  83. err = p.MergeMap(es.Spec.Target.Template.Metadata.Annotations, esv1.TemplateTargetAnnotations)
  84. if err != nil {
  85. return fmt.Errorf(errExecTpl, err)
  86. }
  87. return nil
  88. }
  89. // setMetadata sets Labels and Annotations to the given secret.
  90. func setMetadata(secret *v1.Secret, es *esv1.ExternalSecret) error {
  91. // ensure that Labels and Annotations are not nil
  92. // so it is safe to merge them
  93. if secret.Labels == nil {
  94. secret.Labels = make(map[string]string)
  95. }
  96. if secret.Annotations == nil {
  97. secret.Annotations = make(map[string]string)
  98. }
  99. // remove any existing labels managed by this external secret
  100. // this is to ensure that we don't have any stale labels
  101. labelKeys, err := templating.GetManagedLabelKeys(secret, es.Name)
  102. if err != nil {
  103. return err
  104. }
  105. for _, key := range labelKeys {
  106. delete(secret.ObjectMeta.Labels, key)
  107. }
  108. annotationKeys, err := templating.GetManagedAnnotationKeys(secret, es.Name)
  109. if err != nil {
  110. return err
  111. }
  112. for _, key := range annotationKeys {
  113. delete(secret.ObjectMeta.Annotations, key)
  114. }
  115. // if no template is defined, copy labels and annotations from the ExternalSecret
  116. if es.Spec.Target.Template == nil {
  117. utils.MergeStringMap(secret.ObjectMeta.Labels, es.ObjectMeta.Labels)
  118. utils.MergeStringMap(secret.ObjectMeta.Annotations, es.ObjectMeta.Annotations)
  119. return nil
  120. }
  121. // copy labels and annotations from the template
  122. utils.MergeStringMap(secret.ObjectMeta.Labels, es.Spec.Target.Template.Metadata.Labels)
  123. utils.MergeStringMap(secret.ObjectMeta.Annotations, es.Spec.Target.Template.Metadata.Annotations)
  124. // add finalizers from the template
  125. if secret.ObjectMeta.DeletionTimestamp.IsZero() {
  126. for _, finalizer := range es.Spec.Target.Template.Metadata.Finalizers {
  127. if !controllerutil.ContainsFinalizer(secret, finalizer) {
  128. controllerutil.AddFinalizer(secret, finalizer)
  129. }
  130. }
  131. }
  132. return nil
  133. }