228 lines
7.9 KiB
TypeScript
228 lines
7.9 KiB
TypeScript
import { PropsWithChildren } from 'react';
|
|
import { getWrapper, render, renderHook, screen, waitFor } from 'test/test-utils';
|
|
|
|
import { setupMswServer } from 'app/features/alerting/unified/mockApi';
|
|
import { setFolderAccessControl } from 'app/features/alerting/unified/mocks/server/configure';
|
|
import { MIMIR_DATASOURCE_UID } from 'app/features/alerting/unified/mocks/server/constants';
|
|
import { AlertManagerDataSourceJsonData, AlertManagerImplementation } from 'app/plugins/datasource/alertmanager/types';
|
|
import { AccessControlAction } from 'app/types';
|
|
import { CombinedRule } from 'app/types/unified-alerting';
|
|
|
|
import { getCloudRule, getGrafanaRule, grantUserPermissions, mockDataSource } from '../mocks';
|
|
import { AlertmanagerProvider } from '../state/AlertmanagerContext';
|
|
import { setupDataSources } from '../testSetup/datasources';
|
|
import { DataSourceType, GRAFANA_RULES_SOURCE_NAME } from '../utils/datasource';
|
|
|
|
import {
|
|
AlertRuleAction,
|
|
AlertmanagerAction,
|
|
useAlertmanagerAbilities,
|
|
useAlertmanagerAbility,
|
|
useAllAlertRuleAbilities,
|
|
useAllAlertmanagerAbilities,
|
|
} from './useAbilities';
|
|
|
|
/**
|
|
* This test will write snapshots with a record of the current permissions assigned to actions.
|
|
* We encourage that every change to the snapshot is inspected _very_ thoroughly!
|
|
*/
|
|
describe('alertmanager abilities', () => {
|
|
it("should report Create / Update / Delete actions aren't supported for external vanilla alertmanager", () => {
|
|
setupDataSources(
|
|
mockDataSource<AlertManagerDataSourceJsonData>({
|
|
name: GRAFANA_RULES_SOURCE_NAME,
|
|
type: DataSourceType.Alertmanager,
|
|
jsonData: { implementation: AlertManagerImplementation.prometheus },
|
|
})
|
|
);
|
|
|
|
const { result } = renderHook(() => useAllAlertmanagerAbilities(), {
|
|
wrapper: createAlertmanagerWrapper('does-not-exist'),
|
|
});
|
|
expect(result.current).toMatchSnapshot();
|
|
});
|
|
|
|
it('should report everything is supported for builtin alertmanager', () => {
|
|
setupDataSources(
|
|
mockDataSource<AlertManagerDataSourceJsonData>({
|
|
name: GRAFANA_RULES_SOURCE_NAME,
|
|
type: DataSourceType.Alertmanager,
|
|
})
|
|
);
|
|
|
|
grantUserPermissions([AccessControlAction.AlertingNotificationsRead, AccessControlAction.AlertingInstanceRead]);
|
|
|
|
const { result } = renderHook(() => useAllAlertmanagerAbilities(), {
|
|
wrapper: createAlertmanagerWrapper(GRAFANA_RULES_SOURCE_NAME),
|
|
});
|
|
|
|
Object.values(result.current).forEach(([supported]) => {
|
|
expect(supported).toBe(true);
|
|
});
|
|
|
|
// since we only granted "read" permissions, only those should be allowed
|
|
const { result: viewResult } = renderHook(() => useAlertmanagerAbility(AlertmanagerAction.ViewSilence), {
|
|
wrapper: createAlertmanagerWrapper(GRAFANA_RULES_SOURCE_NAME),
|
|
});
|
|
|
|
const [viewSupported, viewAllowed] = viewResult.current;
|
|
|
|
expect(viewSupported).toBe(true);
|
|
expect(viewAllowed).toBe(true);
|
|
|
|
// editing should not be allowed, but supported
|
|
const { result: editResult } = renderHook(() => useAlertmanagerAbility(AlertmanagerAction.ViewSilence), {
|
|
wrapper: createAlertmanagerWrapper(GRAFANA_RULES_SOURCE_NAME),
|
|
});
|
|
|
|
const [editSupported, editAllowed] = editResult.current;
|
|
|
|
expect(editSupported).toBe(true);
|
|
expect(editAllowed).toBe(true);
|
|
|
|
// record the snapshot to prevent future regressions
|
|
expect(result.current).toMatchSnapshot();
|
|
});
|
|
|
|
it('should report everything except exporting for Mimir alertmanager', () => {
|
|
setupDataSources(
|
|
mockDataSource<AlertManagerDataSourceJsonData>({
|
|
name: MIMIR_DATASOURCE_UID,
|
|
type: DataSourceType.Alertmanager,
|
|
jsonData: {
|
|
implementation: AlertManagerImplementation.mimir,
|
|
},
|
|
})
|
|
);
|
|
|
|
grantUserPermissions([
|
|
AccessControlAction.AlertingNotificationsExternalRead,
|
|
AccessControlAction.AlertingNotificationsExternalWrite,
|
|
AccessControlAction.AlertingInstancesExternalRead,
|
|
AccessControlAction.AlertingInstancesExternalWrite,
|
|
]);
|
|
|
|
const { result } = renderHook(() => useAllAlertmanagerAbilities(), {
|
|
wrapper: createAlertmanagerWrapper('mimir'),
|
|
});
|
|
|
|
expect(result.current).toMatchSnapshot();
|
|
});
|
|
|
|
it('should be able to return multiple abilities', () => {
|
|
setupDataSources(
|
|
mockDataSource<AlertManagerDataSourceJsonData>({
|
|
name: GRAFANA_RULES_SOURCE_NAME,
|
|
type: DataSourceType.Alertmanager,
|
|
})
|
|
);
|
|
|
|
grantUserPermissions([AccessControlAction.AlertingNotificationsRead]);
|
|
|
|
const { result } = renderHook(
|
|
() =>
|
|
useAlertmanagerAbilities([
|
|
AlertmanagerAction.ViewContactPoint,
|
|
AlertmanagerAction.CreateContactPoint,
|
|
AlertmanagerAction.ExportContactPoint,
|
|
]),
|
|
{
|
|
wrapper: createAlertmanagerWrapper(GRAFANA_RULES_SOURCE_NAME),
|
|
}
|
|
);
|
|
|
|
expect(result.current).toHaveLength(3);
|
|
expect(result.current[0]).toStrictEqual([true, true]);
|
|
expect(result.current[1]).toStrictEqual([true, false]);
|
|
expect(result.current[2]).toStrictEqual([true, true]);
|
|
});
|
|
});
|
|
|
|
setupMswServer();
|
|
|
|
const wrapper = () => getWrapper({ renderWithRouter: true });
|
|
|
|
/**
|
|
* Render the hook result in a component so we can more reliably check that the result has settled
|
|
* after API requests. Without this approach, the hook might return `[false, false]` whilst
|
|
* API requests are still loading
|
|
*/
|
|
const RenderActionPermissions = ({ rule, action }: { rule: CombinedRule; action: AlertRuleAction }) => {
|
|
const result = useAllAlertRuleAbilities(rule);
|
|
const [isSupported, isAllowed] = result[action];
|
|
return (
|
|
<>
|
|
{isSupported && 'supported'}
|
|
{isAllowed && 'allowed'}
|
|
</>
|
|
);
|
|
};
|
|
|
|
describe('AlertRule abilities', () => {
|
|
it('should report that all actions are supported for a Grafana Managed alert rule', async () => {
|
|
const rule = getGrafanaRule();
|
|
|
|
const { result } = renderHook(() => useAllAlertRuleAbilities(rule), { wrapper: wrapper() });
|
|
|
|
await waitFor(() => {
|
|
const results = Object.values(result.current);
|
|
|
|
for (const [supported, _allowed] of results) {
|
|
expect(supported).toBe(true);
|
|
}
|
|
});
|
|
|
|
expect(result.current).toMatchSnapshot();
|
|
});
|
|
|
|
it('grants correct silence permissions for folder with silence create permission', async () => {
|
|
setFolderAccessControl({ [AccessControlAction.AlertingSilenceCreate]: true });
|
|
|
|
const rule = getGrafanaRule();
|
|
|
|
render(<RenderActionPermissions rule={rule} action={AlertRuleAction.Silence} />);
|
|
|
|
expect(await screen.findByText(/supported/)).toBeInTheDocument();
|
|
expect(await screen.findByText(/allowed/)).toBeInTheDocument();
|
|
});
|
|
|
|
it('does not grant silence permissions for folder without silence create permission', async () => {
|
|
setFolderAccessControl({ [AccessControlAction.AlertingSilenceCreate]: false });
|
|
|
|
const rule = getGrafanaRule();
|
|
|
|
render(<RenderActionPermissions rule={rule} action={AlertRuleAction.Silence} />);
|
|
|
|
expect(await screen.findByText(/supported/)).toBeInTheDocument();
|
|
expect(screen.queryByText(/allowed/)).not.toBeInTheDocument();
|
|
});
|
|
|
|
it('should report no permissions while we are loading data for cloud rule', async () => {
|
|
const mimirDs = mockDataSource({ uid: 'mimir', name: 'Mimir' });
|
|
setupDataSources(mimirDs);
|
|
|
|
const rule = getCloudRule({}, { rulesSource: mimirDs });
|
|
|
|
const { result } = renderHook(() => useAllAlertRuleAbilities(rule), { wrapper: wrapper() });
|
|
|
|
await waitFor(() => {
|
|
expect(result.current).not.toBeUndefined();
|
|
});
|
|
|
|
expect(result.current).toMatchSnapshot();
|
|
});
|
|
});
|
|
|
|
function createAlertmanagerWrapper(alertmanagerSourceName: string) {
|
|
const ProviderWrapper = getWrapper({ renderWithRouter: true });
|
|
const wrapperComponent = (props: PropsWithChildren) => (
|
|
<ProviderWrapper>
|
|
<AlertmanagerProvider accessType={'notification'} alertmanagerSourceName={alertmanagerSourceName}>
|
|
{props.children}
|
|
</AlertmanagerProvider>
|
|
</ProviderWrapper>
|
|
);
|
|
|
|
return wrapperComponent;
|
|
}
|