diff --git a/apis/vshn/v1/dbaas_vshn_keycloak.go b/apis/vshn/v1/dbaas_vshn_keycloak.go index 7bf6a772a7..f75d5d2d6a 100644 --- a/apis/vshn/v1/dbaas_vshn_keycloak.go +++ b/apis/vshn/v1/dbaas_vshn_keycloak.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" @@ -334,3 +333,11 @@ func (v *VSHNKeycloak) GetPDBLabels() map[string]string { func (v *VSHNKeycloak) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNKeycloak) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + return &StatefulSetManager{} +} + +func (v *VSHNKeycloak) GetWorkloadName() string { + return v.GetName() + "-keycloakx" +} diff --git a/apis/vshn/v1/dbaas_vshn_mariadb.go b/apis/vshn/v1/dbaas_vshn_mariadb.go index 48bfff6fbe..643af44ed1 100644 --- a/apis/vshn/v1/dbaas_vshn_mariadb.go +++ b/apis/vshn/v1/dbaas_vshn_mariadb.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) @@ -273,3 +272,11 @@ func (v *VSHNMariaDB) GetPDBLabels() map[string]string { func (v *VSHNMariaDB) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNMariaDB) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + return &StatefulSetManager{} +} + +func (v *VSHNMariaDB) GetWorkloadName() string { + return v.GetName() +} diff --git a/apis/vshn/v1/dbaas_vshn_postgresql.go b/apis/vshn/v1/dbaas_vshn_postgresql.go index cc601d910f..4a5ec52a92 100644 --- a/apis/vshn/v1/dbaas_vshn_postgresql.go +++ b/apis/vshn/v1/dbaas_vshn_postgresql.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" sgv1 "github.com/vshn/appcat/v4/apis/stackgres/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" @@ -397,3 +396,11 @@ func (v *VSHNPostgreSQL) GetPDBLabels() map[string]string { func (v *VSHNPostgreSQL) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNPostgreSQL) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + return &StatefulSetManager{} +} + +func (v *VSHNPostgreSQL) GetWorkloadName() string { + return v.GetName() +} diff --git a/apis/vshn/v1/dbaas_vshn_redis.go b/apis/vshn/v1/dbaas_vshn_redis.go index 43594b7e02..3703fb2cb2 100644 --- a/apis/vshn/v1/dbaas_vshn_redis.go +++ b/apis/vshn/v1/dbaas_vshn_redis.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) @@ -298,3 +297,11 @@ func (v *VSHNRedis) GetPDBLabels() map[string]string { func (v *VSHNRedis) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNRedis) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + return &StatefulSetManager{} +} + +func (v *VSHNRedis) GetWorkloadName() string { + return "redis-master" +} diff --git a/apis/vshn/v1/non_gen_types.go b/apis/vshn/v1/non_gen_types.go new file mode 100644 index 0000000000..0e8c00993e --- /dev/null +++ b/apis/vshn/v1/non_gen_types.go @@ -0,0 +1,57 @@ +package v1 + +import ( + v1 "k8s.io/api/apps/v1" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +// +kubebuilder:skip +// +kubebuilder:skipclient +// +kubebuilder:skipdeepcopy +// +kubebuilder:object:generate=false +type PodTemplateLabelsManager interface { + SetPodTemplateLabels(map[string]string) + GetPodTemplateLabels() map[string]string + GetObject() client.Object + client.Object +} + +// +kubebuilder:skip +// +kubebuilder:skipclient +// +kubebuilder:skipdeepcopy +// +kubebuilder:object:generate=false +type DeploymentManager struct { + v1.Deployment +} + +// +kubebuilder:skip +// +kubebuilder:skipclient +// +kubebuilder:skipdeepcopy +// +kubebuilder:object:generate=false +type StatefulSetManager struct { + v1.StatefulSet +} + +func (d *DeploymentManager) SetPodTemplateLabels(labels map[string]string) { + d.Spec.Template.SetLabels(labels) +} + +func (s *StatefulSetManager) SetPodTemplateLabels(labels map[string]string) { + s.Spec.Template.SetLabels(labels) +} + +func (d *DeploymentManager) GetPodTemplateLabels() map[string]string { + return d.Spec.Template.GetLabels() +} + +func (s *StatefulSetManager) GetPodTemplateLabels() map[string]string { + return s.Spec.Template.GetLabels() +} + +func (d *DeploymentManager) GetObject() client.Object { + return &d.Deployment +} + +func (d *StatefulSetManager) GetObject() client.Object { + return &d.StatefulSet +} diff --git a/apis/vshn/v1/vshn_minio.go b/apis/vshn/v1/vshn_minio.go index d8978921d3..e18726728b 100644 --- a/apis/vshn/v1/vshn_minio.go +++ b/apis/vshn/v1/vshn_minio.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) @@ -247,3 +246,14 @@ func (v *VSHNMinio) GetPDBLabels() map[string]string { func (v *VSHNMinio) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNMinio) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + if v.GetInstances() == 1 { + return &DeploymentManager{} + } + return &StatefulSetManager{} +} + +func (v *VSHNMinio) GetWorkloadName() string { + return v.GetName() +} diff --git a/apis/vshn/v1/vshn_nextcloud.go b/apis/vshn/v1/vshn_nextcloud.go index dc87debc05..4b846a49a0 100644 --- a/apis/vshn/v1/vshn_nextcloud.go +++ b/apis/vshn/v1/vshn_nextcloud.go @@ -2,7 +2,6 @@ package v1 import ( "fmt" - xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) @@ -304,3 +303,11 @@ func (v *VSHNNextcloud) GetPDBLabels() map[string]string { func (v *VSHNNextcloud) GetSecurity() *Security { return &v.Spec.Parameters.Security } + +func (v *VSHNNextcloud) GetWorkloadPodTemplateLabelsManager() PodTemplateLabelsManager { + return &DeploymentManager{} +} + +func (v *VSHNNextcloud) GetWorkloadName() string { + return v.GetName() +} diff --git a/pkg/comp-functions/functions/common/billing.go b/pkg/comp-functions/functions/common/billing.go new file mode 100644 index 0000000000..d4f5d74a79 --- /dev/null +++ b/pkg/comp-functions/functions/common/billing.go @@ -0,0 +1,53 @@ +package common + +import ( + "context" + "fmt" + v12 "github.com/crossplane/crossplane-runtime/apis/common/v1" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + xkube "github.com/vshn/appcat/v4/apis/kubernetes/v1alpha2" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" + "reflect" + controllerruntime "sigs.k8s.io/controller-runtime" + "strings" +) + +const billingLabel = "appcat.io/billing" + +// InjectBillingLabelToService adds billing label to a service (StatefulSet or Deployment). +// It uses a kube Object to achieve post provisioning labelling +func InjectBillingLabelToService(ctx context.Context, svc *runtime.ServiceRuntime, comp InfoGetter) *xfnproto.Result { + log := controllerruntime.LoggerFrom(ctx) + log.Info("Enabling billing for service", "service", comp.GetName()) + + s := comp.GetWorkloadPodTemplateLabelsManager() + s.SetName(comp.GetWorkloadName()) + s.SetNamespace(comp.GetInstanceNamespace()) + kubeName := comp.GetName() + "-" + getType(s) + + _ = svc.GetObservedKubeObject(s, kubeName) + mp := v12.ManagementPolicies{v12.ManagementActionObserve} + labels := s.GetPodTemplateLabels() + _, exists := labels[billingLabel] + if !s.GetCreationTimestamp().Time.IsZero() { + if !exists { + labels[billingLabel] = "true" + s.SetPodTemplateLabels(labels) + mp = append(mp, v12.ManagementActionCreate, v12.ManagementActionUpdate) + } + } + + err := svc.SetDesiredKubeObject(s.GetObject(), kubeName, func(obj *xkube.Object) { + obj.Spec.ManagementPolicies = mp + }) + + if err != nil && !exists { + runtime.NewWarningResult(fmt.Sprintf("cannot add billing to service object %s", s.GetName())) + } + + return runtime.NewNormalResult("billing enabled") +} + +func getType(myvar interface{}) (res string) { + return strings.ToLower(reflect.TypeOf(myvar).Elem().Field(0).Name) +} diff --git a/pkg/comp-functions/functions/common/interfaces.go b/pkg/comp-functions/functions/common/interfaces.go index d2e193e3f4..31ba70c5f8 100644 --- a/pkg/comp-functions/functions/common/interfaces.go +++ b/pkg/comp-functions/functions/common/interfaces.go @@ -18,6 +18,8 @@ type InfoGetter interface { GetSecurity() *vshnv1.Security InstanceNamespaceInfo GetPDBLabels() map[string]string + GetWorkloadPodTemplateLabelsManager() vshnv1.PodTemplateLabelsManager + GetWorkloadName() string } // InstanceNamespaceInfo provides all the necessary information to create diff --git a/pkg/comp-functions/functions/vshnkeycloak/billing.go b/pkg/comp-functions/functions/vshnkeycloak/billing.go new file mode 100644 index 0000000000..0d81ee5d79 --- /dev/null +++ b/pkg/comp-functions/functions/vshnkeycloak/billing.go @@ -0,0 +1,20 @@ +package vshnkeycloak + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNKeycloak, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnkeycloak/register.go b/pkg/comp-functions/functions/vshnkeycloak/register.go index 33027c2adf..695fa6cd1f 100644 --- a/pkg/comp-functions/functions/vshnkeycloak/register.go +++ b/pkg/comp-functions/functions/vshnkeycloak/register.go @@ -39,6 +39,10 @@ func init() { Name: "pdb", Execute: common.AddPDBSettings[*vshnv1.VSHNKeycloak], }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) } diff --git a/pkg/comp-functions/functions/vshnmariadb/billing.go b/pkg/comp-functions/functions/vshnmariadb/billing.go new file mode 100644 index 0000000000..ed50df3866 --- /dev/null +++ b/pkg/comp-functions/functions/vshnmariadb/billing.go @@ -0,0 +1,20 @@ +package vshnmariadb + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNMariaDB, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnmariadb/register.go b/pkg/comp-functions/functions/vshnmariadb/register.go index ff8dca2cfd..a1deaeeecc 100644 --- a/pkg/comp-functions/functions/vshnmariadb/register.go +++ b/pkg/comp-functions/functions/vshnmariadb/register.go @@ -35,6 +35,10 @@ func init() { Name: "non-sla-prometheus-rules", Execute: nonsla.GenerateNonSLAPromRules[*vshnv1.VSHNMariaDB](nonsla.NewAlertSetBuilder("mariadb", "mariadb").AddAll().GetAlerts()), }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) } diff --git a/pkg/comp-functions/functions/vshnminio/billing.go b/pkg/comp-functions/functions/vshnminio/billing.go new file mode 100644 index 0000000000..feefc28e6a --- /dev/null +++ b/pkg/comp-functions/functions/vshnminio/billing.go @@ -0,0 +1,20 @@ +package vshnminio + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNMinio, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnminio/register.go b/pkg/comp-functions/functions/vshnminio/register.go index e5bbf3b902..4ebf7ddd52 100644 --- a/pkg/comp-functions/functions/vshnminio/register.go +++ b/pkg/comp-functions/functions/vshnminio/register.go @@ -43,6 +43,10 @@ func init() { Name: "pdb", Execute: common.AddPDBSettings[*vshnv1.VSHNMinio], }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) } diff --git a/pkg/comp-functions/functions/vshnnextcloud/billing.go b/pkg/comp-functions/functions/vshnnextcloud/billing.go new file mode 100644 index 0000000000..0fe1d72495 --- /dev/null +++ b/pkg/comp-functions/functions/vshnnextcloud/billing.go @@ -0,0 +1,20 @@ +package vshnnextcloud + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNNextcloud, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnnextcloud/register.go b/pkg/comp-functions/functions/vshnnextcloud/register.go index f784bfce27..180d4758f0 100644 --- a/pkg/comp-functions/functions/vshnnextcloud/register.go +++ b/pkg/comp-functions/functions/vshnnextcloud/register.go @@ -25,6 +25,10 @@ func init() { Name: "backup", Execute: AddBackup, }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) } diff --git a/pkg/comp-functions/functions/vshnpostgres/billing.go b/pkg/comp-functions/functions/vshnpostgres/billing.go new file mode 100644 index 0000000000..40efc47083 --- /dev/null +++ b/pkg/comp-functions/functions/vshnpostgres/billing.go @@ -0,0 +1,20 @@ +package vshnpostgres + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNPostgreSQL, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnpostgres/register.go b/pkg/comp-functions/functions/vshnpostgres/register.go index 1fd323ffca..d0d9ce6bf1 100644 --- a/pkg/comp-functions/functions/vshnpostgres/register.go +++ b/pkg/comp-functions/functions/vshnpostgres/register.go @@ -81,6 +81,10 @@ func init() { Name: "pdb", Execute: common.AddPDBSettings[*vshnv1.VSHNPostgreSQL], }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) } diff --git a/pkg/comp-functions/functions/vshnredis/billing.go b/pkg/comp-functions/functions/vshnredis/billing.go new file mode 100644 index 0000000000..af57a46818 --- /dev/null +++ b/pkg/comp-functions/functions/vshnredis/billing.go @@ -0,0 +1,20 @@ +package vshnredis + +import ( + "context" + "fmt" + xfnproto "github.com/crossplane/function-sdk-go/proto/v1beta1" + v1 "github.com/vshn/appcat/v4/apis/vshn/v1" + "github.com/vshn/appcat/v4/pkg/comp-functions/functions/common" + "github.com/vshn/appcat/v4/pkg/comp-functions/runtime" +) + +// AddServiceBillingLabel adds billingLabel to all the objects of a services that must be billed +func AddServiceBillingLabel(ctx context.Context, comp *v1.VSHNRedis, svc *runtime.ServiceRuntime) *xfnproto.Result { + err := svc.GetObservedComposite(comp) + if err != nil { + return runtime.NewFatalResult(fmt.Errorf("can't get composite: %w", err)) + } + + return common.InjectBillingLabelToService(ctx, svc, comp) +} diff --git a/pkg/comp-functions/functions/vshnredis/register.go b/pkg/comp-functions/functions/vshnredis/register.go index 7fa6953027..79aa603dde 100644 --- a/pkg/comp-functions/functions/vshnredis/register.go +++ b/pkg/comp-functions/functions/vshnredis/register.go @@ -50,6 +50,10 @@ func init() { Name: "non-sla-prometheus-rules", Execute: nonsla.GenerateNonSLAPromRules[*vshnv1.VSHNRedis](nonsla.NewAlertSetBuilder("redis", "redis").AddAll().GetAlerts()), }, + { + Name: "billing", + Execute: AddServiceBillingLabel, + }, }, }) }