package sql import ( "os" "strings" "github.com/prometheus/client_golang/prometheus" "github.com/grafana/authlib/types" infraDB "github.com/grafana/grafana/pkg/infra/db" "github.com/grafana/grafana/pkg/infra/tracing" "github.com/grafana/grafana/pkg/services/sqlstore/migrator" "github.com/grafana/grafana/pkg/setting" "github.com/grafana/grafana/pkg/storage/unified/resource" "github.com/grafana/grafana/pkg/storage/unified/sql/db/dbimpl" ) // Creates a new ResourceServer func NewResourceServer(db infraDB.DB, cfg *setting.Cfg, tracer tracing.Tracer, reg prometheus.Registerer, ac types.AccessClient, searchOptions resource.SearchOptions, storageMetrics *resource.StorageMetrics) (resource.ResourceServer, error) { apiserverCfg := cfg.SectionWithEnvOverrides("grafana-apiserver") opts := resource.ResourceServerOptions{ Tracer: tracer, Blob: resource.BlobConfig{ URL: apiserverCfg.Key("blob_url").MustString(""), }, Reg: reg, } if ac != nil { opts.AccessClient = resource.NewAuthzLimitedClient(ac, resource.AuthzOptions{Tracer: tracer, Registry: reg}) } // Support local file blob if strings.HasPrefix(opts.Blob.URL, "./data/") { dir := strings.Replace(opts.Blob.URL, "./data", cfg.DataPath, 1) err := os.MkdirAll(dir, 0700) if err != nil { return nil, err } opts.Blob.URL = "file:///" + dir } eDB, err := dbimpl.ProvideResourceDB(db, cfg, tracer) if err != nil { return nil, err } isHA := isHighAvailabilityEnabled(cfg.SectionWithEnvOverrides("database")) store, err := NewBackend(BackendOptions{DBProvider: eDB, Tracer: tracer, IsHA: isHA, storageMetrics: storageMetrics}) if err != nil { return nil, err } opts.Backend = store opts.Diagnostics = store opts.Lifecycle = store opts.Search = searchOptions rs, err := resource.NewResourceServer(opts) if err != nil { return nil, err } return rs, nil } // isHighAvailabilityEnabled determines if high availability mode should // be enabled based on database configuration. High availability is enabled // by default except for SQLite databases. func isHighAvailabilityEnabled(dbCfg *setting.DynamicSection) bool { // Check in the config if HA is enabled - by default we always assume a HA setup. isHA := dbCfg.Key("high_availability").MustBool(true) // SQLite is not possible to run in HA, so we force it to false. databaseType := dbCfg.Key("type").String() if databaseType == migrator.SQLite { isHA = false } return isHA }