import { initializeData as _initializeData, InitializeDataSettings, } from 'sentry-test/performance/initializePerformanceData'; import {act, render, screen, userEvent} from 'sentry-test/reactTestingLibrary'; import {MetricsCardinalityProvider} from 'sentry/utils/performance/contexts/metricsCardinality'; import {MEPSettingProvider} from 'sentry/utils/performance/contexts/metricsEnhancedSetting'; import { PageErrorAlert, PageErrorProvider, } from 'sentry/utils/performance/contexts/pageError'; import {PerformanceDisplayProvider} from 'sentry/utils/performance/contexts/performanceDisplayContext'; import {OrganizationContext} from 'sentry/views/organizationContext'; import WidgetContainer from 'sentry/views/performance/landing/widgets/components/widgetContainer'; import {PerformanceWidgetSetting} from 'sentry/views/performance/landing/widgets/widgetDefinitions'; import {ProjectPerformanceType} from 'sentry/views/performance/utils'; const initializeData = (query = {}, rest: InitializeDataSettings = {}) => { const data = _initializeData({ query: {statsPeriod: '7d', environment: ['prod'], project: [-42], ...query}, ...rest, }); data.eventView.additionalConditions.addFilterValues('transaction.op', ['pageload']); return data; }; function WrappedComponent({data, withStaticFilters = false, ...rest}) { return ( ); } const issuesPredicate = (url, options) => url.includes('events') && options.query?.query.includes('error'); describe('Performance > Widgets > WidgetContainer', function () { let wrapper; let eventStatsMock; let eventsTrendsStats; let eventsMock; let issuesListMock; beforeEach(function () { eventStatsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events-stats/`, body: [], }); eventsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events/`, body: { data: [{}], meta: {}, }, match: [(...args) => !issuesPredicate(...args)], }); issuesListMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events/`, body: { data: [ { 'issue.id': 123, transaction: '/issue/:id/', title: 'Error: Something is broken.', 'project.id': 1, count: 3100, issue: 'JAVASCRIPT-ABCD', }, ], }, match: [(...args) => issuesPredicate(...args)], }); eventsTrendsStats = MockApiClient.addMockResponse({ method: 'GET', url: '/organizations/org-slug/events-trends-stats/', body: [], }); 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 () { if (wrapper) { wrapper.unmount(); wrapper = undefined; } }); it('Check requests when changing widget props', function () { const data = initializeData(); wrapper = render( ); expect(eventStatsMock).toHaveBeenCalledTimes(1); // Change eventView reference data.eventView = data.eventView.clone(); wrapper.rerender( ); expect(eventStatsMock).toHaveBeenCalledTimes(1); // Change eventView statsperiod const modifiedData = initializeData({ statsPeriod: '14d', }); wrapper.rerender( ); expect(eventStatsMock).toHaveBeenCalledTimes(2); expect(eventStatsMock).toHaveBeenNthCalledWith( 2, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ interval: '1h', partial: '1', query: 'transaction.op:pageload', statsPeriod: '28d', yAxis: 'tpm()', }), }) ); }); it('Check requests when changing widget props for GenericDiscoverQuery based widget', function () { const data = initializeData(); wrapper = render( ); expect(eventsTrendsStats).toHaveBeenCalledTimes(1); // Change eventView reference data.eventView = data.eventView.clone(); wrapper.rerender( ); expect(eventsTrendsStats).toHaveBeenCalledTimes(1); // Change eventView statsperiod const modifiedData = initializeData({ statsPeriod: '14d', }); wrapper.rerender( ); expect(eventsTrendsStats).toHaveBeenCalledTimes(2); expect(eventsTrendsStats).toHaveBeenNthCalledWith( 2, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ cursor: '0:0:1', environment: ['prod'], field: ['transaction', 'project'], interval: undefined, middle: undefined, noPagination: true, per_page: 3, project: ['-42'], query: 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6', sort: 'trend_percentage()', statsPeriod: '14d', trendFunction: 'avg(transaction.duration)', trendType: 'improved', }), }) ); }); it('should call PageError Provider when errors are present', async function () { const data = initializeData(); eventStatsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events-stats/`, statusCode: 400, body: { detail: 'Request did not work :(', }, }); wrapper = render( ); // Provider update is after request promise. await act(async () => {}); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Transactions Per Minute' ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(await screen.findByTestId('page-error-alert')).toHaveTextContent( 'Request did not work :(' ); }); it('TPM Widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Transactions Per Minute' ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ interval: '1h', partial: '1', query: 'transaction.op:pageload', statsPeriod: '14d', yAxis: 'tpm()', }), }) ); }); it('Failure Rate Widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Failure Rate' ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ interval: '1h', partial: '1', query: 'transaction.op:pageload', statsPeriod: '14d', yAxis: 'failure_rate()', }), }) ); }); it('Widget with MEP enabled and metric meta set to true', async function () { const data = initializeData( {}, { features: ['performance-use-metrics'], } ); eventStatsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events-stats/`, body: { data: [], isMetricsData: true, }, }); eventsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events/`, body: { data: [{}], meta: {isMetricsData: true}, }, }); wrapper = render( ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({dataset: 'metrics'}), }) ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ query: expect.objectContaining({dataset: 'metrics'}), }) ); expect(await screen.findByTestId('has-metrics-data-tag')).toHaveTextContent( 'processed' ); }); it('Widget with MEP enabled and metric meta set to undefined', async function () { const data = initializeData( {}, { features: ['performance-use-metrics'], } ); eventStatsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events-stats/`, body: { data: [], isMetricsData: undefined, }, }); wrapper = render( ); expect(await screen.findByTestId('no-metrics-data-tag')).toBeInTheDocument(); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({dataset: 'metrics'}), }) ); }); it('Widget with MEP enabled and metric meta set to false', async function () { const data = initializeData( {}, { features: ['performance-use-metrics'], } ); eventStatsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events-stats/`, body: { data: [], isMetricsData: false, }, }); eventsMock = MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/events/`, body: { data: [{}], meta: {isMetricsData: false}, }, }); wrapper = render( ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({dataset: 'metrics'}), }) ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenCalledWith( expect.anything(), expect.objectContaining({ query: expect.objectContaining({dataset: 'metrics'}), }) ); expect(await screen.findByTestId('has-metrics-data-tag')).toHaveTextContent( 'indexed' ); }); it('User misery Widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'User Misery' ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ interval: '1h', partial: '1', query: 'transaction.op:pageload', statsPeriod: '14d', yAxis: 'user_misery()', }), }) ); }); it('Worst LCP widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Worst LCP Web Vitals' ); expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All'); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: [ 'transaction', 'title', 'project.id', 'count_web_vitals(measurements.lcp, poor)', 'count_web_vitals(measurements.lcp, meh)', 'count_web_vitals(measurements.lcp, good)', ], per_page: 4, project: ['-42'], query: 'transaction.op:pageload', sort: '-count_web_vitals(measurements.lcp, poor)', statsPeriod: '7d', }), }) ); }); it('Worst LCP widget - MEP', async function () { const data = initializeData( {}, { features: ['performance-use-metrics'], } ); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Worst LCP Web Vitals' ); expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All'); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: [ 'transaction', 'title', 'project.id', 'count_web_vitals(measurements.lcp, poor)', 'count_web_vitals(measurements.lcp, meh)', 'count_web_vitals(measurements.lcp, good)', ], per_page: 4, project: ['-42'], query: 'transaction.op:pageload !transaction:"<< unparameterized >>"', sort: '-count_web_vitals(measurements.lcp, poor)', statsPeriod: '7d', }), }) ); }); it('Worst FCP widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Worst FCP Web Vitals' ); expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All'); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: [ 'transaction', 'title', 'project.id', 'count_web_vitals(measurements.fcp, poor)', 'count_web_vitals(measurements.fcp, meh)', 'count_web_vitals(measurements.fcp, good)', ], per_page: 4, project: ['-42'], query: 'transaction.op:pageload', sort: '-count_web_vitals(measurements.fcp, poor)', statsPeriod: '7d', }), }) ); }); it('Worst FID widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Worst FID Web Vitals' ); expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All'); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: [ 'transaction', 'title', 'project.id', 'count_web_vitals(measurements.fid, poor)', 'count_web_vitals(measurements.fid, meh)', 'count_web_vitals(measurements.fid, good)', ], per_page: 4, project: ['-42'], query: 'transaction.op:pageload', sort: '-count_web_vitals(measurements.fid, poor)', statsPeriod: '7d', }), }) ); }); it('LCP Histogram Widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'LCP Distribution' ); // TODO(k-fish): Add histogram mock }); it('FCP Histogram Widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'FCP Distribution' ); // TODO(k-fish): Add histogram mock }); it('Most errors widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Related Errors' ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: ['transaction', 'project.id', 'failure_count()'], per_page: 3, project: ['-42'], query: 'transaction.op:pageload failure_count():>0', sort: '-failure_count()', statsPeriod: '7d', }), }) ); }); it('Most related issues widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Related Issues' ); expect(issuesListMock).toHaveBeenCalledTimes(1); expect(issuesListMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: ['issue', 'transaction', 'title', 'project.id', 'count()'], per_page: 3, project: ['-42'], query: 'event.type:error !tags[transaction]:"" count():>0', sort: '-count()', statsPeriod: '7d', }), }) ); }); it('Switching from issues to errors widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Related Issues' ); expect(issuesListMock).toHaveBeenCalledTimes(1); wrapper.rerender( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Related Errors' ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventStatsMock).toHaveBeenCalledTimes(1); }); it('Most improved trends widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Improved' ); expect(eventsTrendsStats).toHaveBeenCalledTimes(1); expect(eventsTrendsStats).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: ['transaction', 'project'], interval: undefined, middle: undefined, per_page: 3, project: ['-42'], query: 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6', sort: 'trend_percentage()', statsPeriod: '7d', trendFunction: 'avg(transaction.duration)', trendType: 'improved', }), }) ); }); it('Most regressed trends widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Regressed' ); expect(eventsTrendsStats).toHaveBeenCalledTimes(1); expect(eventsTrendsStats).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: ['prod'], field: ['transaction', 'project'], interval: undefined, middle: undefined, per_page: 3, project: ['-42'], query: 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6', sort: '-trend_percentage()', statsPeriod: '7d', trendFunction: 'avg(transaction.duration)', trendType: 'regression', }), }) ); }); it('Most slow frames widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Slow Frames' ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ cursor: '0:0:1', environment: ['prod'], field: ['transaction', 'project.id', 'epm()', 'avg(measurements.frames_slow)'], noPagination: true, per_page: 3, project: ['-42'], query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_slow):>0', sort: '-avg(measurements.frames_slow)', statsPeriod: '7d', }), }) ); expect(await screen.findByTestId('empty-state')).toBeInTheDocument(); }); it('Most slow frames widget - MEP', async function () { const data = initializeData( {}, { features: ['performance-use-metrics'], } ); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Slow Frames' ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ cursor: '0:0:1', environment: ['prod'], field: ['transaction', 'project.id', 'epm()', 'avg(measurements.frames_slow)'], noPagination: true, per_page: 3, project: ['-42'], query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_slow):>0', sort: '-avg(measurements.frames_slow)', statsPeriod: '7d', }), }) ); expect(await screen.findByTestId('empty-state')).toBeInTheDocument(); }); it('Most frozen frames widget', async function () { const data = initializeData(); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Most Frozen Frames' ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ cursor: '0:0:1', environment: ['prod'], field: [ 'transaction', 'project.id', 'epm()', 'avg(measurements.frames_frozen)', ], noPagination: true, per_page: 3, project: ['-42'], query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_frozen):>0', sort: '-avg(measurements.frames_frozen)', statsPeriod: '7d', }), }) ); expect(await screen.findByTestId('empty-state')).toBeInTheDocument(); }); it('Able to change widget type from menu', async function () { const data = initializeData(); const setRowChartSettings = jest.fn(() => {}); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Failure Rate' ); expect(eventStatsMock).toHaveBeenCalledTimes(1); expect(setRowChartSettings).toHaveBeenCalledTimes(0); await userEvent.click(await screen.findByLabelText('More')); await userEvent.click(await screen.findByText('User Misery')); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'User Misery' ); expect(eventStatsMock).toHaveBeenCalledTimes(2); expect(setRowChartSettings).toHaveBeenCalledTimes(1); }); it('Chart settings passed from the row are disabled in the menu', async function () { const data = initializeData(); const setRowChartSettings = jest.fn(() => {}); wrapper = render( ); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Failure Rate' ); // Open context menu await userEvent.click(await screen.findByLabelText('More')); // Check that the the "User Misery" option is disabled by clicking on it, // expecting that the selected option doesn't change const userMiseryOption = await screen.findByRole('option', {name: 'User Misery'}); await userEvent.click(userMiseryOption); expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent( 'Failure Rate' ); }); });