123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- import {MemberFixture} from 'sentry-fixture/member';
- import {MonitorFixture} from 'sentry-fixture/monitor';
- import {OrganizationFixture} from 'sentry-fixture/organization';
- import {TeamFixture} from 'sentry-fixture/team';
- import {UserFixture} from 'sentry-fixture/user';
- import {initializeOrg} from 'sentry-test/initializeOrg';
- import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
- import selectEvent from 'sentry-test/selectEvent';
- import {useMembers} from 'sentry/utils/useMembers';
- import useProjects from 'sentry/utils/useProjects';
- import {useTeams} from 'sentry/utils/useTeams';
- import MonitorForm from 'sentry/views/monitors/components/monitorForm';
- import {ScheduleType} from 'sentry/views/monitors/types';
- jest.mock('sentry/utils/useProjects');
- jest.mock('sentry/utils/useTeams');
- jest.mock('sentry/utils/useMembers');
- describe('MonitorForm', function () {
- const organization = OrganizationFixture({features: ['issue-platform']});
- const member = MemberFixture({user: UserFixture({name: 'John Smith'})});
- const team = TeamFixture({slug: 'test-team'});
- const {project, routerContext} = initializeOrg({organization});
- beforeEach(() => {
- jest.mocked(useProjects).mockReturnValue({
- fetchError: null,
- fetching: false,
- hasMore: false,
- initiallyLoaded: false,
- onSearch: jest.fn(),
- placeholders: [],
- projects: [project],
- });
- jest.mocked(useTeams).mockReturnValue({
- fetchError: null,
- fetching: false,
- hasMore: false,
- initiallyLoaded: false,
- loadMore: jest.fn(),
- onSearch: jest.fn(),
- teams: [team],
- });
- jest.mocked(useMembers).mockReturnValue({
- fetchError: null,
- fetching: false,
- hasMore: false,
- initiallyLoaded: false,
- loadMore: jest.fn(),
- onSearch: jest.fn(),
- members: [member.user!],
- });
- });
- it('displays human readable schedule', async function () {
- render(
- <MonitorForm
- apiMethod="POST"
- apiEndpoint={`/organizations/${organization.slug}/monitors/`}
- onSubmitSuccess={jest.fn()}
- />,
- {context: routerContext, organization}
- );
- const schedule = screen.getByRole('textbox', {name: 'Crontab Schedule'});
- await userEvent.clear(schedule);
- await userEvent.type(schedule, '5 * * * *');
- expect(screen.getByText('"At 5 minutes past the hour"')).toBeInTheDocument();
- });
- it('submits a new monitor', async function () {
- const mockHandleSubmitSuccess = jest.fn();
- const apiEndpont = `/organizations/${organization.slug}/monitors/`;
- render(
- <MonitorForm
- apiMethod="POST"
- apiEndpoint={apiEndpont}
- onSubmitSuccess={mockHandleSubmitSuccess}
- submitLabel="Add Monitor"
- />,
- {context: routerContext, organization}
- );
- await userEvent.type(screen.getByRole('textbox', {name: 'Name'}), 'My Monitor');
- await selectEvent.select(
- screen.getByRole('textbox', {name: 'Project'}),
- project.slug
- );
- const schedule = screen.getByRole('textbox', {name: 'Crontab Schedule'});
- await userEvent.clear(schedule);
- await userEvent.type(schedule, '5 * * * *');
- await selectEvent.select(
- screen.getByRole('textbox', {name: 'Timezone'}),
- 'Los Angeles'
- );
- await userEvent.type(screen.getByRole('spinbutton', {name: 'Grace Period'}), '5');
- await userEvent.type(screen.getByRole('spinbutton', {name: 'Max Runtime'}), '20');
- await userEvent.type(
- screen.getByRole('spinbutton', {name: 'Failure Tolerance'}),
- '4'
- );
- await userEvent.type(
- screen.getByRole('spinbutton', {name: 'Recovery Tolerance'}),
- '2'
- );
- const notifySelect = screen.getByRole('textbox', {name: 'Notify'});
- await selectEvent.openMenu(notifySelect);
- expect(
- screen.getByRole('menuitemcheckbox', {name: 'John Smith'})
- ).toBeInTheDocument();
- expect(
- screen.getByRole('menuitemcheckbox', {name: '#test-team'})
- ).toBeInTheDocument();
- await selectEvent.select(notifySelect, 'John Smith');
- const submitMock = MockApiClient.addMockResponse({
- url: apiEndpont,
- method: 'POST',
- });
- await userEvent.click(screen.getByRole('button', {name: 'Add Monitor'}));
- const config = {
- checkin_margin: '5',
- max_runtime: '20',
- failure_issue_threshold: '4',
- recovery_threshold: '2',
- schedule: '5 * * * *',
- schedule_type: 'crontab',
- timezone: 'America/Los_Angeles',
- };
- const alertRule = {
- environment: undefined,
- targets: [{targetIdentifier: 1, targetType: 'Member'}],
- };
- expect(submitMock).toHaveBeenCalledWith(
- expect.anything(),
- expect.objectContaining({
- data: {
- name: 'My Monitor',
- project: 'project-slug',
- type: 'cron_job',
- config,
- alertRule,
- },
- })
- );
- expect(mockHandleSubmitSuccess).toHaveBeenCalled();
- });
- it('prefills with an existing monitor', async function () {
- const monitor = MonitorFixture({project});
- const apiEndpont = `/organizations/${organization.slug}/monitors/${monitor.slug}/`;
- if (monitor.config.schedule_type !== ScheduleType.CRONTAB) {
- throw new Error('Fixture is not crontab');
- }
- render(
- <MonitorForm
- monitor={monitor}
- apiMethod="POST"
- apiEndpoint={apiEndpont}
- onSubmitSuccess={jest.fn()}
- submitLabel="Edit Monitor"
- />,
- {context: routerContext, organization}
- );
- // Name and slug
- expect(screen.getByRole('textbox', {name: 'Name'})).toHaveValue(monitor.name);
- expect(screen.getByRole('textbox', {name: 'Slug'})).toHaveValue(monitor.slug);
- // Project
- expect(screen.getByRole('textbox', {name: 'Project'})).toBeDisabled();
- expect(screen.getByText(project.slug)).toBeInTheDocument();
- // Schedule type
- await selectEvent.openMenu(screen.getByRole('textbox', {name: 'Schedule Type'}));
- const crontabOption = screen.getByRole('menuitemradio', {name: 'Crontab'});
- expect(crontabOption).toBeChecked();
- await userEvent.click(crontabOption);
- // Schedule value
- expect(screen.getByRole('textbox', {name: 'Crontab Schedule'})).toHaveValue(
- monitor.config.schedule
- );
- // Schedule timezone
- await selectEvent.openMenu(screen.getByRole('textbox', {name: 'Timezone'}));
- const losAngelesOption = screen.getByRole('menuitemradio', {name: 'Los Angeles'});
- expect(losAngelesOption).toBeChecked();
- await userEvent.click(losAngelesOption);
- // Margins
- expect(screen.getByRole('spinbutton', {name: 'Grace Period'})).toHaveValue(5);
- expect(screen.getByRole('spinbutton', {name: 'Max Runtime'})).toHaveValue(10);
- // Tolerances
- expect(screen.getByRole('spinbutton', {name: 'Failure Tolerance'})).toHaveValue(2);
- expect(screen.getByRole('spinbutton', {name: 'Recovery Tolerance'})).toHaveValue(2);
- // Alert rule configuration
- await selectEvent.openMenu(screen.getByRole('textbox', {name: 'Notify'}));
- const memberOption = screen.getByRole('menuitemcheckbox', {name: member.user?.name});
- expect(memberOption).toBeChecked();
- await userEvent.keyboard('{Escape}');
- const submitMock = MockApiClient.addMockResponse({
- url: apiEndpont,
- method: 'POST',
- });
- // Monitor form is not submitable until something is changed
- const submitButton = screen.getByRole('button', {name: 'Edit Monitor'});
- expect(submitButton).toBeDisabled();
- // Change Failure Tolerance
- await userEvent.clear(screen.getByRole('spinbutton', {name: 'Failure Tolerance'}));
- await userEvent.type(
- screen.getByRole('spinbutton', {name: 'Failure Tolerance'}),
- '10'
- );
- await userEvent.click(submitButton);
- // XXX(epurkhiser): When the values are loaded directly from the
- // monitor they come in as numbers, when changed via the toggles they
- // are translated to strings :(
- const config = {
- max_runtime: monitor.config.max_runtime,
- checkin_margin: monitor.config.checkin_margin,
- recovery_threshold: monitor.config.recovery_threshold,
- schedule: monitor.config.schedule,
- schedule_type: monitor.config.schedule_type,
- timezone: monitor.config.timezone,
- failure_issue_threshold: '10',
- };
- const alertRule = {
- environment: undefined,
- targets: [{targetIdentifier: 1, targetType: 'Member'}],
- };
- expect(submitMock).toHaveBeenCalledWith(
- expect.anything(),
- expect.objectContaining({
- data: {
- name: monitor.name,
- slug: monitor.slug,
- project: monitor.project.slug,
- type: 'cron_job',
- config,
- alertRule,
- },
- })
- );
- });
- });
|