mirror of https://github.com/grafana/grafana
Plugins: Introduce `LoadingStrategy` for frontend loading logic (#92392)
* do it all * feat(plugins): move loadingStrategy to ds pluginMeta and add to plugin settings endpoint * support child plugins and update tests * use relative path for nested plugins * feat(plugins): support nested plugins in the plugin loader cache by extracting pluginId from path * feat(grafana-data): add plugin loading strategy to plugin meta and export * feat(plugins): pass down loadingStrategy to fe plugin loader * refactor(plugins): make PluginLoadingStrategy an enum * feat(plugins): add the loading strategy to the fe plugin loader cache * feat(plugins): load fe plugin js assets as script tags based on be loadingStrategy * add more tests * feat(plugins): add loading strategy to plugin preloader * feat(plugins): make loadingStrategy a maybe and provide fetch fallback * test(alerting): update config.apps mocks to include loadingStrategy * fix format --------- Co-authored-by: Jack Westbrook <jack.westbrook@gmail.com>pull/93100/head
parent
d61530941a
commit
2c47d246fc
@ -0,0 +1,81 @@ |
|||||||
|
package pluginassets |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
|
||||||
|
"github.com/Masterminds/semver/v3" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/infra/log" |
||||||
|
"github.com/grafana/grafana/pkg/plugins" |
||||||
|
"github.com/grafana/grafana/pkg/plugins/pluginscdn" |
||||||
|
"github.com/grafana/grafana/pkg/services/pluginsintegration/pluginstore" |
||||||
|
"github.com/grafana/grafana/pkg/setting" |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
CreatePluginVersionCfgKey = "create_plugin_version" |
||||||
|
CreatePluginVersionScriptSupportEnabled = "4.15.0" |
||||||
|
) |
||||||
|
|
||||||
|
var ( |
||||||
|
scriptLoadingMinSupportedVersion = semver.MustParse(CreatePluginVersionScriptSupportEnabled) |
||||||
|
) |
||||||
|
|
||||||
|
func ProvideService(cfg *setting.Cfg, cdn *pluginscdn.Service) *Service { |
||||||
|
return &Service{ |
||||||
|
cfg: cfg, |
||||||
|
cdn: cdn, |
||||||
|
log: log.New("pluginassets"), |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
type Service struct { |
||||||
|
cfg *setting.Cfg |
||||||
|
cdn *pluginscdn.Service |
||||||
|
log log.Logger |
||||||
|
} |
||||||
|
|
||||||
|
// LoadingStrategy calculates the loading strategy for a plugin.
|
||||||
|
// If a plugin has plugin setting `create_plugin_version` >= 4.15.0, set loadingStrategy to "script".
|
||||||
|
// If a plugin is not loaded via the CDN and is not Angular, set loadingStrategy to "script".
|
||||||
|
// Otherwise, set loadingStrategy to "fetch".
|
||||||
|
func (s *Service) LoadingStrategy(_ context.Context, p pluginstore.Plugin) plugins.LoadingStrategy { |
||||||
|
if pCfg, ok := s.cfg.PluginSettings[p.ID]; ok { |
||||||
|
if s.compatibleCreatePluginVersion(pCfg) { |
||||||
|
return plugins.LoadingStrategyScript |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// If the plugin has a parent, check the parent's create_plugin_version setting
|
||||||
|
if p.Parent != nil { |
||||||
|
if pCfg, ok := s.cfg.PluginSettings[p.Parent.ID]; ok { |
||||||
|
if s.compatibleCreatePluginVersion(pCfg) { |
||||||
|
return plugins.LoadingStrategyScript |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if !s.cndEnabled(p) && !p.Angular.Detected { |
||||||
|
return plugins.LoadingStrategyScript |
||||||
|
} |
||||||
|
|
||||||
|
return plugins.LoadingStrategyFetch |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) compatibleCreatePluginVersion(ps map[string]string) bool { |
||||||
|
if cpv, ok := ps[CreatePluginVersionCfgKey]; ok { |
||||||
|
createPluginVer, err := semver.NewVersion(cpv) |
||||||
|
if err != nil { |
||||||
|
s.log.Warn("Failed to parse create plugin version setting as semver", "version", cpv, "error", err) |
||||||
|
} else { |
||||||
|
if !createPluginVer.LessThan(scriptLoadingMinSupportedVersion) { |
||||||
|
return true |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return false |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) cndEnabled(p pluginstore.Plugin) bool { |
||||||
|
return s.cdn.PluginSupported(p.ID) || p.Class == plugins.ClassCDN |
||||||
|
} |
@ -0,0 +1,168 @@ |
|||||||
|
package pluginassets |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
"testing" |
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/infra/log" |
||||||
|
"github.com/grafana/grafana/pkg/plugins" |
||||||
|
"github.com/grafana/grafana/pkg/plugins/config" |
||||||
|
"github.com/grafana/grafana/pkg/plugins/pluginscdn" |
||||||
|
"github.com/grafana/grafana/pkg/services/pluginsintegration/pluginstore" |
||||||
|
"github.com/grafana/grafana/pkg/setting" |
||||||
|
) |
||||||
|
|
||||||
|
func TestService_Calculate(t *testing.T) { |
||||||
|
const pluginID = "grafana-test-datasource" |
||||||
|
|
||||||
|
const ( |
||||||
|
incompatVersion = "4.14.0" |
||||||
|
compatVersion = CreatePluginVersionScriptSupportEnabled |
||||||
|
futureVersion = "5.0.0" |
||||||
|
) |
||||||
|
|
||||||
|
tcs := []struct { |
||||||
|
name string |
||||||
|
pluginSettings setting.PluginSettings |
||||||
|
plugin pluginstore.Plugin |
||||||
|
expected plugins.LoadingStrategy |
||||||
|
}{ |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when create-plugin version is compatible and plugin is not angular", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: compatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when parent create-plugin version is compatible and plugin is not angular", |
||||||
|
pluginSettings: newPluginSettings("parent-datasource", map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: compatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false, func(p pluginstore.Plugin) pluginstore.Plugin { |
||||||
|
p.Parent = &pluginstore.ParentPlugin{ID: "parent-datasource"} |
||||||
|
return p |
||||||
|
}), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when create-plugin version is future compatible and plugin is not angular", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: futureVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when create-plugin version is not provided, plugin is not angular and is not configured as CDN enabled", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
// NOTE: cdn key is not set
|
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when create-plugin version is not compatible, plugin is not angular, is not configured as CDN enabled and does not have the CDN class", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: incompatVersion, |
||||||
|
// NOTE: cdn key is not set
|
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false, func(p pluginstore.Plugin) pluginstore.Plugin { |
||||||
|
p.Class = plugins.ClassExternal |
||||||
|
return p |
||||||
|
}), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyFetch when create-plugin version is not compatible, plugin is not angular, is configured as CDN enabled and does not have the CDN class", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
"cdn": "true", |
||||||
|
CreatePluginVersionCfgKey: incompatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false, func(p pluginstore.Plugin) pluginstore.Plugin { |
||||||
|
p.Class = plugins.ClassExternal |
||||||
|
return p |
||||||
|
}), |
||||||
|
expected: plugins.LoadingStrategyFetch, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyFetch when create-plugin version is not compatible and plugin is angular", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: incompatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, true), |
||||||
|
expected: plugins.LoadingStrategyFetch, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyFetch when create-plugin version is not compatible, plugin is not angular and plugin is configured as CDN enabled", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
"cdn": "true", |
||||||
|
CreatePluginVersionCfgKey: incompatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false), |
||||||
|
expected: plugins.LoadingStrategyFetch, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyFetch when create-plugin version is not compatible, plugin is not angular and has the CDN class", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: incompatVersion, |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false, func(p pluginstore.Plugin) pluginstore.Plugin { |
||||||
|
p.Class = plugins.ClassCDN |
||||||
|
return p |
||||||
|
}), |
||||||
|
expected: plugins.LoadingStrategyFetch, |
||||||
|
}, |
||||||
|
{ |
||||||
|
name: "Expected LoadingStrategyScript when plugin setting create-plugin version is badly formatted, plugin is not configured as CDN enabled and does not have the CDN class", |
||||||
|
pluginSettings: newPluginSettings(pluginID, map[string]string{ |
||||||
|
CreatePluginVersionCfgKey: "invalidSemver", |
||||||
|
}), |
||||||
|
plugin: newPlugin(pluginID, false), |
||||||
|
expected: plugins.LoadingStrategyScript, |
||||||
|
}, |
||||||
|
} |
||||||
|
for _, tc := range tcs { |
||||||
|
t.Run(tc.name, func(t *testing.T) { |
||||||
|
s := &Service{ |
||||||
|
cfg: newCfg(tc.pluginSettings), |
||||||
|
cdn: pluginscdn.ProvideService(&config.PluginManagementCfg{ |
||||||
|
PluginsCDNURLTemplate: "http://cdn.example.com", // required for cdn.PluginSupported check
|
||||||
|
PluginSettings: tc.pluginSettings, |
||||||
|
}), |
||||||
|
log: log.NewNopLogger(), |
||||||
|
} |
||||||
|
|
||||||
|
got := s.LoadingStrategy(context.Background(), tc.plugin) |
||||||
|
assert.Equal(t, tc.expected, got, "unexpected loading strategy") |
||||||
|
}) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func newPlugin(pluginID string, angular bool, cbs ...func(p pluginstore.Plugin) pluginstore.Plugin) pluginstore.Plugin { |
||||||
|
p := pluginstore.Plugin{ |
||||||
|
JSONData: plugins.JSONData{ |
||||||
|
ID: pluginID, |
||||||
|
}, |
||||||
|
Angular: plugins.AngularMeta{Detected: angular}, |
||||||
|
} |
||||||
|
for _, cb := range cbs { |
||||||
|
p = cb(p) |
||||||
|
} |
||||||
|
return p |
||||||
|
} |
||||||
|
|
||||||
|
func newCfg(ps setting.PluginSettings) *setting.Cfg { |
||||||
|
return &setting.Cfg{ |
||||||
|
PluginSettings: ps, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func newPluginSettings(pluginID string, kv map[string]string) setting.PluginSettings { |
||||||
|
return setting.PluginSettings{ |
||||||
|
pluginID: kv, |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue