generator.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package resolvers
  13. import (
  14. "context"
  15. "fmt"
  16. "reflect"
  17. apiextensions "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
  18. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  19. "k8s.io/apimachinery/pkg/runtime"
  20. "k8s.io/apimachinery/pkg/runtime/schema"
  21. "k8s.io/apimachinery/pkg/types"
  22. "sigs.k8s.io/controller-runtime/pkg/client"
  23. "sigs.k8s.io/controller-runtime/pkg/reconcile"
  24. esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
  25. genv1alpha1 "github.com/external-secrets/external-secrets/apis/generators/v1alpha1"
  26. )
  27. // these errors are explicitly defined so we can detect them with `errors.Is()`.
  28. var (
  29. // ErrUnableToGetGenerator is returned when a generator reference cannot be resolved.
  30. ErrUnableToGetGenerator = fmt.Errorf("unable to get generator")
  31. )
  32. // GeneratorRef resolves a generator reference to a generator implementation.
  33. func GeneratorRef(ctx context.Context, cl client.Client, scheme *runtime.Scheme, namespace string, generatorRef *esv1beta1.GeneratorRef) (genv1alpha1.Generator, *apiextensions.JSON, error) {
  34. generator, jsonObj, err := getGenerator(ctx, cl, scheme, namespace, generatorRef)
  35. if err != nil {
  36. return nil, nil, fmt.Errorf("%w: %w", ErrUnableToGetGenerator, err)
  37. }
  38. return generator, jsonObj, nil
  39. }
  40. func getGenerator(ctx context.Context, cl client.Client, scheme *runtime.Scheme, namespace string, generatorRef *esv1beta1.GeneratorRef) (genv1alpha1.Generator, *apiextensions.JSON, error) {
  41. // get a GVK from the generatorRef
  42. gv, err := schema.ParseGroupVersion(generatorRef.APIVersion)
  43. if err != nil {
  44. return nil, nil, reconcile.TerminalError(fmt.Errorf("generatorRef has invalid APIVersion: %w", err))
  45. }
  46. gvk := schema.GroupVersionKind{
  47. Group: gv.Group,
  48. Version: gv.Version,
  49. Kind: generatorRef.Kind,
  50. }
  51. // fail if the GVK does not use the generator group
  52. if gvk.Group != genv1alpha1.Group {
  53. return nil, nil, reconcile.TerminalError(fmt.Errorf("generatorRef may only reference the generators group, but got %s", gvk.Group))
  54. }
  55. // get a client Object from the GVK
  56. t, exists := scheme.AllKnownTypes()[gvk]
  57. if !exists {
  58. return nil, nil, reconcile.TerminalError(fmt.Errorf("generatorRef references unknown GVK %s", gvk))
  59. }
  60. obj := reflect.New(t).Interface().(client.Object)
  61. // this interface provides the Generate() method used by the controller
  62. // NOTE: all instances of a generator kind use the same instance of this interface
  63. var generator genv1alpha1.Generator
  64. // ClusterGenerator is a special case because it's a cluster-scoped resource
  65. // to use it, we create a "virtual" namespaced generator for the current namespace, as if one existed in the API
  66. if gvk.Kind == genv1alpha1.ClusterGeneratorKind {
  67. clusterGenerator := obj.(*genv1alpha1.ClusterGenerator)
  68. // get the cluster generator resource from the API
  69. // NOTE: it's important that we use the structured client so we use the cache
  70. err = cl.Get(ctx, client.ObjectKey{Name: generatorRef.Name}, clusterGenerator)
  71. if err != nil {
  72. return nil, nil, err
  73. }
  74. // convert the cluster generator to a virtual namespaced generator object
  75. obj, err = clusterGeneratorToVirtual(clusterGenerator)
  76. if err != nil {
  77. return nil, nil, reconcile.TerminalError(fmt.Errorf("invalid ClusterGenerator: %w", err))
  78. }
  79. // get the generator interface
  80. var ok bool
  81. generator, ok = genv1alpha1.GetGeneratorByName(string(clusterGenerator.Spec.Kind))
  82. if !ok {
  83. return nil, nil, reconcile.TerminalError(fmt.Errorf("ClusterGenerator has unknown kind %s", clusterGenerator.Spec.Kind))
  84. }
  85. } else {
  86. // get the generator resource from the API
  87. // NOTE: it's important that we use the structured client so we use the cache
  88. err = cl.Get(ctx, types.NamespacedName{
  89. Name: generatorRef.Name,
  90. Namespace: namespace,
  91. }, obj)
  92. if err != nil {
  93. return nil, nil, err
  94. }
  95. // get the generator interface
  96. var ok bool
  97. generator, ok = genv1alpha1.GetGeneratorByName(gvk.Kind)
  98. if !ok {
  99. return nil, nil, reconcile.TerminalError(fmt.Errorf("generatorRef has unknown kind %s", gvk.Kind))
  100. }
  101. }
  102. // convert the generator to unstructured object
  103. u := &unstructured.Unstructured{}
  104. u.Object, err = runtime.DefaultUnstructuredConverter.ToUnstructured(obj)
  105. if err != nil {
  106. return nil, nil, err
  107. }
  108. // convert the unstructured object to JSON
  109. // NOTE: we do this for backwards compatibility with how this API works, not because it's a good idea
  110. // we should refactor the generator API to use the normal typed objects
  111. jsonObj, err := u.MarshalJSON()
  112. if err != nil {
  113. return nil, nil, err
  114. }
  115. return generator, &apiextensions.JSON{Raw: jsonObj}, nil
  116. }
  117. // clusterGeneratorToVirtual converts a ClusterGenerator to a "virtual" namespaced generator that doesn't actually exist in the API.
  118. func clusterGeneratorToVirtual(gen *genv1alpha1.ClusterGenerator) (client.Object, error) {
  119. switch gen.Spec.Kind {
  120. case genv1alpha1.GeneratorKindACRAccessToken:
  121. if gen.Spec.Generator.ACRAccessTokenSpec == nil {
  122. return nil, fmt.Errorf("when kind is %s, ACRAccessTokenSpec must be set", gen.Spec.Kind)
  123. }
  124. return &genv1alpha1.ACRAccessToken{
  125. Spec: *gen.Spec.Generator.ACRAccessTokenSpec,
  126. }, nil
  127. case genv1alpha1.GeneratorKindECRAuthorizationToken:
  128. if gen.Spec.Generator.ECRAuthorizationTokenSpec == nil {
  129. return nil, fmt.Errorf("when kind is %s, ECRAuthorizationTokenSpec must be set", gen.Spec.Kind)
  130. }
  131. return &genv1alpha1.ECRAuthorizationToken{
  132. Spec: *gen.Spec.Generator.ECRAuthorizationTokenSpec,
  133. }, nil
  134. case genv1alpha1.GeneratorKindFake:
  135. if gen.Spec.Generator.FakeSpec == nil {
  136. return nil, fmt.Errorf("when kind is %s, FakeSpec must be set", gen.Spec.Kind)
  137. }
  138. return &genv1alpha1.Fake{
  139. Spec: *gen.Spec.Generator.FakeSpec,
  140. }, nil
  141. case genv1alpha1.GeneratorKindGCRAccessToken:
  142. if gen.Spec.Generator.GCRAccessTokenSpec == nil {
  143. return nil, fmt.Errorf("when kind is %s, GCRAccessTokenSpec must be set", gen.Spec.Kind)
  144. }
  145. return &genv1alpha1.GCRAccessToken{
  146. Spec: *gen.Spec.Generator.GCRAccessTokenSpec,
  147. }, nil
  148. case genv1alpha1.GeneratorKindGithubAccessToken:
  149. if gen.Spec.Generator.GithubAccessTokenSpec == nil {
  150. return nil, fmt.Errorf("when kind is %s, GithubAccessTokenSpec must be set", gen.Spec.Kind)
  151. }
  152. return &genv1alpha1.GithubAccessToken{
  153. Spec: *gen.Spec.Generator.GithubAccessTokenSpec,
  154. }, nil
  155. case genv1alpha1.GeneratorKindQuayAccessToken:
  156. if gen.Spec.Generator.QuayAccessTokenSpec == nil {
  157. return nil, fmt.Errorf("when kind is %s, QuayAccessTokenSpec must be set", gen.Spec.Kind)
  158. }
  159. return &genv1alpha1.QuayAccessToken{
  160. Spec: *gen.Spec.Generator.QuayAccessTokenSpec,
  161. }, nil
  162. case genv1alpha1.GeneratorKindPassword:
  163. if gen.Spec.Generator.PasswordSpec == nil {
  164. return nil, fmt.Errorf("when kind is %s, PasswordSpec must be set", gen.Spec.Kind)
  165. }
  166. return &genv1alpha1.Password{
  167. Spec: *gen.Spec.Generator.PasswordSpec,
  168. }, nil
  169. case genv1alpha1.GeneratorKindSTSSessionToken:
  170. if gen.Spec.Generator.STSSessionTokenSpec == nil {
  171. return nil, fmt.Errorf("when kind is %s, STSSessionTokenSpec must be set", gen.Spec.Kind)
  172. }
  173. return &genv1alpha1.STSSessionToken{
  174. Spec: *gen.Spec.Generator.STSSessionTokenSpec,
  175. }, nil
  176. case genv1alpha1.GeneratorKindUUID:
  177. if gen.Spec.Generator.UUIDSpec == nil {
  178. return nil, fmt.Errorf("when kind is %s, UUIDSpec must be set", gen.Spec.Kind)
  179. }
  180. return &genv1alpha1.UUID{
  181. Spec: *gen.Spec.Generator.UUIDSpec,
  182. }, nil
  183. case genv1alpha1.GeneratorKindVaultDynamicSecret:
  184. if gen.Spec.Generator.VaultDynamicSecretSpec == nil {
  185. return nil, fmt.Errorf("when kind is %s, VaultDynamicSecretSpec must be set", gen.Spec.Kind)
  186. }
  187. return &genv1alpha1.VaultDynamicSecret{
  188. Spec: *gen.Spec.Generator.VaultDynamicSecretSpec,
  189. }, nil
  190. case genv1alpha1.GeneratorKindWebhook:
  191. if gen.Spec.Generator.WebhookSpec == nil {
  192. return nil, fmt.Errorf("when kind is %s, WebhookSpec must be set", gen.Spec.Kind)
  193. }
  194. return &genv1alpha1.Webhook{
  195. Spec: *gen.Spec.Generator.WebhookSpec,
  196. }, nil
  197. default:
  198. return nil, fmt.Errorf("unknown kind %s", gen.Spec.Kind)
  199. }
  200. }