import moment from 'moment';
import {GroupFixture} from 'sentry-fixture/group';
import {MemberFixture} from 'sentry-fixture/member';
import {OrganizationFixture} from 'sentry-fixture/organization';
import {ProjectFixture} from 'sentry-fixture/project';
import {ProjectAlertRuleFixture} from 'sentry-fixture/projectAlertRule';
import {initializeOrg} from 'sentry-test/initializeOrg';
import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
import ProjectsStore from 'sentry/stores/projectsStore';
import {browserHistory} from 'sentry/utils/browserHistory';
import AlertRuleDetails from './ruleDetails';
describe('AlertRuleDetails', () => {
const context = initializeOrg();
const organization = context.organization;
const project = ProjectFixture();
const rule = ProjectAlertRuleFixture({
lastTriggered: moment().subtract(2, 'day').format(),
});
const member = MemberFixture();
const createWrapper = (props: any = {}, newContext?: any, org = organization) => {
const router = newContext ? newContext.router : context.router;
const routerContext = newContext ? newContext.routerContext : context.routerContext;
return render(
,
{context: routerContext, organization: org}
);
};
beforeEach(() => {
browserHistory.push = jest.fn();
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
body: rule,
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/stats/`,
body: [],
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/group-history/`,
body: [
{
count: 1,
group: GroupFixture(),
lastTriggered: moment('Apr 11, 2019 1:08:59 AM UTC').format(),
eventId: 'eventId',
},
],
headers: {
Link:
'; rel="previous"; results="false"; cursor="0:0:1", ' +
'; rel="next"; results="true"; cursor="0:100:0"',
},
});
MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/projects/`,
body: [project],
});
MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/users/`,
body: [member],
});
ProjectsStore.init();
ProjectsStore.loadInitialData([project]);
});
afterEach(() => {
ProjectsStore.reset();
MockApiClient.clearMockResponses();
});
it('displays alert rule with list of issues', async () => {
createWrapper();
expect(await screen.findAllByText('My alert rule')).toHaveLength(2);
expect(await screen.findByText('RequestError:')).toBeInTheDocument();
expect(screen.getByText('Apr 11, 2019 1:08:59 AM UTC')).toBeInTheDocument();
expect(screen.getByText('RequestError:')).toHaveAttribute(
'href',
expect.stringMatching(
RegExp(
`/organizations/${organization.slug}/issues/${
GroupFixture().id
}/events/eventId.*`
)
)
);
});
it('should allow paginating results', async () => {
createWrapper();
expect(await screen.findByLabelText('Next')).toBeEnabled();
await userEvent.click(screen.getByLabelText('Next'));
expect(browserHistory.push).toHaveBeenCalledWith({
pathname: '/mock-pathname/',
query: {
cursor: '0:100:0',
},
});
});
it('should reset pagination cursor on date change', async () => {
createWrapper();
const dateSelector = await screen.findByText('7D');
expect(dateSelector).toBeInTheDocument();
await userEvent.click(dateSelector);
await userEvent.click(screen.getByRole('option', {name: 'Last 24 hours'}));
expect(context.router.push).toHaveBeenCalledWith(
expect.objectContaining({
query: {
pageStatsPeriod: '24h',
cursor: undefined,
pageEnd: undefined,
pageStart: undefined,
pageUtc: undefined,
},
})
);
});
it('should show the time since last triggered in sidebar', async () => {
createWrapper();
expect(await screen.findAllByText('Last Triggered')).toHaveLength(2);
expect(screen.getByText('2 days ago')).toBeInTheDocument();
});
it('renders not found on 404', async () => {
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
statusCode: 404,
body: {},
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
createWrapper();
expect(
await screen.findByText('The alert rule you were looking for was not found.')
).toBeInTheDocument();
});
it('renders incompatible rule filter', async () => {
const incompatibleRule = ProjectAlertRuleFixture({
conditions: [
{id: 'sentry.rules.conditions.first_seen_event.FirstSeenEventCondition'},
{id: 'sentry.rules.conditions.regression_event.RegressionEventCondition'},
],
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
body: incompatibleRule,
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
createWrapper();
expect(
await screen.findByText(
'The conditions in this alert rule conflict and might not be working properly.'
)
).toBeInTheDocument();
});
it('incompatible rule banner hidden for good rule', async () => {
createWrapper();
expect(await screen.findAllByText('My alert rule')).toHaveLength(2);
expect(
screen.queryByText(
'The conditions in this alert rule conflict and might not be working properly.'
)
).not.toBeInTheDocument();
});
it('rule disabled banner because of missing actions and hides some actions', async () => {
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
body: ProjectAlertRuleFixture({
actions: [],
status: 'disabled',
}),
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
createWrapper();
expect(
await screen.findByText(
'This alert is disabled due to missing actions. Please edit the alert rule to enable this alert.'
)
).toBeInTheDocument();
expect(screen.getByRole('button', {name: 'Edit to enable'})).toBeInTheDocument();
expect(screen.getByRole('button', {name: 'Duplicate'})).toBeDisabled();
expect(screen.getByRole('button', {name: 'Mute for me'})).toBeDisabled();
});
it('rule disabled banner generic', async () => {
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
body: ProjectAlertRuleFixture({
status: 'disabled',
}),
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
createWrapper();
expect(
await screen.findByText(
'This alert is disabled due to its configuration and needs to be edited to be enabled.'
)
).toBeInTheDocument();
});
it('rule to be disabled can opt out', async () => {
const disabledRule = ProjectAlertRuleFixture({
disableDate: moment().add(1, 'day').format(),
disableReason: 'noisy',
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${disabledRule.id}/`,
body: disabledRule,
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
const updateMock = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${disabledRule.id}/`,
method: 'PUT',
});
createWrapper();
expect(
await screen.findByText(/This alert is scheduled to be disabled/)
).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'click here'}));
expect(updateMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({data: {...disabledRule, optOutExplicit: true}})
);
expect(
screen.queryByText(/This alert is scheduled to be disabled/)
).not.toBeInTheDocument();
});
it('disabled rule can be re-enabled', async () => {
const disabledRule = ProjectAlertRuleFixture({
status: 'disabled',
disableDate: moment().subtract(1, 'day').format(),
disableReason: 'noisy',
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${disabledRule.id}/`,
body: disabledRule,
match: [MockApiClient.matchQuery({expand: 'lastTriggered'})],
});
const enableMock = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${disabledRule.id}/enable/`,
method: 'PUT',
});
createWrapper();
expect(
await screen.findByText(/This alert was disabled due to lack of activity/)
).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'click here'}));
expect(enableMock).toHaveBeenCalled();
expect(
screen.queryByText(/This alert was disabled due to lack of activity/)
).not.toBeInTheDocument();
expect(screen.queryByText(/This alert is disabled/)).not.toBeInTheDocument();
});
it('renders the mute button and can mute/unmute alerts', async () => {
const postRequest = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/snooze/`,
method: 'POST',
});
const deleteRequest = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/snooze/`,
method: 'DELETE',
});
createWrapper();
expect(await screen.findByText('Mute for everyone')).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'Mute for everyone'}));
expect(postRequest).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({data: {target: 'everyone'}})
);
expect(await screen.findByText('Unmute')).toBeInTheDocument();
await userEvent.click(screen.getByRole('button', {name: 'Unmute'}));
expect(deleteRequest).toHaveBeenCalledTimes(1);
});
it('mutes alert if query parameter is set', async () => {
const request = MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/snooze/`,
method: 'POST',
});
const contextWithQueryParam = initializeOrg({
router: {
location: {query: {mute: '1'}},
},
});
createWrapper({}, contextWithQueryParam);
expect(await screen.findByText('Unmute')).toBeInTheDocument();
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
data: {target: 'everyone'},
})
);
});
it('mute button is disabled if no alerts:write permission', async () => {
const orgWithoutAccess = OrganizationFixture({
access: [],
});
const contextWithoutAccess = initializeOrg({
organization: orgWithoutAccess,
});
createWrapper({}, contextWithoutAccess, orgWithoutAccess);
expect(await screen.findByRole('button', {name: 'Mute for everyone'})).toBeDisabled();
});
it('inserts user email into rule notify action', async () => {
// Alert rule with "send a notification to member" action
const sendNotificationRule = ProjectAlertRuleFixture({
actions: [
{
id: 'sentry.mail.actions.NotifyEmailAction',
name: 'Send a notification to Member and if none can be found then send a notification to ActiveMembers',
targetIdentifier: member.id,
targetType: 'Member',
},
],
});
MockApiClient.addMockResponse({
url: `/projects/${organization.slug}/${project.slug}/rules/${rule.id}/`,
body: sendNotificationRule,
});
createWrapper();
expect(
await screen.findByText(`Send a notification to ${member.email}`)
).toBeInTheDocument();
});
});