import merge from 'lodash/merge'; import {GroupFixture} from 'sentry-fixture/group'; import {GroupStatsFixture} from 'sentry-fixture/groupStats'; import {LocationFixture} from 'sentry-fixture/locationFixture'; import {MemberFixture} from 'sentry-fixture/member'; import {OrganizationFixture} from 'sentry-fixture/organization'; import {ProjectFixture} from 'sentry-fixture/project'; import {SearchFixture} from 'sentry-fixture/search'; import {TagsFixture} from 'sentry-fixture/tags'; import {initializeOrg} from 'sentry-test/initializeOrg'; import { act, render, screen, userEvent, waitFor, waitForElementToBeRemoved, } from 'sentry-test/reactTestingLibrary'; import {textWithMarkupMatcher} from 'sentry-test/utils'; import StreamGroup from 'sentry/components/stream/group'; import {DEFAULT_QUERY} from 'sentry/constants'; import ProjectsStore from 'sentry/stores/projectsStore'; import TagStore from 'sentry/stores/tagStore'; import {SavedSearchVisibility} from 'sentry/types/group'; import {browserHistory} from 'sentry/utils/browserHistory'; import localStorageWrapper from 'sentry/utils/localStorage'; import * as parseLinkHeader from 'sentry/utils/parseLinkHeader'; import IssueListWithStores, {IssueListOverview} from 'sentry/views/issueList/overview'; // Mock jest.mock('sentry/views/issueList/actions', () => jest.fn(() => null)); jest.mock('sentry/components/stream/group', () => jest.fn(() => null)); const DEFAULT_LINKS_HEADER = '; rel="previous"; results="false"; cursor="1443575731:0:1", ' + '; rel="next"; results="true"; cursor="1443575000:0:0"'; const project = ProjectFixture({ id: '3559', name: 'Foo Project', slug: 'project-slug', firstEvent: new Date().toISOString(), }); const {organization, projects, router} = initializeOrg({ organization: { id: '1337', slug: 'org-slug', features: ['global-views'], access: [], }, router: { location: {query: {}, search: ''}, params: {}, }, projects: [project], }); const routerProps = { params: router.params, location: router.location, }; function getSearchInput() { const input = screen.getAllByRole('combobox', {name: 'Add a search term'}).at(-1); expect(input).toBeInTheDocument(); return input!; } describe('IssueList', function () { let props: any; const tags = TagsFixture(); const group = GroupFixture({project}); const groupStats = GroupStatsFixture(); const savedSearch = SearchFixture({ id: '789', query: 'is:unresolved TypeError', sort: 'date', name: 'Unresolved TypeErrors', }); let fetchMembersRequest: jest.Mock; const api = new MockApiClient(); const parseLinkHeaderSpy = jest.spyOn(parseLinkHeader, 'default'); beforeEach(function () { // The tests fail because we have a "component update was not wrapped in act" error. // It should be safe to ignore this error, but we should remove the mock once we move to react testing library // eslint-disable-next-line no-console jest.spyOn(console, 'error').mockImplementation(jest.fn()); Object.defineProperty(Element.prototype, 'clientWidth', {value: 1000}); MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [group], headers: { Link: DEFAULT_LINKS_HEADER, }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues-stats/', body: [groupStats], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/recent-searches/', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/recent-searches/', method: 'POST', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues-count/', method: 'GET', body: [{}], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/processingissues/', method: 'GET', body: [ { project: 'test-project', numIssues: 1, hasIssues: true, lastSeen: '2019-01-16T15:39:11.081Z', }, ], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/tags/', method: 'GET', body: tags, }); fetchMembersRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/users/', method: 'GET', body: [MemberFixture({projects: [project.slug]})], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/sent-first-event/', body: {sentFirstEvent: true}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: [project], }); TagStore.init?.(); props = { api, savedSearchLoading: false, savedSearches: [savedSearch], useOrgSavedSearches: true, selection: { projects: [parseInt(projects[0].id, 10)], environments: [], datetime: {period: '14d'}, }, location: {query: {query: DEFAULT_QUERY}, search: `query=${DEFAULT_QUERY}`}, params: {}, organization, tags: tags.reduce>((acc, tag) => { acc[tag.key] = tag; return acc; }, {}), }; }); afterEach(function () { jest.clearAllMocks(); MockApiClient.clearMockResponses(); localStorageWrapper.clear(); }); describe('withStores and feature flags', function () { let savedSearchesRequest: jest.Mock; let recentSearchesRequest: jest.Mock; let issuesRequest: jest.Mock; beforeEach(function () { jest.mocked(StreamGroup).mockClear(); recentSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/recent-searches/', method: 'GET', body: [], }); savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch], }); issuesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [group], headers: { Link: DEFAULT_LINKS_HEADER, }, }); }); it('loads group rows with default query (no pinned queries, async and no query in URL)', async function () { render(, {router}); // Loading saved searches await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); expect(savedSearchesRequest).toHaveBeenCalledTimes(1); await screen.findByRole('grid', {name: 'Create a search query'}); expect(screen.getByRole('row', {name: 'is:unresolved'})).toBeInTheDocument(); expect(screen.getByRole('button', {name: /custom search/i})).toBeInTheDocument(); await userEvent.click(getSearchInput()); // auxillary requests being made await waitFor(() => { expect(recentSearchesRequest).toHaveBeenCalledTimes(1); }); expect(fetchMembersRequest).toHaveBeenCalledTimes(1); // primary /issues/ request expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with default query data: expect.stringContaining('is%3Aunresolved'), }) ); }); it('loads with query in URL and pinned queries', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ savedSearch, SearchFixture({ id: '123', name: 'My Pinned Search', isPinned: true, query: 'is:resolved', }), ], }); const location = LocationFixture({query: {query: 'level:foo'}}); render(, { router: {location}, }); await waitFor(() => { // Main /issues/ request expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with default query data: expect.stringContaining('level%3Afoo'), }) ); }); expect(screen.getByRole('row', {name: 'level:foo'})).toBeInTheDocument(); // Tab shows "custom search" expect(screen.getByRole('button', {name: 'Custom Search'})).toBeInTheDocument(); }); it('loads with a pinned custom query', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ savedSearch, SearchFixture({ id: '123', name: 'My Pinned Search', isPinned: true, isGlobal: false, query: 'is:resolved', }), ], }); render(, {router}); await waitFor(() => { expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with default query data: expect.stringContaining('is%3Aresolved'), }) ); }); expect(screen.getByRole('row', {name: 'is:resolved'})).toBeInTheDocument(); // Organization saved search selector should have default saved search selected expect(screen.getByRole('button', {name: 'My Default Search'})).toBeInTheDocument(); }); it('loads with a saved query', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ SearchFixture({ id: '123', name: 'Assigned to Me', isPinned: false, isGlobal: true, query: 'assigned:me', sort: 'trends', type: 0, }), ], }); const localRouter = {params: {searchId: '123'}}; render(, { router: localRouter, }); await waitFor(() => { expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with default query data: expect.stringContaining('assigned%3Ame') && expect.stringContaining('sort=trends'), }) ); }); expect(screen.getByRole('row', {name: 'assigned:me'})).toBeInTheDocument(); // Organization saved search selector should have default saved search selected expect(screen.getByRole('button', {name: 'Assigned to Me'})).toBeInTheDocument(); }); it('loads with a query in URL', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ SearchFixture({ id: '123', name: 'Assigned to Me', isPinned: false, isGlobal: true, query: 'assigned:me', type: 0, }), ], }); const localRouter = {location: {query: {query: 'level:error'}}}; render(, { router, }); await waitFor(() => { expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with default query data: expect.stringContaining('level%3Aerror'), }) ); }); expect(screen.getByRole('row', {name: 'level:error'})).toBeInTheDocument(); // Organization saved search selector should have default saved search selected expect(screen.getByRole('button', {name: 'Custom Search'})).toBeInTheDocument(); }); it('loads with an empty query in URL', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ SearchFixture({ id: '123', name: 'My Pinned Search', isPinned: true, isGlobal: false, query: 'is:resolved', }), ], }); render( , {router} ); await waitFor(() => { expect(issuesRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ // Should be called with empty query data: expect.stringContaining(''), }) ); }); expect(screen.getByRole('row', {name: 'is:resolved'})).toBeInTheDocument(); // Organization saved search selector should have default saved search selected expect(screen.getByRole('button', {name: 'My Default Search'})).toBeInTheDocument(); }); it('caches the search results', async function () { issuesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [...new Array(25)].map((_, i) => ({id: i})), headers: { Link: DEFAULT_LINKS_HEADER, 'X-Hits': '500', 'X-Max-Hits': '1000', }, }); const defaultProps = { ...props, ...routerProps, useOrgSavedSearches: true, selection: { projects: [], environments: [], datetime: {period: '14d'}, }, organization: OrganizationFixture({ features: ['issue-stream-performance'], }), }; const {unmount} = render(, { router, organization: defaultProps.organization, }); expect( await screen.findByText(textWithMarkupMatcher('1-25 of 500')) ).toBeInTheDocument(); expect(issuesRequest).toHaveBeenCalledTimes(1); unmount(); // Mount component again, getting from cache render(, { router, organization: defaultProps.organization, }); expect( await screen.findByText(textWithMarkupMatcher('1-25 of 500')) ).toBeInTheDocument(); expect(issuesRequest).toHaveBeenCalledTimes(1); }); it('1 search', async function () { const localSavedSearch = {...savedSearch, projectId: null}; savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [localSavedSearch], }); render(, { router, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); await userEvent.click(screen.getByRole('button', {name: /custom search/i})); await userEvent.click(screen.getByRole('button', {name: localSavedSearch.name})); expect(browserHistory.push).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/searches/789/', }) ); }); it('clears a saved search when a custom one is entered', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [ savedSearch, SearchFixture({ id: '123', name: 'Pinned search', isPinned: true, isGlobal: false, query: 'is:resolved', }), ], }); render(, { router, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); await screen.findByRole('grid', {name: 'Create a search query'}); await userEvent.click(screen.getByRole('button', {name: 'Clear search query'})); await userEvent.click(getSearchInput()); await userEvent.keyboard('dogs{Enter}'); expect(browserHistory.push).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/', query: { environment: [], project: [], referrer: 'issue-list', sort: '', query: 'dogs', statsPeriod: '14d', }, }) ); }); it('pins a custom query', async function () { const pinnedSearch = { id: '666', name: 'My Pinned Search', query: 'assigned:me level:fatal', sort: 'date', isPinned: true, visibility: SavedSearchVisibility.ORGANIZATION, }; savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch], }); const {rerender} = render(, { router, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); await screen.findByRole('grid', {name: 'Create a search query'}); await userEvent.click(screen.getByRole('button', {name: 'Clear search query'})); await userEvent.click(getSearchInput()); await userEvent.paste('assigned:me level:fatal'); await userEvent.keyboard('{Enter}'); expect(browserHistory.push as jest.Mock).toHaveBeenCalledWith( expect.objectContaining({ query: expect.objectContaining({ query: 'assigned:me level:fatal', }), }) ); await tick(); const routerWithQuery = {location: {query: {query: 'assigned:me level:fatal'}}}; rerender(); expect(screen.getByRole('button', {name: 'Custom Search'})).toBeInTheDocument(); MockApiClient.clearMockResponses(); const createPin = MockApiClient.addMockResponse({ url: '/organizations/org-slug/pinned-searches/', method: 'PUT', body: pinnedSearch, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch, pinnedSearch], }); await userEvent.click(screen.getByLabelText(/Set as Default/i)); await waitFor(() => { expect(createPin).toHaveBeenCalled(); expect(browserHistory.replace).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/searches/666/', query: { referrer: 'search-bar', }, search: '', }) ); }); }); it('unpins a custom query', async function () { const pinnedSearch = SearchFixture({ id: '666', name: 'My Pinned Search', query: 'assigned:me level:fatal', sort: 'date', isPinned: true, visibility: SavedSearchVisibility.ORGANIZATION, }); savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [pinnedSearch], }); const deletePin = MockApiClient.addMockResponse({ url: '/organizations/org-slug/pinned-searches/', method: 'DELETE', }); const routerWithSavedSearch = { params: {searchId: pinnedSearch.id}, }; render(, { router: routerWithSavedSearch, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); expect(screen.getByRole('button', {name: 'My Default Search'})).toBeInTheDocument(); await userEvent.click(screen.getByLabelText(/Remove Default/i)); await waitFor(() => { expect(deletePin).toHaveBeenCalled(); }); await waitFor(() => { expect(browserHistory.replace).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/', }) ); }); }); it('pins a saved query', async function () { const assignedToMe = SearchFixture({ id: '234', name: 'Assigned to Me', isPinned: false, isGlobal: true, query: 'assigned:me', sort: 'date', type: 0, }); savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch, assignedToMe], }); const createPin = MockApiClient.addMockResponse({ url: '/organizations/org-slug/pinned-searches/', method: 'PUT', body: { ...savedSearch, isPinned: true, }, }); const routerWithSavedSearch = {params: {searchId: '789'}}; render(, { router: routerWithSavedSearch, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); expect(screen.getByRole('button', {name: savedSearch.name})).toBeInTheDocument(); await userEvent.click(screen.getByLabelText(/set as default/i)); await waitFor(() => { expect(createPin).toHaveBeenCalled(); expect(browserHistory.replace).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/searches/789/', }) ); }); }); it('pinning search should keep project selected', async function () { savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [savedSearch], }); const {router: newRouter} = initializeOrg({ router: { location: { query: { project: ['123'], environment: ['prod'], query: 'assigned:me level:fatal', }, }, }, }); render( , {router: newRouter} ); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); const createPin = MockApiClient.addMockResponse({ url: '/organizations/org-slug/pinned-searches/', method: 'PUT', body: { ...savedSearch, id: '666', name: 'My Pinned Search', query: 'assigned:me level:fatal', sort: 'date', isPinned: true, }, }); await userEvent.click(screen.getByLabelText(/set as default/i)); await waitFor(() => { expect(createPin).toHaveBeenCalled(); expect(browserHistory.replace).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/searches/666/', query: expect.objectContaining({ project: ['123'], environment: ['prod'], query: 'assigned:me level:fatal', referrer: 'search-bar', }), }) ); }); }); it('unpinning search should keep project selected', async function () { const localSavedSearch = { ...savedSearch, id: '666', isPinned: true, query: 'assigned:me level:fatal', }; savedSearchesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/searches/', body: [localSavedSearch], }); const deletePin = MockApiClient.addMockResponse({ url: '/organizations/org-slug/pinned-searches/', method: 'DELETE', }); const {router: newRouter} = initializeOrg( merge({}, router, { router: { location: { query: { project: ['123'], environment: ['prod'], query: 'assigned:me level:fatal', }, }, params: {searchId: '666'}, }, }) ); render( , {router: newRouter} ); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); await userEvent.click(screen.getByLabelText(/Remove Default/i)); await waitFor(() => { expect(deletePin).toHaveBeenCalled(); expect(browserHistory.replace).toHaveBeenLastCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/issues/', query: expect.objectContaining({ project: ['123'], environment: ['prod'], query: 'assigned:me level:fatal', referrer: 'search-bar', }), }) ); }); }); it('does not allow pagination to "previous" while on first page and resets cursors when navigating back to initial page', async function () { const {rerender} = render(, { router, }); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); expect(screen.getByRole('button', {name: 'Previous'})).toBeDisabled(); issuesRequest = MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [group], headers: { Link: '; rel="previous"; results="true"; cursor="1443575000:0:1", ; rel="next"; results="true"; cursor="1443574000:0:0"', }, }); await userEvent.click(screen.getByRole('button', {name: 'Next'})); let pushArgs = { pathname: '/organizations/org-slug/issues/', query: { cursor: '1443575000:0:0', page: 1, environment: [], project: [], query: DEFAULT_QUERY, statsPeriod: '14d', referrer: 'issue-list', }, }; await waitFor(() => { expect(browserHistory.push).toHaveBeenLastCalledWith(pushArgs); }); rerender(); expect(screen.getByRole('button', {name: 'Previous'})).toBeEnabled(); // Click next again await userEvent.click(screen.getByRole('button', {name: 'Next'})); pushArgs = { pathname: '/organizations/org-slug/issues/', query: { cursor: '1443574000:0:0', page: 2, environment: [], project: [], query: DEFAULT_QUERY, statsPeriod: '14d', referrer: 'issue-list', }, }; await waitFor(() => { expect(browserHistory.push).toHaveBeenLastCalledWith(pushArgs); }); rerender(); // Click previous await userEvent.click(screen.getByRole('button', {name: 'Previous'})); pushArgs = { pathname: '/organizations/org-slug/issues/', query: { cursor: '1443575000:0:1', page: 1, environment: [], project: [], query: DEFAULT_QUERY, statsPeriod: '14d', referrer: 'issue-list', }, }; await waitFor(() => { expect(browserHistory.push).toHaveBeenLastCalledWith(pushArgs); }); rerender(); // Click previous back to initial page await userEvent.click(screen.getByRole('button', {name: 'Previous'})); await waitFor(() => { // cursor is undefined because "prev" cursor is === initial "next" cursor expect(browserHistory.push).toHaveBeenLastCalledWith({ pathname: '/organizations/org-slug/issues/', query: { cursor: undefined, environment: [], page: undefined, project: [], query: DEFAULT_QUERY, statsPeriod: '14d', referrer: 'issue-list', }, }); }); }); }); describe('transitionTo', function () { it('pushes to history when query is updated', async function () { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [], headers: { Link: DEFAULT_LINKS_HEADER, }, }); render(, { router, }); await userEvent.click(screen.getByRole('button', {name: 'Clear search query'})); await userEvent.click(getSearchInput()); await userEvent.paste('is:ignored'); await userEvent.keyboard('{enter}'); await waitFor(() => { expect(browserHistory.push).toHaveBeenCalledWith({ pathname: '/organizations/org-slug/issues/', query: { environment: [], project: [parseInt(project.id, 10)], query: 'is:ignored', statsPeriod: '14d', referrer: 'issue-list', }, }); }); }); }); it('fetches tags and members', async function () { render(, {router}); await waitFor(() => { expect(fetchMembersRequest).toHaveBeenCalled(); }); }); describe('componentDidUpdate fetching groups', function () { let fetchDataMock: jest.Mock; beforeEach(function () { fetchDataMock = MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [group], headers: { Link: DEFAULT_LINKS_HEADER, }, }); fetchDataMock.mockReset(); }); it('fetches data on selection change', async function () { const {rerender} = render(, { router, }); rerender( ); await waitFor(() => { expect(fetchDataMock).toHaveBeenCalled(); }); }); it('fetches data on savedSearch change', async function () { const {rerender} = render(, { router, }); rerender( ); await waitFor(() => { expect(fetchDataMock).toHaveBeenCalled(); }); }); it('uses correct statsPeriod when fetching issues list and no datetime given', async function () { const {rerender} = render(, { router, }); const selection = {projects: [99], environments: [], datetime: {}}; rerender(); await waitFor(() => { expect(fetchDataMock).toHaveBeenLastCalledWith( '/organizations/org-slug/issues/', expect.objectContaining({ data: 'collapse=stats&collapse=unhandled&expand=owners&expand=inbox&limit=25&project=99&query=is%3Aunresolved%20issue.priority%3A%5Bhigh%2C%20medium%5D&savedSearch=1&shortIdLookup=1&statsPeriod=14d', }) ); }); }); }); describe('componentDidUpdate fetching members', function () { it('fetches memberlist on project change', async function () { const {rerender} = render(, { router, }); // Called during componentDidMount await waitFor(() => { expect(fetchMembersRequest).toHaveBeenCalled(); }); const selection = { projects: [99], environments: [], datetime: {period: '24h'}, }; rerender(); await waitFor(() => { expect(fetchMembersRequest).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ query: { project: selection.projects.map(p => p.toString()), }, }) ); }); }); }); describe('render states', function () { it('displays the loading icon when saved searches are loading', async function () { render(, { router, }); expect(await screen.findByTestId('loading-indicator')).toBeInTheDocument(); }); it('displays an error when issues fail to load', async function () { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', status: 500, statusCode: 500, }); render(, { router, }); expect(await screen.findByTestId('loading-error')).toBeInTheDocument(); }); it('displays congrats robots animation with only default query', async function () { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [], headers: { Link: DEFAULT_LINKS_HEADER, }, }); render(, {router}); expect( await screen.findByText(/We couldn't find any issues that matched your filters/i) ).toBeInTheDocument(); }); it('displays an empty resultset with a non-default query', async function () { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [], headers: { Link: DEFAULT_LINKS_HEADER, }, }); render(, {router}); await screen.findByRole('grid', {name: 'Create a search query'}); await userEvent.click(getSearchInput()); await userEvent.keyboard('foo{enter}'); expect( await screen.findByText(/We couldn't find any issues that matched your filters/i) ).toBeInTheDocument(); }); }); describe('Error Robot', function () { const createWrapper = async (moreProps: any) => { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [], headers: { Link: DEFAULT_LINKS_HEADER, }, }); const defaultProps = { ...props, useOrgSavedSearches: true, selection: { projects: [], environments: [], datetime: {period: '14d'}, }, ...merge({}, routerProps, { params: {}, location: {query: {query: DEFAULT_QUERY}, search: 'query=is:unresolved'}, }), organization: OrganizationFixture(), ...moreProps, }; render(, {router}); await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator')); }; it('displays when no projects selected and all projects user is member of, async does not have first event', async function () { const projectsBody = [ ProjectFixture({ id: '1', slug: 'foo', isMember: true, firstEvent: null, }), ProjectFixture({ id: '2', slug: 'bar', isMember: true, firstEvent: null, }), ProjectFixture({ id: '3', slug: 'baz', isMember: true, firstEvent: null, }), ]; MockApiClient.addMockResponse({ url: '/organizations/org-slug/sent-first-event/', query: { is_member: true, }, body: {sentFirstEvent: false}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: projectsBody, }); MockApiClient.addMockResponse({ url: '/projects/org-slug/foo/issues/', body: [], }); await createWrapper({ organization: OrganizationFixture(), }); expect(await screen.findByTestId('awaiting-events')).toBeInTheDocument(); }); it('does not display when no projects selected and any projects have a first event', async function () { const projectsBody = [ ProjectFixture({ id: '1', slug: 'foo', isMember: true, firstEvent: null, }), ProjectFixture({ id: '2', slug: 'bar', isMember: true, firstEvent: new Date().toISOString(), }), ProjectFixture({ id: '3', slug: 'baz', isMember: true, firstEvent: null, }), ]; MockApiClient.addMockResponse({ url: '/organizations/org-slug/sent-first-event/', query: { is_member: true, }, body: {sentFirstEvent: true}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: projectsBody, }); await createWrapper({ organization: OrganizationFixture(), }); expect(screen.queryByTestId('awaiting-events')).not.toBeInTheDocument(); }); it('displays when all selected projects do not have first event', async function () { const projectsBody = [ ProjectFixture({ id: '1', slug: 'foo', isMember: true, firstEvent: null, }), ProjectFixture({ id: '2', slug: 'bar', isMember: true, firstEvent: null, }), ProjectFixture({ id: '3', slug: 'baz', isMember: true, firstEvent: null, }), ]; MockApiClient.addMockResponse({ url: '/organizations/org-slug/sent-first-event/', query: { project: [1, 2], }, body: {sentFirstEvent: false}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: projectsBody, }); MockApiClient.addMockResponse({ url: '/projects/org-slug/foo/issues/', body: [], }); await createWrapper({ selection: { projects: [1, 2], environments: [], datetime: {period: '14d'}, }, organization: OrganizationFixture(), }); expect(await screen.findByTestId('awaiting-events')).toBeInTheDocument(); }); it('does not display when any selected projects have first event', async function () { const projectsBody = [ ProjectFixture({ id: '1', slug: 'foo', isMember: true, firstEvent: null, }), ProjectFixture({ id: '2', slug: 'bar', isMember: true, firstEvent: new Date().toISOString(), }), ProjectFixture({ id: '3', slug: 'baz', isMember: true, firstEvent: new Date().toISOString(), }), ]; MockApiClient.addMockResponse({ url: '/organizations/org-slug/sent-first-event/', query: { project: [1, 2], }, body: {sentFirstEvent: true}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: projectsBody, }); await createWrapper({ selection: { projects: [1, 2], environments: [], datetime: {period: '14d'}, }, organization: OrganizationFixture(), }); expect(screen.queryByTestId('awaiting-events')).not.toBeInTheDocument(); }); }); it('displays a count that represents the current page', async function () { MockApiClient.addMockResponse({ url: '/organizations/org-slug/issues/', body: [...new Array(25)].map((_, i) => ({id: i})), headers: { Link: DEFAULT_LINKS_HEADER, 'X-Hits': '500', 'X-Max-Hits': '1000', }, }); parseLinkHeaderSpy.mockReturnValue({ next: { results: true, cursor: '', href: '', }, previous: { results: false, cursor: '', href: '', }, }); props = { ...props, location: { query: { cursor: 'some cursor', page: 1, }, }, }; const {router: newRouter} = initializeOrg(); const {rerender} = render(, { router: newRouter, }); await waitFor(() => { expect(screen.getByText(textWithMarkupMatcher('1-25 of 500'))).toBeInTheDocument(); }); parseLinkHeaderSpy.mockReturnValue({ next: { results: true, cursor: '', href: '', }, previous: { results: true, cursor: '', href: '', }, }); rerender(); await waitFor(() => { expect(screen.getByText(textWithMarkupMatcher('26-50 of 500'))).toBeInTheDocument(); }); }); describe('project low trends queue alert', function () { const {router: newRouter} = initializeOrg(); beforeEach(function () { act(() => ProjectsStore.reset()); }); it('does not render event processing alert', async function () { act(() => ProjectsStore.loadInitialData([project])); render(, { router: newRouter, }); await waitFor(() => { expect(screen.queryByText(/event processing/i)).not.toBeInTheDocument(); }); }); describe('renders alert', function () { it('for one project', async function () { act(() => ProjectsStore.loadInitialData([ {...project, eventProcessing: {symbolicationDegraded: true}}, ]) ); render(, {router}); await waitFor(() => { expect( screen.getByText(/Event Processing for this project is currently degraded/i) ).toBeInTheDocument(); }); }); it('for multiple projects', async function () { const projectBar = ProjectFixture({ id: '3560', name: 'Bar Project', slug: 'project-slug-bar', }); act(() => ProjectsStore.loadInitialData([ { ...project, slug: 'project-slug', eventProcessing: {symbolicationDegraded: true}, }, { ...projectBar, slug: 'project-slug-bar', eventProcessing: {symbolicationDegraded: true}, }, ]) ); render( , { router: newRouter, } ); await waitFor(() => { expect( screen.getByText( textWithMarkupMatcher( 'Event Processing for the project-slug, project-slug-bar projects is currently degraded.' ) ) ).toBeInTheDocument(); }); }); }); }); });