diff --git a/public/app/features/alerting/unified/components/rule-editor/AnnotationsField.test.tsx b/public/app/features/alerting/unified/components/rule-editor/AnnotationsField.test.tsx
index 5ca9a009cc221..c9b25fe59b5a9 100644
--- a/public/app/features/alerting/unified/components/rule-editor/AnnotationsField.test.tsx
+++ b/public/app/features/alerting/unified/components/rule-editor/AnnotationsField.test.tsx
@@ -296,37 +296,37 @@ describe('AnnotationsField', function () {
expect(annotationKeyElements[1]).toHaveTextContent('Panel ID');
expect(annotationValueElements[1]).toHaveTextContent('3');
});
+ });
+});
- it('should render warning icon for panels of type other than graph and timeseries', async function () {
- mockSearchApiResponse(server, [
- mockDashboardSearchItem({ title: 'My dashboard', uid: 'dash-test-uid', type: DashboardSearchItemType.DashDB }),
- ]);
-
- mockGetDashboardResponse(
- mockDashboardDto({
- title: 'My dashboard',
- uid: 'dash-test-uid',
- panels: [
- { id: 1, title: 'First panel', type: 'bar' },
- { id: 2, title: 'Second panel', type: 'graph' },
- ],
- })
- );
+it('should render warning icon for panels of type other than graph and timeseries', async function () {
+ mockSearchApiResponse(server, [
+ mockDashboardSearchItem({ title: 'My dashboard', uid: 'dash-test-uid', type: DashboardSearchItemType.DashDB }),
+ ]);
+
+ mockGetDashboardResponse(
+ mockDashboardDto({
+ title: 'My dashboard',
+ uid: 'dash-test-uid',
+ panels: [
+ { id: 1, title: 'First panel', type: 'bar' },
+ { id: 2, title: 'Second panel', type: 'graph' },
+ ],
+ })
+ );
- const user = userEvent.setup();
+ const user = userEvent.setup();
- render();
+ render();
- const { dialog } = ui.dashboardPicker;
+ const { dialog } = ui.dashboardPicker;
- await user.click(ui.setDashboardButton.get());
- await user.click(await findByTitle(dialog.get(), 'My dashboard'));
+ await user.click(ui.setDashboardButton.get());
+ await user.click(await findByTitle(dialog.get(), 'My dashboard'));
- const warnedPanel = await findByRole(dialog.get(), 'button', { name: /First panel/ });
+ const warnedPanel = await findByRole(dialog.get(), 'button', { name: /First panel/ });
- expect(getByTestId(warnedPanel, 'warning-icon')).toBeInTheDocument();
- });
- });
+ expect(getByTestId(warnedPanel, 'warning-icon')).toBeInTheDocument();
});
function mockGetDashboardResponse(dashboard: DashboardDTO) {
diff --git a/public/app/features/alerting/unified/utils/access-control.ts b/public/app/features/alerting/unified/utils/access-control.ts
index d71134a46598a..46ca371882ccb 100644
--- a/public/app/features/alerting/unified/utils/access-control.ts
+++ b/public/app/features/alerting/unified/utils/access-control.ts
@@ -1,7 +1,8 @@
+import { getConfig } from 'app/core/config';
import { contextSrv } from 'app/core/services/context_srv';
import { AccessControlAction } from 'app/types';
-import { isGrafanaRulesSource } from './datasource';
+import { GRAFANA_RULES_SOURCE_NAME, isGrafanaRulesSource } from './datasource';
type RulesSourceType = 'grafana' | 'external';
@@ -128,3 +129,12 @@ export function getRulesAccess() {
contextSrv.hasPermission(provisioningPermissions.readSecrets),
};
}
+
+export function getCreateAlertInMenuAvailability() {
+ const { unifiedAlertingEnabled } = getConfig();
+ const hasRuleReadPermissions = contextSrv.hasPermission(getRulesPermissions(GRAFANA_RULES_SOURCE_NAME).read);
+ const hasRuleUpdatePermissions = contextSrv.hasPermission(getRulesPermissions(GRAFANA_RULES_SOURCE_NAME).update);
+ const isAlertingAvailableForRead = unifiedAlertingEnabled && hasRuleReadPermissions;
+
+ return isAlertingAvailableForRead && hasRuleUpdatePermissions;
+}
diff --git a/public/app/features/dashboard/utils/getPanelMenu.test.ts b/public/app/features/dashboard/utils/getPanelMenu.test.ts
index 2afb76ce4d443..ad194684d10a4 100644
--- a/public/app/features/dashboard/utils/getPanelMenu.test.ts
+++ b/public/app/features/dashboard/utils/getPanelMenu.test.ts
@@ -12,8 +12,10 @@ import {
} from '@grafana/data';
import { AngularComponent, getPluginLinkExtensions } from '@grafana/runtime';
import config from 'app/core/config';
+import { grantUserPermissions } from 'app/features/alerting/unified/mocks';
import * as actions from 'app/features/explore/state/main';
import { setStore } from 'app/store/store';
+import { AccessControlAction } from 'app/types';
import { PanelModel } from '../state';
import { createDashboardModelFixture } from '../state/__fixtures__/dashboardFixtures';
@@ -23,6 +25,7 @@ import { getPanelMenu } from './getPanelMenu';
jest.mock('app/core/services/context_srv', () => ({
contextSrv: {
hasAccessToExplore: () => true,
+ hasPermission: jest.fn(),
},
}));
@@ -38,6 +41,8 @@ describe('getPanelMenu()', () => {
beforeEach(() => {
getPluginLinkExtensionsMock.mockRestore();
getPluginLinkExtensionsMock.mockReturnValue({ extensions: [] });
+ grantUserPermissions([AccessControlAction.AlertingRuleRead, AccessControlAction.AlertingRuleUpdate]);
+ config.unifiedAlertingEnabled = false;
});
it('should return the correct panel menu items', () => {
@@ -619,4 +624,57 @@ describe('getPanelMenu()', () => {
expect(windowOpen).toHaveBeenLastCalledWith(`${testSubUrl}${testUrl}`);
});
});
+ describe('Alerting menu', () => {
+ it('should render Create alert menu item if user has permissions to read and update alerts ', () => {
+ const panel = new PanelModel({});
+
+ const dashboard = createDashboardModelFixture({});
+ config.unifiedAlertingEnabled = true;
+ grantUserPermissions([AccessControlAction.AlertingRuleRead, AccessControlAction.AlertingRuleUpdate]);
+ const menuItems = getPanelMenu(dashboard, panel);
+ const moreSubMenu = menuItems.find((i) => i.text === 'More...')?.subMenu;
+
+ expect(moreSubMenu).toEqual(
+ expect.arrayContaining([
+ expect.objectContaining({
+ text: 'Create alert',
+ }),
+ ])
+ );
+ });
+
+ it('should not render Create alert menu item, if user does not have permissions to update alerts ', () => {
+ const panel = new PanelModel({});
+ const dashboard = createDashboardModelFixture({});
+
+ grantUserPermissions([AccessControlAction.AlertingRuleRead]);
+ config.unifiedAlertingEnabled = true;
+
+ const menuItems = getPanelMenu(dashboard, panel);
+
+ const moreSubMenu = menuItems.find((i) => i.text === 'More...')?.subMenu;
+
+ expect(moreSubMenu).toEqual(
+ expect.arrayContaining([
+ expect.not.objectContaining({
+ text: 'Create alert',
+ }),
+ ])
+ );
+ });
+ it('should not render Create alert menu item, if user does not have permissions to read update alerts ', () => {
+ const panel = new PanelModel({});
+
+ const dashboard = createDashboardModelFixture({});
+ grantUserPermissions([]);
+ config.unifiedAlertingEnabled = true;
+
+ const menuItems = getPanelMenu(dashboard, panel);
+
+ const moreSubMenu = menuItems.find((i) => i.text === 'More...')?.subMenu;
+ const createAlertOption = moreSubMenu?.find((i) => i.text === 'Create alert')?.subMenu;
+
+ expect(createAlertOption).toBeUndefined();
+ });
+ });
});
diff --git a/public/app/features/dashboard/utils/getPanelMenu.ts b/public/app/features/dashboard/utils/getPanelMenu.ts
index f5725cc1f3602..0894368211d6c 100644
--- a/public/app/features/dashboard/utils/getPanelMenu.ts
+++ b/public/app/features/dashboard/utils/getPanelMenu.ts
@@ -3,14 +3,16 @@ import {
PanelMenuItem,
PluginExtensionLink,
PluginExtensionPoints,
+ urlUtil,
type PluginExtensionPanelContext,
} from '@grafana/data';
-import { AngularComponent, locationService, reportInteraction, getPluginLinkExtensions } from '@grafana/runtime';
+import { AngularComponent, getPluginLinkExtensions, locationService, reportInteraction } from '@grafana/runtime';
import { PanelCtrl } from 'app/angular/panel/panel_ctrl';
import config from 'app/core/config';
import { t } from 'app/core/internationalization';
import { contextSrv } from 'app/core/services/context_srv';
import { getExploreUrl } from 'app/core/utils/explore';
+import { panelToRuleFormValues } from 'app/features/alerting/unified/utils/rule-form';
import { DashboardModel } from 'app/features/dashboard/state/DashboardModel';
import { PanelModel } from 'app/features/dashboard/state/PanelModel';
import {
@@ -28,6 +30,7 @@ import { truncateTitle } from 'app/features/plugins/extensions/utils';
import { SHARED_DASHBOARD_QUERY } from 'app/plugins/datasource/dashboard';
import { store } from 'app/store/store';
+import { getCreateAlertInMenuAvailability } from '../../alerting/unified/utils/access-control';
import { navigateToExplore } from '../../explore/state/main';
import { getTimeSrv } from '../services/TimeSrv';
@@ -202,8 +205,27 @@ export function getPanelMenu(
subMenu: inspectMenu,
});
+ const createAlert = async () => {
+ const formValues = await panelToRuleFormValues(panel, dashboard);
+
+ const ruleFormUrl = urlUtil.renderUrl('/alerting/new', {
+ defaults: JSON.stringify(formValues),
+ returnTo: location.pathname + location.search,
+ });
+
+ locationService.push(ruleFormUrl);
+ };
+
+ const onCreateAlert = (event: React.MouseEvent) => {
+ event.preventDefault();
+ createAlert();
+ reportInteraction('dashboards_panelheader_menu', { item: 'create-alert' });
+ };
+
const subMenu: PanelMenuItem[] = [];
const canEdit = dashboard.canEditPanel(panel);
+ const isCreateAlertMenuOptionAvailable = getCreateAlertInMenuAvailability();
+
if (!(panel.isViewing || panel.isEditing)) {
if (canEdit) {
subMenu.push({
@@ -237,6 +259,13 @@ export function getPanelMenu(
}
}
+ if (isCreateAlertMenuOptionAvailable) {
+ subMenu.push({
+ text: t('panel.header-menu.create-alert', `Create alert`),
+ onClick: onCreateAlert,
+ });
+ }
+
// add old angular panel options
if (angularComponent) {
const scope = angularComponent.getScope();
@@ -273,6 +302,12 @@ export function getPanelMenu(
// When editing hide most actions
if (panel.isEditing) {
subMenu.length = 0;
+ if (isCreateAlertMenuOptionAvailable) {
+ subMenu.push({
+ text: t('panel.header-menu.create-alert', `Create alert`),
+ onClick: onCreateAlert,
+ });
+ }
}
if (canEdit && panel.plugin && !panel.plugin.meta.skipDataQuery) {
diff --git a/public/locales/de-DE/grafana.json b/public/locales/de-DE/grafana.json
index fe5d7e7981df8..d2bd547ab4db6 100644
--- a/public/locales/de-DE/grafana.json
+++ b/public/locales/de-DE/grafana.json
@@ -773,6 +773,7 @@
"panel": {
"header-menu": {
"copy": "Kopieren",
+ "create-alert": "",
"create-library-panel": "Bibliotheksleiste erstellen",
"duplicate": "Duplikat",
"edit": "Bearbeiten",
diff --git a/public/locales/en-US/grafana.json b/public/locales/en-US/grafana.json
index 29c575e4c4ba0..d0369489aa91d 100644
--- a/public/locales/en-US/grafana.json
+++ b/public/locales/en-US/grafana.json
@@ -773,6 +773,7 @@
"panel": {
"header-menu": {
"copy": "Copy",
+ "create-alert": "Create alert",
"create-library-panel": "Create library panel",
"duplicate": "Duplicate",
"edit": "Edit",
diff --git a/public/locales/es-ES/grafana.json b/public/locales/es-ES/grafana.json
index ca9fd8b7530a0..d80df14b05f09 100644
--- a/public/locales/es-ES/grafana.json
+++ b/public/locales/es-ES/grafana.json
@@ -779,6 +779,7 @@
"panel": {
"header-menu": {
"copy": "Copiar",
+ "create-alert": "",
"create-library-panel": "Crear panel de librería",
"duplicate": "Duplicar",
"edit": "Editar",
diff --git a/public/locales/fr-FR/grafana.json b/public/locales/fr-FR/grafana.json
index e887ae1378b3a..f703114f1ff70 100644
--- a/public/locales/fr-FR/grafana.json
+++ b/public/locales/fr-FR/grafana.json
@@ -779,6 +779,7 @@
"panel": {
"header-menu": {
"copy": "Copier",
+ "create-alert": "",
"create-library-panel": "Créer un panneau Bibliothèque",
"duplicate": "Dupliquer",
"edit": "Modifier",
diff --git a/public/locales/pseudo-LOCALE/grafana.json b/public/locales/pseudo-LOCALE/grafana.json
index c1860822a0084..172d2a2111438 100644
--- a/public/locales/pseudo-LOCALE/grafana.json
+++ b/public/locales/pseudo-LOCALE/grafana.json
@@ -773,6 +773,7 @@
"panel": {
"header-menu": {
"copy": "Cőpy",
+ "create-alert": "Cřęäŧę äľęřŧ",
"create-library-panel": "Cřęäŧę ľįþřäřy päʼnęľ",
"duplicate": "Đūpľįčäŧę",
"edit": "Ēđįŧ",
diff --git a/public/locales/zh-Hans/grafana.json b/public/locales/zh-Hans/grafana.json
index 8f7810c189efa..9e5da4eb6d7dc 100644
--- a/public/locales/zh-Hans/grafana.json
+++ b/public/locales/zh-Hans/grafana.json
@@ -767,6 +767,7 @@
"panel": {
"header-menu": {
"copy": "复制",
+ "create-alert": "",
"create-library-panel": "创建库面板",
"duplicate": "复制",
"edit": "编辑",