2022-10-29 18:15:50 +00:00
|
|
|
/*
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package externalsecret
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-05-16 06:59:26 +00:00
|
|
|
"encoding/json"
|
2022-10-29 18:15:50 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
v1 "k8s.io/api/core/v1"
|
|
|
|
apiextensions "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
|
|
|
|
|
|
|
esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
|
|
|
|
genv1alpha1 "github.com/external-secrets/external-secrets/apis/generators/v1alpha1"
|
2022-11-29 19:04:46 +00:00
|
|
|
"github.com/external-secrets/external-secrets/pkg/controllers/secretstore"
|
2024-04-25 09:36:11 +00:00
|
|
|
"github.com/external-secrets/external-secrets/pkg/utils"
|
2024-10-02 06:43:00 +00:00
|
|
|
"github.com/external-secrets/external-secrets/pkg/utils/resolvers"
|
2024-04-25 09:36:11 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
// Loading registered generators.
|
|
|
|
_ "github.com/external-secrets/external-secrets/pkg/generator/register"
|
|
|
|
_ "github.com/external-secrets/external-secrets/pkg/provider/register"
|
|
|
|
)
|
|
|
|
|
|
|
|
// getProviderSecretData returns the provider's secret data with the provided ExternalSecret.
|
|
|
|
func (r *Reconciler) getProviderSecretData(ctx context.Context, externalSecret *esv1beta1.ExternalSecret) (map[string][]byte, error) {
|
|
|
|
// We MUST NOT create multiple instances of a provider client (mostly due to limitations with GCP)
|
|
|
|
// Clientmanager keeps track of the client instances
|
|
|
|
// that are created during the fetching process and closes clients
|
|
|
|
// if needed.
|
2022-11-29 19:04:46 +00:00
|
|
|
mgr := secretstore.NewManager(r.Client, r.ControllerClass, r.EnableFloodGate)
|
2022-10-29 18:15:50 +00:00
|
|
|
defer mgr.Close(ctx)
|
|
|
|
|
|
|
|
providerData := make(map[string][]byte)
|
|
|
|
for i, remoteRef := range externalSecret.Spec.DataFrom {
|
|
|
|
var secretMap map[string][]byte
|
|
|
|
var err error
|
|
|
|
|
|
|
|
if remoteRef.Find != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
secretMap, err = r.handleFindAllSecrets(ctx, externalSecret, remoteRef, mgr)
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("error processing spec.dataFrom[%d].find, err: %w", i, err)
|
|
|
|
}
|
2022-10-29 18:15:50 +00:00
|
|
|
} else if remoteRef.Extract != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
secretMap, err = r.handleExtractSecrets(ctx, externalSecret, remoteRef, mgr)
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("error processing spec.dataFrom[%d].extract, err: %w", i, err)
|
|
|
|
}
|
2022-10-29 18:15:50 +00:00
|
|
|
} else if remoteRef.SourceRef != nil && remoteRef.SourceRef.GeneratorRef != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
secretMap, err = r.handleGenerateSecrets(ctx, externalSecret.Namespace, remoteRef)
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("error processing spec.dataFrom[%d].sourceRef.generatorRef, err: %w", i, err)
|
|
|
|
}
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
if errors.Is(err, esv1beta1.NoSecretErr) && externalSecret.Spec.Target.DeletionPolicy != esv1beta1.DeletionPolicyRetain {
|
2024-12-06 21:22:59 +00:00
|
|
|
r.recorder.Eventf(externalSecret, v1.EventTypeNormal, esv1beta1.ReasonMissingProviderSecret, eventMissingProviderSecret, i)
|
2022-10-29 18:15:50 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
providerData = utils.MergeByteMap(providerData, secretMap)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, secretRef := range externalSecret.Spec.Data {
|
2024-12-06 21:22:59 +00:00
|
|
|
err := r.handleSecretData(ctx, *externalSecret, secretRef, providerData, mgr)
|
2022-10-29 18:15:50 +00:00
|
|
|
if errors.Is(err, esv1beta1.NoSecretErr) && externalSecret.Spec.Target.DeletionPolicy != esv1beta1.DeletionPolicyRetain {
|
2024-12-06 21:22:59 +00:00
|
|
|
r.recorder.Eventf(externalSecret, v1.EventTypeNormal, esv1beta1.ReasonMissingProviderSecret, eventMissingProviderSecretKey, i, secretRef.RemoteRef.Key)
|
2022-10-29 18:15:50 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf("error processing spec.data[%d] (key: %s), err: %w", i, secretRef.RemoteRef.Key, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return providerData, nil
|
|
|
|
}
|
|
|
|
|
2024-12-06 21:22:59 +00:00
|
|
|
func (r *Reconciler) handleSecretData(ctx context.Context, externalSecret esv1beta1.ExternalSecret, secretRef esv1beta1.ExternalSecretData, providerData map[string][]byte, cmgr *secretstore.Manager) error {
|
2023-11-02 13:37:59 +00:00
|
|
|
client, err := cmgr.Get(ctx, externalSecret.Spec.SecretStoreRef, externalSecret.Namespace, toStoreGenSourceRef(secretRef.SourceRef))
|
2022-10-29 18:15:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// get a single secret from the store
|
2022-10-29 18:15:50 +00:00
|
|
|
secretData, err := client.GetSecret(ctx, secretRef.RemoteRef)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// decode the secret if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretData, err = utils.Decode(secretRef.RemoteRef.DecodingStrategy, secretData)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return fmt.Errorf(errDecode, secretRef.RemoteRef.DecodingStrategy, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// store the secret data
|
2022-10-29 18:15:50 +00:00
|
|
|
providerData[secretRef.SecretKey] = secretData
|
2024-12-06 21:22:59 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-02 13:37:59 +00:00
|
|
|
func toStoreGenSourceRef(ref *esv1beta1.StoreSourceRef) *esv1beta1.StoreGeneratorSourceRef {
|
|
|
|
if ref == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return &esv1beta1.StoreGeneratorSourceRef{
|
|
|
|
SecretStoreRef: &ref.SecretStoreRef,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-06 21:22:59 +00:00
|
|
|
func (r *Reconciler) handleGenerateSecrets(ctx context.Context, namespace string, remoteRef esv1beta1.ExternalSecretDataFromRemoteRef) (map[string][]byte, error) {
|
2024-11-28 08:28:21 +00:00
|
|
|
gen, obj, err := resolvers.GeneratorRef(ctx, r.Client, r.Scheme, namespace, remoteRef.SourceRef.GeneratorRef)
|
2022-10-29 18:15:50 +00:00
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, err
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// use the generator
|
2024-10-02 06:43:00 +00:00
|
|
|
secretMap, err := gen.Generate(ctx, obj, r.Client, namespace)
|
2022-10-29 18:15:50 +00:00
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errGenerate, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// rewrite the keys if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err = utils.RewriteMap(remoteRef.Rewrite, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errRewrite, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// validate the keys
|
|
|
|
err = utils.ValidateKeys(secretMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf(errInvalidKeys, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
return secretMap, err
|
|
|
|
}
|
|
|
|
|
2024-12-06 21:22:59 +00:00
|
|
|
//nolint:dupl
|
|
|
|
func (r *Reconciler) handleExtractSecrets(ctx context.Context, externalSecret *esv1beta1.ExternalSecret, remoteRef esv1beta1.ExternalSecretDataFromRemoteRef, cmgr *secretstore.Manager) (map[string][]byte, error) {
|
2022-10-29 18:15:50 +00:00
|
|
|
client, err := cmgr.Get(ctx, externalSecret.Spec.SecretStoreRef, externalSecret.Namespace, remoteRef.SourceRef)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// get multiple secrets from the store
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err := client.GetSecretMap(ctx, *remoteRef.Extract)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// rewrite the keys if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err = utils.RewriteMap(remoteRef.Rewrite, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errRewrite, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
if len(remoteRef.Rewrite) == 0 {
|
|
|
|
secretMap, err = utils.ConvertKeys(remoteRef.Extract.ConversionStrategy, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errConvert, remoteRef.Extract.ConversionStrategy, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// validate the keys
|
|
|
|
err = utils.ValidateKeys(secretMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf(errInvalidKeys, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// decode the secrets if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err = utils.DecodeMap(remoteRef.Extract.DecodingStrategy, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errDecode, remoteRef.Extract.DecodingStrategy, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
2022-10-29 18:15:50 +00:00
|
|
|
return secretMap, err
|
|
|
|
}
|
|
|
|
|
2024-12-06 21:22:59 +00:00
|
|
|
//nolint:dupl
|
|
|
|
func (r *Reconciler) handleFindAllSecrets(ctx context.Context, externalSecret *esv1beta1.ExternalSecret, remoteRef esv1beta1.ExternalSecretDataFromRemoteRef, cmgr *secretstore.Manager) (map[string][]byte, error) {
|
2022-10-29 18:15:50 +00:00
|
|
|
client, err := cmgr.Get(ctx, externalSecret.Spec.SecretStoreRef, externalSecret.Namespace, remoteRef.SourceRef)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// get all secrets from the store that match the selector
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err := client.GetAllSecrets(ctx, *remoteRef.Find)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// rewrite the keys if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err = utils.RewriteMap(remoteRef.Rewrite, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errRewrite, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
if len(remoteRef.Rewrite) == 0 {
|
|
|
|
secretMap, err = utils.ConvertKeys(remoteRef.Find.ConversionStrategy, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errConvert, remoteRef.Find.ConversionStrategy, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// validate the keys
|
|
|
|
err = utils.ValidateKeys(secretMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf(errInvalidKeys, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
2024-12-06 21:22:59 +00:00
|
|
|
|
|
|
|
// decode the secrets if needed
|
2022-10-29 18:15:50 +00:00
|
|
|
secretMap, err = utils.DecodeMap(remoteRef.Find.DecodingStrategy, secretMap)
|
|
|
|
if err != nil {
|
2024-12-06 21:22:59 +00:00
|
|
|
return nil, fmt.Errorf(errDecode, remoteRef.Find.DecodingStrategy, err)
|
2022-10-29 18:15:50 +00:00
|
|
|
}
|
|
|
|
return secretMap, err
|
|
|
|
}
|
2023-05-16 06:59:26 +00:00
|
|
|
|
|
|
|
func shouldSkipGenerator(r *Reconciler, generatorDef *apiextensions.JSON) (bool, error) {
|
|
|
|
var genControllerClass genv1alpha1.ControllerClassResource
|
|
|
|
err := json.Unmarshal(generatorDef.Raw, &genControllerClass)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
var controllerClass = genControllerClass.Spec.ControllerClass
|
|
|
|
return controllerClass != "" && controllerClass != r.ControllerClass, nil
|
|
|
|
}
|