import {EventFixture} from 'sentry-fixture/event'; import {EventsStatsFixture} from 'sentry-fixture/events'; import {GroupFixture} from 'sentry-fixture/group'; import {LocationFixture} from 'sentry-fixture/locationFixture'; import {OrganizationFixture} from 'sentry-fixture/organization'; import {ProjectFixture} from 'sentry-fixture/project'; import {RouterFixture} from 'sentry-fixture/routerFixture'; import {TagsFixture} from 'sentry-fixture/tags'; import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary'; import PageFiltersStore from 'sentry/stores/pageFiltersStore'; import ProjectsStore from 'sentry/stores/projectsStore'; import {EventGraph} from 'sentry/views/issueDetails/streamline/eventGraph'; import {EventDetailsHeader} from './eventDetailsHeader'; describe('EventGraph', () => { const organization = OrganizationFixture(); const project = ProjectFixture({ environments: ['production', 'staging', 'development'], }); const group = GroupFixture(); const event = EventFixture({id: 'event-id'}); const persistantQuery = `issue:${group.shortId}`; const defaultProps = {group, event, project}; let mockEventStats: jest.Mock; beforeEach(() => { MockApiClient.clearMockResponses(); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/issues/${group.id}/tags/`, body: TagsFixture(), method: 'GET', }); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/releases/stats/`, body: [], }); PageFiltersStore.init(); PageFiltersStore.onInitializeUrlState( { projects: [], environments: [], datetime: {start: null, end: null, period: '14d', utc: null}, }, new Set(['environments']) ); ProjectsStore.loadInitialData([project]); mockEventStats = MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/events-stats/`, body: {'count()': EventsStatsFixture(), 'count_unique(user)': EventsStatsFixture()}, method: 'GET', }); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/events/`, body: {data: [{'count_unique(user)': 21}]}, }); }); it('displays allows toggling data sets', async function () { render(, {organization}); expect(await screen.findByTestId('event-graph-loading')).not.toBeInTheDocument(); const eventsToggle = screen.getByRole('button', { name: 'Toggle graph series - Events', }); const usersToggle = screen.getByRole('button', {name: 'Toggle graph series - Users'}); expect(eventsToggle).toHaveTextContent('444'); expect(usersToggle).toHaveTextContent('21'); // Defaults to events graph expect(eventsToggle).toBeDisabled(); expect(usersToggle).toBeEnabled(); // Switch to users graph await userEvent.click(usersToggle); expect(eventsToggle).toBeEnabled(); expect(usersToggle).toBeDisabled(); // Another click should do nothing await userEvent.click(usersToggle); expect(eventsToggle).toBeEnabled(); expect(usersToggle).toBeDisabled(); // Switch back to events await userEvent.click(eventsToggle); expect(eventsToggle).toBeDisabled(); expect(usersToggle).toBeEnabled(); }); it('renders the graph using a discover event stats query', async function () { render(, {organization}); expect(await screen.findByTestId('event-graph-loading')).not.toBeInTheDocument(); expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: { dataset: 'errors', environment: [], field: expect.anything(), partial: 1, interval: '4h', per_page: 50, project: [project.id], query: persistantQuery, referrer: 'issue_details.streamline_graph', statsPeriod: '14d', yAxis: ['count()', 'count_unique(user)'], }, }) ); }); it('allows filtering by environment, and shows unfiltered stats', async function () { render(, {organization}); expect(await screen.findByTestId('event-graph-loading')).not.toBeInTheDocument(); await userEvent.click(screen.getByRole('button', {name: 'All Envs'})); await userEvent.click(screen.getByRole('row', {name: 'production'})); expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: expect.objectContaining({ environment: ['production'], }), }) ); // Also makes request without environment filter expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: expect.objectContaining({ environment: [], }), }) ); }); it('updates query from location param change', async function () { const [tagKey, tagValue] = ['user.email', 'leander.rodrigues@sentry.io']; const locationQuery = { query: { query: `${tagKey}:${tagValue}`, }, }; const router = RouterFixture({ location: LocationFixture(locationQuery), }); render(, {organization, router}); expect(await screen.findByTestId('event-graph-loading')).not.toBeInTheDocument(); expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: expect.objectContaining({ query: [persistantQuery, locationQuery.query.query].join(' '), }), }) ); // Also makes request without tag filter expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: expect.objectContaining({ query: persistantQuery, }), }) ); }); it('allows filtering by date', async function () { render(, {organization}); expect(await screen.findByTestId('event-graph-loading')).not.toBeInTheDocument(); await userEvent.click(screen.getByRole('button', {name: '14D'})); await userEvent.click(await screen.findByRole('option', {name: 'Last 7 days'})); expect(mockEventStats).toHaveBeenCalledWith( '/organizations/org-slug/events-stats/', expect.objectContaining({ query: expect.objectContaining({ statsPeriod: '7d', }), }) ); }); it('displays error messages from bad queries', async function () { const errorMessage = 'wrong, try again'; const mockStats = MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/events-stats/`, body: {detail: errorMessage}, method: 'GET', statusCode: 400, }); render(, {organization}); await screen.findByRole('button', {name: '14D'}); expect(mockStats).toHaveBeenCalled(); expect(screen.getByText(RegExp(errorMessage))).toBeInTheDocument(); // Omit the graph expect(screen.queryByRole('figure')).not.toBeInTheDocument(); }); });