Files
mariadb-operator/api/v1alpha1/maxscale_webhook.go
2024-02-27 18:26:58 +01:00

187 lines
5.0 KiB
Go

package v1alpha1
import (
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/util/validation/field"
"k8s.io/utils/ptr"
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
"sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
var maxscaleLogger = logf.Log.WithName("maxscale")
func (r *MaxScale) SetupWebhookWithManager(mgr ctrl.Manager) error {
return ctrl.NewWebhookManagedBy(mgr).
For(r).
Complete()
}
//+kubebuilder:webhook:path=/validate-k8s-mariadb-com-v1alpha1-maxscale,mutating=false,failurePolicy=fail,sideEffects=None,groups=k8s.mariadb.com,resources=maxscales,verbs=create;update,versions=v1alpha1,name=vmaxscale.kb.io,admissionReviewVersions=v1
var _ webhook.Validator = &MaxScale{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
func (r *MaxScale) ValidateCreate() (admission.Warnings, error) {
maxscaleLogger.V(1).Info("Validate create", "name", r.Name)
validateFns := []func() error{
r.validateAuth,
r.validateCreateServerSources,
r.validateServers,
r.validateMonitor,
r.validateServices,
r.validatePodDisruptionBudget,
}
for _, fn := range validateFns {
if err := fn(); err != nil {
return nil, err
}
}
return nil, nil
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
func (r *MaxScale) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
maxscaleLogger.V(1).Info("Validate update", "name", r.Name)
oldMaxScale := old.(*MaxScale)
if err := inmutableWebhook.ValidateUpdate(r, oldMaxScale); err != nil {
return nil, err
}
validateFns := []func() error{
r.validateAuth,
r.validateServerSources,
r.validateServers,
r.validateMonitor,
r.validateServices,
r.validatePodDisruptionBudget,
}
for _, fn := range validateFns {
if err := fn(); err != nil {
return nil, err
}
}
return nil, nil
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
func (r *MaxScale) ValidateDelete() (admission.Warnings, error) {
return nil, nil
}
func (r *MaxScale) validateAuth() error {
if ptr.Deref(r.Spec.Auth.Generate, false) && r.Spec.MariaDBRef == nil {
return field.Invalid(
field.NewPath("spec").Child("auth").Child("generate").Child("enabled"),
r.Spec.MariaDBRef,
"'spec.auth.generate' can only be enabled when 'spec.mariaDbRef' is set",
)
}
return nil
}
func (r *MaxScale) validateCreateServerSources() error {
if err := r.validateServerSources(); err != nil {
return err
}
if r.Spec.MariaDBRef != nil && r.Spec.Servers != nil {
return field.Invalid(
field.NewPath("spec").Child("mariaDbRef"),
r.Spec.MariaDBRef,
"'spec.mariaDbRef' and 'spec.servers' cannot be specified simultaneously",
)
}
return nil
}
func (r *MaxScale) validateServerSources() error {
if r.Spec.MariaDBRef == nil && r.Spec.Servers == nil {
return field.Invalid(
field.NewPath("spec").Child("mariaDbRef"),
r.Spec.MariaDBRef,
"'spec.mariaDbRef' or 'spec.servers' must be defined",
)
}
return nil
}
func (r *MaxScale) validateServers() error {
idx := r.ServerIndex()
if len(idx) != len(r.Spec.Servers) {
return field.Invalid(
field.NewPath("spec").Child("servers"),
r.Spec.Servers,
"server names must be unique",
)
}
addresses := make(map[string]struct{})
for _, srv := range r.Spec.Servers {
addresses[srv.Address] = struct{}{}
}
if len(addresses) != len(r.Spec.Servers) {
return field.Invalid(
field.NewPath("spec").Child("servers"),
r.Spec.Servers,
"server addresses must be unique",
)
}
return nil
}
func (r *MaxScale) validateMonitor() error {
if r.Spec.MariaDBRef == nil && r.Spec.Monitor.Module == "" {
return field.Invalid(
field.NewPath("spec").Child("monitor").Child("module"),
r.Spec.Monitor.Module,
"'spec.monitor.module' must be provided when 'spec.mariaDbRef' is not defined",
)
}
if r.Spec.Monitor.Module != "" {
if err := r.Spec.Monitor.Module.Validate(); err != nil {
return field.Invalid(
field.NewPath("spec").Child("monitor").Child("module"),
r.Spec.Monitor.Module,
err.Error(),
)
}
}
return nil
}
func (r *MaxScale) validateServices() error {
idx := r.ServiceIndex()
if len(idx) != len(r.Spec.Services) {
return field.Invalid(
field.NewPath("spec").Child("services"),
r.Spec.Services,
"service names must be unique",
)
}
ports := make(map[int]struct{})
for _, svc := range r.Spec.Services {
ports[int(svc.Listener.Port)] = struct{}{}
}
if len(ports) != len(r.Spec.Services) {
return field.Invalid(
field.NewPath("spec").Child("services"),
r.Spec.Services,
"service listener ports must be unique",
)
}
return nil
}
func (r *MaxScale) validatePodDisruptionBudget() error {
if r.Spec.PodDisruptionBudget == nil {
return nil
}
if err := r.Spec.PodDisruptionBudget.Validate(); err != nil {
return field.Invalid(
field.NewPath("spec").Child("podDisruptionBudget"),
r.Spec.PodDisruptionBudget,
err.Error(),
)
}
return nil
}