| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- /*
- 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 secretmanager
- import (
- "bytes"
- "errors"
- "fmt"
- "maps"
- "cloud.google.com/go/secretmanager/apiv1/secretmanagerpb"
- "github.com/tidwall/sjson"
- esv1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1"
- "github.com/external-secrets/external-secrets/pkg/utils/metadata"
- )
- type PushSecretMetadataMergePolicy string
- const (
- PushSecretMetadataMergePolicyReplace PushSecretMetadataMergePolicy = "Replace"
- PushSecretMetadataMergePolicyMerge PushSecretMetadataMergePolicy = "Merge"
- )
- type PushSecretMetadataSpec struct {
- Annotations map[string]string `json:"annotations,omitempty"`
- Labels map[string]string `json:"labels,omitempty"`
- Topics []string `json:"topics,omitempty"`
- MergePolicy PushSecretMetadataMergePolicy `json:"mergePolicy,omitempty"`
- CMEKKeyName string `json:"cmekKeyName,omitempty"`
- ReplicationLocation string `json:"replicationLocation,omitempty"`
- }
- func newPushSecretBuilder(payload []byte, data esv1.PushSecretData) (pushSecretBuilder, error) {
- if data.GetProperty() == "" {
- return &psBuilder{
- payload: payload,
- pushSecretData: data,
- }, nil
- }
- if data.GetMetadata() != nil {
- return nil, errors.New("cannot specify metadata and property at the same time")
- }
- return &propertyPSBuilder{
- payload: payload,
- pushSecretData: data,
- }, nil
- }
- type pushSecretBuilder interface {
- buildMetadata(annotations, labels map[string]string, topics []*secretmanagerpb.Topic) (map[string]string, map[string]string, []string, error)
- needUpdate(original []byte) bool
- buildData(original []byte) ([]byte, error)
- }
- type psBuilder struct {
- payload []byte
- pushSecretData esv1.PushSecretData
- }
- func (b *psBuilder) buildMetadata(_, labels map[string]string, _ []*secretmanagerpb.Topic) (map[string]string, map[string]string, []string, error) {
- if manager, ok := labels[managedByKey]; !ok || manager != managedByValue {
- return nil, nil, nil, fmt.Errorf("secret %v is not managed by external secrets", b.pushSecretData.GetRemoteKey())
- }
- var meta *metadata.PushSecretMetadata[PushSecretMetadataSpec]
- if b.pushSecretData.GetMetadata() != nil {
- var err error
- meta, err = metadata.ParseMetadataParameters[PushSecretMetadataSpec](b.pushSecretData.GetMetadata())
- if err != nil {
- return nil, nil, nil, fmt.Errorf("failed to parse PushSecret metadata: %w", err)
- }
- }
- var spec PushSecretMetadataSpec
- if meta != nil {
- spec = meta.Spec
- }
- newLabels := map[string]string{}
- maps.Copy(newLabels, spec.Labels)
- if spec.MergePolicy == PushSecretMetadataMergePolicyMerge {
- // Keep labels from the existing GCP Secret Manager Secret
- maps.Copy(newLabels, labels)
- }
- newLabels[managedByKey] = managedByValue
- return spec.Annotations, newLabels, spec.Topics, nil
- }
- func (b *psBuilder) needUpdate(original []byte) bool {
- if original == nil {
- return true
- }
- return !bytes.Equal(b.payload, original)
- }
- func (b *psBuilder) buildData(_ []byte) ([]byte, error) {
- return b.payload, nil
- }
- type propertyPSBuilder struct {
- payload []byte
- pushSecretData esv1.PushSecretData
- }
- func (b *propertyPSBuilder) buildMetadata(annotations, labels map[string]string, topics []*secretmanagerpb.Topic) (map[string]string, map[string]string, []string, error) {
- newAnnotations := map[string]string{}
- newLabels := map[string]string{}
- if annotations != nil {
- newAnnotations = annotations
- }
- if labels != nil {
- newLabels = labels
- }
- newLabels[managedByKey] = managedByValue
- result := make([]string, 0, len(topics))
- for _, t := range topics {
- result = append(result, t.Name)
- }
- return newAnnotations, newLabels, result, nil
- }
- func (b *propertyPSBuilder) needUpdate(original []byte) bool {
- if original == nil {
- return true
- }
- val := getDataByProperty(original, b.pushSecretData.GetProperty())
- return !val.Exists() || val.String() != string(b.payload)
- }
- func (b *propertyPSBuilder) buildData(original []byte) ([]byte, error) {
- var base []byte
- if original != nil {
- base = original
- }
- return sjson.SetBytes(base, b.pushSecretData.GetProperty(), b.payload)
- }
|