import {EventsStatsFixture} from 'sentry-fixture/events';
import {GroupFixture} from 'sentry-fixture/group';
import {IncidentFixture} from 'sentry-fixture/incident';
import {MetricRuleFixture} from 'sentry-fixture/metricRule';
import {ProjectFixture} from 'sentry-fixture/project';
import {initializeOrg} from 'sentry-test/initializeOrg';
import {act, render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
import ProjectsStore from 'sentry/stores/projectsStore';
import {trackAnalytics} from 'sentry/utils/analytics';
import MetricAlertDetails from 'sentry/views/alerts/rules/metric/details';
import {Dataset} from 'sentry/views/alerts/rules/metric/types';
jest.mock('sentry/utils/analytics');
describe('MetricAlertDetails', () => {
const project = ProjectFixture({slug: 'earth', platform: 'javascript'});
beforeEach(() => {
act(() => ProjectsStore.loadInitialData([project]));
MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [project],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/users/',
body: [],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
body: EventsStatsFixture(),
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/issues/?end=2017-10-17T02%3A41%3A20&groupStatsPeriod=auto&limit=5&project=2&query=event.type%3Aerror&sort=freq&start=2017-10-10T02%3A41%3A20',
body: [GroupFixture()],
});
});
afterEach(() => {
act(() => ProjectsStore.reset());
jest.resetAllMocks();
MockApiClient.clearMockResponses();
});
it('renders', async () => {
const {routerContext, organization, routerProps} = initializeOrg();
const incident = IncidentFixture();
const rule = MetricRuleFixture({
projects: [project.slug],
latestIncident: incident,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/alert-rules/${rule.id}/`,
body: rule,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/incidents/`,
body: [incident],
});
render(
,
{context: routerContext, organization}
);
expect(await screen.findAllByText(rule.name)).toHaveLength(2);
expect(screen.getByText('Change alert status to Resolved')).toBeInTheDocument();
expect(screen.getByText(`#${incident.identifier}`)).toBeInTheDocument();
// Related issues
expect(screen.getByTestId('group')).toBeInTheDocument();
expect(trackAnalytics).toHaveBeenCalledWith(
'alert_rule_details.viewed',
expect.objectContaining({
rule_id: Number(rule.id),
alert: '',
})
);
});
it('renders selected incident', async () => {
const {routerContext, organization, router, routerProps} = initializeOrg();
const rule = MetricRuleFixture({projects: [project.slug]});
const incident = IncidentFixture();
MockApiClient.addMockResponse({
url: `/organizations/org-slug/alert-rules/${rule.id}/`,
body: rule,
});
const incidentMock = MockApiClient.addMockResponse({
url: `/organizations/org-slug/incidents/${incident.id}/`,
body: incident,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/incidents/`,
body: [incident],
});
// Related issues to the selected incident
const issuesRequest = MockApiClient.addMockResponse({
url: '/organizations/org-slug/issues/?end=2016-04-26T19%3A44%3A05&groupStatsPeriod=auto&limit=5&project=2&query=event.type%3Aerror&sort=freq&start=2016-03-29T19%3A44%3A05',
body: [GroupFixture()],
});
render(
,
{context: routerContext, organization}
);
expect(await screen.findAllByText(rule.name)).toHaveLength(2);
// Related issues
expect(screen.getByTestId('group')).toBeInTheDocument();
expect(trackAnalytics).toHaveBeenCalledWith(
'alert_rule_details.viewed',
expect.objectContaining({
rule_id: Number(rule.id),
alert: '321',
})
);
expect(incidentMock).toHaveBeenCalled();
expect(issuesRequest).toHaveBeenCalled();
});
it('renders mute button for metric alert', async () => {
const {routerContext, organization, routerProps} = initializeOrg();
const incident = IncidentFixture();
const rule = MetricRuleFixture({
projects: [project.slug],
latestIncident: incident,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/alert-rules/${rule.id}/`,
body: rule,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/incidents/`,
body: [incident],
});
const postRequest = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/alert-rules/${rule.id}/snooze/`,
method: 'POST',
});
const deleteRequest = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/alert-rules/${rule.id}/snooze/`,
method: 'DELETE',
});
render(
,
{context: routerContext, organization}
);
expect(await screen.findByText('Mute for me')).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'Mute for me'}));
expect(postRequest).toHaveBeenCalledTimes(1);
expect(await screen.findByText('Unmute')).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'Unmute'}));
expect(deleteRequest).toHaveBeenCalledTimes(1);
});
it('renders open in discover button with dataset=errors for is:unresolved query', async () => {
const {routerContext, organization, routerProps} = initializeOrg({
organization: {features: ['discover-basic', 'metric-alert-ignore-archived']},
});
const rule = MetricRuleFixture({
projects: [project.slug],
dataset: Dataset.ERRORS,
query: 'is:unresolved',
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/alert-rules/${rule.id}/`,
body: rule,
});
MockApiClient.addMockResponse({
url: `/organizations/org-slug/incidents/`,
body: [],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/issues/?end=2017-10-17T02%3A41%3A20&groupStatsPeriod=auto&limit=5&project=2&query=event.type%3Aerror%20is%3Aunresolved&sort=freq&start=2017-10-10T02%3A41%3A20',
body: [],
});
render(
,
{context: routerContext, organization}
);
expect(await screen.findAllByText(rule.name)).toHaveLength(2);
expect(screen.getByRole('button', {name: 'Open in Discover'})).toHaveAttribute(
'href',
expect.stringContaining('dataset=errors')
);
});
});