mirror of
https://github.com/grafana/grafana.git
synced 2025-07-28 23:32:18 +08:00
148 lines
4.3 KiB
Go
148 lines
4.3 KiB
Go
package app
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/grafana/grafana-app-sdk/app"
|
|
"github.com/grafana/grafana-app-sdk/k8s"
|
|
"github.com/grafana/grafana-app-sdk/logging"
|
|
"github.com/grafana/grafana-app-sdk/resource"
|
|
"github.com/grafana/grafana-app-sdk/simple"
|
|
advisorv0alpha1 "github.com/grafana/grafana/apps/advisor/pkg/apis/advisor/v0alpha1"
|
|
"github.com/grafana/grafana/apps/advisor/pkg/app/checkregistry"
|
|
"github.com/grafana/grafana/apps/advisor/pkg/app/checks"
|
|
"github.com/grafana/grafana/apps/advisor/pkg/app/checkscheduler"
|
|
"github.com/grafana/grafana/apps/advisor/pkg/app/checktyperegisterer"
|
|
"github.com/grafana/grafana/pkg/apimachinery/identity"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
)
|
|
|
|
func New(cfg app.Config) (app.App, error) {
|
|
// Read config
|
|
specificConfig, ok := cfg.SpecificConfig.(checkregistry.AdvisorAppConfig)
|
|
if !ok {
|
|
return nil, fmt.Errorf("invalid config type")
|
|
}
|
|
checkRegistry := specificConfig.CheckRegistry
|
|
log := logging.DefaultLogger.With("app", "advisor.app")
|
|
|
|
// Prepare storage client
|
|
clientGenerator := k8s.NewClientRegistry(cfg.KubeConfig, k8s.ClientConfig{})
|
|
client, err := clientGenerator.ClientFor(advisorv0alpha1.CheckKind())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
typesClient, err := clientGenerator.ClientFor(advisorv0alpha1.CheckTypeKind())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Initialize checks
|
|
checkMap := map[string]checks.Check{}
|
|
for _, c := range checkRegistry.Checks() {
|
|
checkMap[c.ID()] = c
|
|
}
|
|
|
|
simpleConfig := simple.AppConfig{
|
|
Name: "advisor",
|
|
KubeConfig: cfg.KubeConfig,
|
|
InformerConfig: simple.AppInformerConfig{
|
|
ErrorHandler: func(ctx context.Context, err error) {
|
|
log.WithContext(ctx).Error("Informer processing error", "error", err)
|
|
},
|
|
},
|
|
ManagedKinds: []simple.AppManagedKind{
|
|
{
|
|
Kind: advisorv0alpha1.CheckKind(),
|
|
Validator: &simple.Validator{
|
|
ValidateFunc: func(ctx context.Context, req *app.AdmissionRequest) error {
|
|
if req.Object != nil {
|
|
check, err := getCheck(req.Object, checkMap)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if req.Action == resource.AdmissionActionCreate {
|
|
go func() {
|
|
logger := log.WithContext(ctx).With("check", check.ID())
|
|
logger.Debug("Processing check", "namespace", req.Object.GetNamespace())
|
|
requester, err := identity.GetRequester(ctx)
|
|
if err != nil {
|
|
logger.Error("Error getting requester", "error", err)
|
|
return
|
|
}
|
|
ctx = identity.WithServiceIdentityContext(context.WithoutCancel(ctx), requester.GetOrgID())
|
|
err = processCheck(ctx, logger, client, typesClient, req.Object, check)
|
|
if err != nil {
|
|
logger.Error("Error processing check", "error", err)
|
|
}
|
|
}()
|
|
}
|
|
if req.Action == resource.AdmissionActionUpdate {
|
|
go func() {
|
|
logger := log.WithContext(ctx).With("check", check.ID())
|
|
logger.Debug("Updating check", "namespace", req.Object.GetNamespace(), "name", req.Object.GetName())
|
|
requester, err := identity.GetRequester(ctx)
|
|
if err != nil {
|
|
logger.Error("Error getting requester", "error", err)
|
|
return
|
|
}
|
|
ctx = identity.WithServiceIdentityContext(context.WithoutCancel(ctx), requester.GetOrgID())
|
|
err = processCheckRetry(ctx, logger, client, typesClient, req.Object, check)
|
|
if err != nil {
|
|
logger.Error("Error processing check retry", "error", err)
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
return nil
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Kind: advisorv0alpha1.CheckTypeKind(),
|
|
},
|
|
},
|
|
}
|
|
|
|
a, err := simple.NewApp(simpleConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = a.ValidateManifest(cfg.ManifestData)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Save check types as resources
|
|
ctr, err := checktyperegisterer.New(cfg, log)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
a.AddRunnable(ctr)
|
|
|
|
// Start scheduler
|
|
csch, err := checkscheduler.New(cfg, log)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
a.AddRunnable(csch)
|
|
|
|
return a, nil
|
|
}
|
|
|
|
func GetKinds() map[schema.GroupVersion][]resource.Kind {
|
|
gv := schema.GroupVersion{
|
|
// Group and version are the same for all checks
|
|
Group: advisorv0alpha1.CheckKind().Group(),
|
|
Version: advisorv0alpha1.CheckKind().Version(),
|
|
}
|
|
return map[schema.GroupVersion][]resource.Kind{
|
|
gv: {
|
|
advisorv0alpha1.CheckKind(),
|
|
advisorv0alpha1.CheckTypeKind(),
|
|
},
|
|
}
|
|
}
|