import {EventFixture} from 'sentry-fixture/event'; import {EventAttachmentFixture} from 'sentry-fixture/eventAttachment'; import {GroupFixture} from 'sentry-fixture/group'; import {LocationFixture} from 'sentry-fixture/locationFixture'; import {RouterFixture} from 'sentry-fixture/routerFixture'; import {initializeOrg} from 'sentry-test/initializeOrg'; import {render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary'; import * as useMedia from 'sentry/utils/useMedia'; import {SectionKey, useEventDetails} from 'sentry/views/issueDetails/streamline/context'; import {Tab, TabPaths} from 'sentry/views/issueDetails/types'; import {IssueEventNavigation} from './eventNavigation'; jest.mock('sentry/views/issueDetails/streamline/context'); describe('EventNavigation', () => { const {organization, router} = initializeOrg(); const group = GroupFixture({id: 'group-id'}); const testEvent = EventFixture({ id: 'event-id', size: 7, dateCreated: '2019-03-20T00:00:00.000Z', errors: [], entries: [], tags: [ {key: 'environment', value: 'dev'}, {key: 'replayId', value: 'replay-id'}, ], previousEventID: 'prev-event-id', nextEventID: 'next-event-id', }); const defaultProps: React.ComponentProps = { event: testEvent, group, query: undefined, }; beforeEach(() => { jest.resetAllMocks(); jest.mocked(useEventDetails).mockReturnValue({ sectionData: { highlights: {key: SectionKey.HIGHLIGHTS}, tags: {key: SectionKey.TAGS}, replay: {key: SectionKey.REPLAY}, }, dispatch: jest.fn(), }); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/issues/${group.id}/tags/`, body: [], }); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/issues/${group.id}/attachments/`, body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/replay-count/', body: {}, }); }); describe('all events buttons', () => { it('renders the all events controls', () => { const allEventsRouter = RouterFixture({ params: {groupId: group.id}, routes: [{path: TabPaths[Tab.EVENTS]}], location: LocationFixture({ pathname: `/organizations/${organization.slug}/issues/${group.id}/events/`, }), }); render(, {router: allEventsRouter}); const discoverButton = screen.getByLabelText('Open in Discover'); expect(discoverButton).toBeInTheDocument(); expect(discoverButton).toHaveAttribute( 'href', expect.stringContaining(`/organizations/${organization.slug}/discover/results/`) ); const closeButton = screen.getByRole('button', {name: 'Return to event details'}); expect(closeButton).toBeInTheDocument(); expect(closeButton).toHaveAttribute( 'href', expect.stringContaining(`/organizations/${organization.slug}/issues/${group.id}/`) ); }); }); describe('recommended event tabs', () => { it('can navigate to the oldest event', async () => { jest.spyOn(useMedia, 'default').mockReturnValue(true); render(, {router}); await userEvent.click(await screen.findByRole('tab', {name: 'First'})); expect(router.push).toHaveBeenCalledWith({ pathname: '/organizations/org-slug/issues/group-id/events/oldest/', query: {referrer: 'oldest-event'}, }); }); it('can navigate to the latest event', async () => { jest.spyOn(useMedia, 'default').mockReturnValue(true); render(, {router}); await userEvent.click(await screen.findByRole('tab', {name: 'Last'})); expect(router.push).toHaveBeenCalledWith({ pathname: '/organizations/org-slug/issues/group-id/events/latest/', query: {referrer: 'latest-event'}, }); }); it('can navigate to the recommended event', async () => { jest.spyOn(useMedia, 'default').mockReturnValue(true); const recommendedEventRouter = RouterFixture({ params: {eventId: 'latest'}, location: LocationFixture({ pathname: `/organizations/org-slug/issues/group-id/events/latest/`, }), }); render(, { router: recommendedEventRouter, }); await userEvent.click(await screen.findByRole('tab', {name: 'Rec.'})); expect(recommendedEventRouter.push).toHaveBeenCalledWith({ pathname: '/organizations/org-slug/issues/group-id/events/recommended/', query: {referrer: 'recommended-event'}, }); }); }); it('can navigate next/previous events', async () => { render(); expect(await screen.findByRole('button', {name: 'Previous Event'})).toHaveAttribute( 'href', `/organizations/org-slug/issues/group-id/events/prev-event-id/?referrer=previous-event` ); expect(screen.getByRole('button', {name: 'Next Event'})).toHaveAttribute( 'href', `/organizations/org-slug/issues/group-id/events/next-event-id/?referrer=next-event` ); }); it('can preload next/previous events', async () => { const event = EventFixture({ nextEventID: 'next-event-id', previousEventID: 'prev-event-id', }); const mockNextEvent = MockApiClient.addMockResponse({ url: `/organizations/org-slug/issues/group-id/events/next-event-id/`, body: EventFixture(), }); const mockPreviousEvent = MockApiClient.addMockResponse({ url: `/organizations/org-slug/issues/group-id/events/prev-event-id/`, body: EventFixture(), }); render(); expect(mockNextEvent).not.toHaveBeenCalled(); expect(mockPreviousEvent).not.toHaveBeenCalled(); await userEvent.hover(await screen.findByRole('button', {name: 'Next Event'})); await waitFor(() => expect(mockNextEvent).toHaveBeenCalled()); expect(mockPreviousEvent).not.toHaveBeenCalled(); await userEvent.hover(screen.getByRole('button', {name: 'Previous Event'})); await waitFor(() => expect(mockPreviousEvent).toHaveBeenCalled()); }); describe('counts', () => { it('renders default counts', async () => { render(); await userEvent.click(screen.getByRole('button', {name: 'Select issue content'})); expect( await screen.findByRole('menuitemradio', {name: 'Attachments 0'}) ).toBeInTheDocument(); expect( screen.getByRole('menuitemradio', {name: 'Events 327k'}) ).toBeInTheDocument(); expect(screen.getByRole('menuitemradio', {name: 'Replays 0'})).toBeInTheDocument(); expect(screen.getByRole('menuitemradio', {name: 'Feedback 0'})).toBeInTheDocument(); }); it('renders 1 attachment', async () => { MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/issues/${group.id}/attachments/`, body: [EventAttachmentFixture()], }); render(); await userEvent.click(screen.getByRole('button', {name: 'Select issue content'})); expect( await screen.findByRole('menuitemradio', {name: 'Attachments 1'}) ).toBeInTheDocument(); }); it('renders 50+ attachments', async () => { MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/issues/${group.id}/attachments/`, body: [EventAttachmentFixture()], headers: { // Assumes there is more than 50 attachments if there is a next page Link: '; rel="previous"; results="false"; cursor="0:0:1", ; rel="next"; results="true"; cursor="0:20:0"', }, }); render(); await userEvent.click(screen.getByRole('button', {name: 'Select issue content'})); expect( await screen.findByRole('menuitemradio', {name: 'Attachments 50+'}) ).toBeInTheDocument(); }); }); });