1
0
Fork 0
mirror of https://github.com/external-secrets/external-secrets.git synced 2024-12-14 11:57:59 +00:00

feat: status conditions (#25)

* feat: implement es ready condition

Co-authored-by: Kellin <kellinmcavoy@gmail.com>
This commit is contained in:
Moritz Johner 2021-02-15 21:51:38 +01:00 committed by GitHub
parent 87cfc51216
commit 89c56c269f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 313 additions and 10 deletions

View file

@ -141,19 +141,28 @@ type ExternalSecretStatusCondition struct {
LastTransitionTime metav1.Time `json:"lastTransitionTime,omitempty"` LastTransitionTime metav1.Time `json:"lastTransitionTime,omitempty"`
} }
const (
// ConditionReasonSecretSynced indicates that the secrets was synced.
ConditionReasonSecretSynced = "SecretSynced"
// ConditionReasonSecretSyncedError indicates that there was an error syncing the secret.
ConditionReasonSecretSyncedError = "SecretSyncedError"
)
type ExternalSecretStatus struct { type ExternalSecretStatus struct {
// +optional // +nullable
// refreshTime is the time and date the external secret was fetched and // refreshTime is the time and date the external secret was fetched and
// the target secret updated // the target secret updated
RefreshTime metav1.Time `json:"refreshTime"` RefreshTime metav1.Time `json:"refreshTime,omitempty"`
// +optional // +optional
Conditions []ExternalSecretStatusCondition `json:"conditions"` Conditions []ExternalSecretStatusCondition `json:"conditions,omitempty"`
} }
// +kubebuilder:object:root=true // +kubebuilder:object:root=true
// ExternalSecret is the Schema for the external-secrets API. // ExternalSecret is the Schema for the external-secrets API.
// +kubebuilder:subresource:status
// +kubebuilder:resource:scope=Namespaced,categories={externalsecrets},shortName=es
type ExternalSecret struct { type ExternalSecret struct {
metav1.TypeMeta `json:",inline"` metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"` metav1.ObjectMeta `json:"metadata,omitempty"`

View file

@ -8,9 +8,13 @@ metadata:
spec: spec:
group: external-secrets.io group: external-secrets.io
names: names:
categories:
- externalsecrets
kind: ExternalSecret kind: ExternalSecret
listKind: ExternalSecretList listKind: ExternalSecretList
plural: externalsecrets plural: externalsecrets
shortNames:
- es
singular: externalsecret singular: externalsecret
scope: Namespaced scope: Namespaced
versions: versions:
@ -152,11 +156,14 @@ spec:
description: refreshTime is the time and date the external secret description: refreshTime is the time and date the external secret
was fetched and the target secret updated was fetched and the target secret updated
format: date-time format: date-time
nullable: true
type: string type: string
type: object type: object
type: object type: object
served: true served: true
storage: true storage: true
subresources:
status: {}
status: status:
acceptedNames: acceptedNames:
kind: "" kind: ""

View file

@ -64,7 +64,7 @@ func (r *Reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu
secret := &corev1.Secret{ secret := &corev1.Secret{
ObjectMeta: metav1.ObjectMeta{ ObjectMeta: metav1.ObjectMeta{
Name: externalSecret.Name, Name: externalSecret.Spec.Target.Name,
Namespace: externalSecret.Namespace, Namespace: externalSecret.Namespace,
}, },
} }
@ -108,9 +108,22 @@ func (r *Reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu
if err != nil { if err != nil {
log.Error(err, "could not reconcile ExternalSecret") log.Error(err, "could not reconcile ExternalSecret")
conditionSynced := NewExternalSecretCondition(esv1alpha1.ExternalSecretReady, corev1.ConditionFalse, esv1alpha1.ConditionReasonSecretSynced, err.Error())
SetExternalSecretCondition(&externalSecret.Status, *conditionSynced)
err = r.Status().Update(ctx, &externalSecret)
if err != nil {
log.Error(err, "unable to update status")
}
return ctrl.Result{RequeueAfter: requeueAfter}, nil return ctrl.Result{RequeueAfter: requeueAfter}, nil
} }
conditionSynced := NewExternalSecretCondition(esv1alpha1.ExternalSecretReady, corev1.ConditionTrue, esv1alpha1.ConditionReasonSecretSynced, "Secret was synced")
SetExternalSecretCondition(&externalSecret.Status, *conditionSynced)
externalSecret.Status.RefreshTime = metav1.NewTime(time.Now())
err = r.Status().Update(ctx, &externalSecret)
if err != nil {
log.Error(err, "unable to update status")
}
return ctrl.Result{}, nil return ctrl.Result{}, nil
} }

View file

@ -0,0 +1,190 @@
/*
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"
"fmt"
"time"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
"k8s.io/apimachinery/pkg/util/wait"
"sigs.k8s.io/controller-runtime/pkg/client"
esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
"github.com/external-secrets/external-secrets/pkg/provider/fake"
"github.com/external-secrets/external-secrets/pkg/provider/schema"
)
var fakeProvider *fake.Client
var _ = Describe("ExternalSecret controller", func() {
const (
ExternalSecretName = "test-es"
ExternalSecretStore = "test-store"
ExternalSecretTargetSecretName = "test-secret"
timeout = time.Second * 5
interval = time.Millisecond * 250
)
var ExternalSecretNamespace string
BeforeEach(func() {
var err error
ExternalSecretNamespace, err = CreateNamespace("test-ns", k8sClient)
Expect(err).ToNot(HaveOccurred())
Expect(k8sClient.Create(context.Background(), &esv1alpha1.SecretStore{
ObjectMeta: metav1.ObjectMeta{
Name: ExternalSecretStore,
Namespace: ExternalSecretNamespace,
},
Spec: esv1alpha1.SecretStoreSpec{
Provider: &esv1alpha1.SecretStoreProvider{
AWSSM: &esv1alpha1.AWSSMProvider{},
},
},
})).To(Succeed())
})
AfterEach(func() {
Expect(k8sClient.Delete(context.Background(), &v1.Namespace{
ObjectMeta: metav1.ObjectMeta{
Name: ExternalSecretNamespace,
},
}, client.PropagationPolicy(metav1.DeletePropagationBackground)), client.GracePeriodSeconds(0)).To(Succeed())
Expect(k8sClient.Delete(context.Background(), &esv1alpha1.SecretStore{
ObjectMeta: metav1.ObjectMeta{
Name: ExternalSecretStore,
Namespace: ExternalSecretNamespace,
},
}, client.PropagationPolicy(metav1.DeletePropagationBackground)), client.GracePeriodSeconds(0)).To(Succeed())
})
Context("When updating ExternalSecret Status", func() {
It("should set the condition eventually", func() {
By("creating an ExternalSecret")
ctx := context.Background()
es := &esv1alpha1.ExternalSecret{
ObjectMeta: metav1.ObjectMeta{
Name: ExternalSecretName,
Namespace: ExternalSecretNamespace,
},
Spec: esv1alpha1.ExternalSecretSpec{
SecretStoreRef: esv1alpha1.SecretStoreRef{
Name: ExternalSecretStore,
},
Target: esv1alpha1.ExternalSecretTarget{
Name: ExternalSecretTargetSecretName,
},
},
}
Expect(k8sClient.Create(ctx, es)).Should(Succeed())
esLookupKey := types.NamespacedName{Name: ExternalSecretName, Namespace: ExternalSecretNamespace}
createdES := &esv1alpha1.ExternalSecret{}
Eventually(func() bool {
err := k8sClient.Get(ctx, esLookupKey, createdES)
if err != nil {
return false
}
cond := GetExternalSecretCondition(createdES.Status, esv1alpha1.ExternalSecretReady)
if cond == nil || cond.Status != v1.ConditionTrue {
return false
}
return true
}, timeout, interval).Should(BeTrue())
})
})
Context("When syncing ExternalSecret value", func() {
It("should set the secret value", func() {
By("creating an ExternalSecret")
ctx := context.Background()
const targetProp = "targetProperty"
const secretVal = "someValue"
es := &esv1alpha1.ExternalSecret{
ObjectMeta: metav1.ObjectMeta{
Name: ExternalSecretName,
Namespace: ExternalSecretNamespace,
},
Spec: esv1alpha1.ExternalSecretSpec{
SecretStoreRef: esv1alpha1.SecretStoreRef{
Name: ExternalSecretStore,
},
Target: esv1alpha1.ExternalSecretTarget{
Name: ExternalSecretTargetSecretName,
},
Data: []esv1alpha1.ExternalSecretData{
{
SecretKey: targetProp,
RemoteRef: esv1alpha1.ExternalSecretDataRemoteRef{
Key: "barz",
Property: "bang",
},
},
},
},
}
fakeProvider.WithGetSecret([]byte(secretVal), nil)
Expect(k8sClient.Create(ctx, es)).Should(Succeed())
secretLookupKey := types.NamespacedName{
Name: ExternalSecretTargetSecretName,
Namespace: ExternalSecretNamespace}
syncedSecret := &v1.Secret{}
Eventually(func() bool {
err := k8sClient.Get(ctx, secretLookupKey, syncedSecret)
if err != nil {
return false
}
v := syncedSecret.Data[targetProp]
return string(v) == secretVal
}, timeout, interval).Should(BeTrue())
})
})
})
// CreateNamespace creates a new namespace in the cluster.
func CreateNamespace(baseName string, c client.Client) (string, error) {
genName := fmt.Sprintf("ctrl-test-%v", baseName)
ns := &v1.Namespace{
ObjectMeta: metav1.ObjectMeta{
GenerateName: genName,
},
}
var err error
err = wait.Poll(time.Second, 10*time.Second, func() (bool, error) {
err = c.Create(context.Background(), ns)
if err != nil {
return false, nil
}
return true, nil
})
if err != nil {
return "", err
}
return ns.Name, nil
}
func init() {
fakeProvider = fake.New()
schema.ForceRegister(fakeProvider, &esv1alpha1.SecretStoreProvider{
AWSSM: &esv1alpha1.AWSSMProvider{},
})
}

View file

@ -22,6 +22,7 @@ import (
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
"k8s.io/client-go/kubernetes/scheme" "k8s.io/client-go/kubernetes/scheme"
"k8s.io/client-go/rest" "k8s.io/client-go/rest"
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/envtest" "sigs.k8s.io/controller-runtime/pkg/envtest"
"sigs.k8s.io/controller-runtime/pkg/envtest/printer" "sigs.k8s.io/controller-runtime/pkg/envtest/printer"
@ -52,7 +53,7 @@ var _ = BeforeSuite(func(done Done) {
By("bootstrapping test environment") By("bootstrapping test environment")
testEnv = &envtest.Environment{ testEnv = &envtest.Environment{
CRDDirectoryPaths: []string{filepath.Join("..", "config", "crd", "bases")}, CRDDirectoryPaths: []string{filepath.Join("..", "..", "..", "config", "crd", "bases")},
} }
var err error var err error
@ -63,14 +64,30 @@ var _ = BeforeSuite(func(done Done) {
err = esv1alpha1.AddToScheme(scheme.Scheme) err = esv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred()) Expect(err).NotTo(HaveOccurred())
err = esv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
// +kubebuilder:scaffold:scheme // +kubebuilder:scaffold:scheme
k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme}) k8sManager, err := ctrl.NewManager(cfg, ctrl.Options{
Scheme: scheme.Scheme,
})
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
// do not use k8sManager.GetClient()
// see https://github.com/kubernetes-sigs/controller-runtime/issues/343#issuecomment-469435686
k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme})
Expect(k8sClient).ToNot(BeNil()) Expect(k8sClient).ToNot(BeNil())
Expect(err).ToNot(HaveOccurred())
err = (&Reconciler{
Client: k8sClient,
Scheme: k8sManager.GetScheme(),
Log: ctrl.Log.WithName("controllers").WithName("ExternalSecrets"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
go func() {
defer GinkgoRecover()
Expect(k8sManager.Start(ctrl.SetupSignalHandler())).ToNot(HaveOccurred())
}()
close(done) close(done)
}, 60) }, 60)

View file

@ -0,0 +1,67 @@
/*
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 (
v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
)
func NewExternalSecretCondition(condType esv1alpha1.ExternalSecretConditionType, status v1.ConditionStatus, reason, message string) *esv1alpha1.ExternalSecretStatusCondition {
return &esv1alpha1.ExternalSecretStatusCondition{
Type: condType,
Status: status,
LastTransitionTime: metav1.Now(),
Reason: reason,
Message: message,
}
}
// GetExternalSecretCondition returns the condition with the provided type.
func GetExternalSecretCondition(status esv1alpha1.ExternalSecretStatus, condType esv1alpha1.ExternalSecretConditionType) *esv1alpha1.ExternalSecretStatusCondition {
for i := range status.Conditions {
c := status.Conditions[i]
if c.Type == condType {
return &c
}
}
return nil
}
// SetExternalSecretCondition updates the external secret to include the provided
// condition.
func SetExternalSecretCondition(status *esv1alpha1.ExternalSecretStatus, condition esv1alpha1.ExternalSecretStatusCondition) {
currentCond := GetExternalSecretCondition(*status, condition.Type)
if currentCond != nil && currentCond.Status == condition.Status && currentCond.Reason == condition.Reason {
return
}
// Do not update lastTransitionTime if the status of the condition doesn't change.
if currentCond != nil && currentCond.Status == condition.Status {
condition.LastTransitionTime = currentCond.LastTransitionTime
}
status.Conditions = append(filterOutCondition(status.Conditions, condition.Type), condition)
}
func filterOutCondition(conditions []esv1alpha1.ExternalSecretStatusCondition, condType esv1alpha1.ExternalSecretConditionType) []esv1alpha1.ExternalSecretStatusCondition {
newConditions := make([]esv1alpha1.ExternalSecretStatusCondition, 0, len(conditions))
for _, c := range conditions {
if c.Type == condType {
continue
}
newConditions = append(newConditions, c)
}
return newConditions
}

View file

@ -46,7 +46,7 @@ func New() *Client {
} }
v.NewFn = func(context.Context, esv1alpha1.GenericStore, client.Client, string) (provider.Provider, error) { v.NewFn = func(context.Context, esv1alpha1.GenericStore, client.Client, string) (provider.Provider, error) {
return nil, nil return v, nil
} }
return v return v