mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
Plugins: Extend panel menu with commands from plugins (#63802)
* feat(plugins): introduce dashboard panel menu placement for adding menu items
* test: add test for getPanelMenu()
* added an unique identifier for each extension.
* added context to getPluginExtensions.
* wip
* Wip
* wiwip
* Wip
* feat: WWWIIIIPPPP 🧨
* Wip
* Renamed some of the types to align a bit better.
* added limit to how many extensions a plugin can register per placement.
* decreased number of items to 2
* will trim the lenght of titles to max 25 chars.
* wrapping configure function with error handling.
* added error handling for all scenarios.
* moved extension menu items to the bottom of the more sub menu.
* added tests for configuring the title.
* minor refactorings.
* changed so you need to specify the full path in package.json.
* wip
* removed unused type.
* big refactor to make things simpler and to centralize all configure error/validation handling.
* added missing import.
* fixed failing tests.
* fixed tests.
* revert(extensions): remove static extensions config in favour of registering via AppPlugin APIs
* removed the compose that didn't work for some reason.
* added tests just to verify that validation and error handling is tied together in configuration function.
* adding some more values to the context.
* draft validation.
* added missing tests for getPanelMenu.
* added more tests.
* refactor(extensions): move logic for validating extension link config to function
* Fixed ts errors.
* Started to add structure for supporting commands.
* fixed tests.
* adding commands to the registry
* tests: group test cases in describe blocks
* tests: add a little bit more refactoring to the tests
* tests: add a test case for checking correct placements
* feat: first version of the command handler
* feat: register panel menu items with commands
* refactor: make the 'configure' function not optional on `PluginExtensionRegistryItem`
* Wip
* Wip
* Wip
* added test to verify the default configure function.
* added some more tests to verify that commands have the proper error handling for its configure function.
* tests: fix TS errors in tests
* tests: add auxiliary functions
* refactor: small refactoring in tests
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* refactor: refactoring tests for registryFactory
* draft of wrapping command handler with error handling.
* refactor: refactoring tests for registryFactory
* added test for edge case.
* replaced the registry item with a configure function.
* renamed the configure function type.
* refactoring of the registryfactory.
* added tests for handler error handling.
* fixed issue with assert function.
* added comment about the limited type.
* Update public/app/features/plugins/extensions/errorHandling.test.ts
Co-authored-by: Levente Balogh <balogh.levente.hu@gmail.com>
* Update public/app/features/plugins/extensions/errorHandling.test.ts
Co-authored-by: Levente Balogh <balogh.levente.hu@gmail.com>
* Update public/app/features/plugins/extensions/errorHandling.test.ts
Co-authored-by: Levente Balogh <balogh.levente.hu@gmail.com>
* added missing tests.
---------
Co-authored-by: Jack Westbrook <jack.westbrook@gmail.com>
Co-authored-by: Levente Balogh <balogh.levente.hu@gmail.com>
This commit is contained in:
parent
7aca818aae
commit
b63c56903d
@ -334,7 +334,8 @@ exports[`better eslint`] = {
|
||||
],
|
||||
"packages/grafana-data/src/types/app.ts:5381": [
|
||||
[0, 0, 0, "Unexpected any. Specify a different type.", "0"],
|
||||
[0, 0, 0, "Do not use any type assertions.", "1"]
|
||||
[0, 0, 0, "Do not use any type assertions.", "1"],
|
||||
[0, 0, 0, "Do not use any type assertions.", "2"]
|
||||
],
|
||||
"packages/grafana-data/src/types/config.ts:5381": [
|
||||
[0, 0, 0, "Unexpected any. Specify a different type.", "0"]
|
||||
|
@ -3,7 +3,7 @@ import { ComponentType } from 'react';
|
||||
import { KeyValue } from './data';
|
||||
import { NavModel } from './navModel';
|
||||
import { PluginMeta, GrafanaPlugin, PluginIncludeType } from './plugin';
|
||||
import { extensionLinkConfigIsValid, PluginExtensionLink } from './pluginExtensions';
|
||||
import { extensionLinkConfigIsValid, type PluginExtensionCommand, type PluginExtensionLink } from './pluginExtensions';
|
||||
|
||||
/**
|
||||
* @public
|
||||
@ -51,23 +51,32 @@ export interface AppPluginMeta<T extends KeyValue = KeyValue> extends PluginMeta
|
||||
}
|
||||
|
||||
/**
|
||||
* These types are towards the plugin developer when extending Grafana or other
|
||||
* plugins from the module.ts
|
||||
* The `configure()` function can only update certain properties of the extension, and due to this
|
||||
* it only receives a subset of the original extension object.
|
||||
*/
|
||||
export type AppConfigureExtension<T, C = object> = (extension: T, context: C) => Partial<T> | undefined;
|
||||
|
||||
export type AppPluginExtensionLink = Pick<PluginExtensionLink, 'description' | 'path' | 'title'>;
|
||||
|
||||
export type AppPluginExtensionCommand = Pick<PluginExtensionCommand, 'description' | 'title'>;
|
||||
|
||||
export type AppPluginExtensionLinkConfig<C extends object = object> = {
|
||||
title: string;
|
||||
description: string;
|
||||
placement: string;
|
||||
path: string;
|
||||
configure?: AppConfigureExtension<AppPluginExtensionLink, C>;
|
||||
configure?: (extension: AppPluginExtensionLink, context?: C) => Partial<AppPluginExtensionLink> | undefined;
|
||||
};
|
||||
|
||||
export type AppPluginExtensionCommandConfig<C extends object = object> = {
|
||||
title: string;
|
||||
description: string;
|
||||
placement: string;
|
||||
handler: (context?: C) => void;
|
||||
configure?: (extension: AppPluginExtensionCommand, context?: C) => Partial<AppPluginExtensionCommand> | undefined;
|
||||
};
|
||||
|
||||
export class AppPlugin<T extends KeyValue = KeyValue> extends GrafanaPlugin<AppPluginMeta<T>> {
|
||||
private linkExtensions: AppPluginExtensionLinkConfig[] = [];
|
||||
private commandExtensions: AppPluginExtensionCommandConfig[] = [];
|
||||
|
||||
// Content under: /a/${plugin-id}/*
|
||||
root?: ComponentType<AppRootProps<T>>;
|
||||
@ -113,6 +122,10 @@ export class AppPlugin<T extends KeyValue = KeyValue> extends GrafanaPlugin<AppP
|
||||
return this.linkExtensions;
|
||||
}
|
||||
|
||||
get extensionCommands(): AppPluginExtensionCommandConfig[] {
|
||||
return this.commandExtensions;
|
||||
}
|
||||
|
||||
configureExtensionLink<C extends object>(config: AppPluginExtensionLinkConfig<C>) {
|
||||
const { path, description, title, placement } = config;
|
||||
|
||||
@ -124,6 +137,11 @@ export class AppPlugin<T extends KeyValue = KeyValue> extends GrafanaPlugin<AppP
|
||||
this.linkExtensions.push(config as AppPluginExtensionLinkConfig);
|
||||
return this;
|
||||
}
|
||||
|
||||
configureExtensionCommand<C extends object>(config: AppPluginExtensionCommandConfig<C>) {
|
||||
this.commandExtensions.push(config as AppPluginExtensionCommandConfig);
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -56,5 +56,9 @@ export {
|
||||
type PluginExtension,
|
||||
type PluginExtensionLink,
|
||||
isPluginExtensionLink,
|
||||
assertPluginExtensionLink,
|
||||
type PluginExtensionCommand,
|
||||
isPluginExtensionCommand,
|
||||
assertPluginExtensionCommand,
|
||||
PluginExtensionTypes,
|
||||
} from './pluginExtensions';
|
||||
|
@ -4,6 +4,7 @@
|
||||
|
||||
export enum PluginExtensionTypes {
|
||||
link = 'link',
|
||||
command = 'command',
|
||||
}
|
||||
|
||||
export type PluginExtension = {
|
||||
@ -18,10 +19,41 @@ export type PluginExtensionLink = PluginExtension & {
|
||||
path: string;
|
||||
};
|
||||
|
||||
export function isPluginExtensionLink(extension: PluginExtension): extension is PluginExtensionLink {
|
||||
export type PluginExtensionCommand = PluginExtension & {
|
||||
type: PluginExtensionTypes.command;
|
||||
callHandlerWithContext: () => void;
|
||||
};
|
||||
|
||||
export function isPluginExtensionLink(extension: PluginExtension | undefined): extension is PluginExtensionLink {
|
||||
if (!extension) {
|
||||
return false;
|
||||
}
|
||||
return extension.type === PluginExtensionTypes.link && 'path' in extension;
|
||||
}
|
||||
|
||||
export function assertPluginExtensionLink(
|
||||
extension: PluginExtension | undefined
|
||||
): asserts extension is PluginExtensionLink {
|
||||
if (!isPluginExtensionLink(extension)) {
|
||||
throw new Error(`extension is not a link extension`);
|
||||
}
|
||||
}
|
||||
|
||||
export function isPluginExtensionCommand(extension: PluginExtension | undefined): extension is PluginExtensionCommand {
|
||||
if (!extension) {
|
||||
return false;
|
||||
}
|
||||
return extension.type === PluginExtensionTypes.command;
|
||||
}
|
||||
|
||||
export function assertPluginExtensionCommand(
|
||||
extension: PluginExtension | undefined
|
||||
): asserts extension is PluginExtensionCommand {
|
||||
if (!isPluginExtensionCommand(extension)) {
|
||||
throw new Error(`extension is not a command extension`);
|
||||
}
|
||||
}
|
||||
|
||||
export function extensionLinkConfigIsValid(props: {
|
||||
path?: string;
|
||||
description?: string;
|
||||
|
@ -11,7 +11,6 @@ export * from './appEvents';
|
||||
export {
|
||||
type PluginExtensionRegistry,
|
||||
type PluginExtensionRegistryItem,
|
||||
type RegistryConfigureExtension,
|
||||
setPluginsExtensionRegistry,
|
||||
} from './pluginExtensions/registry';
|
||||
export {
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { isPluginExtensionLink, PluginExtension, PluginExtensionLink, PluginExtensionTypes } from '@grafana/data';
|
||||
import { assertPluginExtensionLink, PluginExtensionLink, PluginExtensionTypes } from '@grafana/data';
|
||||
|
||||
import { getPluginExtensions } from './extensions';
|
||||
import { PluginExtensionRegistryItem, setPluginsExtensionRegistry } from './registry';
|
||||
@ -33,7 +33,7 @@ describe('getPluginExtensions', () => {
|
||||
const { extensions } = getPluginExtensions({ placement });
|
||||
const [extension] = extensions;
|
||||
|
||||
assertLinkExtension(extension);
|
||||
assertPluginExtensionLink(extension);
|
||||
|
||||
expect(extension.path).toBe(`/a/${pluginId}/declare-incident`);
|
||||
expect(extensions.length).toBe(1);
|
||||
@ -43,7 +43,7 @@ describe('getPluginExtensions', () => {
|
||||
const { extensions } = getPluginExtensions({ placement });
|
||||
const [extension] = extensions;
|
||||
|
||||
assertLinkExtension(extension);
|
||||
assertPluginExtensionLink(extension);
|
||||
|
||||
expect(extension.description).toBe('Declaring an incident in the app');
|
||||
expect(extensions.length).toBe(1);
|
||||
@ -53,13 +53,13 @@ describe('getPluginExtensions', () => {
|
||||
const { extensions } = getPluginExtensions({ placement });
|
||||
const [extension] = extensions;
|
||||
|
||||
assertLinkExtension(extension);
|
||||
assertPluginExtensionLink(extension);
|
||||
|
||||
expect(extension.title).toBe('Declare incident');
|
||||
expect(extensions.length).toBe(1);
|
||||
});
|
||||
|
||||
it('should return an empty array when extensions cannot be found', () => {
|
||||
it('should return an empty array when extensions can be found', () => {
|
||||
const { extensions } = getPluginExtensions({
|
||||
placement: 'plugins/not-installed-app/news',
|
||||
});
|
||||
@ -72,17 +72,8 @@ describe('getPluginExtensions', () => {
|
||||
function createRegistryLinkItem(
|
||||
link: Omit<PluginExtensionLink, 'type'>
|
||||
): PluginExtensionRegistryItem<PluginExtensionLink> {
|
||||
return {
|
||||
configure: undefined,
|
||||
extension: {
|
||||
...link,
|
||||
type: PluginExtensionTypes.link,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
function assertLinkExtension(extension: PluginExtension): asserts extension is PluginExtensionLink {
|
||||
if (!isPluginExtensionLink(extension)) {
|
||||
throw new Error(`extension is not a link extension`);
|
||||
}
|
||||
return (context?: object) => ({
|
||||
...link,
|
||||
type: PluginExtensionTypes.link,
|
||||
});
|
||||
}
|
||||
|
@ -16,15 +16,10 @@ export function getPluginExtensions<T extends object = {}>(
|
||||
): PluginExtensionsResult {
|
||||
const { placement, context } = options;
|
||||
const registry = getPluginsExtensionRegistry();
|
||||
const items = registry[placement] ?? [];
|
||||
const configureFuncs = registry[placement] ?? [];
|
||||
|
||||
const extensions = items.reduce<PluginExtension[]>((result, item) => {
|
||||
if (!context || !item.configure) {
|
||||
result.push(item.extension);
|
||||
return result;
|
||||
}
|
||||
|
||||
const extension = item.configure(context);
|
||||
const extensions = configureFuncs.reduce<PluginExtension[]>((result, configure) => {
|
||||
const extension = configure(context);
|
||||
if (extension) {
|
||||
result.push(extension);
|
||||
}
|
||||
|
@ -1,14 +1,9 @@
|
||||
import { PluginExtension } from '@grafana/data';
|
||||
|
||||
export type RegistryConfigureExtension<T extends PluginExtension = PluginExtension, C extends object = object> = (
|
||||
context: C
|
||||
export type PluginExtensionRegistryItem<T extends PluginExtension = PluginExtension, C extends object = object> = (
|
||||
context?: C
|
||||
) => T | undefined;
|
||||
|
||||
export type PluginExtensionRegistryItem<T extends PluginExtension = PluginExtension, C extends object = object> = {
|
||||
extension: T;
|
||||
configure?: RegistryConfigureExtension<T, C>;
|
||||
};
|
||||
|
||||
export type PluginExtensionRegistry = Record<string, PluginExtensionRegistryItem[]>;
|
||||
|
||||
let registry: PluginExtensionRegistry | undefined;
|
||||
|
@ -2,7 +2,6 @@ import { PanelMenuItem, PluginExtension, PluginExtensionLink, PluginExtensionTyp
|
||||
import {
|
||||
PluginExtensionPanelContext,
|
||||
PluginExtensionRegistryItem,
|
||||
RegistryConfigureExtension,
|
||||
setPluginsExtensionRegistry,
|
||||
} from '@grafana/runtime';
|
||||
import { LoadingState } from '@grafana/schema';
|
||||
@ -194,7 +193,7 @@ describe('getPanelMenu()', () => {
|
||||
});
|
||||
|
||||
it('should use extension for panel menu returned by configure function', () => {
|
||||
const configure = () => ({
|
||||
const configure: PluginExtensionRegistryItem<PluginExtensionLink> = () => ({
|
||||
title: 'Wohoo',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'Declaring an incident in the app',
|
||||
@ -334,7 +333,7 @@ describe('getPanelMenu()', () => {
|
||||
});
|
||||
|
||||
it('should pass context that can not be edited in configure function', () => {
|
||||
const configure = (context: PluginExtensionPanelContext) => {
|
||||
const configure: PluginExtensionRegistryItem<PluginExtensionLink> = (context) => {
|
||||
// trying to change values in the context
|
||||
// @ts-ignore
|
||||
context.pluginId = 'changed';
|
||||
@ -507,18 +506,10 @@ describe('getPanelMenu()', () => {
|
||||
});
|
||||
});
|
||||
|
||||
function createRegistryItem<T extends PluginExtension>(
|
||||
function createRegistryItem<T extends PluginExtension, C extends object = object>(
|
||||
extension: T,
|
||||
configure?: (context: PluginExtensionPanelContext) => T | undefined
|
||||
): PluginExtensionRegistryItem<T> {
|
||||
if (!configure) {
|
||||
return {
|
||||
extension,
|
||||
};
|
||||
}
|
||||
|
||||
return {
|
||||
extension,
|
||||
configure: configure as RegistryConfigureExtension<T>,
|
||||
};
|
||||
configure?: PluginExtensionRegistryItem<T, C>
|
||||
): PluginExtensionRegistryItem<T, C> {
|
||||
const defaultConfigure = () => extension;
|
||||
return configure || defaultConfigure;
|
||||
}
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { isPluginExtensionLink, PanelMenuItem } from '@grafana/data';
|
||||
import { isPluginExtensionCommand, isPluginExtensionLink, PanelMenuItem } from '@grafana/data';
|
||||
import {
|
||||
AngularComponent,
|
||||
getDataSourceSrv,
|
||||
@ -297,6 +297,15 @@ export function getPanelMenu(
|
||||
text: truncateTitle(extension.title, 25),
|
||||
href: extension.path,
|
||||
});
|
||||
continue;
|
||||
}
|
||||
|
||||
if (isPluginExtensionCommand(extension)) {
|
||||
subMenu.push({
|
||||
text: truncateTitle(extension.title, 25),
|
||||
onClick: extension.callHandlerWithContext,
|
||||
});
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,79 +1,122 @@
|
||||
import { AppConfigureExtension, AppPluginExtensionLink } from '@grafana/data';
|
||||
import { AppPluginExtensionLink } from '@grafana/data';
|
||||
|
||||
import { createErrorHandling } from './errorHandling';
|
||||
import { handleErrorsInConfigure, handleErrorsInHandler } from './errorHandling';
|
||||
import type { CommandHandlerFunc, ConfigureFunc } from './types';
|
||||
|
||||
describe('extension error handling', () => {
|
||||
const pluginId = 'grafana-basic-app';
|
||||
const errorHandler = createErrorHandling<AppPluginExtensionLink>({
|
||||
pluginId: pluginId,
|
||||
title: 'Go to page one',
|
||||
logger: jest.fn(),
|
||||
});
|
||||
describe('error handling for extensions', () => {
|
||||
describe('error handling for configure', () => {
|
||||
const pluginId = 'grafana-basic-app';
|
||||
const errorHandler = handleErrorsInConfigure<AppPluginExtensionLink>({
|
||||
pluginId: pluginId,
|
||||
title: 'Go to page one',
|
||||
logger: jest.fn(),
|
||||
});
|
||||
|
||||
const context = {};
|
||||
const extension: AppPluginExtensionLink = {
|
||||
title: 'Go to page one',
|
||||
description: 'Will navigate the user to page one',
|
||||
path: `/a/${pluginId}/one`,
|
||||
};
|
||||
const context = {};
|
||||
const extension: AppPluginExtensionLink = {
|
||||
title: 'Go to page one',
|
||||
description: 'Will navigate the user to page one',
|
||||
path: `/a/${pluginId}/one`,
|
||||
};
|
||||
|
||||
it('should return configured link if configure is successful', () => {
|
||||
const configureWithErrorHandling = errorHandler(() => {
|
||||
return {
|
||||
it('should return configured link if configure is successful', () => {
|
||||
const configureWithErrorHandling = errorHandler(() => {
|
||||
return {
|
||||
title: 'This is a new title',
|
||||
};
|
||||
});
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toEqual({
|
||||
title: 'This is a new title',
|
||||
};
|
||||
});
|
||||
});
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
it('should return undefined if configure throws error', () => {
|
||||
const configureWithErrorHandling = errorHandler(() => {
|
||||
throw new Error();
|
||||
});
|
||||
|
||||
expect(configured).toEqual({
|
||||
title: 'This is a new title',
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should return undefined if configure is promise/async-based', () => {
|
||||
const promisebased = (async () => {}) as ConfigureFunc<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(promisebased);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should return undefined if configure is not a function', () => {
|
||||
const objectbased = {} as ConfigureFunc<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(objectbased);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should return undefined if configure returns other than an object', () => {
|
||||
const returnString = (() => '') as ConfigureFunc<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(returnString);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should return undefined if configure returns undefined', () => {
|
||||
const returnUndefined = () => undefined;
|
||||
const configureWithErrorHandling = errorHandler(returnUndefined);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
});
|
||||
|
||||
it('should return undefined if configure throws error', () => {
|
||||
const configureWithErrorHandling = errorHandler(() => {
|
||||
throw new Error();
|
||||
describe('error handling for command handler', () => {
|
||||
const pluginId = 'grafana-basic-app';
|
||||
const errorHandler = handleErrorsInHandler({
|
||||
pluginId: pluginId,
|
||||
title: 'open modal',
|
||||
logger: jest.fn(),
|
||||
});
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
it('should be called successfully when handler is a normal synchronous function', () => {
|
||||
const handler = jest.fn();
|
||||
const handlerWithErrorHandling = errorHandler(handler);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
handlerWithErrorHandling();
|
||||
|
||||
it('should return undefined if configure is promise/async-based', () => {
|
||||
const promisebased = (async () => {}) as AppConfigureExtension<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(promisebased);
|
||||
expect(handler).toBeCalled();
|
||||
});
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
it('should not error out even if the handler throws an error', () => {
|
||||
const handlerWithErrorHandling = errorHandler(() => {
|
||||
throw new Error();
|
||||
});
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
expect(handlerWithErrorHandling).not.toThrowError();
|
||||
});
|
||||
|
||||
it('should return undefined if configure is not a function', () => {
|
||||
const objectbased = {} as AppConfigureExtension<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(objectbased);
|
||||
it('should be called successfully when handler is an async function / promise', () => {
|
||||
const promisebased = (async () => {}) as CommandHandlerFunc;
|
||||
const configureWithErrorHandling = errorHandler(promisebased);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
expect(configureWithErrorHandling).not.toThrowError();
|
||||
});
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
it('should be called successfully when handler is not a function', () => {
|
||||
const objectbased = {} as CommandHandlerFunc;
|
||||
const configureWithErrorHandling = errorHandler(objectbased);
|
||||
|
||||
it('should return undefined if configure returns other than an object', () => {
|
||||
const returnString = (() => '') as AppConfigureExtension<AppPluginExtensionLink>;
|
||||
const configureWithErrorHandling = errorHandler(returnString);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should return undefined if configure returns undefined', () => {
|
||||
const returnUndefined = () => undefined;
|
||||
const configureWithErrorHandling = errorHandler(returnUndefined);
|
||||
|
||||
const configured = configureWithErrorHandling(extension, context);
|
||||
|
||||
expect(configured).toBeUndefined();
|
||||
expect(configureWithErrorHandling).not.toThrowError();
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@ -1,6 +1,6 @@
|
||||
import { isFunction, isObject } from 'lodash';
|
||||
|
||||
import type { AppConfigureExtension } from '@grafana/data';
|
||||
import type { CommandHandlerFunc, ConfigureFunc } from './types';
|
||||
|
||||
type Options = {
|
||||
pluginId: string;
|
||||
@ -8,10 +8,10 @@ type Options = {
|
||||
logger: (msg: string, error?: unknown) => void;
|
||||
};
|
||||
|
||||
export function createErrorHandling<T>(options: Options) {
|
||||
export function handleErrorsInConfigure<T>(options: Options) {
|
||||
const { pluginId, title, logger } = options;
|
||||
|
||||
return (configure: AppConfigureExtension<T>): AppConfigureExtension<T> => {
|
||||
return (configure: ConfigureFunc<T>): ConfigureFunc<T> => {
|
||||
return function handleErrors(extension, context) {
|
||||
try {
|
||||
if (!isFunction(configure)) {
|
||||
@ -41,3 +41,32 @@ export function createErrorHandling<T>(options: Options) {
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
export function handleErrorsInHandler(options: Options) {
|
||||
const { pluginId, title, logger } = options;
|
||||
|
||||
return (handler: CommandHandlerFunc): CommandHandlerFunc => {
|
||||
return function handleErrors(context) {
|
||||
try {
|
||||
if (!isFunction(handler)) {
|
||||
logger(`[Plugins] ${pluginId} provided invalid handler function for command extension '${title}'.`);
|
||||
return;
|
||||
}
|
||||
|
||||
const result = handler(context);
|
||||
if (result instanceof Promise) {
|
||||
logger(
|
||||
`[Plugins] ${pluginId} provided an unsupported async/promise-based handler function for command extension '${title}'.`
|
||||
);
|
||||
result.catch(() => {});
|
||||
return;
|
||||
}
|
||||
|
||||
return result;
|
||||
} catch (error) {
|
||||
logger(`[Plugins] ${pluginId} thow an error while handling command extension '${title}'`, error);
|
||||
return;
|
||||
}
|
||||
};
|
||||
};
|
||||
}
|
||||
|
@ -0,0 +1,15 @@
|
||||
export class PlacementsPerPlugin {
|
||||
private counter: Record<string, number> = {};
|
||||
private limit = 2;
|
||||
|
||||
allowedToAdd(placement: string): boolean {
|
||||
const count = this.counter[placement] ?? 0;
|
||||
|
||||
if (count >= this.limit) {
|
||||
return false;
|
||||
}
|
||||
|
||||
this.counter[placement] = count + 1;
|
||||
return true;
|
||||
}
|
||||
}
|
@ -1,15 +1,27 @@
|
||||
import { PluginExtensionTypes } from '@grafana/data';
|
||||
import {
|
||||
AppPluginExtensionCommandConfig,
|
||||
AppPluginExtensionLinkConfig,
|
||||
assertPluginExtensionCommand,
|
||||
PluginExtensionTypes,
|
||||
} from '@grafana/data';
|
||||
import { PluginExtensionRegistry } from '@grafana/runtime';
|
||||
|
||||
import { createPluginExtensionRegistry } from './registryFactory';
|
||||
|
||||
const validateLink = jest.fn((configure, extension, context) => configure?.(extension, context));
|
||||
const errorHandler = jest.fn((configure, extension, context) => configure?.(extension, context));
|
||||
const configureErrorHandler = jest.fn((configure, extension, context) => configure?.(extension, context));
|
||||
const commandErrorHandler = jest.fn((configure, context) => configure?.(context));
|
||||
|
||||
jest.mock('./errorHandling', () => ({
|
||||
...jest.requireActual('./errorHandling'),
|
||||
createErrorHandling: jest.fn(() => {
|
||||
handleErrorsInConfigure: jest.fn(() => {
|
||||
return jest.fn((configure) => {
|
||||
return jest.fn((extension, context) => errorHandler(configure, extension, context));
|
||||
return jest.fn((extension, context) => configureErrorHandler(configure, extension, context));
|
||||
});
|
||||
}),
|
||||
handleErrorsInHandler: jest.fn(() => {
|
||||
return jest.fn((configure) => {
|
||||
return jest.fn((context) => commandErrorHandler(configure, context));
|
||||
});
|
||||
}),
|
||||
}));
|
||||
@ -23,304 +35,489 @@ jest.mock('./validateLink', () => ({
|
||||
}),
|
||||
}));
|
||||
|
||||
describe('Creating extensions registry', () => {
|
||||
describe('createPluginExtensionRegistry()', () => {
|
||||
beforeEach(() => {
|
||||
validateLink.mockClear();
|
||||
errorHandler.mockClear();
|
||||
configureErrorHandler.mockClear();
|
||||
commandErrorHandler.mockClear();
|
||||
});
|
||||
|
||||
it('should register an extension', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const numberOfPlacements = Object.keys(registry).length;
|
||||
const extensions = registry['grafana/dashboard/panel/menu'];
|
||||
|
||||
expect(numberOfPlacements).toBe(1);
|
||||
expect(extensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -68154691,
|
||||
},
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it('should register extensions from one plugin with multiple placements', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
{
|
||||
placement: 'plugins/grafana-slo-app/slo-breached',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const numberOfPlacements = Object.keys(registry).length;
|
||||
const panelExtensions = registry['grafana/dashboard/panel/menu'];
|
||||
const sloExtensions = registry['plugins/grafana-slo-app/slo-breached'];
|
||||
|
||||
expect(numberOfPlacements).toBe(2);
|
||||
expect(panelExtensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -68154691,
|
||||
},
|
||||
},
|
||||
]);
|
||||
expect(sloExtensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -1638987831,
|
||||
},
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it('should register extensions from multiple plugins with multiple placements', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
{
|
||||
placement: 'plugins/grafana-slo-app/slo-breached',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
pluginId: 'grafana-monitoring-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open Incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/grafana-monitoring-app/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const numberOfPlacements = Object.keys(registry).length;
|
||||
const panelExtensions = registry['grafana/dashboard/panel/menu'];
|
||||
const sloExtensions = registry['plugins/grafana-slo-app/slo-breached'];
|
||||
|
||||
expect(numberOfPlacements).toBe(2);
|
||||
expect(panelExtensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -68154691,
|
||||
},
|
||||
},
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open Incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/grafana-monitoring-app/incidents/declare',
|
||||
key: -540306829,
|
||||
},
|
||||
},
|
||||
]);
|
||||
|
||||
expect(sloExtensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -1638987831,
|
||||
},
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it('should register maximum 2 extensions per plugin and placement', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident 2',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident 3',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const numberOfPlacements = Object.keys(registry).length;
|
||||
const panelExtensions = registry['grafana/dashboard/panel/menu'];
|
||||
|
||||
expect(numberOfPlacements).toBe(1);
|
||||
expect(panelExtensions).toEqual([
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -68154691,
|
||||
},
|
||||
},
|
||||
{
|
||||
configure: undefined,
|
||||
extension: {
|
||||
title: 'Open incident 2',
|
||||
type: PluginExtensionTypes.link,
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
key: -1072147569,
|
||||
},
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it('should not register extensions with invalid path configured', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/incidents/declare',
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const numberOfPlacements = Object.keys(registry).length;
|
||||
expect(numberOfPlacements).toBe(0);
|
||||
});
|
||||
|
||||
it('should wrap configure function with link extension validator', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const extensions = registry['grafana/dashboard/panel/menu'];
|
||||
const [extension] = extensions;
|
||||
|
||||
const context = {};
|
||||
const configurable = {
|
||||
describe('when registering links', () => {
|
||||
const placement1 = 'grafana/dashboard/panel/menu';
|
||||
const placement2 = 'plugins/grafana-slo-app/slo-breached';
|
||||
const pluginId = 'belugacdn-app';
|
||||
// Sample link configurations that can be used in tests
|
||||
const linkConfig = {
|
||||
placement: placement1,
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
};
|
||||
|
||||
extension?.configure?.(context);
|
||||
it('should register a link extension', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [linkConfig],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
expect(validateLink).toBeCalledWith(expect.any(Function), configurable, context);
|
||||
shouldHaveExtensionsAtPlacement({ configs: [linkConfig], placement: placement1, registry });
|
||||
});
|
||||
|
||||
it('should only register a link extension to a single placement', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [linkConfig],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 1);
|
||||
expect(registry[placement1]).toBeDefined();
|
||||
});
|
||||
|
||||
it('should register link extensions from one plugin with multiple placements', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{ ...linkConfig, placement: placement1 },
|
||||
{ ...linkConfig, placement: placement2 },
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 2);
|
||||
shouldHaveExtensionsAtPlacement({ placement: placement1, configs: [linkConfig], registry });
|
||||
shouldHaveExtensionsAtPlacement({ placement: placement2, configs: [linkConfig], registry });
|
||||
});
|
||||
|
||||
it('should register link extensions from multiple plugins with multiple placements', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{ ...linkConfig, placement: placement1 },
|
||||
{ ...linkConfig, placement: placement2 },
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
{
|
||||
pluginId: 'grafana-monitoring-app',
|
||||
linkExtensions: [
|
||||
{ ...linkConfig, placement: placement1, path: '/a/grafana-monitoring-app/incidents/declare' },
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 2);
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: placement1,
|
||||
configs: [linkConfig, { ...linkConfig, path: '/a/grafana-monitoring-app/incidents/declare' }],
|
||||
registry,
|
||||
});
|
||||
shouldHaveExtensionsAtPlacement({ placement: placement2, configs: [linkConfig], registry });
|
||||
});
|
||||
|
||||
it('should register maximum 2 extensions per plugin and placement', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{ ...linkConfig, title: 'Link 1' },
|
||||
{ ...linkConfig, title: 'Link 2' },
|
||||
{ ...linkConfig, title: 'Link 3' },
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 1);
|
||||
|
||||
// The 3rd link is being ignored
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: linkConfig.placement,
|
||||
configs: [
|
||||
{ ...linkConfig, title: 'Link 1' },
|
||||
{ ...linkConfig, title: 'Link 2' },
|
||||
],
|
||||
registry,
|
||||
});
|
||||
});
|
||||
|
||||
it('should not register link extensions with invalid path configured', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{
|
||||
...linkConfig,
|
||||
path: '/incidents/declare', // invalid path, should always be prefixed with the plugin id
|
||||
},
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 0);
|
||||
});
|
||||
|
||||
it('should add default configure function when none provided via extension config', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [linkConfig],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[linkConfig.placement];
|
||||
const configured = configure();
|
||||
|
||||
// The default configure() function returns the same extension config
|
||||
expect(configured).toEqual({
|
||||
key: expect.any(Number),
|
||||
type: PluginExtensionTypes.link,
|
||||
title: linkConfig.title,
|
||||
description: linkConfig.description,
|
||||
path: linkConfig.path,
|
||||
});
|
||||
});
|
||||
|
||||
it('should wrap the configure function with link extension validator', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{
|
||||
...linkConfig,
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[linkConfig.placement];
|
||||
const context = {};
|
||||
const configurable = {
|
||||
title: linkConfig.title,
|
||||
description: linkConfig.description,
|
||||
path: linkConfig.path,
|
||||
};
|
||||
|
||||
configure(context);
|
||||
|
||||
expect(validateLink).toBeCalledWith(expect.any(Function), configurable, context);
|
||||
});
|
||||
|
||||
it('should wrap configure function with extension error handling', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{
|
||||
...linkConfig,
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[linkConfig.placement];
|
||||
const context = {};
|
||||
const configurable = {
|
||||
title: linkConfig.title,
|
||||
description: linkConfig.description,
|
||||
path: linkConfig.path,
|
||||
};
|
||||
|
||||
configure(context);
|
||||
|
||||
expect(configureErrorHandler).toBeCalledWith(expect.any(Function), configurable, context);
|
||||
});
|
||||
|
||||
it('should return undefined if returned by the provided extension config', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [
|
||||
{
|
||||
...linkConfig,
|
||||
configure: () => undefined,
|
||||
},
|
||||
],
|
||||
commandExtensions: [],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[linkConfig.placement];
|
||||
const context = {};
|
||||
|
||||
expect(configure(context)).toBeUndefined();
|
||||
});
|
||||
});
|
||||
|
||||
it('should wrap configure function with extension error handling', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId: 'belugacdn-app',
|
||||
linkExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const extensions = registry['grafana/dashboard/panel/menu'];
|
||||
const [extension] = extensions;
|
||||
|
||||
const context = {};
|
||||
const configurable = {
|
||||
// Command extensions
|
||||
// ------------------
|
||||
describe('when registering commands', () => {
|
||||
const pluginId = 'belugacdn-app';
|
||||
// Sample command configurations to be used in tests
|
||||
const commandConfig1 = {
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
path: '/a/belugacdn-app/incidents/declare',
|
||||
handler: () => {},
|
||||
};
|
||||
const commandConfig2 = {
|
||||
placement: 'plugins/grafana-slo-app/slo-breached',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
handler: () => {},
|
||||
};
|
||||
|
||||
extension?.configure?.(context);
|
||||
it('should register a command extension', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [commandConfig1],
|
||||
},
|
||||
]);
|
||||
|
||||
expect(errorHandler).toBeCalledWith(expect.any(Function), configurable, context);
|
||||
shouldHaveNumberOfPlacements(registry, 1);
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: commandConfig1.placement,
|
||||
configs: [commandConfig1],
|
||||
registry,
|
||||
});
|
||||
});
|
||||
|
||||
it('should register command extensions from a SINGLE PLUGIN with MULTIPLE PLACEMENTS', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [commandConfig1, commandConfig2],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 2);
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: commandConfig1.placement,
|
||||
configs: [commandConfig1],
|
||||
registry,
|
||||
});
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: commandConfig2.placement,
|
||||
configs: [commandConfig2],
|
||||
registry,
|
||||
});
|
||||
});
|
||||
|
||||
it('should register command extensions from MULTIPLE PLUGINS with MULTIPLE PLACEMENTS', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [commandConfig1, commandConfig2],
|
||||
},
|
||||
{
|
||||
pluginId: 'grafana-monitoring-app',
|
||||
linkExtensions: [],
|
||||
commandExtensions: [commandConfig1],
|
||||
},
|
||||
]);
|
||||
|
||||
shouldHaveNumberOfPlacements(registry, 2);
|
||||
|
||||
// Both plugins register commands to the same placement
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: commandConfig1.placement,
|
||||
configs: [commandConfig1, commandConfig1],
|
||||
registry,
|
||||
});
|
||||
|
||||
// The 'beluga-cdn-app' plugin registers a command to an other placement as well
|
||||
shouldHaveExtensionsAtPlacement({
|
||||
placement: commandConfig2.placement,
|
||||
configs: [commandConfig2],
|
||||
registry,
|
||||
});
|
||||
});
|
||||
|
||||
it('should add default configure function when none is provided via the extension config', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [commandConfig1],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[commandConfig1.placement];
|
||||
const configured = configure();
|
||||
|
||||
// The default configure() function returns the extension config as is
|
||||
expect(configured).toEqual({
|
||||
type: PluginExtensionTypes.command,
|
||||
key: expect.any(Number),
|
||||
title: commandConfig1.title,
|
||||
description: commandConfig1.description,
|
||||
callHandlerWithContext: expect.any(Function),
|
||||
});
|
||||
});
|
||||
|
||||
it('should wrap the configure function with error handling', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [
|
||||
{
|
||||
...commandConfig1,
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[commandConfig1.placement];
|
||||
const context = {};
|
||||
const configurable = {
|
||||
title: commandConfig1.title,
|
||||
description: commandConfig2.description,
|
||||
};
|
||||
|
||||
configure(context);
|
||||
|
||||
// The error handler is wrapping (decorating) the configure function, so it can provide standard error messages
|
||||
expect(configureErrorHandler).toBeCalledWith(expect.any(Function), configurable, context);
|
||||
});
|
||||
|
||||
it('should return undefined if returned by the provided extension config', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [
|
||||
{
|
||||
...commandConfig1,
|
||||
configure: () => undefined,
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const [configure] = registry[commandConfig1.placement];
|
||||
const context = {};
|
||||
|
||||
expect(configure(context)).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should wrap handler function with extension error handling', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
handler: () => {},
|
||||
configure: () => ({}),
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const extensions = registry['grafana/dashboard/panel/menu'];
|
||||
const [configure] = extensions;
|
||||
const context = {};
|
||||
const extension = configure?.(context);
|
||||
|
||||
assertPluginExtensionCommand(extension);
|
||||
|
||||
extension.callHandlerWithContext();
|
||||
|
||||
expect(commandErrorHandler).toBeCalledWith(expect.any(Function), context);
|
||||
});
|
||||
|
||||
it('should wrap handler function with extension error handling when no configure function is added', () => {
|
||||
const registry = createPluginExtensionRegistry([
|
||||
{
|
||||
pluginId,
|
||||
linkExtensions: [],
|
||||
commandExtensions: [
|
||||
{
|
||||
placement: 'grafana/dashboard/panel/menu',
|
||||
title: 'Open incident',
|
||||
description: 'You can create an incident from this context',
|
||||
handler: () => {},
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
|
||||
const extensions = registry['grafana/dashboard/panel/menu'];
|
||||
const [configure] = extensions;
|
||||
const context = {};
|
||||
const extension = configure?.(context);
|
||||
|
||||
assertPluginExtensionCommand(extension);
|
||||
|
||||
extension.callHandlerWithContext();
|
||||
|
||||
expect(commandErrorHandler).toBeCalledWith(expect.any(Function), context);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
// Checks the number of total placements in the registry
|
||||
function shouldHaveNumberOfPlacements(registry: PluginExtensionRegistry, numberOfPlacements: number) {
|
||||
expect(Object.keys(registry).length).toBe(numberOfPlacements);
|
||||
}
|
||||
|
||||
// Checks if the registry has exactly the same extensions at the expected placement
|
||||
function shouldHaveExtensionsAtPlacement({
|
||||
configs,
|
||||
placement,
|
||||
registry,
|
||||
}: {
|
||||
configs: Array<AppPluginExtensionLinkConfig | AppPluginExtensionCommandConfig>;
|
||||
placement: string;
|
||||
registry: PluginExtensionRegistry;
|
||||
}) {
|
||||
const extensions = registry[placement].map((configure) => configure());
|
||||
|
||||
expect(extensions).toEqual(
|
||||
configs.map((extension) => {
|
||||
// Command extension
|
||||
if ('handler' in extension) {
|
||||
return {
|
||||
key: expect.any(Number),
|
||||
title: extension.title,
|
||||
description: extension.description,
|
||||
type: PluginExtensionTypes.command,
|
||||
callHandlerWithContext: expect.any(Function),
|
||||
};
|
||||
}
|
||||
|
||||
// Link extension
|
||||
return {
|
||||
key: expect.any(Number),
|
||||
title: extension.title,
|
||||
description: extension.description,
|
||||
type: PluginExtensionTypes.link,
|
||||
path: extension.path,
|
||||
};
|
||||
})
|
||||
);
|
||||
}
|
||||
|
@ -1,41 +1,40 @@
|
||||
import {
|
||||
AppConfigureExtension,
|
||||
AppPluginExtensionLink,
|
||||
AppPluginExtensionLinkConfig,
|
||||
PluginExtensionLink,
|
||||
type AppPluginExtensionCommand,
|
||||
type AppPluginExtensionCommandConfig,
|
||||
type AppPluginExtensionLink,
|
||||
type AppPluginExtensionLinkConfig,
|
||||
type PluginExtension,
|
||||
type PluginExtensionCommand,
|
||||
type PluginExtensionLink,
|
||||
PluginExtensionTypes,
|
||||
} from '@grafana/data';
|
||||
import type {
|
||||
PluginExtensionRegistry,
|
||||
PluginExtensionRegistryItem,
|
||||
RegistryConfigureExtension,
|
||||
} from '@grafana/runtime';
|
||||
import type { PluginExtensionRegistry, PluginExtensionRegistryItem } from '@grafana/runtime';
|
||||
|
||||
import { PluginPreloadResult } from '../pluginPreloader';
|
||||
import type { PluginPreloadResult } from '../pluginPreloader';
|
||||
|
||||
import { createErrorHandling } from './errorHandling';
|
||||
import { handleErrorsInHandler, handleErrorsInConfigure } from './errorHandling';
|
||||
import { PlacementsPerPlugin } from './placementsPerPlugin';
|
||||
import { ConfigureFunc } from './types';
|
||||
import { createLinkValidator, isValidLinkPath } from './validateLink';
|
||||
|
||||
export function createPluginExtensionRegistry(preloadResults: PluginPreloadResult[]): PluginExtensionRegistry {
|
||||
const registry: PluginExtensionRegistry = {};
|
||||
|
||||
for (const result of preloadResults) {
|
||||
const pluginPlacementCount: Record<string, number> = {};
|
||||
const { pluginId, linkExtensions, error } = result;
|
||||
const { pluginId, linkExtensions, commandExtensions, error } = result;
|
||||
|
||||
if (!Array.isArray(linkExtensions) || error) {
|
||||
if (error) {
|
||||
continue;
|
||||
}
|
||||
|
||||
for (const extension of linkExtensions) {
|
||||
const placement = extension.placement;
|
||||
const placementsPerPlugin = new PlacementsPerPlugin();
|
||||
const configs = [...linkExtensions, ...commandExtensions];
|
||||
|
||||
pluginPlacementCount[placement] = (pluginPlacementCount[placement] ?? 0) + 1;
|
||||
const item = createRegistryLink(pluginId, extension);
|
||||
for (const config of configs) {
|
||||
const placement = config.placement;
|
||||
const item = createRegistryItem(pluginId, config);
|
||||
|
||||
// If there was an issue initialising the plugin, skip adding its extensions to the registry
|
||||
// or if the plugin already have placed 2 items at the extension point.
|
||||
if (!item || pluginPlacementCount[placement] > 2) {
|
||||
if (!item || !placementsPerPlugin.allowedToAdd(placement)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -55,41 +54,21 @@ export function createPluginExtensionRegistry(preloadResults: PluginPreloadResul
|
||||
return Object.freeze(registry);
|
||||
}
|
||||
|
||||
function createRegistryLink(
|
||||
function createRegistryItem(
|
||||
pluginId: string,
|
||||
config: AppPluginExtensionLinkConfig
|
||||
): PluginExtensionRegistryItem<PluginExtensionLink> | undefined {
|
||||
if (!isValidLinkPath(pluginId, config.path)) {
|
||||
return undefined;
|
||||
config: AppPluginExtensionCommandConfig | AppPluginExtensionLinkConfig
|
||||
): PluginExtensionRegistryItem | undefined {
|
||||
if ('handler' in config) {
|
||||
return createCommandRegistryItem(pluginId, config);
|
||||
}
|
||||
|
||||
const id = `${pluginId}${config.placement}${config.title}`;
|
||||
const extension = Object.freeze({
|
||||
type: PluginExtensionTypes.link,
|
||||
title: config.title,
|
||||
description: config.description,
|
||||
key: hashKey(id),
|
||||
path: config.path,
|
||||
});
|
||||
|
||||
return Object.freeze({
|
||||
extension: extension,
|
||||
configure: createLinkConfigure(pluginId, config, extension),
|
||||
});
|
||||
return createLinkRegistryItem(pluginId, config);
|
||||
}
|
||||
|
||||
function hashKey(key: string): number {
|
||||
return Array.from(key).reduce((s, c) => (Math.imul(31, s) + c.charCodeAt(0)) | 0, 0);
|
||||
}
|
||||
|
||||
function createLinkConfigure(
|
||||
function createCommandRegistryItem(
|
||||
pluginId: string,
|
||||
config: AppPluginExtensionLinkConfig,
|
||||
extension: PluginExtensionLink
|
||||
): RegistryConfigureExtension<PluginExtensionLink> | undefined {
|
||||
if (!config.configure) {
|
||||
return undefined;
|
||||
}
|
||||
config: AppPluginExtensionCommandConfig
|
||||
): PluginExtensionRegistryItem<PluginExtensionCommand> | undefined {
|
||||
const configure = config.configure ?? defaultConfigure;
|
||||
|
||||
const options = {
|
||||
pluginId: pluginId,
|
||||
@ -97,36 +76,102 @@ function createLinkConfigure(
|
||||
logger: console.warn,
|
||||
};
|
||||
|
||||
const mapper = mapToRegistryType(extension);
|
||||
const validator = createLinkValidator(options);
|
||||
const errorHandler = createErrorHandling<AppPluginExtensionLink>(options);
|
||||
const catchErrorsInHandler = handleErrorsInHandler(options);
|
||||
const handler = catchErrorsInHandler(config.handler);
|
||||
|
||||
return mapper(validator(errorHandler(config.configure)));
|
||||
}
|
||||
const extensionFactory = createCommandFactory(pluginId, config, handler);
|
||||
|
||||
function mapToRegistryType(
|
||||
extension: PluginExtensionLink
|
||||
): (configure: AppConfigureExtension<AppPluginExtensionLink>) => RegistryConfigureExtension<PluginExtensionLink> {
|
||||
const configurable: AppPluginExtensionLink = {
|
||||
title: extension.title,
|
||||
description: extension.description,
|
||||
path: extension.path,
|
||||
const configurable: AppPluginExtensionCommand = {
|
||||
title: config.title,
|
||||
description: config.description,
|
||||
};
|
||||
|
||||
return (configure) => {
|
||||
return function mapper(context: object): PluginExtensionLink | undefined {
|
||||
const configured = configure(configurable, context);
|
||||
const mapper = mapToConfigure<PluginExtensionCommand, AppPluginExtensionCommand>(extensionFactory, configurable);
|
||||
const catchErrorsInConfigure = handleErrorsInConfigure<AppPluginExtensionCommand>(options);
|
||||
|
||||
if (!configured) {
|
||||
return mapper(catchErrorsInConfigure(configure));
|
||||
}
|
||||
|
||||
function createLinkRegistryItem(
|
||||
pluginId: string,
|
||||
config: AppPluginExtensionLinkConfig
|
||||
): PluginExtensionRegistryItem<PluginExtensionLink> | undefined {
|
||||
if (!isValidLinkPath(pluginId, config.path)) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
const configure = config.configure ?? defaultConfigure;
|
||||
const options = { pluginId: pluginId, title: config.title, logger: console.warn };
|
||||
|
||||
const extensionFactory = createLinkFactory(pluginId, config);
|
||||
|
||||
const configurable: AppPluginExtensionLink = {
|
||||
title: config.title,
|
||||
description: config.description,
|
||||
path: config.path,
|
||||
};
|
||||
|
||||
const mapper = mapToConfigure<PluginExtensionLink, AppPluginExtensionLink>(extensionFactory, configurable);
|
||||
const withConfigureErrorHandling = handleErrorsInConfigure<AppPluginExtensionLink>(options);
|
||||
const validateLink = createLinkValidator(options);
|
||||
|
||||
return mapper(validateLink(withConfigureErrorHandling(configure)));
|
||||
}
|
||||
|
||||
function createLinkFactory(pluginId: string, config: AppPluginExtensionLinkConfig) {
|
||||
return (override: Partial<AppPluginExtensionLink>, context?: object): PluginExtensionLink => {
|
||||
const title = override?.title ?? config.title;
|
||||
const description = override?.description ?? config.description;
|
||||
const path = override?.path ?? config.path;
|
||||
|
||||
return Object.freeze({
|
||||
type: PluginExtensionTypes.link,
|
||||
title: title,
|
||||
description: description,
|
||||
path: path,
|
||||
key: hashKey(`${pluginId}${config.placement}${title}`),
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
function createCommandFactory(
|
||||
pluginId: string,
|
||||
config: AppPluginExtensionCommandConfig,
|
||||
handler: (context?: object) => void
|
||||
) {
|
||||
return (override: Partial<AppPluginExtensionCommand>, context?: object): PluginExtensionCommand => {
|
||||
const title = override?.title ?? config.title;
|
||||
const description = override?.description ?? config.description;
|
||||
|
||||
return Object.freeze({
|
||||
type: PluginExtensionTypes.command,
|
||||
title: title,
|
||||
description: description,
|
||||
key: hashKey(`${pluginId}${config.placement}${title}`),
|
||||
callHandlerWithContext: () => handler(context),
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
function mapToConfigure<T extends PluginExtension, C>(
|
||||
commandFactory: (override: Partial<C>, context?: object) => T | undefined,
|
||||
configurable: C
|
||||
): (configure: ConfigureFunc<C>) => PluginExtensionRegistryItem<T> {
|
||||
return (configure) => {
|
||||
return function mapToExtension(context?: object): T | undefined {
|
||||
const override = configure(configurable, context);
|
||||
if (!override) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
return {
|
||||
...extension,
|
||||
title: configured.title ?? extension.title,
|
||||
description: configured.description ?? extension.description,
|
||||
path: configured.path ?? extension.path,
|
||||
};
|
||||
return commandFactory(override, context);
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
function hashKey(key: string): number {
|
||||
return Array.from(key).reduce((s, c) => (Math.imul(31, s) + c.charCodeAt(0)) | 0, 0);
|
||||
}
|
||||
|
||||
function defaultConfigure() {
|
||||
return {};
|
||||
}
|
||||
|
4
public/app/features/plugins/extensions/types.ts
Normal file
4
public/app/features/plugins/extensions/types.ts
Normal file
@ -0,0 +1,4 @@
|
||||
import type { AppPluginExtensionCommandConfig } from '@grafana/data';
|
||||
|
||||
export type CommandHandlerFunc = AppPluginExtensionCommandConfig['handler'];
|
||||
export type ConfigureFunc<T> = (extension: T, context?: object) => Partial<T> | undefined;
|
@ -1,4 +1,6 @@
|
||||
import type { AppConfigureExtension, AppPluginExtensionLink } from '@grafana/data';
|
||||
import type { AppPluginExtensionLink } from '@grafana/data';
|
||||
|
||||
import type { ConfigureFunc } from './types';
|
||||
|
||||
type Options = {
|
||||
pluginId: string;
|
||||
@ -9,7 +11,7 @@ type Options = {
|
||||
export function createLinkValidator(options: Options) {
|
||||
const { pluginId, title, logger } = options;
|
||||
|
||||
return (configure: AppConfigureExtension<AppPluginExtensionLink>): AppConfigureExtension<AppPluginExtensionLink> => {
|
||||
return (configure: ConfigureFunc<AppPluginExtensionLink>): ConfigureFunc<AppPluginExtensionLink> => {
|
||||
return function validateLink(link, context) {
|
||||
const configured = configure(link, context);
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { AppPluginExtensionLinkConfig } from '@grafana/data';
|
||||
import type { AppPluginExtensionCommandConfig, AppPluginExtensionLinkConfig } from '@grafana/data';
|
||||
import type { AppPluginConfig } from '@grafana/runtime';
|
||||
|
||||
import * as pluginLoader from './plugin_loader';
|
||||
@ -6,6 +6,7 @@ import * as pluginLoader from './plugin_loader';
|
||||
export type PluginPreloadResult = {
|
||||
pluginId: string;
|
||||
linkExtensions: AppPluginExtensionLinkConfig[];
|
||||
commandExtensions: AppPluginExtensionCommandConfig[];
|
||||
error?: unknown;
|
||||
};
|
||||
|
||||
@ -18,10 +19,10 @@ async function preload(config: AppPluginConfig): Promise<PluginPreloadResult> {
|
||||
const { path, version, id: pluginId } = config;
|
||||
try {
|
||||
const { plugin } = await pluginLoader.importPluginModule(path, version);
|
||||
const { linkExtensions = [] } = plugin;
|
||||
return { pluginId, linkExtensions };
|
||||
const { linkExtensions = [], commandExtensions = [] } = plugin;
|
||||
return { pluginId, linkExtensions, commandExtensions };
|
||||
} catch (error) {
|
||||
console.error(`[Plugins] Failed to preload plugin: ${path} (version: ${version})`, error);
|
||||
return { pluginId, linkExtensions: [], error };
|
||||
return { pluginId, linkExtensions: [], commandExtensions: [], error };
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user