import {browserHistory} from 'react-router'; import {initializeData as _initializeData} from 'sentry-test/performance/initializePerformanceData'; import {render, screen, userEvent, within} from 'sentry-test/reactTestingLibrary'; import ProjectsStore from 'sentry/stores/projectsStore'; import EventView from 'sentry/utils/discover/eventView'; import {MEPSettingProvider} from 'sentry/utils/performance/contexts/metricsEnhancedSetting'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import {OrganizationContext} from 'sentry/views/organizationContext'; import Table from 'sentry/views/performance/table'; const FEATURES = ['performance-view']; const initializeData = (settings = {}, features: string[] = []) => { const projects = [ TestStubs.Project({id: '1', slug: '1'}), TestStubs.Project({id: '2', slug: '2'}), ]; return _initializeData({ features: [...FEATURES, ...features], projects, selectedProject: projects[0], ...settings, }); }; const WrappedComponent = ({data, ...rest}) => { return ( ); }; function mockEventView(data) { const eventView = new EventView({ id: '1', name: 'my query', fields: [ { field: 'team_key_transaction', }, { field: 'transaction', }, { field: 'project', }, { field: 'tpm()', }, { field: 'p50()', }, { field: 'p95()', }, { field: 'failure_rate()', }, { field: 'apdex()', }, { field: 'count_unique(user)', }, { field: 'count_miserable(user)', }, { field: 'user_misery()', }, ], sorts: [{field: 'tpm ', kind: 'desc'}], query: 'event.type:transaction transaction:/api*', project: [data.projects[0].id, data.projects[1].id], start: '2019-10-01T00:00:00', end: '2019-10-02T00:00:00', statsPeriod: '14d', environment: [], additionalConditions: new MutableSearch(''), createdBy: undefined, interval: undefined, display: '', team: [], topEvents: undefined, yAxis: undefined, }); return eventView; } describe('Performance > Table', function () { let eventsMock; beforeEach(function () { browserHistory.push = jest.fn(); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', body: [], }); const eventsMetaFieldsMock = { user: 'string', transaction: 'string', project: 'string', tpm: 'number', p50: 'number', p95: 'number', failure_rate: 'number', apdex: 'number', count_unique_user: 'number', count_miserable_user: 'number', user_misery: 'number', }; const eventsBodyMock = [ { team_key_transaction: 1, transaction: '/apple/cart', project: '2', user: 'uhoh@example.com', tpm: 30, p50: 100, p95: 500, failure_rate: 0.1, apdex: 0.6, count_unique_user: 1000, count_miserable_user: 122, user_misery: 0.114, project_threshold_config: ['duration', 300], }, { team_key_transaction: 0, transaction: '/apple/checkout', project: '1', user: 'uhoh@example.com', tpm: 30, p50: 100, p95: 500, failure_rate: 0.1, apdex: 0.6, count_unique_user: 1000, count_miserable_user: 122, user_misery: 0.114, project_threshold_config: ['duration', 300], }, ]; eventsMock = MockApiClient.addMockResponse({ url: '/organizations/org-slug/events/', body: { meta: {fields: eventsMetaFieldsMock}, data: eventsBodyMock, }, }); MockApiClient.addMockResponse({ method: 'GET', url: `/organizations/org-slug/key-transactions-list/`, body: [], }); }); afterEach(function () { MockApiClient.clearMockResponses(); }); describe('with events', function () { it('renders correct cell actions without feature', async function () { const data = initializeData({ query: 'event.type:transaction transaction:/api*', }); ProjectsStore.loadInitialData(data.organization.projects); render( , {context: data.routerContext} ); const rows = await screen.findAllByTestId('grid-body-row'); const transactionCells = within(rows[0]).getAllByTestId('grid-body-cell'); const transactionCell = transactionCells[1]; const link = within(transactionCell).getByRole('link', {name: '/apple/cart'}); expect(link).toHaveAttribute( 'href', '/organizations/org-slug/performance/summary/?end=2019-10-02T00%3A00%3A00&project=2&query=&referrer=performance-transaction-summary&start=2019-10-01T00%3A00%3A00&statsPeriod=14d&transaction=%2Fapple%2Fcart&unselectedSeries=p100%28%29' ); const cellActionContainers = screen.getAllByTestId('cell-action-container'); expect(cellActionContainers).toHaveLength(18); // 9 cols x 2 rows userEvent.hover(cellActionContainers[8]); const cellActions = await screen.findByTestId('cell-action'); expect(cellActions).toBeInTheDocument(); userEvent.click(cellActions); expect(await screen.findByTestId('add-to-filter')).toBeInTheDocument(); expect(screen.getByTestId('exclude-from-filter')).toBeInTheDocument(); userEvent.hover(cellActionContainers[0]); // Transaction name const transactionCellActions = await screen.findAllByTestId('cell-action'); expect(transactionCellActions[0]).toBeInTheDocument(); userEvent.click(transactionCellActions[0]); expect(browserHistory.push).toHaveBeenCalledTimes(0); userEvent.click(screen.getByTestId('add-to-filter')); expect(browserHistory.push).toHaveBeenCalledTimes(1); expect(browserHistory.push).toHaveBeenNthCalledWith(1, { pathname: undefined, query: expect.objectContaining({ query: 'transaction:/apple/cart', }), }); }); it('hides cell actions when withStaticFilters is true', function () { const data = initializeData({ query: 'event.type:transaction transaction:/api*', }); render( ); const cellActionContainers = screen.queryByTestId('cell-action-container'); expect(cellActionContainers).not.toBeInTheDocument(); }); it('sends MEP param when setting enabled', function () { const data = initializeData( { query: 'event.type:transaction transaction:/api*', }, ['performance-use-metrics'] ); render( ); expect(eventsMock).toHaveBeenCalledTimes(1); expect(eventsMock).toHaveBeenNthCalledWith( 1, expect.anything(), expect.objectContaining({ query: expect.objectContaining({ environment: [], field: [ 'team_key_transaction', 'transaction', 'project', 'tpm()', 'p50()', 'p95()', 'failure_rate()', 'apdex()', 'count_unique(user)', 'count_miserable(user)', 'user_misery()', ], dataset: 'metrics', per_page: 50, project: ['1', '2'], query: 'event.type:transaction transaction:/api*', referrer: 'api.performance.landing-table', sort: '-team_key_transaction', statsPeriod: '14d', }), }) ); }); }); });