Files
grafana/pkg/setting/setting_plugins_test.go
Syerikjan Kh b38662e985 Plugins: Add preinstall_sync config - allow plugins to be preinstalled synchronously (#105292)
* feat: preinstall_sync config - process and installation logic

* ref: add preinstall_sync list to preinstalled plugins of frontendsettings

* fix: conf blank line for sections

* ref: remove plugins async flag, and rename PreinstallPlugins

* docs: default installed plugin list
2025-05-16 10:45:50 -04:00

275 lines
8.3 KiB
Go

package setting
import (
"strings"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestPluginSettings(t *testing.T) {
cfg := NewCfg()
sec, err := cfg.Raw.NewSection("plugin")
require.NoError(t, err)
_, err = sec.NewKey("key", "value")
require.NoError(t, err)
sec, err = cfg.Raw.NewSection("plugin.plugin")
require.NoError(t, err)
_, err = sec.NewKey("key1", "value1")
require.NoError(t, err)
_, err = sec.NewKey("key2", "value2")
require.NoError(t, err)
sec, err = cfg.Raw.NewSection("plugin.plugin2")
require.NoError(t, err)
_, err = sec.NewKey("key3", "value3")
require.NoError(t, err)
_, err = sec.NewKey("key4", "value4")
require.NoError(t, err)
sec, err = cfg.Raw.NewSection("other")
require.NoError(t, err)
_, err = sec.NewKey("keySomething", "whatever")
require.NoError(t, err)
ps := extractPluginSettings(cfg.Raw.Sections())
require.Len(t, ps, 2)
require.Len(t, ps["plugin"], 2)
require.Equal(t, ps["plugin"]["key1"], "value1")
require.Equal(t, ps["plugin"]["key2"], "value2")
require.Len(t, ps["plugin2"], 2)
require.Equal(t, ps["plugin2"]["key3"], "value3")
require.Equal(t, ps["plugin2"]["key4"], "value4")
}
func Test_readPluginSettings(t *testing.T) {
t.Run("should parse separated plugin ids", func(t *testing.T) {
for _, tc := range []struct {
name string
f func(ids ...string) string
}{
{
name: "commas",
f: func(ids ...string) string {
return strings.Join(ids, ",")
},
},
{
name: "commas and a space",
f: func(ids ...string) string {
return strings.Join(ids, ", ")
},
},
{
name: "spaces",
f: func(ids ...string) string {
return strings.Join(ids, " ")
},
},
{
name: "json-like",
f: func(ids ...string) string {
return `["` + strings.Join(ids, `","`) + `"]`
},
},
} {
t.Run(tc.name, func(t *testing.T) {
cfg := NewCfg()
sec, err := cfg.Raw.NewSection("plugins")
require.NoError(t, err)
_, err = sec.NewKey("disable_plugins", tc.f("plugin1", "plugin2"))
require.NoError(t, err)
_, err = sec.NewKey("plugin_catalog_hidden_plugins", tc.f("plugin3"))
require.NoError(t, err)
_, err = sec.NewKey("hide_angular_deprecation", tc.f("a", "b", "c"))
require.NoError(t, err)
err = cfg.readPluginSettings(cfg.Raw)
require.NoError(t, err)
require.Equal(t, []string{"plugin1", "plugin2"}, cfg.DisablePlugins)
require.Equal(t, []string{"plugin3", "plugin1", "plugin2"}, cfg.PluginCatalogHiddenPlugins)
require.Equal(t, []string{"a", "b", "c"}, cfg.HideAngularDeprecation)
})
}
})
t.Run("when plugins.preinstall_sync is defined", func(t *testing.T) {
tests := []struct {
name string
rawInput string
expected []InstallPlugin
disablePlugins string
disablePreinstall bool
}{
{
name: "should add the plugin to the sync list - not contain default plugins like async list",
rawInput: "plugin1",
expected: []InstallPlugin{
{ID: "plugin1", Version: "", URL: ""},
},
},
{
name: "it should remove the disabled plugin",
rawInput: "plugin1,plugin2",
disablePlugins: "plugin1",
expected: []InstallPlugin{{ID: "plugin2"}},
},
{
name: "should not process at all when preinstall is disabled",
rawInput: "plugin1",
disablePreinstall: true,
expected: nil,
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
cfg := NewCfg()
sec, err := cfg.Raw.NewSection("plugins")
require.NoError(t, err)
_, err = sec.NewKey("preinstall_sync", tc.rawInput)
require.NoError(t, err)
if tc.disablePreinstall {
_, err = sec.NewKey("preinstall_disabled", "true")
require.NoError(t, err)
}
if tc.disablePlugins != "" {
_, err = sec.NewKey("disable_plugins", tc.disablePlugins)
require.NoError(t, err)
}
err = cfg.readPluginSettings(cfg.Raw)
require.NoError(t, err)
assert.ElementsMatch(t, cfg.PreinstallPluginsSync, tc.expected)
})
}
})
t.Run("when plugins.preinstall is defined", func(t *testing.T) {
defaultPreinstallPluginsList := make([]InstallPlugin, 0, len(defaultPreinstallPlugins))
defaultPreinstallPluginsIDs := []string{}
for _, p := range defaultPreinstallPlugins {
defaultPreinstallPluginsList = append(defaultPreinstallPluginsList, p)
defaultPreinstallPluginsIDs = append(defaultPreinstallPluginsIDs, p.ID)
}
tests := []struct {
name string
rawInput string
rawInputSync string
disablePreinstall bool
expected []InstallPlugin
expectedSync []InstallPlugin
disableAsync bool
disablePlugins string
}{
{
name: "should add the default preinstalled plugin",
rawInput: "",
expected: defaultPreinstallPluginsList,
},
{
name: "should add the default preinstalled plugin and the one defined",
rawInput: "plugin1",
expected: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "", ""}),
},
{
name: "should add the default preinstalled plugin and the one defined with version",
rawInput: "plugin1@1.0.0",
expected: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "1.0.0", ""}),
},
{
name: "it should remove the disabled plugin",
rawInput: "plugin1",
disablePlugins: "plugin1",
expected: defaultPreinstallPluginsList,
},
{
name: "it should remove default plugins",
rawInput: "",
disablePlugins: strings.Join(defaultPreinstallPluginsIDs, ","),
expected: nil,
},
{
name: "should ignore input when preinstall is disabled",
rawInput: "plugin1",
disablePreinstall: true,
expected: nil,
},
{
name: "should mark preinstall as sync",
rawInput: "plugin1",
disableAsync: true,
expected: nil,
expectedSync: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "", ""}),
},
{
name: "should parse a plugin with version and URL",
rawInput: "plugin1@1.0.1@https://example.com/plugin1.tar.gz",
expected: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "1.0.1", "https://example.com/plugin1.tar.gz"}),
},
{
name: "should parse a plugin with URL",
rawInput: "plugin1@@https://example.com/plugin1.tar.gz",
expected: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "", "https://example.com/plugin1.tar.gz"}),
},
{
name: "when preinstall_async is false, should add all plugins to preinstall_sync",
rawInput: "plugin1",
rawInputSync: "plugin2",
disableAsync: true,
expected: nil,
expectedSync: append(defaultPreinstallPluginsList, InstallPlugin{"plugin1", "", ""}, InstallPlugin{"plugin2", "", ""}),
},
{
name: "should overwrite default when user pins a version",
rawInput: "grafana-pyroscope-app@4.0.0",
expected: func() []InstallPlugin {
var plugins []InstallPlugin
for _, p := range defaultPreinstallPlugins {
if p.ID == "grafana-pyroscope-app" {
plugins = append(plugins, InstallPlugin{"grafana-pyroscope-app", "4.0.0", ""})
} else {
plugins = append(plugins, p)
}
}
return plugins
}(),
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
cfg := NewCfg()
sec, err := cfg.Raw.NewSection("plugins")
require.NoError(t, err)
_, err = sec.NewKey("preinstall", tc.rawInput)
require.NoError(t, err)
if tc.rawInputSync != "" {
_, err = sec.NewKey("preinstall_sync", tc.rawInputSync)
require.NoError(t, err)
}
if tc.disablePreinstall {
_, err = sec.NewKey("preinstall_disabled", "true")
require.NoError(t, err)
}
if tc.disableAsync {
_, err = sec.NewKey("preinstall_async", "false")
require.NoError(t, err)
}
if tc.disablePlugins != "" {
_, err = sec.NewKey("disable_plugins", tc.disablePlugins)
require.NoError(t, err)
}
err = cfg.readPluginSettings(cfg.Raw)
require.NoError(t, err)
assert.ElementsMatch(t, cfg.PreinstallPluginsAsync, tc.expected)
if tc.expectedSync != nil {
assert.ElementsMatch(t, cfg.PreinstallPluginsSync, tc.expectedSync)
}
})
}
})
}