utils.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678
  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 utils
  13. import (
  14. "bytes"
  15. "context"
  16. "crypto/md5" //nolint:gosec
  17. "crypto/x509"
  18. "encoding/base64"
  19. "encoding/json"
  20. "encoding/pem"
  21. "errors"
  22. "fmt"
  23. "net"
  24. "net/url"
  25. "reflect"
  26. "regexp"
  27. "strconv"
  28. "strings"
  29. tpl "text/template"
  30. "time"
  31. "unicode"
  32. "github.com/go-logr/logr"
  33. corev1 "k8s.io/api/core/v1"
  34. apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
  35. "sigs.k8s.io/controller-runtime/pkg/client"
  36. esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
  37. esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
  38. esmeta "github.com/external-secrets/external-secrets/apis/meta/v1"
  39. "github.com/external-secrets/external-secrets/pkg/template/v2"
  40. "github.com/external-secrets/external-secrets/pkg/utils/resolvers"
  41. )
  42. const (
  43. errParse = "unable to parse transform template: %s"
  44. errExecute = "unable to execute transform template: %s"
  45. )
  46. var (
  47. errKeyNotFound = errors.New("key not found")
  48. unicodeRegex = regexp.MustCompile(`_U([0-9a-fA-F]{4,5})_`)
  49. )
  50. // JSONMarshal takes an interface and returns a new escaped and encoded byte slice.
  51. func JSONMarshal(t any) ([]byte, error) {
  52. buffer := &bytes.Buffer{}
  53. encoder := json.NewEncoder(buffer)
  54. encoder.SetEscapeHTML(false)
  55. err := encoder.Encode(t)
  56. return bytes.TrimRight(buffer.Bytes(), "\n"), err
  57. }
  58. // MergeByteMap merges map of byte slices.
  59. func MergeByteMap(dst, src map[string][]byte) map[string][]byte {
  60. for k, v := range src {
  61. dst[k] = v
  62. }
  63. return dst
  64. }
  65. func RewriteMap(operations []esv1beta1.ExternalSecretRewrite, in map[string][]byte) (map[string][]byte, error) {
  66. out := in
  67. var err error
  68. for i, op := range operations {
  69. if op.Regexp != nil {
  70. out, err = RewriteRegexp(*op.Regexp, out)
  71. if err != nil {
  72. return nil, fmt.Errorf("failed rewriting regexp operation[%v]: %w", i, err)
  73. }
  74. }
  75. if op.Transform != nil {
  76. out, err = RewriteTransform(*op.Transform, out)
  77. if err != nil {
  78. return nil, fmt.Errorf("failed rewriting transform operation[%v]: %w", i, err)
  79. }
  80. }
  81. }
  82. return out, nil
  83. }
  84. // RewriteRegexp rewrites a single Regexp Rewrite Operation.
  85. func RewriteRegexp(operation esv1beta1.ExternalSecretRewriteRegexp, in map[string][]byte) (map[string][]byte, error) {
  86. out := make(map[string][]byte)
  87. re, err := regexp.Compile(operation.Source)
  88. if err != nil {
  89. return nil, err
  90. }
  91. for key, value := range in {
  92. newKey := re.ReplaceAllString(key, operation.Target)
  93. out[newKey] = value
  94. }
  95. return out, nil
  96. }
  97. // RewriteTransform applies string transformation on each secret key name to rewrite.
  98. func RewriteTransform(operation esv1beta1.ExternalSecretRewriteTransform, in map[string][]byte) (map[string][]byte, error) {
  99. out := make(map[string][]byte)
  100. for key, value := range in {
  101. data := map[string][]byte{
  102. "value": []byte(key),
  103. }
  104. result, err := transform(operation.Template, data)
  105. if err != nil {
  106. return nil, err
  107. }
  108. newKey := string(result)
  109. out[newKey] = value
  110. }
  111. return out, nil
  112. }
  113. func transform(val string, data map[string][]byte) ([]byte, error) {
  114. strValData := make(map[string]string, len(data))
  115. for k := range data {
  116. strValData[k] = string(data[k])
  117. }
  118. t, err := tpl.New("transform").
  119. Funcs(template.FuncMap()).
  120. Parse(val)
  121. if err != nil {
  122. return nil, fmt.Errorf(errParse, err)
  123. }
  124. buf := bytes.NewBuffer(nil)
  125. err = t.Execute(buf, strValData)
  126. if err != nil {
  127. return nil, fmt.Errorf(errExecute, err)
  128. }
  129. return buf.Bytes(), nil
  130. }
  131. // DecodeMap decodes values from a secretMap.
  132. func DecodeMap(strategy esv1beta1.ExternalSecretDecodingStrategy, in map[string][]byte) (map[string][]byte, error) {
  133. out := make(map[string][]byte, len(in))
  134. for k, v := range in {
  135. val, err := Decode(strategy, v)
  136. if err != nil {
  137. return nil, fmt.Errorf("failure decoding key %v: %w", k, err)
  138. }
  139. out[k] = val
  140. }
  141. return out, nil
  142. }
  143. func Decode(strategy esv1beta1.ExternalSecretDecodingStrategy, in []byte) ([]byte, error) {
  144. switch strategy {
  145. case esv1beta1.ExternalSecretDecodeBase64:
  146. out, err := base64.StdEncoding.DecodeString(string(in))
  147. if err != nil {
  148. return nil, err
  149. }
  150. return out, nil
  151. case esv1beta1.ExternalSecretDecodeBase64URL:
  152. out, err := base64.URLEncoding.DecodeString(string(in))
  153. if err != nil {
  154. return nil, err
  155. }
  156. return out, nil
  157. case esv1beta1.ExternalSecretDecodeNone:
  158. return in, nil
  159. // default when stored version is v1alpha1
  160. case "":
  161. return in, nil
  162. case esv1beta1.ExternalSecretDecodeAuto:
  163. out, err := Decode(esv1beta1.ExternalSecretDecodeBase64, in)
  164. if err != nil {
  165. out, err := Decode(esv1beta1.ExternalSecretDecodeBase64URL, in)
  166. if err != nil {
  167. return Decode(esv1beta1.ExternalSecretDecodeNone, in)
  168. }
  169. return out, nil
  170. }
  171. return out, nil
  172. default:
  173. return nil, fmt.Errorf("decoding strategy %v is not supported", strategy)
  174. }
  175. }
  176. // ValidateKeys checks if the keys in the secret map are valid keys for a Kubernetes secret.
  177. func ValidateKeys(log logr.Logger, in map[string][]byte) error {
  178. for key := range in {
  179. keyLength := len(key)
  180. if keyLength == 0 {
  181. delete(in, key)
  182. log.V(1).Info("key was deleted from the secret output because it did not exist upstream", "key", key)
  183. continue
  184. }
  185. if keyLength > 253 {
  186. return fmt.Errorf("key has length %d but max is 253: (following is truncated): %s", keyLength, key[:253])
  187. }
  188. for _, c := range key {
  189. if !unicode.IsLetter(c) && !unicode.IsNumber(c) && c != '-' && c != '.' && c != '_' {
  190. return fmt.Errorf("key has invalid character %c, only alphanumeric, '-', '.' and '_' are allowed: %s", c, key)
  191. }
  192. }
  193. }
  194. return nil
  195. }
  196. // ConvertKeys converts a secret map into a valid key.
  197. // Replaces any non-alphanumeric characters depending on convert strategy.
  198. func ConvertKeys(strategy esv1beta1.ExternalSecretConversionStrategy, in map[string][]byte) (map[string][]byte, error) {
  199. out := make(map[string][]byte, len(in))
  200. for k, v := range in {
  201. key := convert(strategy, k)
  202. if _, exists := out[key]; exists {
  203. return nil, fmt.Errorf("secret name collision during conversion: %s", key)
  204. }
  205. out[key] = v
  206. }
  207. return out, nil
  208. }
  209. func convert(strategy esv1beta1.ExternalSecretConversionStrategy, str string) string {
  210. rs := []rune(str)
  211. newName := make([]string, len(rs))
  212. for rk, rv := range rs {
  213. if !unicode.IsNumber(rv) &&
  214. !unicode.IsLetter(rv) &&
  215. rv != '-' &&
  216. rv != '.' &&
  217. rv != '_' {
  218. switch strategy {
  219. case esv1beta1.ExternalSecretConversionDefault:
  220. newName[rk] = "_"
  221. case esv1beta1.ExternalSecretConversionUnicode:
  222. newName[rk] = fmt.Sprintf("_U%04x_", rv)
  223. default:
  224. newName[rk] = string(rv)
  225. }
  226. } else {
  227. newName[rk] = string(rv)
  228. }
  229. }
  230. return strings.Join(newName, "")
  231. }
  232. // ReverseKeys reverses a secret map into a valid key map as expected by push secrets.
  233. // Replaces the unicode encoded representation characters back to the actual unicode character depending on convert strategy.
  234. func ReverseKeys(strategy esv1alpha1.PushSecretConversionStrategy, in map[string][]byte) (map[string][]byte, error) {
  235. out := make(map[string][]byte, len(in))
  236. for k, v := range in {
  237. key := reverse(strategy, k)
  238. if _, exists := out[key]; exists {
  239. return nil, fmt.Errorf("secret name collision during conversion: %s", key)
  240. }
  241. out[key] = v
  242. }
  243. return out, nil
  244. }
  245. func reverse(strategy esv1alpha1.PushSecretConversionStrategy, str string) string {
  246. switch strategy {
  247. case esv1alpha1.PushSecretConversionReverseUnicode:
  248. matches := unicodeRegex.FindAllStringSubmatchIndex(str, -1)
  249. for i := len(matches) - 1; i >= 0; i-- {
  250. match := matches[i]
  251. start := match[0]
  252. end := match[1]
  253. unicodeHex := str[match[2]:match[3]]
  254. unicodeInt, err := strconv.ParseInt(unicodeHex, 16, 32)
  255. if err != nil {
  256. continue // Skip invalid unicode representations
  257. }
  258. unicodeChar := fmt.Sprintf("%c", unicodeInt)
  259. str = str[:start] + unicodeChar + str[end:]
  260. }
  261. return str
  262. case esv1alpha1.PushSecretConversionNone:
  263. return str
  264. default:
  265. return str
  266. }
  267. }
  268. // MergeStringMap performs a deep clone from src to dest.
  269. func MergeStringMap(dest, src map[string]string) {
  270. for k, v := range src {
  271. dest[k] = v
  272. }
  273. }
  274. var (
  275. ErrUnexpectedKey = errors.New("unexpected key in data")
  276. ErrSecretType = errors.New("can not handle secret value with type")
  277. )
  278. func GetByteValueFromMap(data map[string]any, key string) ([]byte, error) {
  279. v, ok := data[key]
  280. if !ok {
  281. return nil, fmt.Errorf("%w: %s", ErrUnexpectedKey, key)
  282. }
  283. return GetByteValue(v)
  284. }
  285. func GetByteValue(v any) ([]byte, error) {
  286. switch t := v.(type) {
  287. case string:
  288. return []byte(t), nil
  289. case map[string]any:
  290. return json.Marshal(t)
  291. case []string:
  292. return []byte(strings.Join(t, "\n")), nil
  293. case json.RawMessage:
  294. return t, nil
  295. case []byte:
  296. return t, nil
  297. // also covers int and float32 due to json.Marshal
  298. case float64:
  299. return []byte(strconv.FormatFloat(t, 'f', -1, 64)), nil
  300. case json.Number:
  301. return []byte(t.String()), nil
  302. case []any:
  303. return json.Marshal(t)
  304. case bool:
  305. return []byte(strconv.FormatBool(t)), nil
  306. case nil:
  307. return []byte(nil), nil
  308. default:
  309. return nil, fmt.Errorf("%w: %T", ErrSecretType, t)
  310. }
  311. }
  312. // IsNil checks if an Interface is nil.
  313. func IsNil(i any) bool {
  314. if i == nil {
  315. return true
  316. }
  317. value := reflect.ValueOf(i)
  318. if value.Type().Kind() == reflect.Ptr {
  319. return value.IsNil()
  320. }
  321. return false
  322. }
  323. // ObjectHash calculates md5 sum of the data contained in the secret.
  324. //
  325. //nolint:gosec
  326. func ObjectHash(object any) string {
  327. textualVersion := fmt.Sprintf("%+v", object)
  328. return fmt.Sprintf("%x", md5.Sum([]byte(textualVersion)))
  329. }
  330. func ErrorContains(out error, want string) bool {
  331. if out == nil {
  332. return want == ""
  333. }
  334. if want == "" {
  335. return false
  336. }
  337. return strings.Contains(out.Error(), want)
  338. }
  339. var (
  340. errNamespaceNotAllowed = errors.New("namespace should either be empty or match the namespace of the SecretStore for a namespaced SecretStore")
  341. errRequireNamespace = errors.New("cluster scope requires namespace")
  342. )
  343. // ValidateSecretSelector just checks if the namespace field is present/absent
  344. // depending on the secret store type.
  345. // We MUST NOT check the name or key property here. It MAY be defaulted by the provider.
  346. func ValidateSecretSelector(store esv1beta1.GenericStore, ref esmeta.SecretKeySelector) error {
  347. clusterScope := store.GetObjectKind().GroupVersionKind().Kind == esv1beta1.ClusterSecretStoreKind
  348. if clusterScope && ref.Namespace == nil {
  349. return errRequireNamespace
  350. }
  351. if !clusterScope && ref.Namespace != nil && *ref.Namespace != store.GetNamespace() {
  352. return errNamespaceNotAllowed
  353. }
  354. return nil
  355. }
  356. // ValidateReferentSecretSelector allows
  357. // cluster scoped store without namespace
  358. // this should replace above ValidateServiceAccountSelector once all providers
  359. // support referent auth.
  360. func ValidateReferentSecretSelector(store esv1beta1.GenericStore, ref esmeta.SecretKeySelector) error {
  361. clusterScope := store.GetObjectKind().GroupVersionKind().Kind == esv1beta1.ClusterSecretStoreKind
  362. if !clusterScope && ref.Namespace != nil && *ref.Namespace != store.GetNamespace() {
  363. return errNamespaceNotAllowed
  364. }
  365. return nil
  366. }
  367. // ValidateServiceAccountSelector just checks if the namespace field is present/absent
  368. // depending on the secret store type.
  369. // We MUST NOT check the name or key property here. It MAY be defaulted by the provider.
  370. func ValidateServiceAccountSelector(store esv1beta1.GenericStore, ref esmeta.ServiceAccountSelector) error {
  371. clusterScope := store.GetObjectKind().GroupVersionKind().Kind == esv1beta1.ClusterSecretStoreKind
  372. if clusterScope && ref.Namespace == nil {
  373. return errRequireNamespace
  374. }
  375. if !clusterScope && ref.Namespace != nil && *ref.Namespace != store.GetNamespace() {
  376. return errNamespaceNotAllowed
  377. }
  378. return nil
  379. }
  380. // ValidateReferentServiceAccountSelector allows
  381. // cluster scoped store without namespace
  382. // this should replace above ValidateServiceAccountSelector once all providers
  383. // support referent auth.
  384. func ValidateReferentServiceAccountSelector(store esv1beta1.GenericStore, ref esmeta.ServiceAccountSelector) error {
  385. clusterScope := store.GetObjectKind().GroupVersionKind().Kind == esv1beta1.ClusterSecretStoreKind
  386. if !clusterScope && ref.Namespace != nil && *ref.Namespace != store.GetNamespace() {
  387. return errNamespaceNotAllowed
  388. }
  389. return nil
  390. }
  391. func NetworkValidate(endpoint string, timeout time.Duration) error {
  392. hostname, err := url.Parse(endpoint)
  393. if err != nil {
  394. return fmt.Errorf("could not parse url: %w", err)
  395. }
  396. host := hostname.Hostname()
  397. port := hostname.Port()
  398. if port == "" {
  399. port = "443"
  400. }
  401. url := fmt.Sprintf("%v:%v", host, port)
  402. conn, err := net.DialTimeout("tcp", url, timeout)
  403. if err != nil {
  404. return fmt.Errorf("error accessing external store: %w", err)
  405. }
  406. defer conn.Close()
  407. return nil
  408. }
  409. func Deref[V any](v *V) V {
  410. if v == nil {
  411. // Create zero value
  412. var res V
  413. return res
  414. }
  415. return *v
  416. }
  417. func Ptr[T any](i T) *T {
  418. return &i
  419. }
  420. func ConvertToType[T any](obj any) (T, error) {
  421. var v T
  422. data, err := json.Marshal(obj)
  423. if err != nil {
  424. return v, fmt.Errorf("failed to marshal object: %w", err)
  425. }
  426. if err = json.Unmarshal(data, &v); err != nil {
  427. return v, fmt.Errorf("failed to unmarshal object: %w", err)
  428. }
  429. return v, nil
  430. }
  431. // FetchValueFromMetadata fetches a key from a metadata if it exists. It will recursively look in
  432. // embedded values as well. Must be a unique key, otherwise it will just return the first
  433. // occurrence.
  434. func FetchValueFromMetadata[T any](key string, data *apiextensionsv1.JSON, def T) (t T, _ error) {
  435. if data == nil {
  436. return def, nil
  437. }
  438. m := map[string]any{}
  439. if err := json.Unmarshal(data.Raw, &m); err != nil {
  440. return t, fmt.Errorf("failed to parse JSON raw data: %w", err)
  441. }
  442. v, err := dig[T](key, m)
  443. if err != nil {
  444. if errors.Is(err, errKeyNotFound) {
  445. return def, nil
  446. }
  447. }
  448. return v, nil
  449. }
  450. func dig[T any](key string, data map[string]any) (t T, _ error) {
  451. if v, ok := data[key]; ok {
  452. c, k := v.(T)
  453. if !k {
  454. return t, fmt.Errorf("failed to convert value to the desired type; was: %T", v)
  455. }
  456. return c, nil
  457. }
  458. for _, v := range data {
  459. if ty, ok := v.(map[string]any); ok {
  460. return dig[T](key, ty)
  461. }
  462. }
  463. return t, errKeyNotFound
  464. }
  465. func CompareStringAndByteSlices(valueString *string, valueByte []byte) bool {
  466. if valueString == nil {
  467. return false
  468. }
  469. return bytes.Equal(valueByte, []byte(*valueString))
  470. }
  471. func ExtractSecretData(data esv1beta1.PushSecretData, secret *corev1.Secret) ([]byte, error) {
  472. var (
  473. err error
  474. value []byte
  475. ok bool
  476. )
  477. if data.GetSecretKey() == "" {
  478. decodedMap := make(map[string]string)
  479. for k, v := range secret.Data {
  480. decodedMap[k] = string(v)
  481. }
  482. value, err = JSONMarshal(decodedMap)
  483. if err != nil {
  484. return nil, fmt.Errorf("failed to marshal secret data: %w", err)
  485. }
  486. } else {
  487. value, ok = secret.Data[data.GetSecretKey()]
  488. if !ok {
  489. return nil, fmt.Errorf("failed to find secret key in secret with key: %s", data.GetSecretKey())
  490. }
  491. }
  492. return value, nil
  493. }
  494. // CreateCertOpts contains options for a cert pool creation.
  495. type CreateCertOpts struct {
  496. CABundle []byte
  497. CAProvider *esv1beta1.CAProvider
  498. StoreKind string
  499. Namespace string
  500. Client client.Client
  501. }
  502. // FetchCACertFromSource creates a CertPool using either a CABundle directly, or
  503. // a ConfigMap / Secret.
  504. func FetchCACertFromSource(ctx context.Context, opts CreateCertOpts) ([]byte, error) {
  505. if len(opts.CABundle) == 0 && opts.CAProvider == nil {
  506. return nil, nil
  507. }
  508. if len(opts.CABundle) > 0 {
  509. pem, err := base64decode(opts.CABundle)
  510. if err != nil {
  511. return nil, fmt.Errorf("failed to decode ca bundle: %w", err)
  512. }
  513. return pem, nil
  514. }
  515. if opts.CAProvider != nil &&
  516. opts.StoreKind == esv1beta1.ClusterSecretStoreKind &&
  517. opts.CAProvider.Namespace == nil {
  518. return nil, errors.New("missing namespace on caProvider secret")
  519. }
  520. switch opts.CAProvider.Type {
  521. case esv1beta1.CAProviderTypeSecret:
  522. cert, err := getCertFromSecret(ctx, opts.Client, opts.CAProvider, opts.StoreKind, opts.Namespace)
  523. if err != nil {
  524. return nil, fmt.Errorf("failed to get cert from secret: %w", err)
  525. }
  526. return cert, nil
  527. case esv1beta1.CAProviderTypeConfigMap:
  528. cert, err := getCertFromConfigMap(ctx, opts.Namespace, opts.Client, opts.CAProvider)
  529. if err != nil {
  530. return nil, fmt.Errorf("failed to get cert from configmap: %w", err)
  531. }
  532. return cert, nil
  533. }
  534. return nil, fmt.Errorf("unsupported CA provider type: %s", opts.CAProvider.Type)
  535. }
  536. func base64decode(cert []byte) ([]byte, error) {
  537. if c, err := parseCertificateBytes(cert); err == nil {
  538. return c, nil
  539. }
  540. // try decoding and test for validity again...
  541. certificate, err := Decode(esv1beta1.ExternalSecretDecodeAuto, cert)
  542. if err != nil {
  543. return nil, fmt.Errorf("failed to decode base64: %w", err)
  544. }
  545. return parseCertificateBytes(certificate)
  546. }
  547. func parseCertificateBytes(certBytes []byte) ([]byte, error) {
  548. block, _ := pem.Decode(certBytes)
  549. if block == nil {
  550. return nil, errors.New("failed to parse the new certificate, not valid pem data")
  551. }
  552. if _, err := x509.ParseCertificate(block.Bytes); err != nil {
  553. return nil, fmt.Errorf("failed to validate certificate: %w", err)
  554. }
  555. return certBytes, nil
  556. }
  557. func getCertFromSecret(ctx context.Context, c client.Client, provider *esv1beta1.CAProvider, storeKind, namespace string) ([]byte, error) {
  558. secretRef := esmeta.SecretKeySelector{
  559. Name: provider.Name,
  560. Key: provider.Key,
  561. }
  562. if provider.Namespace != nil {
  563. secretRef.Namespace = provider.Namespace
  564. }
  565. cert, err := resolvers.SecretKeyRef(ctx, c, storeKind, namespace, &secretRef)
  566. if err != nil {
  567. return nil, fmt.Errorf("failed to resolve secret key ref: %w", err)
  568. }
  569. return []byte(cert), nil
  570. }
  571. func getCertFromConfigMap(ctx context.Context, namespace string, c client.Client, provider *esv1beta1.CAProvider) ([]byte, error) {
  572. objKey := client.ObjectKey{
  573. Name: provider.Name,
  574. Namespace: namespace,
  575. }
  576. if provider.Namespace != nil {
  577. objKey.Namespace = *provider.Namespace
  578. }
  579. configMapRef := &corev1.ConfigMap{}
  580. err := c.Get(ctx, objKey, configMapRef)
  581. if err != nil {
  582. return nil, fmt.Errorf("failed to get caProvider secret %s: %w", objKey.Name, err)
  583. }
  584. val, ok := configMapRef.Data[provider.Key]
  585. if !ok {
  586. return nil, fmt.Errorf("failed to get caProvider configMap %s -> %s", objKey.Name, provider.Key)
  587. }
  588. return []byte(val), nil
  589. }