import {browserHistory} from 'react-router'; import {QueryClient, QueryClientProvider} from '@tanstack/react-query'; import {initializeOrg} from 'sentry-test/initializeOrg'; import {act, render, screen, userEvent} from 'sentry-test/reactTestingLibrary'; import * as pageFilters from 'sentry/actionCreators/pageFilters'; import OrganizationStore from 'sentry/stores/organizationStore'; import ProjectsStore from 'sentry/stores/projectsStore'; import TeamStore from 'sentry/stores/teamStore'; import {MEPSettingProvider} from 'sentry/utils/performance/contexts/metricsEnhancedSetting'; import {OrganizationContext} from 'sentry/views/organizationContext'; import PerformanceContent from 'sentry/views/performance/content'; import {DEFAULT_MAX_DURATION} from 'sentry/views/performance/trends/utils'; import {RouteContext} from 'sentry/views/routeContext'; const FEATURES = ['performance-view']; function WrappedComponent({organization, router}) { const client = new QueryClient(); return ( ); } function initializeData(projects, query, features = FEATURES) { const organization = TestStubs.Organization({ features, projects, }); const initialData = initializeOrg({ ...initializeOrg(), organization, router: { location: { pathname: '/test', query: query || {}, }, }, }); act(() => void OrganizationStore.onUpdate(initialData.organization, {replace: true})); act(() => ProjectsStore.loadInitialData(initialData.organization.projects)); return initialData; } function initializeTrendsData(query, addDefaultQuery = true) { const projects = [ TestStubs.Project({id: '1', firstTransactionEvent: false}), TestStubs.Project({id: '2', firstTransactionEvent: true}), ]; const organization = TestStubs.Organization({ FEATURES, projects, }); const otherTrendsQuery = addDefaultQuery ? { query: `tpm():>0.01 transaction.duration:>0 transaction.duration:<${DEFAULT_MAX_DURATION}`, } : {}; const initialData = initializeOrg({ ...initializeOrg(), organization, router: { location: { pathname: '/test', query: { ...otherTrendsQuery, ...query, }, }, }, }); act(() => ProjectsStore.loadInitialData(initialData.organization.projects)); return initialData; } describe('Performance > Content', function () { beforeEach(function () { act(() => void TeamStore.loadInitialData([], false, null)); browserHistory.push = jest.fn(); jest.spyOn(pageFilters, 'updateDateTime'); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/tags/', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-stats/', body: {data: [[123, []]]}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-histogram/', body: {'transaction.duration': [{bin: 0, count: 1000}]}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/users/', body: [], }); 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/sdk-updates/', body: [], }); MockApiClient.addMockResponse({ url: '/prompts-activity/', body: {}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events/', body: { meta: { fields: { user: 'string', transaction: 'string', 'project.id': 'integer', 'tpm()': 'number', 'p50()': 'number', 'p95()': 'number', 'failure_rate()': 'number', 'apdex(300)': 'number', 'count_unique(user)': 'number', 'count_miserable(user,300)': 'number', 'user_misery(300)': 'number', }, }, data: [ { transaction: '/apple/cart', 'project.id': 1, user: 'uhoh@example.com', 'tpm()': 30, 'p50()': 100, 'p95()': 500, 'failure_rate()': 0.1, 'apdex(300)': 0.6, 'count_unique(user)': 1000, 'count_miserable(user,300)': 122, 'user_misery(300)': 0.114, }, ], }, match: [ (_, options) => { if (!options.hasOwnProperty('query')) { return false; } if (!options.query?.hasOwnProperty('field')) { return false; } return !options.query?.field.includes('team_key_transaction'); }, ], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events/', body: { meta: { fields: { user: 'string', transaction: 'string', 'project.id': 'integer', 'tpm()': 'number', 'p50()': 'number', 'p95()': 'number', 'failure_rate()': 'number', 'apdex(300)': 'number', 'count_unique(user)': 'number', 'count_miserable(user,300)': 'number', 'user_misery(300)': 'number', }, }, data: [ { team_key_transaction: 1, transaction: '/apple/cart', 'project.id': 1, user: 'uhoh@example.com', 'tpm()': 30, 'p50()': 100, 'p95()': 500, 'failure_rate()': 0.1, 'apdex(300)': 0.6, 'count_unique(user)': 1000, 'count_miserable(user,300)': 122, 'user_misery(300)': 0.114, }, { team_key_transaction: 0, transaction: '/apple/checkout', 'project.id': 1, user: 'uhoh@example.com', 'tpm()': 30, 'p50()': 100, 'p95()': 500, 'failure_rate()': 0.1, 'apdex(300)': 0.6, 'count_unique(user)': 1000, 'count_miserable(user,300)': 122, 'user_misery(300)': 0.114, }, ], }, match: [ (_, options) => { if (!options.hasOwnProperty('query')) { return false; } if (!options.query?.hasOwnProperty('field')) { return false; } return options.query?.field.includes('team_key_transaction'); }, ], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-meta/', body: { count: 2, }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-trends/', body: { stats: {}, events: {meta: {}, data: []}, }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-trends-stats/', body: { stats: {}, events: {meta: {}, data: []}, }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/events-vitals/', body: { 'measurements.lcp': { poor: 1, meh: 2, good: 3, total: 6, p75: 4500, }, }, }); MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/key-transactions-list/`, body: [], }); }); afterEach(function () { MockApiClient.clearMockResponses(); act(() => ProjectsStore.reset()); // TODO: This was likely a defensive check added due to a previous isolation issue, it can possibly be removed. // @ts-ignore pageFilters.updateDateTime.mockRestore(); }); it('renders basic UI elements', async function () { const projects = [TestStubs.Project({firstTransactionEvent: true})]; const data = initializeData(projects, {}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(screen.getByTestId('performance-table')).toBeInTheDocument(); expect(screen.queryByText('Pinpoint problems')).not.toBeInTheDocument(); }); it('renders onboarding state when the selected project has no events', async function () { const projects = [ TestStubs.Project({id: 1, firstTransactionEvent: false}), TestStubs.Project({id: 2, firstTransactionEvent: true}), ]; const data = initializeData(projects, {project: [1]}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(screen.queryByText('Pinpoint problems')).toBeInTheDocument(); expect(screen.queryByTestId('performance-table')).not.toBeInTheDocument(); }); it('does not render onboarding for "my projects"', async function () { const projects = [ TestStubs.Project({id: '1', firstTransactionEvent: false}), TestStubs.Project({id: '2', firstTransactionEvent: true}), ]; const data = initializeData(projects, {project: ['-1']}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(screen.queryByText('Pinpoint problems')).not.toBeInTheDocument(); }); it('forwards conditions to transaction summary', async function () { const projects = [TestStubs.Project({id: '1', firstTransactionEvent: true})]; const data = initializeData(projects, {project: ['1'], query: 'sentry:yes'}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); const link = screen.getByRole('link', {name: '/apple/cart'}); userEvent.click(link); expect(data.router.push).toHaveBeenCalledWith( expect.objectContaining({ query: expect.objectContaining({ transaction: '/apple/cart', query: 'sentry:yes', }), }) ); }); it('Default period for trends does not call updateDateTime', async function () { const data = initializeTrendsData({query: 'tag:value'}, false); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(pageFilters.updateDateTime).toHaveBeenCalledTimes(0); }); it('Navigating to trends does not modify statsPeriod when already set', async function () { const data = initializeTrendsData({ query: `tpm():>0.005 transaction.duration:>10 transaction.duration:<${DEFAULT_MAX_DURATION} api`, statsPeriod: '24h', }); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); const link = screen.getByRole('button', {name: 'View Trends'}); userEvent.click(link); expect(pageFilters.updateDateTime).toHaveBeenCalledTimes(0); expect(browserHistory.push).toHaveBeenCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/performance/trends/', query: { query: `tpm():>0.005 transaction.duration:>10 transaction.duration:<${DEFAULT_MAX_DURATION}`, statsPeriod: '24h', }, }) ); }); it('Default page (transactions) without trends feature will not update filters if none are set', async function () { const projects = [ TestStubs.Project({id: 1, firstTransactionEvent: false}), TestStubs.Project({id: 2, firstTransactionEvent: true}), ]; const data = initializeData(projects, {view: undefined}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(browserHistory.push).toHaveBeenCalledTimes(0); }); it('Default page (transactions) with trends feature will not update filters if none are set', async function () { const data = initializeTrendsData({view: undefined}, false); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(browserHistory.push).toHaveBeenCalledTimes(0); }); it('Tags are replaced with trends default query if navigating to trends', async function () { const data = initializeTrendsData({query: 'device.family:Mac'}, false); render(, { context: data.routerContext, }); const trendsLinks = await screen.findAllByTestId('landing-header-trends'); userEvent.click(trendsLinks[0]); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(browserHistory.push).toHaveBeenCalledWith( expect.objectContaining({ pathname: '/organizations/org-slug/performance/trends/', query: { query: `tpm():>0.01 transaction.duration:>0 transaction.duration:<${DEFAULT_MAX_DURATION}`, }, }) ); }); it('Display Create Sample Transaction Button', async function () { const projects = [ TestStubs.Project({id: 1, firstTransactionEvent: false}), TestStubs.Project({id: 2, firstTransactionEvent: false}), ]; const data = initializeData(projects, {view: undefined}); render(, { context: data.routerContext, }); expect(await screen.findByTestId('performance-landing-v3')).toBeInTheDocument(); expect(screen.queryByTestId('create-sample-transaction-btn')).toBeInTheDocument(); }); });