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();
});
});