2021-03-08 00:02:49 -06:00
|
|
|
package manager
|
2015-02-27 06:45:00 -06:00
|
|
|
|
|
|
|
import (
|
2020-05-04 03:57:55 -05:00
|
|
|
"context"
|
2020-10-21 05:39:41 -05:00
|
|
|
"errors"
|
2020-05-04 03:57:55 -05:00
|
|
|
"fmt"
|
2015-02-27 06:45:00 -06:00
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
|
|
"github.com/grafana/grafana/pkg/models"
|
2021-03-08 00:02:49 -06:00
|
|
|
"github.com/grafana/grafana/pkg/plugins"
|
2020-05-04 03:57:55 -05:00
|
|
|
"github.com/grafana/grafana/pkg/plugins/backendplugin"
|
2015-11-27 03:04:43 -06:00
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
2020-05-04 03:57:55 -05:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
2015-11-27 03:17:27 -06:00
|
|
|
"gopkg.in/ini.v1"
|
2015-02-27 06:45:00 -06:00
|
|
|
)
|
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
func TestPluginManager_Init(t *testing.T) {
|
|
|
|
t.Run("Base case", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginSettings = setting.PluginSettings{
|
|
|
|
"nginx-app": map[string]string{
|
|
|
|
"path": "testdata/test-app",
|
2020-05-04 03:57:55 -05:00
|
|
|
},
|
2021-03-10 05:41:29 -06:00
|
|
|
}
|
|
|
|
})
|
2018-04-27 08:11:55 -05:00
|
|
|
err := pm.Init()
|
2020-05-04 03:57:55 -05:00
|
|
|
require.NoError(t, err)
|
2015-02-27 06:45:00 -06:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
assert.Empty(t, pm.scanningErrors)
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.Greater(t, len(pm.dataSources), 1)
|
2021-03-18 07:53:01 -05:00
|
|
|
assert.Greater(t, len(pm.panels), 1)
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.Equal(t, "app/plugins/datasource/graphite/module", pm.dataSources["graphite"].Module)
|
2021-03-18 07:53:01 -05:00
|
|
|
assert.NotEmpty(t, pm.apps)
|
|
|
|
assert.Equal(t, "public/plugins/test-app/img/logo_large.png", pm.apps["test-app"].Info.Logos.Large)
|
|
|
|
assert.Equal(t, "public/plugins/test-app/img/screenshot2.png", pm.apps["test-app"].Info.Screenshots[1].Path)
|
2020-05-04 03:57:55 -05:00
|
|
|
})
|
2016-01-09 16:34:20 -06:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
t.Run("With external back-end plugin lacking signature", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/unsigned"
|
|
|
|
})
|
2020-05-04 03:57:55 -05:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test" is unsigned`)}, pm.scanningErrors)
|
2015-02-27 06:45:00 -06:00
|
|
|
})
|
2016-01-08 13:57:58 -06:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
t.Run("With external unsigned back-end plugin and configuration disabling signature check of this plugin", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/unsigned"
|
|
|
|
pm.Cfg.PluginsAllowUnsigned = []string{"test"}
|
|
|
|
})
|
2020-04-14 11:04:27 -05:00
|
|
|
err := pm.Init()
|
2020-05-04 03:57:55 -05:00
|
|
|
require.NoError(t, err)
|
2016-03-10 12:57:48 -06:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
assert.Empty(t, pm.scanningErrors)
|
2016-01-08 13:57:58 -06:00
|
|
|
})
|
|
|
|
|
2020-12-11 05:57:57 -06:00
|
|
|
t.Run("With external back-end plugin with invalid v1 signature", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/invalid-v1-signature"
|
|
|
|
})
|
2020-05-04 03:57:55 -05:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
2019-09-24 00:54:28 -05:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test" has an invalid signature`)}, pm.scanningErrors)
|
2019-09-24 00:54:28 -05:00
|
|
|
})
|
2020-05-04 10:39:20 -05:00
|
|
|
|
2020-05-12 08:48:24 -05:00
|
|
|
t.Run("With external back-end plugin lacking files listed in manifest", func(t *testing.T) {
|
|
|
|
fm := &fakeBackendPluginManager{}
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/lacking-files"
|
|
|
|
pm.BackendPluginManager = fm
|
|
|
|
})
|
2020-05-12 08:48:24 -05:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test"'s signature has been modified`)}, pm.scanningErrors)
|
|
|
|
})
|
|
|
|
|
2020-05-04 10:39:20 -05:00
|
|
|
t.Run("Transform plugins should be ignored when expressions feature is off", func(t *testing.T) {
|
|
|
|
fm := fakeBackendPluginManager{}
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/behind-feature-flag"
|
|
|
|
pm.BackendPluginManager = &fm
|
|
|
|
})
|
2020-05-04 10:39:20 -05:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Empty(t, pm.scanningErrors)
|
2020-05-12 08:48:24 -05:00
|
|
|
assert.Empty(t, fm.registeredPlugins)
|
2020-05-04 10:39:20 -05:00
|
|
|
})
|
|
|
|
|
2020-10-21 05:39:41 -05:00
|
|
|
t.Run("With nested plugin duplicating parent", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/duplicate-plugins"
|
|
|
|
})
|
2020-10-21 05:39:41 -05:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Len(t, pm.scanningErrors, 1)
|
2021-03-08 00:02:49 -06:00
|
|
|
assert.True(t, errors.Is(pm.scanningErrors[0], plugins.DuplicatePluginError{}))
|
2020-10-21 05:39:41 -05:00
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
|
|
|
|
t.Run("With external back-end plugin with valid v2 signature", func(t *testing.T) {
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(manager *PluginManager) {
|
|
|
|
manager.Cfg.PluginsPath = "testdata/valid-v2-signature"
|
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Empty(t, pm.scanningErrors)
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
const pluginID = "test"
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.NotNil(t, pm.plugins[pluginID])
|
|
|
|
assert.Equal(t, "datasource", pm.plugins[pluginID].Type)
|
|
|
|
assert.Equal(t, "Test", pm.plugins[pluginID].Name)
|
|
|
|
assert.Equal(t, pluginID, pm.plugins[pluginID].Id)
|
|
|
|
assert.Equal(t, "1.0.0", pm.plugins[pluginID].Info.Version)
|
|
|
|
assert.Equal(t, plugins.PluginSignatureValid, pm.plugins[pluginID].Signature)
|
|
|
|
assert.Equal(t, plugins.GrafanaType, pm.plugins[pluginID].SignatureType)
|
|
|
|
assert.Equal(t, "Grafana Labs", pm.plugins[pluginID].SignatureOrg)
|
|
|
|
assert.False(t, pm.plugins[pluginID].IsCorePlugin)
|
2020-12-11 05:57:57 -06:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("With back-end plugin with invalid v2 private signature (mismatched root URL)", func(t *testing.T) {
|
|
|
|
origAppURL := setting.AppUrl
|
|
|
|
t.Cleanup(func() {
|
|
|
|
setting.AppUrl = origAppURL
|
|
|
|
})
|
|
|
|
setting.AppUrl = "http://localhost:1234"
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/valid-v2-pvt-signature"
|
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test" has an invalid signature`)}, pm.scanningErrors)
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.Nil(t, pm.plugins[("test")])
|
2020-12-11 05:57:57 -06:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("With back-end plugin with valid v2 private signature", func(t *testing.T) {
|
|
|
|
origAppURL := setting.AppUrl
|
|
|
|
t.Cleanup(func() {
|
|
|
|
setting.AppUrl = origAppURL
|
|
|
|
})
|
|
|
|
setting.AppUrl = "http://localhost:3000/"
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/valid-v2-pvt-signature"
|
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Empty(t, pm.scanningErrors)
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
const pluginID = "test"
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.NotNil(t, pm.plugins[pluginID])
|
|
|
|
assert.Equal(t, "datasource", pm.plugins[pluginID].Type)
|
|
|
|
assert.Equal(t, "Test", pm.plugins[pluginID].Name)
|
|
|
|
assert.Equal(t, pluginID, pm.plugins[pluginID].Id)
|
|
|
|
assert.Equal(t, "1.0.0", pm.plugins[pluginID].Info.Version)
|
|
|
|
assert.Equal(t, plugins.PluginSignatureValid, pm.plugins[pluginID].Signature)
|
|
|
|
assert.Equal(t, plugins.PrivateType, pm.plugins[pluginID].SignatureType)
|
|
|
|
assert.Equal(t, "Will Browne", pm.plugins[pluginID].SignatureOrg)
|
|
|
|
assert.False(t, pm.plugins[pluginID].IsCorePlugin)
|
2020-12-11 05:57:57 -06:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("With back-end plugin with modified v2 signature (missing file from plugin dir)", func(t *testing.T) {
|
|
|
|
origAppURL := setting.AppUrl
|
|
|
|
t.Cleanup(func() {
|
|
|
|
setting.AppUrl = origAppURL
|
|
|
|
})
|
|
|
|
setting.AppUrl = "http://localhost:3000/"
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/invalid-v2-signature"
|
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test"'s signature has been modified`)}, pm.scanningErrors)
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.Nil(t, pm.plugins[("test")])
|
2020-12-11 05:57:57 -06:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("With back-end plugin with modified v2 signature (unaccounted file in plugin dir)", func(t *testing.T) {
|
|
|
|
origAppURL := setting.AppUrl
|
|
|
|
t.Cleanup(func() {
|
|
|
|
setting.AppUrl = origAppURL
|
|
|
|
})
|
|
|
|
setting.AppUrl = "http://localhost:3000/"
|
|
|
|
|
2021-03-10 05:41:29 -06:00
|
|
|
pm := createManager(t, func(pm *PluginManager) {
|
|
|
|
pm.Cfg.PluginsPath = "testdata/invalid-v2-signature-2"
|
|
|
|
})
|
2020-12-11 05:57:57 -06:00
|
|
|
err := pm.Init()
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, []error{fmt.Errorf(`plugin "test"'s signature has been modified`)}, pm.scanningErrors)
|
2021-03-17 10:06:10 -05:00
|
|
|
assert.Nil(t, pm.plugins[("test")])
|
2020-12-11 05:57:57 -06:00
|
|
|
})
|
2020-05-04 03:57:55 -05:00
|
|
|
}
|
2019-09-24 00:54:28 -05:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
func TestPluginManager_IsBackendOnlyPlugin(t *testing.T) {
|
|
|
|
pluginScanner := &PluginScanner{}
|
2019-09-24 00:54:28 -05:00
|
|
|
|
2020-05-04 03:57:55 -05:00
|
|
|
type testCase struct {
|
|
|
|
name string
|
|
|
|
isBackendOnly bool
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range []testCase{
|
|
|
|
{name: "renderer", isBackendOnly: true},
|
|
|
|
{name: "app", isBackendOnly: false},
|
|
|
|
} {
|
|
|
|
t.Run(fmt.Sprintf("Plugin %s", c.name), func(t *testing.T) {
|
|
|
|
result := pluginScanner.IsBackendOnlyPlugin(c.name)
|
|
|
|
|
|
|
|
assert.Equal(t, c.isBackendOnly, result)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type fakeBackendPluginManager struct {
|
2021-03-08 00:02:49 -06:00
|
|
|
backendplugin.Manager
|
|
|
|
|
2020-05-12 08:48:24 -05:00
|
|
|
registeredPlugins []string
|
2020-05-04 03:57:55 -05:00
|
|
|
}
|
|
|
|
|
2020-06-11 09:14:05 -05:00
|
|
|
func (f *fakeBackendPluginManager) Register(pluginID string, factory backendplugin.PluginFactoryFunc) error {
|
|
|
|
f.registeredPlugins = append(f.registeredPlugins, pluginID)
|
2020-05-04 03:57:55 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-05-04 10:39:20 -05:00
|
|
|
func (f *fakeBackendPluginManager) StartPlugin(ctx context.Context, pluginID string) error {
|
2020-05-04 03:57:55 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-06-11 09:14:05 -05:00
|
|
|
func (f *fakeBackendPluginManager) CollectMetrics(ctx context.Context, pluginID string) (*backend.CollectMetricsResult, error) {
|
2020-05-04 03:57:55 -05:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2020-06-11 09:14:05 -05:00
|
|
|
func (f *fakeBackendPluginManager) CheckHealth(ctx context.Context, pCtx backend.PluginContext) (*backend.CheckHealthResult, error) {
|
2020-05-04 03:57:55 -05:00
|
|
|
return nil, nil
|
|
|
|
}
|
2019-09-24 00:54:28 -05:00
|
|
|
|
2020-05-04 10:39:20 -05:00
|
|
|
func (f *fakeBackendPluginManager) CallResource(pluginConfig backend.PluginContext, ctx *models.ReqContext, path string) {
|
2015-02-27 06:45:00 -06:00
|
|
|
}
|
2021-03-10 05:41:29 -06:00
|
|
|
|
|
|
|
func createManager(t *testing.T, cbs ...func(*PluginManager)) *PluginManager {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
staticRootPath, err := filepath.Abs("../../../public/")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-03-18 07:53:01 -05:00
|
|
|
pm := newManager(&setting.Cfg{
|
|
|
|
Raw: ini.Empty(),
|
|
|
|
Env: setting.Prod,
|
|
|
|
StaticRootPath: staticRootPath,
|
|
|
|
})
|
|
|
|
pm.BackendPluginManager = &fakeBackendPluginManager{}
|
2021-03-10 05:41:29 -06:00
|
|
|
for _, cb := range cbs {
|
|
|
|
cb(pm)
|
|
|
|
}
|
|
|
|
|
|
|
|
return pm
|
|
|
|
}
|