2022-05-03 18:23:34 +02:00
|
|
|
package certmanager
|
2021-06-08 12:37:19 -07:00
|
|
|
|
|
|
|
import (
|
2022-09-30 13:24:47 +02:00
|
|
|
"context"
|
2021-06-08 12:37:19 -07:00
|
|
|
"time"
|
|
|
|
|
2022-10-07 16:21:37 +02:00
|
|
|
"github.com/go-logr/logr"
|
2022-09-30 13:24:47 +02:00
|
|
|
"github.com/kyverno/kyverno/pkg/controllers"
|
2022-05-02 12:58:04 +02:00
|
|
|
"github.com/kyverno/kyverno/pkg/tls"
|
2022-10-07 16:21:37 +02:00
|
|
|
controllerutils "github.com/kyverno/kyverno/pkg/utils/controller"
|
2022-12-21 21:30:45 +01:00
|
|
|
retryutils "github.com/kyverno/kyverno/pkg/utils/retry"
|
2022-05-17 16:14:31 +02:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
2022-10-07 16:21:37 +02:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/labels"
|
2022-05-17 17:51:03 +02:00
|
|
|
corev1informers "k8s.io/client-go/informers/core/v1"
|
|
|
|
corev1listers "k8s.io/client-go/listers/core/v1"
|
2022-10-07 16:21:37 +02:00
|
|
|
"k8s.io/client-go/util/workqueue"
|
2021-06-08 12:37:19 -07:00
|
|
|
)
|
|
|
|
|
2022-10-07 09:38:38 +02:00
|
|
|
const (
|
|
|
|
// Workers is the number of workers for this controller
|
|
|
|
Workers = 1
|
|
|
|
ControllerName = "certmanager-controller"
|
2022-10-07 16:21:37 +02:00
|
|
|
maxRetries = 10
|
2022-10-07 09:38:38 +02:00
|
|
|
)
|
2022-10-03 09:55:59 +02:00
|
|
|
|
2022-05-03 18:23:34 +02:00
|
|
|
type controller struct {
|
2022-10-13 11:46:05 +02:00
|
|
|
renewer tls.CertRenewer
|
2022-10-07 16:21:37 +02:00
|
|
|
|
|
|
|
// listers
|
2023-06-22 14:14:53 +02:00
|
|
|
caLister corev1listers.SecretLister
|
|
|
|
tlsLister corev1listers.SecretLister
|
2022-10-07 16:21:37 +02:00
|
|
|
|
|
|
|
// queue
|
2024-08-28 19:09:58 +02:00
|
|
|
queue workqueue.TypedRateLimitingInterface[any]
|
2023-06-22 14:14:53 +02:00
|
|
|
caEnqueue controllerutils.EnqueueFunc
|
|
|
|
tlsEnqueue controllerutils.EnqueueFunc
|
2023-08-28 17:34:37 +05:30
|
|
|
|
2023-08-28 16:05:49 +02:00
|
|
|
caSecretName string
|
|
|
|
tlsSecretName string
|
|
|
|
namespace string
|
2021-06-08 12:37:19 -07:00
|
|
|
}
|
|
|
|
|
2023-06-22 14:14:53 +02:00
|
|
|
func NewController(
|
|
|
|
caInformer corev1informers.SecretInformer,
|
|
|
|
tlsInformer corev1informers.SecretInformer,
|
|
|
|
certRenewer tls.CertRenewer,
|
2023-08-28 16:05:49 +02:00
|
|
|
caSecretName string,
|
|
|
|
tlsSecretName string,
|
2023-08-28 17:34:37 +05:30
|
|
|
namespace string,
|
2023-06-22 14:14:53 +02:00
|
|
|
) controllers.Controller {
|
2024-08-28 19:09:58 +02:00
|
|
|
queue := workqueue.NewNamedRateLimitingQueue(workqueue.DefaultTypedControllerRateLimiter[any](), ControllerName)
|
2023-08-31 22:08:29 +02:00
|
|
|
caEnqueue, _, _ := controllerutils.AddDefaultEventHandlers(logger, caInformer.Informer(), queue)
|
|
|
|
tlsEnqueue, _, _ := controllerutils.AddDefaultEventHandlers(logger, tlsInformer.Informer(), queue)
|
2022-10-07 16:21:37 +02:00
|
|
|
c := controller{
|
2023-08-28 16:05:49 +02:00
|
|
|
renewer: certRenewer,
|
|
|
|
caLister: caInformer.Lister(),
|
|
|
|
tlsLister: tlsInformer.Lister(),
|
|
|
|
queue: queue,
|
2023-08-31 22:08:29 +02:00
|
|
|
caEnqueue: caEnqueue,
|
|
|
|
tlsEnqueue: tlsEnqueue,
|
2023-08-28 16:05:49 +02:00
|
|
|
caSecretName: caSecretName,
|
|
|
|
tlsSecretName: tlsSecretName,
|
|
|
|
namespace: namespace,
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
|
|
|
return &c
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *controller) Run(ctx context.Context, workers int) {
|
|
|
|
// we need to enqueue our secrets in case they don't exist yet in the cluster
|
|
|
|
// this way we ensure the reconcile happens (hence renewal/creation)
|
2023-06-22 14:14:53 +02:00
|
|
|
if err := c.tlsEnqueue(&corev1.Secret{
|
2022-10-07 16:21:37 +02:00
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
2023-08-28 17:34:37 +05:30
|
|
|
Namespace: c.namespace,
|
2023-08-28 16:05:49 +02:00
|
|
|
Name: c.tlsSecretName,
|
2022-10-07 16:21:37 +02:00
|
|
|
},
|
|
|
|
}); err != nil {
|
2023-08-28 16:05:49 +02:00
|
|
|
logger.Error(err, "failed to enqueue secret", "name", c.tlsSecretName)
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
2023-06-22 14:14:53 +02:00
|
|
|
if err := c.caEnqueue(&corev1.Secret{
|
2022-10-07 16:21:37 +02:00
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
2023-08-28 17:34:37 +05:30
|
|
|
Namespace: c.namespace,
|
2023-08-28 16:05:49 +02:00
|
|
|
Name: c.caSecretName,
|
2022-10-07 16:21:37 +02:00
|
|
|
},
|
|
|
|
}); err != nil {
|
2023-08-28 16:05:49 +02:00
|
|
|
logger.Error(err, "failed to enqueue CA secret", "name", c.caSecretName)
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
2022-10-19 10:54:48 +02:00
|
|
|
controllerutils.Run(ctx, logger, ControllerName, time.Second, c.queue, workers, maxRetries, c.reconcile, c.ticker)
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *controller) reconcile(ctx context.Context, logger logr.Logger, key, namespace, name string) error {
|
2023-08-28 17:34:37 +05:30
|
|
|
if namespace != c.namespace {
|
2022-10-07 16:21:37 +02:00
|
|
|
return nil
|
2021-06-08 12:37:19 -07:00
|
|
|
}
|
2023-08-28 16:05:49 +02:00
|
|
|
if name != c.caSecretName && name != c.tlsSecretName {
|
2022-10-07 16:21:37 +02:00
|
|
|
return nil
|
|
|
|
}
|
2023-02-22 10:04:17 +01:00
|
|
|
return c.renewCertificates(ctx)
|
2021-06-08 12:37:19 -07:00
|
|
|
}
|
|
|
|
|
2022-10-19 10:54:48 +02:00
|
|
|
func (c *controller) ticker(ctx context.Context, logger logr.Logger) {
|
2022-09-30 13:24:47 +02:00
|
|
|
certsRenewalTicker := time.NewTicker(tls.CertRenewalInterval)
|
|
|
|
defer certsRenewalTicker.Stop()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-certsRenewalTicker.C:
|
2023-06-22 14:14:53 +02:00
|
|
|
{
|
|
|
|
list, err := c.caLister.List(labels.Everything())
|
|
|
|
if err == nil {
|
|
|
|
for _, secret := range list {
|
|
|
|
if err := c.caEnqueue(secret); err != nil {
|
|
|
|
logger.Error(err, "failed to enqueue secret", "name", secret.Name)
|
|
|
|
}
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
2023-06-22 14:14:53 +02:00
|
|
|
} else {
|
|
|
|
logger.Error(err, "falied to list secrets")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
|
|
|
list, err := c.tlsLister.List(labels.Everything())
|
|
|
|
if err == nil {
|
|
|
|
for _, secret := range list {
|
|
|
|
if err := c.tlsEnqueue(secret); err != nil {
|
|
|
|
logger.Error(err, "failed to enqueue secret", "name", secret.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logger.Error(err, "falied to list secrets")
|
2022-10-07 16:21:37 +02:00
|
|
|
}
|
2022-09-30 13:24:47 +02:00
|
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-22 10:04:17 +01:00
|
|
|
func (c *controller) renewCertificates(ctx context.Context) error {
|
|
|
|
if err := retryutils.RetryFunc(ctx, time.Second, 5*time.Second, logger, "failed to renew CA", c.renewer.RenewCA)(); err != nil {
|
2022-05-12 16:07:25 +02:00
|
|
|
return err
|
|
|
|
}
|
2023-02-22 10:04:17 +01:00
|
|
|
if err := retryutils.RetryFunc(ctx, time.Second, 5*time.Second, logger, "failed to renew TLS", c.renewer.RenewTLS)(); err != nil {
|
2022-05-12 16:07:25 +02:00
|
|
|
return err
|
2022-05-11 10:05:13 +02:00
|
|
|
}
|
|
|
|
return nil
|
2021-06-08 12:37:19 -07:00
|
|
|
}
|