mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
* Provisioning: check sanity across all dashboard readers Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Apply suggestions from code review Co-authored-by: Emil Tullstedt <emil.tullstedt@grafana.com> Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> * Refactor of duplicateValidator and fix issues according to commentaries Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Apply suggestions from code review Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> * Remove newDuplicateEntries function Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Change folderUid in logs to folderUID Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Restrict write access for readers, which are provisioning duplicate dashboards Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Fix file reader after rebasing onto master Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Apply suggestions from code review Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> * Format file_reader Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Apply suggestions from code review Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> * Apply suggestions from code review Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> * Fix lint problem Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> * Apply suggestions from code review Co-authored-by: Marcus Efraimsson <marcus.efraimsson@gmail.com> Signed-off-by: m.nabokikh <maksim.nabokikh@flant.com> Co-authored-by: Emil Tullstedt <emil.tullstedt@grafana.com> Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com> Co-authored-by: Marcus Efraimsson <marcus.efraimsson@gmail.com>
142 lines
4.4 KiB
Go
142 lines
4.4 KiB
Go
package dashboards
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
|
|
"github.com/grafana/grafana/pkg/bus"
|
|
"github.com/grafana/grafana/pkg/dashboards"
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
"github.com/grafana/grafana/pkg/models"
|
|
"github.com/grafana/grafana/pkg/util/errutil"
|
|
)
|
|
|
|
// DashboardProvisioner is responsible for syncing dashboard from disk to
|
|
// Grafana's database.
|
|
type DashboardProvisioner interface {
|
|
Provision() error
|
|
PollChanges(ctx context.Context)
|
|
GetProvisionerResolvedPath(name string) string
|
|
GetAllowUIUpdatesFromConfig(name string) bool
|
|
CleanUpOrphanedDashboards()
|
|
}
|
|
|
|
// DashboardProvisionerFactory creates DashboardProvisioners based on input
|
|
type DashboardProvisionerFactory func(string, dashboards.Store) (DashboardProvisioner, error)
|
|
|
|
// Provisioner is responsible for syncing dashboard from disk to Grafana's database.
|
|
type Provisioner struct {
|
|
log log.Logger
|
|
fileReaders []*FileReader
|
|
configs []*config
|
|
duplicateValidator duplicateValidator
|
|
}
|
|
|
|
// New returns a new DashboardProvisioner
|
|
func New(configDirectory string, store dashboards.Store) (DashboardProvisioner, error) {
|
|
logger := log.New("provisioning.dashboard")
|
|
cfgReader := &configReader{path: configDirectory, log: logger}
|
|
configs, err := cfgReader.readConfig()
|
|
if err != nil {
|
|
return nil, errutil.Wrap("Failed to read dashboards config", err)
|
|
}
|
|
|
|
fileReaders, err := getFileReaders(configs, logger, store)
|
|
if err != nil {
|
|
return nil, errutil.Wrap("Failed to initialize file readers", err)
|
|
}
|
|
|
|
d := &Provisioner{
|
|
log: logger,
|
|
fileReaders: fileReaders,
|
|
configs: configs,
|
|
duplicateValidator: newDuplicateValidator(logger, fileReaders),
|
|
}
|
|
|
|
return d, nil
|
|
}
|
|
|
|
// Provision scans the disk for dashboards and updates
|
|
// the database with the latest versions of those dashboards.
|
|
func (provider *Provisioner) Provision() error {
|
|
for _, reader := range provider.fileReaders {
|
|
if err := reader.walkDisk(); err != nil {
|
|
if os.IsNotExist(err) {
|
|
// don't stop the provisioning service in case the folder is missing. The folder can appear after the startup
|
|
provider.log.Warn("Failed to provision config", "name", reader.Cfg.Name, "error", err)
|
|
return nil
|
|
}
|
|
|
|
return errutil.Wrapf(err, "Failed to provision config %v", reader.Cfg.Name)
|
|
}
|
|
}
|
|
|
|
provider.duplicateValidator.validate()
|
|
return nil
|
|
}
|
|
|
|
// CleanUpOrphanedDashboards deletes provisioned dashboards missing a linked reader.
|
|
func (provider *Provisioner) CleanUpOrphanedDashboards() {
|
|
currentReaders := make([]string, len(provider.fileReaders))
|
|
|
|
for index, reader := range provider.fileReaders {
|
|
currentReaders[index] = reader.Cfg.Name
|
|
}
|
|
|
|
if err := bus.Dispatch(&models.DeleteOrphanedProvisionedDashboardsCommand{ReaderNames: currentReaders}); err != nil {
|
|
provider.log.Warn("Failed to delete orphaned provisioned dashboards", "err", err)
|
|
}
|
|
}
|
|
|
|
// PollChanges starts polling for changes in dashboard definition files. It creates a goroutine for each provider
|
|
// defined in the config.
|
|
func (provider *Provisioner) PollChanges(ctx context.Context) {
|
|
for _, reader := range provider.fileReaders {
|
|
go reader.pollChanges(ctx)
|
|
}
|
|
|
|
go provider.duplicateValidator.Run(ctx)
|
|
}
|
|
|
|
// GetProvisionerResolvedPath returns resolved path for the specified provisioner name. Can be used to generate
|
|
// relative path to provisioning file from it's external_id.
|
|
func (provider *Provisioner) GetProvisionerResolvedPath(name string) string {
|
|
for _, reader := range provider.fileReaders {
|
|
if reader.Cfg.Name == name {
|
|
return reader.resolvedPath()
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// GetAllowUIUpdatesFromConfig return if a dashboard provisioner allows updates from the UI
|
|
func (provider *Provisioner) GetAllowUIUpdatesFromConfig(name string) bool {
|
|
for _, config := range provider.configs {
|
|
if config.Name == name {
|
|
return config.AllowUIUpdates
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func getFileReaders(configs []*config, logger log.Logger, store dashboards.Store) ([]*FileReader, error) {
|
|
var readers []*FileReader
|
|
|
|
for _, config := range configs {
|
|
switch config.Type {
|
|
case "file":
|
|
fileReader, err := NewDashboardFileReader(config, logger.New("type", config.Type, "name", config.Name),
|
|
store)
|
|
if err != nil {
|
|
return nil, errutil.Wrapf(err, "Failed to create file reader for config %v", config.Name)
|
|
}
|
|
readers = append(readers, fileReader)
|
|
default:
|
|
return nil, fmt.Errorf("type %s is not supported", config.Type)
|
|
}
|
|
}
|
|
|
|
return readers, nil
|
|
}
|