externalsecret_controller_manifest.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. /*
  2. Copyright © The ESO 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. 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. "github.com/go-logr/logr"
  19. v1 "k8s.io/api/core/v1"
  20. apierrors "k8s.io/apimachinery/pkg/api/errors"
  21. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  22. "k8s.io/apimachinery/pkg/runtime/schema"
  23. "sigs.k8s.io/controller-runtime/pkg/client"
  24. esv1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1"
  25. "github.com/external-secrets/external-secrets/pkg/controllers/templating"
  26. "github.com/external-secrets/external-secrets/runtime/template"
  27. )
  28. // isGenericTarget checks if the ExternalSecret targets a generic resource.
  29. func isGenericTarget(es *esv1.ExternalSecret) bool {
  30. return es.Spec.Target.Manifest != nil
  31. }
  32. // validateGenericTarget validates that generic targets are properly configured.
  33. func (r *Reconciler) validateGenericTarget(log logr.Logger, es *esv1.ExternalSecret) error {
  34. if !r.AllowGenericTargets {
  35. return fmt.Errorf("generic targets are disabled. Enable with --unsafe-allow-generic-targets flag")
  36. }
  37. manifest := es.Spec.Target.Manifest
  38. if manifest.APIVersion == "" {
  39. return fmt.Errorf("target.manifest.apiVersion is required")
  40. }
  41. if manifest.Kind == "" {
  42. return fmt.Errorf("target.manifest.kind is required")
  43. }
  44. log.Info("Warning: Using generic target. Make sure access policies and encryption are properly configured.",
  45. "apiVersion", manifest.APIVersion,
  46. "kind", manifest.Kind,
  47. "name", getTargetName(es))
  48. return nil
  49. }
  50. // getTargetGVK returns the GroupVersionKind for the target resource.
  51. func getTargetGVK(es *esv1.ExternalSecret) schema.GroupVersionKind {
  52. manifest := es.Spec.Target.Manifest
  53. gv, _ := schema.ParseGroupVersion(manifest.APIVersion)
  54. return schema.GroupVersionKind{
  55. Group: gv.Group,
  56. Version: gv.Version,
  57. Kind: manifest.Kind,
  58. }
  59. }
  60. // getTargetName returns the name of the target resource.
  61. func getTargetName(es *esv1.ExternalSecret) string {
  62. if es.Spec.Target.Name != "" {
  63. return es.Spec.Target.Name
  64. }
  65. return es.Name
  66. }
  67. // getGenericResource retrieves a generic resource using the controller-runtime client.
  68. func (r *Reconciler) getGenericResource(ctx context.Context, log logr.Logger, es *esv1.ExternalSecret) (*unstructured.Unstructured, error) {
  69. gvk := getTargetGVK(es)
  70. resource := &unstructured.Unstructured{}
  71. resource.SetGroupVersionKind(gvk)
  72. err := r.Client.Get(ctx, client.ObjectKey{
  73. Namespace: es.Namespace,
  74. Name: getTargetName(es),
  75. }, resource)
  76. if err != nil {
  77. if apierrors.IsNotFound(err) {
  78. log.V(1).Info("target resource does not exist", "gvk", gvk.String(), "name", getTargetName(es))
  79. return nil, err
  80. }
  81. return nil, fmt.Errorf("failed to get target resource: %w", err)
  82. }
  83. return resource, nil
  84. }
  85. func (r *Reconciler) createGenericResource(ctx context.Context, log logr.Logger, es *esv1.ExternalSecret, obj *unstructured.Unstructured) error {
  86. gvk := getTargetGVK(es)
  87. // Check if resource already exists
  88. existing := &unstructured.Unstructured{}
  89. existing.SetGroupVersionKind(gvk)
  90. err := r.Client.Get(ctx, client.ObjectKey{
  91. Namespace: es.Namespace,
  92. Name: getTargetName(es),
  93. }, existing)
  94. if err != nil {
  95. if !apierrors.IsNotFound(err) {
  96. return fmt.Errorf("failed to check if target resource exists: %w", err)
  97. }
  98. } else {
  99. return fmt.Errorf("target resource with name %s already exists", getTargetName(es))
  100. }
  101. log.Info("creating target resource", "gvk", gvk.String(), "name", getTargetName(es))
  102. err = r.Client.Create(ctx, obj)
  103. if err != nil {
  104. return fmt.Errorf("failed to create target resource: %w", err)
  105. }
  106. r.recorder.Event(es, v1.EventTypeNormal, "Created", fmt.Sprintf("Created %s %s", gvk.Kind, getTargetName(es)))
  107. return nil
  108. }
  109. func (r *Reconciler) updateGenericResource(ctx context.Context, log logr.Logger, es *esv1.ExternalSecret, existing *unstructured.Unstructured) error {
  110. gvk := getTargetGVK(es)
  111. log.Info("updating target resource", "gvk", gvk.String(), "name", getTargetName(es))
  112. err := r.Client.Update(ctx, existing)
  113. if err != nil {
  114. return fmt.Errorf("failed to update target resource: %w", err)
  115. }
  116. r.recorder.Event(es, v1.EventTypeNormal, "Updated", fmt.Sprintf("Updated %s %s", gvk.Kind, getTargetName(es)))
  117. return nil
  118. }
  119. // deleteGenericResource deletes a generic resource.
  120. func (r *Reconciler) deleteGenericResource(ctx context.Context, log logr.Logger, es *esv1.ExternalSecret) error {
  121. if !r.AllowGenericTargets || !isGenericTarget(es) {
  122. return nil
  123. }
  124. gvk := getTargetGVK(es)
  125. obj := &unstructured.Unstructured{}
  126. obj.SetGroupVersionKind(gvk)
  127. obj.SetNamespace(es.Namespace)
  128. obj.SetName(getTargetName(es))
  129. log.Info("deleting target resource", "gvk", gvk.String(), "name", getTargetName(es))
  130. err := r.Client.Delete(ctx, obj)
  131. if err != nil && !apierrors.IsNotFound(err) {
  132. return fmt.Errorf("failed to delete target resource: %w", err)
  133. }
  134. r.recorder.Event(es, v1.EventTypeNormal, "Deleted", fmt.Sprintf("Deleted %s %s", gvk.Kind, getTargetName(es)))
  135. return nil
  136. }
  137. // applyTemplateToManifest renders templates for generic resources and returns an unstructured object.
  138. // If existingObj is provided, templates will be applied to it (for merge behavior).
  139. // Otherwise, a new object is created.
  140. func (r *Reconciler) applyTemplateToManifest(ctx context.Context, es *esv1.ExternalSecret, dataMap map[string][]byte, existingObj *unstructured.Unstructured) (*unstructured.Unstructured, error) {
  141. var obj *unstructured.Unstructured
  142. if existingObj != nil {
  143. // use the existing object for merge behavior if it exists
  144. obj = existingObj.DeepCopy()
  145. } else {
  146. gvk := getTargetGVK(es)
  147. obj = &unstructured.Unstructured{}
  148. obj.SetGroupVersionKind(gvk)
  149. obj.SetName(getTargetName(es))
  150. obj.SetNamespace(es.Namespace)
  151. switch gvk.Kind {
  152. case "ConfigMap", "Secret":
  153. obj.Object["data"] = map[string]any{}
  154. default:
  155. obj.Object["spec"] = map[string]any{}
  156. }
  157. }
  158. labels := obj.GetLabels()
  159. if labels == nil {
  160. labels = make(map[string]string)
  161. }
  162. annotations := obj.GetAnnotations()
  163. if annotations == nil {
  164. annotations = make(map[string]string)
  165. }
  166. srcLabels, srcAnnotations := es.ObjectMeta.Labels, es.ObjectMeta.Annotations
  167. if es.Spec.Target.Template != nil {
  168. srcLabels = es.Spec.Target.Template.Metadata.Labels
  169. srcAnnotations = es.Spec.Target.Template.Metadata.Annotations
  170. }
  171. maps.Copy(labels, srcLabels)
  172. maps.Copy(annotations, srcAnnotations)
  173. labels[esv1.LabelManaged] = esv1.LabelManagedValue
  174. obj.SetLabels(labels)
  175. obj.SetAnnotations(annotations)
  176. var result *unstructured.Unstructured
  177. var err error
  178. if es.Spec.Target.Template == nil {
  179. result = r.createSimpleManifest(obj, dataMap)
  180. } else {
  181. result, err = r.renderTemplatedManifest(ctx, es, obj, dataMap)
  182. }
  183. if err != nil {
  184. return nil, err
  185. }
  186. ann := result.GetAnnotations()
  187. if ann == nil {
  188. ann = make(map[string]string)
  189. }
  190. hash, err := genericTargetContentHash(result)
  191. if err != nil {
  192. return nil, fmt.Errorf("failed to hash target %q content: %w", es.Spec.Target.Name, err)
  193. }
  194. ann[esv1.AnnotationDataHash] = hash
  195. result.SetAnnotations(ann)
  196. if err := r.applyOwnership(es, result); err != nil {
  197. return nil, err
  198. }
  199. return result, nil
  200. }
  201. // createSimpleManifest creates a simple resource without templates (e.g., ConfigMap with data field).
  202. func (r *Reconciler) createSimpleManifest(obj *unstructured.Unstructured, dataMap map[string][]byte) *unstructured.Unstructured {
  203. // For ConfigMaps and similar resources, put data in .data field
  204. if obj.GetKind() == "ConfigMap" {
  205. data := make(map[string]string)
  206. for k, v := range dataMap {
  207. data[k] = string(v)
  208. }
  209. obj.Object["data"] = data
  210. return obj
  211. }
  212. // For other resources, put in spec.data or just data
  213. data := make(map[string]string)
  214. for k, v := range dataMap {
  215. data[k] = string(v)
  216. }
  217. if obj.Object["spec"] == nil {
  218. obj.Object["spec"] = make(map[string]any)
  219. }
  220. spec := obj.Object["spec"].(map[string]any)
  221. spec["data"] = data
  222. return obj
  223. }
  224. // renderTemplatedManifest renders templates for a custom resource.
  225. func (r *Reconciler) renderTemplatedManifest(ctx context.Context, es *esv1.ExternalSecret, obj *unstructured.Unstructured, dataMap map[string][]byte) (*unstructured.Unstructured, error) {
  226. execute, err := template.EngineForVersion(es.Spec.Target.Template.EngineVersion)
  227. if err != nil {
  228. return nil, fmt.Errorf("failed to get template engine: %w", err)
  229. }
  230. // Handle templateFrom entries
  231. for _, tplFrom := range es.Spec.Target.Template.TemplateFrom {
  232. targetPath := tplFrom.Target
  233. if targetPath == "" {
  234. targetPath = esv1.TemplateTargetData
  235. }
  236. if tplFrom.Literal != nil {
  237. // Execute template directly against the unstructured object
  238. out := make(map[string][]byte)
  239. out[*tplFrom.Literal] = []byte(*tplFrom.Literal)
  240. if err := execute(out, dataMap, esv1.TemplateScopeKeysAndValues, targetPath, obj); err != nil {
  241. return nil, fmt.Errorf("failed to execute literal template: %w", err)
  242. }
  243. }
  244. if tplFrom.ConfigMap != nil || tplFrom.Secret != nil {
  245. // Parser still uses v1.Secret, so collect data and apply via template engine to the end result.
  246. tempSecret := &v1.Secret{Data: make(map[string][]byte)}
  247. p := templating.Parser{
  248. Client: r.Client,
  249. TargetSecret: tempSecret,
  250. DataMap: dataMap,
  251. Exec: execute,
  252. }
  253. if tplFrom.ConfigMap != nil {
  254. if err := p.MergeConfigMap(ctx, es.Namespace, tplFrom); err != nil {
  255. return nil, fmt.Errorf("failed to merge configmap template: %w", err)
  256. }
  257. }
  258. if tplFrom.Secret != nil {
  259. if err := p.MergeSecret(ctx, es.Namespace, tplFrom); err != nil {
  260. return nil, fmt.Errorf("failed to merge secret template: %w", err)
  261. }
  262. }
  263. // apply collected data to the target object
  264. if err := execute(tempSecret.Data, dataMap, esv1.TemplateScopeValues, targetPath, obj); err != nil {
  265. return nil, fmt.Errorf("failed to apply merged templates to path %s: %w", targetPath, err)
  266. }
  267. }
  268. }
  269. // Handle template.data entries
  270. if len(es.Spec.Target.Template.Data) > 0 {
  271. tplMap := make(map[string][]byte)
  272. for k, v := range es.Spec.Target.Template.Data {
  273. tplMap[k] = []byte(v)
  274. }
  275. if err := execute(tplMap, dataMap, esv1.TemplateScopeValues, esv1.TemplateTargetData, obj); err != nil {
  276. return nil, fmt.Errorf("failed to execute template.data: %w", err)
  277. }
  278. }
  279. return obj, nil
  280. }