import {browserHistory} from 'react-router';
import {QueryClient, QueryClientProvider} from '@tanstack/react-query';
import {addMetricsDataMock} from 'sentry-test/performance/addMetricsDataMock';
import {initializeData} from 'sentry-test/performance/initializePerformanceData';
import {
act,
render,
screen,
userEvent,
waitFor,
waitForElementToBeRemoved,
} from 'sentry-test/reactTestingLibrary';
import TeamStore from 'sentry/stores/teamStore';
import {MetricsCardinalityProvider} from 'sentry/utils/performance/contexts/metricsCardinality';
import {OrganizationContext} from 'sentry/views/organizationContext';
import {generatePerformanceEventView} from 'sentry/views/performance/data';
import {PerformanceLanding} from 'sentry/views/performance/landing';
import {REACT_NATIVE_COLUMN_TITLES} from 'sentry/views/performance/landing/data';
import {LandingDisplayField} from 'sentry/views/performance/landing/utils';
const searchHandlerMock = jest.fn();
const WrappedComponent = ({data, withStaticFilters = false}) => {
const eventView = generatePerformanceEventView(
data.router.location,
data.projects,
{
withStaticFilters,
},
data.organization
);
const client = new QueryClient();
return (
{}}
setError={() => {}}
withStaticFilters={withStaticFilters}
/>
);
};
describe('Performance > Landing > Index', function () {
let eventStatsMock: jest.Mock;
let eventsMock: jest.Mock;
let wrapper: any;
act(() => void TeamStore.loadInitialData([], false, null));
beforeEach(function () {
// @ts-ignore no-console
// eslint-disable-next-line no-console
jest.spyOn(console, 'error').mockImplementation(jest.fn());
MockApiClient.addMockResponse({
url: '/organizations/org-slug/sdk-updates/',
body: [],
});
MockApiClient.addMockResponse({
url: '/prompts-activity/',
body: {},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [],
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/key-transactions-list/`,
body: [],
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/legacy-key-transactions-count/`,
body: [],
});
eventStatsMock = MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/events-stats/`,
body: [],
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/events-trends-stats/`,
body: [],
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/events-histogram/`,
body: [],
});
eventsMock = MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/events/`,
body: {
meta: {
fields: {
id: 'string',
},
},
data: [
{
id: '1234',
},
],
},
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/metrics-compatibility/`,
body: [],
});
MockApiClient.addMockResponse({
method: 'GET',
url: `/organizations/org-slug/metrics-compatibility-sums/`,
body: [],
});
});
afterEach(function () {
MockApiClient.clearMockResponses();
jest.clearAllMocks();
jest.restoreAllMocks();
if (wrapper) {
wrapper.unmount();
wrapper = undefined;
}
});
it('renders basic UI elements', function () {
const data = initializeData();
wrapper = render(, data.routerContext);
expect(screen.getByTestId('performance-landing-v3')).toBeInTheDocument();
});
it('renders frontend pageload view', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.FRONTEND_PAGELOAD},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('frontend-pageload-view')).toBeInTheDocument();
expect(screen.getByTestId('performance-table')).toBeInTheDocument();
const titles = screen.getAllByTestId('performance-widget-title');
expect(titles).toHaveLength(5);
expect(titles[0]).toHaveTextContent('Worst LCP Web Vitals');
expect(titles[1]).toHaveTextContent('Worst FCP Web Vitals');
expect(titles[2]).toHaveTextContent('p75 LCP');
expect(titles[3]).toHaveTextContent('LCP Distribution');
expect(titles[4]).toHaveTextContent('FCP Distribution');
});
it('renders frontend other view', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.FRONTEND_OTHER},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('performance-table')).toBeInTheDocument();
});
it('renders backend view', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.BACKEND},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('performance-table')).toBeInTheDocument();
});
it('renders mobile view', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.MOBILE},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('performance-table')).toBeInTheDocument();
});
it('renders react-native table headers in mobile view', async function () {
const project = TestStubs.Project({platform: 'react-native'});
const projects = [project];
const data = initializeData({
query: {landingDisplay: LandingDisplayField.MOBILE},
selectedProject: project.id,
projects,
});
wrapper = render(, data.routerContext);
expect(await screen.findByTestId('performance-table')).toBeInTheDocument();
expect(screen.getByTestId('grid-editable')).toBeInTheDocument();
const columnHeaders = await screen.findAllByTestId('grid-head-cell');
expect(columnHeaders).toHaveLength(REACT_NATIVE_COLUMN_TITLES.length);
for (const [index, title] of columnHeaders.entries()) {
expect(title).toHaveTextContent(REACT_NATIVE_COLUMN_TITLES[index]);
}
});
it('renders all transactions view', async function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.ALL},
});
wrapper = render(, data.routerContext);
expect(await screen.findByTestId('performance-table')).toBeInTheDocument();
await waitFor(() => expect(eventStatsMock).toHaveBeenCalledTimes(1)); // Only one request is made since the query batcher is working.
expect(eventStatsMock).toHaveBeenNthCalledWith(
1,
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
environment: [],
interval: '1h',
partial: '1',
project: [],
query: 'event.type:transaction',
referrer: 'api.performance.generic-widget-chart.user-misery-area',
statsPeriod: '14d',
yAxis: ['user_misery()', 'tpm()', 'failure_rate()'],
}),
})
);
expect(eventsMock).toHaveBeenCalledTimes(3);
const titles = await screen.findAllByTestId('performance-widget-title');
expect(titles).toHaveLength(5);
expect(titles.at(0)).toHaveTextContent('Most Regressed');
expect(titles.at(1)).toHaveTextContent('Most Related Issues');
expect(titles.at(2)).toHaveTextContent('User Misery');
expect(titles.at(3)).toHaveTextContent('Transactions Per Minute');
expect(titles.at(4)).toHaveTextContent('Failure Rate');
});
it('Can switch between landing displays', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.FRONTEND_PAGELOAD, abc: '123'},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('frontend-pageload-view')).toBeInTheDocument();
userEvent.click(screen.getByRole('tab', {name: 'All Transactions'}));
expect(browserHistory.push).toHaveBeenNthCalledWith(
1,
expect.objectContaining({
pathname: data.location.pathname,
query: {query: '', abc: '123'},
})
);
});
it('Updating projects switches performance view', function () {
const data = initializeData({
query: {landingDisplay: LandingDisplayField.FRONTEND_PAGELOAD},
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('frontend-pageload-view')).toBeInTheDocument();
const updatedData = initializeData({
projects: [TestStubs.Project({id: 123, platform: 'unknown'})],
selectedProject: 123,
});
wrapper.rerender(, data.routerContext);
expect(screen.getByTestId('all-transactions-view')).toBeInTheDocument();
});
it('View correctly defaults based on project without url param', function () {
const data = initializeData({
projects: [TestStubs.Project({id: 99, platform: 'javascript-react'})],
selectedProject: 99,
});
wrapper = render(, data.routerContext);
expect(screen.getByTestId('frontend-pageload-view')).toBeInTheDocument();
});
describe('With transaction search feature', function () {
it('does not search for empty string transaction', async function () {
const data = initializeData();
render(, data.routerContext);
await waitForElementToBeRemoved(() => screen.getByTestId('loading-indicator'));
userEvent.type(screen.getByPlaceholderText('Search Transactions'), '{enter}');
expect(searchHandlerMock).toHaveBeenCalledWith('', 'transactionsOnly');
});
it('renders the search bar', async function () {
addMetricsDataMock();
const data = initializeData({
query: {
field: 'test',
},
});
wrapper = render(
,
data.routerContext
);
expect(await screen.findByTestId('transaction-search-bar')).toBeInTheDocument();
});
it('extracts free text from the query', async function () {
const data = initializeData();
wrapper = render(, data.routerContext);
await waitForElementToBeRemoved(() => screen.queryByTestId('loading-indicator'));
expect(await screen.findByPlaceholderText('Search Transactions')).toHaveValue('');
});
});
describe('With span operations widget feature flag', function () {
it('Displays the span operations widget', async function () {
addMetricsDataMock();
const data = initializeData({
features: [
'performance-transaction-name-only-search',
'performance-new-widget-designs',
],
});
wrapper = render(, data.routerContext);
const titles = await screen.findAllByTestId('performance-widget-title');
expect(titles.at(0)).toHaveTextContent('Span Operations');
});
});
});