mirror of
https://github.com/grafana/grafana.git
synced 2024-12-01 21:19:28 -06:00
3c50db328d
* installer is responsible for removing from file system * take plugin as arg * remove resolve step * return plugin in test
186 lines
5.2 KiB
Go
186 lines
5.2 KiB
Go
package manager
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/grafana/grafana/pkg/plugins"
|
|
"github.com/grafana/grafana/pkg/plugins/config"
|
|
"github.com/grafana/grafana/pkg/plugins/log"
|
|
"github.com/grafana/grafana/pkg/plugins/manager/loader"
|
|
"github.com/grafana/grafana/pkg/plugins/manager/registry"
|
|
"github.com/grafana/grafana/pkg/plugins/manager/sources"
|
|
"github.com/grafana/grafana/pkg/plugins/repo"
|
|
"github.com/grafana/grafana/pkg/plugins/storage"
|
|
)
|
|
|
|
var _ plugins.Installer = (*PluginInstaller)(nil)
|
|
|
|
type PluginInstaller struct {
|
|
pluginRepo repo.Service
|
|
pluginStorage storage.ZipExtractor
|
|
pluginStorageDirFunc storage.DirNameGeneratorFunc
|
|
pluginRegistry registry.Service
|
|
pluginLoader loader.Service
|
|
log log.Logger
|
|
}
|
|
|
|
func ProvideInstaller(cfg *config.Cfg, pluginRegistry registry.Service, pluginLoader loader.Service,
|
|
pluginRepo repo.Service) *PluginInstaller {
|
|
return New(pluginRegistry, pluginLoader, pluginRepo,
|
|
storage.FileSystem(log.NewPrettyLogger("installer.fs"), cfg.PluginsPath), storage.SimpleDirNameGeneratorFunc)
|
|
}
|
|
|
|
func New(pluginRegistry registry.Service, pluginLoader loader.Service, pluginRepo repo.Service,
|
|
pluginStorage storage.ZipExtractor, pluginStorageDirFunc storage.DirNameGeneratorFunc) *PluginInstaller {
|
|
return &PluginInstaller{
|
|
pluginLoader: pluginLoader,
|
|
pluginRegistry: pluginRegistry,
|
|
pluginRepo: pluginRepo,
|
|
pluginStorage: pluginStorage,
|
|
pluginStorageDirFunc: pluginStorageDirFunc,
|
|
log: log.New("plugin.installer"),
|
|
}
|
|
}
|
|
|
|
func (m *PluginInstaller) Add(ctx context.Context, pluginID, version string, opts plugins.CompatOpts) error {
|
|
compatOpts, err := repoCompatOpts(opts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var pluginArchive *repo.PluginArchive
|
|
if plugin, exists := m.plugin(ctx, pluginID); exists {
|
|
if plugin.IsCorePlugin() || plugin.IsBundledPlugin() {
|
|
return plugins.ErrInstallCorePlugin
|
|
}
|
|
|
|
if plugin.Info.Version == version {
|
|
return plugins.DuplicateError{
|
|
PluginID: plugin.ID,
|
|
}
|
|
}
|
|
|
|
// get plugin update information to confirm if target update is possible
|
|
pluginArchiveInfo, err := m.pluginRepo.GetPluginArchiveInfo(ctx, pluginID, version, compatOpts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// if existing plugin version is the same as the target update version
|
|
if pluginArchiveInfo.Version == plugin.Info.Version {
|
|
return plugins.DuplicateError{
|
|
PluginID: plugin.ID,
|
|
}
|
|
}
|
|
|
|
if pluginArchiveInfo.URL == "" && pluginArchiveInfo.Version == "" {
|
|
return fmt.Errorf("could not determine update options for %s", pluginID)
|
|
}
|
|
|
|
// remove existing installation of plugin
|
|
err = m.Remove(ctx, plugin.ID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if pluginArchiveInfo.URL != "" {
|
|
pluginArchive, err = m.pluginRepo.GetPluginArchiveByURL(ctx, pluginArchiveInfo.URL, compatOpts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
pluginArchive, err = m.pluginRepo.GetPluginArchive(ctx, pluginID, pluginArchiveInfo.Version, compatOpts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else {
|
|
var err error
|
|
pluginArchive, err = m.pluginRepo.GetPluginArchive(ctx, pluginID, version, compatOpts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
extractedArchive, err := m.pluginStorage.Extract(ctx, pluginID, m.pluginStorageDirFunc, pluginArchive.File)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// download dependency plugins
|
|
pathsToScan := []string{extractedArchive.Path}
|
|
for _, dep := range extractedArchive.Dependencies {
|
|
m.log.Info(fmt.Sprintf("Fetching %s dependencies...", dep.ID))
|
|
d, err := m.pluginRepo.GetPluginArchive(ctx, dep.ID, dep.Version, compatOpts)
|
|
if err != nil {
|
|
return fmt.Errorf("%v: %w", fmt.Sprintf("failed to download plugin %s from repository", dep.ID), err)
|
|
}
|
|
|
|
depArchive, err := m.pluginStorage.Extract(ctx, dep.ID, m.pluginStorageDirFunc, d.File)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
pathsToScan = append(pathsToScan, depArchive.Path)
|
|
}
|
|
|
|
_, err = m.pluginLoader.Load(ctx, sources.NewLocalSource(plugins.ClassExternal, pathsToScan))
|
|
if err != nil {
|
|
m.log.Error("Could not load plugins", "paths", pathsToScan, "error", err)
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (m *PluginInstaller) Remove(ctx context.Context, pluginID string) error {
|
|
plugin, exists := m.plugin(ctx, pluginID)
|
|
if !exists {
|
|
return plugins.ErrPluginNotInstalled
|
|
}
|
|
|
|
if plugin.IsCorePlugin() || plugin.IsBundledPlugin() {
|
|
return plugins.ErrUninstallCorePlugin
|
|
}
|
|
|
|
p, err := m.pluginLoader.Unload(ctx, plugin)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if remover, ok := p.FS.(plugins.FSRemover); ok {
|
|
if err = remover.Remove(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// plugin finds a plugin with `pluginID` from the store
|
|
func (m *PluginInstaller) plugin(ctx context.Context, pluginID string) (*plugins.Plugin, bool) {
|
|
p, exists := m.pluginRegistry.Plugin(ctx, pluginID)
|
|
if !exists {
|
|
return nil, false
|
|
}
|
|
|
|
return p, true
|
|
}
|
|
|
|
func repoCompatOpts(opts plugins.CompatOpts) (repo.CompatOpts, error) {
|
|
os := opts.OS()
|
|
arch := opts.Arch()
|
|
if len(os) == 0 || len(arch) == 0 {
|
|
return repo.CompatOpts{}, errors.New("invalid system compatibility options provided")
|
|
}
|
|
|
|
grafanaVersion := opts.GrafanaVersion()
|
|
if len(grafanaVersion) == 0 {
|
|
return repo.NewSystemCompatOpts(os, arch), nil
|
|
}
|
|
|
|
return repo.NewCompatOpts(grafanaVersion, os, arch), nil
|
|
}
|