import {initializeOrg} from 'sentry-test/initializeOrg'; import { act, render, renderGlobalModal, screen, userEvent, waitFor, } from 'sentry-test/reactTestingLibrary'; import PullRequestLink from 'sentry/components/pullRequestLink'; import ConfigStore from 'sentry/stores/configStore'; import GroupStore from 'sentry/stores/groupStore'; import OrganizationStore from 'sentry/stores/organizationStore'; import ProjectsStore from 'sentry/stores/projectsStore'; import TeamStore from 'sentry/stores/teamStore'; import {GroupActivityType} from 'sentry/types'; import {GroupActivity} from 'sentry/views/issueDetails/groupActivity'; describe('GroupActivity', function () { let project; const dateCreated = '2021-10-01T15:31:38.950115Z'; beforeEach(function () { project = TestStubs.Project(); ProjectsStore.loadInitialData([project]); ConfigStore.init(); ConfigStore.set('user', {id: '123'}); GroupStore.init(); }); afterEach(() => { MockApiClient.clearMockResponses(); jest.clearAllMocks(); }); function createWrapper({activity, organization: additionalOrg} = {}) { const group = TestStubs.Group({ id: '1337', activity: activity ?? [ {type: 'note', id: 'note-1', data: {text: 'Test Note'}, user: TestStubs.User()}, ], project, }); const {organization, routerContext} = initializeOrg({ organization: additionalOrg, group, }); GroupStore.add([group]); TeamStore.loadInitialData([TestStubs.Team({id: '999', slug: 'no-team'})]); OrganizationStore.onUpdate(organization, {replace: true}); return render( , {context: routerContext} ); } it('renders a NoteInput', function () { createWrapper(); expect(screen.getByTestId('activity-note-body')).toBeInTheDocument(); }); it('renders a marked reviewed activity', function () { const user = TestStubs.User({name: 'Samwise'}); createWrapper({ activity: [{type: 'mark_reviewed', id: 'reviewed-1', data: {}, user}], }); expect(screen.getByText('marked this issue as reviewed')).toBeInTheDocument(); expect(screen.getByText(user.name)).toBeInTheDocument(); }); it('renders a pr activity', function () { const user = TestStubs.User({name: 'Test User'}); const repository = TestStubs.Repository(); const pullRequest = TestStubs.PullRequest({message: 'Fixes ISSUE-1'}); createWrapper({ activity: [ { type: 'set_resolved_in_pull_request', id: 'pr-1', data: { pullRequest: { author: 'Test User', version: ( ), repository: {repository}, }, }, user, }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Test User has created a PR for this issue:' ); }); it('renders a assigned to self activity', function () { const user = TestStubs.User({id: '301', name: 'Mark'}); createWrapper({ activity: [ { data: { assignee: user.id, assigneeEmail: user.email, assigneeType: 'user', }, dateCreated: '2021-10-01T15:31:38.950115Z', id: '117', type: 'assigned', user, }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( /Mark assigned this issue to themselves/ ); }); it('renders an assigned via codeowners activity', function () { createWrapper({ activity: [ { data: { assignee: '123', assigneeEmail: 'anotheruser@sentry.io', assigneeType: 'user', integration: 'codeowners', rule: 'path:something/*.py #workflow', }, dateCreated: '2021-10-01T15:31:38.950115Z', id: '117', type: 'assigned', user: null, }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( /Sentry auto-assigned this issue to anotheruser@sentry.io/ ); }); it('renders an assigned via slack activity', function () { const user = TestStubs.User({id: '301', name: 'Mark'}); createWrapper({ activity: [ { data: { assignee: '123', assigneeEmail: 'anotheruser@sentry.io', assigneeType: 'user', integration: 'slack', }, dateCreated: '2021-10-01T15:31:38.950115Z', id: '117', type: 'assigned', user, }, ], }); const item = screen.getAllByTestId('activity-item').at(-1); expect(item).toHaveTextContent(/Mark assigned this issue to anotheruser@sentry.io/); expect(item).toHaveTextContent(/Assigned via Slack/); }); it('resolved in commit with no releases', function () { createWrapper({ activity: [ { type: 'set_resolved_in_commit', id: '123', data: { author: 'hello', commit: { id: 'komal-commit', repository: {}, releases: [], }, }, user: TestStubs.User(), }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar marked this issue as resolved in komal-commit' ); }); it('resolved in commit with one release', function () { createWrapper({ activity: [ { type: 'set_resolved_in_commit', id: '123', data: { author: 'hello', commit: { id: 'komal-commit', repository: {}, releases: [ { dateCreated: '2022-05-01', dateReleased: '2022-05-02', version: 'random', }, ], }, }, user: TestStubs.User(), }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar marked this issue as resolved in komal-commit This commit was released in random' ); }); it('resolved in commit with multiple releases', function () { createWrapper({ activity: [ { type: 'set_resolved_in_commit', id: '123', data: { commit: { id: 'komal-commit', repository: {}, releases: [ { dateCreated: '2022-05-01', dateReleased: '2022-05-02', version: 'random', }, { dateCreated: '2022-06-01', dateReleased: '2022-06-02', version: 'newest', }, { dateCreated: '2021-08-03', dateReleased: '2021-08-03', version: 'oldest-release', }, { dateCreated: '2022-04-21', dateReleased: '2022-04-21', version: 'randomTwo', }, ], }, }, user: TestStubs.User(), }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar marked this issue as resolved in komal-commit This commit was released in oldest-release and 3 others' ); }); it('requests assignees that are not in the team store', async function () { const team = TestStubs.Team({id: '123', name: 'workflow'}); const teamRequest = MockApiClient.addMockResponse({ url: `/organizations/org-slug/teams/`, body: [team], }); createWrapper({ activity: [ { id: '123', user: null, type: 'assigned', data: { assignee: team.id, assigneeEmail: null, assigneeType: 'team', }, dateCreated: '2021-10-28T13:40:10.634821Z', }, ], }); await waitFor(() => expect(teamRequest).toHaveBeenCalledTimes(1)); expect( await screen.findByText(`assigned this issue to #${team.slug}`) ).toBeInTheDocument(); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( /Sentry assigned this issue to #team-slug/ ); }); describe('Delete', function () { let deleteMock; beforeEach(function () { deleteMock = MockApiClient.addMockResponse({ url: '/issues/1337/comments/note-1/', method: 'DELETE', }); ConfigStore.set('user', {id: '123', isSuperuser: true}); }); it('should do nothing if not present in GroupStore', async function () { createWrapper(); renderGlobalModal(); act(() => { // Remove note from group activity GroupStore.removeActivity('1337', 'note-1'); }); await userEvent.click(screen.getByText('Remove')); expect( screen.getByText('Are you sure you wish to delete this comment?') ).toBeInTheDocument(); await userEvent.click(screen.getByRole('button', {name: 'Confirm'})); expect(deleteMock).not.toHaveBeenCalled(); }); it('should remove remove the item from the GroupStore make a DELETE API request', async function () { createWrapper(); renderGlobalModal(); await userEvent.click(screen.getByText('Remove')); expect( screen.getByText('Are you sure you wish to delete this comment?') ).toBeInTheDocument(); await userEvent.click(screen.getByRole('button', {name: 'Confirm'})); expect(deleteMock).toHaveBeenCalledTimes(1); }); }); it('renders ignored', function () { createWrapper({ activity: [ { id: '123', type: GroupActivityType.SET_IGNORED, data: { ignoreUntilEscalating: true, }, user: TestStubs.User(), dateCreated, }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar ignored this issue' ); }); it('renders archived until escalating if org has `escalating-issues-ui` feature', function () { createWrapper({ activity: [ { id: '123', type: GroupActivityType.SET_IGNORED, data: { ignoreUntilEscalating: true, }, user: TestStubs.User(), dateCreated, }, ], organization: {features: ['escalating-issues-ui']}, }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar archived this issue until it escalates' ); }); it('renders escalating with forecast and plural events if org has `escalating-issues-ui` feature', function () { createWrapper({ activity: [ { id: '123', type: GroupActivityType.SET_UNRESOLVED, data: { forecast: 200, }, user: null, dateCreated, }, ], organization: {features: ['escalating-issues-ui']}, }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Sentry flagged this issue as escalating because over 200 events happened in an hour' ); }); it('renders escalating with forecast and singular event if org has `escalating-issues-ui` feature', function () { createWrapper({ activity: [ { id: '123', type: GroupActivityType.SET_UNRESOLVED, data: { forecast: 1, }, user: null, dateCreated, }, ], organization: {features: ['escalating-issues-ui']}, }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Sentry flagged this issue as escalating because over 1 event happened in an hour' ); }); it('renders ignored until it happens x times in time window', function () { createWrapper({ activity: [ { id: '123', type: GroupActivityType.SET_IGNORED, data: { ignoreCount: 400, ignoreWindow: 1, }, user: TestStubs.User(), dateCreated, }, ], }); expect(screen.getAllByTestId('activity-item').at(-1)).toHaveTextContent( 'Foo Bar ignored this issue until it happens 400 time(s) in 1 minute' ); }); });