2020-06-22 23:49:13 +08:00
|
|
|
package plugins
|
|
|
|
|
|
|
|
import (
|
2021-12-03 01:08:59 +08:00
|
|
|
"context"
|
2020-11-19 21:47:17 +08:00
|
|
|
"errors"
|
2025-09-24 20:05:43 +08:00
|
|
|
"fmt"
|
2020-11-19 21:47:17 +08:00
|
|
|
|
2020-06-22 23:49:13 +08:00
|
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
2022-10-05 02:48:02 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/org"
|
2023-03-08 00:22:30 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/pluginsintegration/pluginsettings"
|
2023-09-11 19:59:24 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/pluginsintegration/pluginstore"
|
2020-06-22 23:49:13 +08:00
|
|
|
)
|
|
|
|
|
2025-09-24 20:05:43 +08:00
|
|
|
var (
|
|
|
|
ErrPluginProvisioningNotFound = errors.New("plugin not found")
|
|
|
|
ErrPluginProvisioningAutoEnabled = errors.New("plugin is auto enabled and cannot be disabled")
|
|
|
|
)
|
|
|
|
|
2020-06-22 23:49:13 +08:00
|
|
|
// Provision scans a directory for provisioning config files
|
|
|
|
// and provisions the app in those files.
|
2023-09-11 19:59:24 +08:00
|
|
|
func Provision(ctx context.Context, configDirectory string, pluginStore pluginstore.Store, pluginSettings pluginsettings.Service, orgService org.Service) error {
|
2021-03-18 20:53:01 +08:00
|
|
|
logger := log.New("provisioning.plugins")
|
|
|
|
ap := PluginProvisioner{
|
2022-03-03 18:57:33 +08:00
|
|
|
log: logger,
|
|
|
|
cfgProvider: newConfigReader(logger, pluginStore),
|
|
|
|
pluginSettings: pluginSettings,
|
2022-10-05 02:48:02 +08:00
|
|
|
orgService: orgService,
|
2024-12-16 21:51:04 +08:00
|
|
|
pluginStore: pluginStore,
|
2021-03-18 20:53:01 +08:00
|
|
|
}
|
2021-12-03 01:08:59 +08:00
|
|
|
return ap.applyChanges(ctx, configDirectory)
|
2020-06-22 23:49:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// PluginProvisioner is responsible for provisioning apps based on
|
|
|
|
// configuration read by the `configReader`
|
|
|
|
type PluginProvisioner struct {
|
2022-03-03 18:57:33 +08:00
|
|
|
log log.Logger
|
|
|
|
cfgProvider configReader
|
|
|
|
pluginSettings pluginsettings.Service
|
2022-10-05 02:48:02 +08:00
|
|
|
orgService org.Service
|
2024-12-16 21:51:04 +08:00
|
|
|
pluginStore pluginstore.Store
|
2020-06-22 23:49:13 +08:00
|
|
|
}
|
|
|
|
|
2021-12-03 01:08:59 +08:00
|
|
|
func (ap *PluginProvisioner) apply(ctx context.Context, cfg *pluginsAsConfig) error {
|
2020-06-22 23:49:13 +08:00
|
|
|
for _, app := range cfg.Apps {
|
|
|
|
if app.OrgID == 0 && app.OrgName != "" {
|
2022-10-05 02:48:02 +08:00
|
|
|
getOrgQuery := &org.GetOrgByNameQuery{Name: app.OrgName}
|
|
|
|
res, err := ap.orgService.GetByName(ctx, getOrgQuery)
|
|
|
|
if err != nil {
|
2020-06-22 23:49:13 +08:00
|
|
|
return err
|
|
|
|
}
|
2022-10-05 02:48:02 +08:00
|
|
|
app.OrgID = res.ID
|
2020-06-22 23:49:13 +08:00
|
|
|
} else if app.OrgID < 0 {
|
|
|
|
app.OrgID = 1
|
|
|
|
}
|
|
|
|
|
2024-12-16 21:51:04 +08:00
|
|
|
p, found := ap.pluginStore.Plugin(ctx, app.PluginID)
|
|
|
|
if !found {
|
2025-09-24 20:05:43 +08:00
|
|
|
return fmt.Errorf("%w: %s", ErrPluginProvisioningNotFound, app.PluginID)
|
2024-12-16 21:51:04 +08:00
|
|
|
}
|
|
|
|
if p.AutoEnabled && !app.Enabled {
|
2025-09-24 20:05:43 +08:00
|
|
|
return fmt.Errorf("%w: %s", ErrPluginProvisioningAutoEnabled, app.PluginID)
|
2024-12-16 21:51:04 +08:00
|
|
|
}
|
|
|
|
|
2022-03-19 03:49:13 +08:00
|
|
|
ps, err := ap.pluginSettings.GetPluginSettingByPluginID(ctx, &pluginsettings.GetByPluginIDArgs{
|
|
|
|
OrgID: app.OrgID,
|
|
|
|
PluginID: app.PluginID,
|
|
|
|
})
|
2020-06-22 23:49:13 +08:00
|
|
|
if err != nil {
|
2023-01-24 02:56:20 +08:00
|
|
|
if !errors.Is(err, pluginsettings.ErrPluginSettingNotFound) {
|
2025-09-24 20:05:43 +08:00
|
|
|
return fmt.Errorf("%w: %s", err, app.PluginID)
|
2020-06-22 23:49:13 +08:00
|
|
|
}
|
|
|
|
} else {
|
2022-03-19 03:49:13 +08:00
|
|
|
app.PluginVersion = ps.PluginVersion
|
2020-06-22 23:49:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ap.log.Info("Updating app from configuration ", "type", app.PluginID, "enabled", app.Enabled)
|
2022-03-19 03:49:13 +08:00
|
|
|
if err := ap.pluginSettings.UpdatePluginSetting(ctx, &pluginsettings.UpdateArgs{
|
|
|
|
OrgID: app.OrgID,
|
|
|
|
PluginID: app.PluginID,
|
2020-06-22 23:49:13 +08:00
|
|
|
Enabled: app.Enabled,
|
|
|
|
Pinned: app.Pinned,
|
2022-03-19 03:49:13 +08:00
|
|
|
JSONData: app.JSONData,
|
|
|
|
SecureJSONData: app.SecureJSONData,
|
2020-06-22 23:49:13 +08:00
|
|
|
PluginVersion: app.PluginVersion,
|
2022-03-19 03:49:13 +08:00
|
|
|
}); err != nil {
|
2025-09-24 20:05:43 +08:00
|
|
|
return fmt.Errorf("%w: %s", err, app.PluginID)
|
2020-06-22 23:49:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-03 01:08:59 +08:00
|
|
|
func (ap *PluginProvisioner) applyChanges(ctx context.Context, configPath string) error {
|
2021-12-22 18:02:42 +08:00
|
|
|
configs, err := ap.cfgProvider.readConfig(ctx, configPath)
|
2020-06-22 23:49:13 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cfg := range configs {
|
2021-12-03 01:08:59 +08:00
|
|
|
if err := ap.apply(ctx, cfg); err != nil {
|
2020-06-22 23:49:13 +08:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|