2023-10-11 07:27:20 -05:00
|
|
|
import { ComponentType } from 'react';
|
|
|
|
|
|
|
|
import { PanelPlugin, PanelPluginMeta, PanelProps } from '@grafana/data';
|
2022-04-22 08:33:13 -05:00
|
|
|
import config from 'app/core/config';
|
|
|
|
|
2021-10-13 01:53:36 -05:00
|
|
|
import { getPanelPluginLoadError } from '../panel/components/PanelPluginError';
|
2022-04-22 08:33:13 -05:00
|
|
|
|
2021-10-13 01:53:36 -05:00
|
|
|
import { importPluginModule } from './plugin_loader';
|
|
|
|
|
2022-06-20 07:41:39 -05:00
|
|
|
const promiseCache: Record<string, Promise<PanelPlugin>> = {};
|
|
|
|
const panelPluginCache: Record<string, PanelPlugin> = {};
|
|
|
|
|
|
|
|
export function importPanelPlugin(id: string): Promise<PanelPlugin> {
|
|
|
|
const loaded = promiseCache[id];
|
2021-10-13 01:53:36 -05:00
|
|
|
if (loaded) {
|
|
|
|
return loaded;
|
|
|
|
}
|
|
|
|
|
2023-07-07 04:28:44 -05:00
|
|
|
const meta = getPanelPluginMeta(id);
|
2021-10-13 01:53:36 -05:00
|
|
|
|
|
|
|
if (!meta) {
|
|
|
|
throw new Error(`Plugin ${id} not found`);
|
|
|
|
}
|
|
|
|
|
2022-06-20 07:41:39 -05:00
|
|
|
promiseCache[id] = getPanelPlugin(meta);
|
2023-06-02 12:46:12 -05:00
|
|
|
if (id !== meta.type) {
|
|
|
|
promiseCache[meta.type] = promiseCache[id];
|
|
|
|
}
|
2021-10-13 01:53:36 -05:00
|
|
|
|
2022-06-20 07:41:39 -05:00
|
|
|
return promiseCache[id];
|
2021-10-13 01:53:36 -05:00
|
|
|
}
|
|
|
|
|
2023-07-07 04:28:44 -05:00
|
|
|
export function hasPanelPlugin(id: string): boolean {
|
|
|
|
return !!getPanelPluginMeta(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getPanelPluginMeta(id: string): PanelPluginMeta {
|
2023-09-29 10:20:37 -05:00
|
|
|
const v = config.panels[id];
|
|
|
|
if (!v) {
|
|
|
|
// Check alias values before failing
|
|
|
|
for (const p of Object.values(config.panels)) {
|
|
|
|
if (p.aliasIDs?.includes(id)) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return v;
|
2023-07-07 04:28:44 -05:00
|
|
|
}
|
|
|
|
|
2022-06-20 07:41:39 -05:00
|
|
|
export function importPanelPluginFromMeta(meta: PanelPluginMeta): Promise<PanelPlugin> {
|
2021-10-13 01:53:36 -05:00
|
|
|
return getPanelPlugin(meta);
|
|
|
|
}
|
|
|
|
|
2022-06-20 07:41:39 -05:00
|
|
|
export function syncGetPanelPlugin(id: string): PanelPlugin | undefined {
|
|
|
|
return panelPluginCache[id];
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPanelPlugin(meta: PanelPluginMeta): Promise<PanelPlugin> {
|
2023-06-05 03:51:36 -05:00
|
|
|
return importPluginModule({
|
|
|
|
path: meta.module,
|
|
|
|
version: meta.info?.version,
|
|
|
|
isAngular: meta.angularDetected,
|
|
|
|
pluginId: meta.id,
|
|
|
|
})
|
2022-02-18 05:51:55 -06:00
|
|
|
.then((pluginExports) => {
|
|
|
|
if (pluginExports.plugin) {
|
2023-09-22 09:06:49 -05:00
|
|
|
return pluginExports.plugin;
|
2022-02-18 05:51:55 -06:00
|
|
|
} else if (pluginExports.PanelCtrl) {
|
2022-06-20 07:41:39 -05:00
|
|
|
const plugin = new PanelPlugin(null);
|
2022-02-18 05:51:55 -06:00
|
|
|
plugin.angularPanelCtrl = pluginExports.PanelCtrl;
|
|
|
|
return plugin;
|
|
|
|
}
|
|
|
|
throw new Error('missing export: plugin or PanelCtrl');
|
|
|
|
})
|
2023-10-11 07:27:20 -05:00
|
|
|
.then((plugin: PanelPlugin) => {
|
2022-02-18 05:51:55 -06:00
|
|
|
plugin.meta = meta;
|
2022-06-20 07:41:39 -05:00
|
|
|
panelPluginCache[meta.id] = plugin;
|
2023-10-11 07:27:20 -05:00
|
|
|
|
|
|
|
if (!plugin.panel && plugin.angularPanelCtrl) {
|
|
|
|
plugin.panel = getAngularPanelReactWrapper(plugin);
|
|
|
|
}
|
|
|
|
|
2022-02-18 05:51:55 -06:00
|
|
|
return plugin;
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
// TODO, maybe a different error plugin
|
|
|
|
console.warn('Error loading panel plugin: ' + meta.id, err);
|
|
|
|
return getPanelPluginLoadError(meta, err);
|
|
|
|
});
|
2021-10-13 01:53:36 -05:00
|
|
|
}
|
2023-10-11 07:27:20 -05:00
|
|
|
|
|
|
|
let getAngularPanelReactWrapper = (plugin: PanelPlugin): ComponentType<PanelProps> | null => null;
|
|
|
|
|
|
|
|
export function setAngularPanelReactWrapper(wrapper: typeof getAngularPanelReactWrapper) {
|
|
|
|
getAngularPanelReactWrapper = wrapper;
|
|
|
|
}
|