import {LocationFixture} from 'sentry-fixture/locationFixture'; import {OrganizationFixture} from 'sentry-fixture/organization'; import {initializeOrg} from 'sentry-test/initializeOrg'; import {act, render, screen, userEvent, within} from 'sentry-test/reactTestingLibrary'; import ProjectsStore from 'sentry/stores/projectsStore'; import TagStore from 'sentry/stores/tagStore'; import {browserHistory} from 'sentry/utils/browserHistory'; import type {TableData} from 'sentry/utils/discover/discoverQuery'; import EventView from 'sentry/utils/discover/eventView'; import {SavedQueryDatasets} from 'sentry/utils/discover/types'; import TableView from 'sentry/views/discover/table/tableView'; describe('TableView > CellActions', function () { let initialData: ReturnType; let rows: any; let onChangeShowTags: jest.Mock; const location = LocationFixture({ pathname: '/organizations/org-slug/discover/results/', query: { id: '42', name: 'best query', field: [ 'title', 'transaction', 'count()', 'timestamp', 'release', 'equation|count() + 100', ], sort: ['title'], query: '', project: ['123'], statsPeriod: '14d', environment: ['staging'], yAxis: 'p95', }, }); const eventView = EventView.fromLocation(location); function renderComponent( context: ReturnType, tableData: TableData, view: EventView ) { return render( , {router: context.router} ); } async function openContextMenu(cellIndex: number) { const firstRow = screen.getAllByRole('row')[1]; const emptyValueCell = within(firstRow).getAllByRole('cell')[cellIndex]; await userEvent.click(within(emptyValueCell).getByRole('button', {name: 'Actions'})); } beforeEach(function () { jest.mocked(browserHistory.push).mockReset(); jest.mocked(browserHistory.replace).mockReset(); const organization = OrganizationFixture({ features: ['discover-basic'], }); initialData = initializeOrg({ organization, router: {location}, }); act(() => { ProjectsStore.loadInitialData(initialData.projects); TagStore.reset(); TagStore.loadTagsSuccess([ {name: 'size', key: 'size'}, {name: 'shape', key: 'shape'}, {name: 'direction', key: 'direction'}, ]); }); onChangeShowTags = jest.fn(); rows = { meta: { title: 'string', transaction: 'string', 'count()': 'integer', timestamp: 'date', release: 'string', 'equation[0]': 'integer', }, data: [ { title: 'some title', transaction: '/organizations/', 'count()': 9, timestamp: '2019-05-23T22:12:48+00:00', release: 'v1.0.2', 'equation[0]': 109, }, ], }; MockApiClient.addMockResponse({ url: '/organizations/org-slug/dynamic-sampling/custom-rules/', method: 'GET', statusCode: 204, body: '', }); }); afterEach(() => { ProjectsStore.reset(); }); it('updates sort order on equation fields', function () { const view = eventView.clone(); renderComponent(initialData, rows, view); const equationCell = screen.getByRole('columnheader', {name: 'count() + 100'}); const sortLink = within(equationCell).getByRole('link'); expect(sortLink).toHaveAttribute( 'href', '/organizations/org-slug/discover/results/?environment=staging&field=title&field=transaction&field=count%28%29&field=timestamp&field=release&field=equation%7Ccount%28%29%20%2B%20100&id=42&name=best%20query&project=123&query=&sort=-equation%7Ccount%28%29%20%2B%20100&statsPeriod=14d&yAxis=p95' ); }); it('updates sort order on non-equation fields', function () { const view = eventView.clone(); renderComponent(initialData, rows, view); const transactionCell = screen.getByRole('columnheader', {name: 'transaction'}); const sortLink = within(transactionCell).getByRole('link'); expect(sortLink).toHaveAttribute( 'href', '/organizations/org-slug/discover/results/?environment=staging&field=title&field=transaction&field=count%28%29&field=timestamp&field=release&field=equation%7Ccount%28%29%20%2B%20100&id=42&name=best%20query&project=123&query=&sort=-transaction&statsPeriod=14d&yAxis=p95' ); }); it('handles add cell action on null value', async function () { rows.data[0].title = null; renderComponent(initialData, rows, eventView); await openContextMenu(1); await userEvent.click(screen.getByRole('menuitemradio', {name: 'Add to filter'})); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: '!has:title', }), }); }); it('handles add cell action on null value replace has condition', async function () { rows.data[0].title = null; const view = eventView.clone(); view.query = 'tag:value has:title'; renderComponent(initialData, rows, view); await openContextMenu(1); await userEvent.click(screen.getByRole('menuitemradio', {name: 'Add to filter'})); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'tag:value !has:title', }), }); }); it('handles add cell action on string value replace negation', async function () { const view = eventView.clone(); view.query = 'tag:value !title:nope'; renderComponent(initialData, rows, view); await openContextMenu(1); await userEvent.click(screen.getByRole('menuitemradio', {name: 'Add to filter'})); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'tag:value title:"some title"', }), }); }); it('handles add cell action with multiple y axis', async function () { location.query.yAxis = ['count()', 'failure_count()']; renderComponent(initialData, rows, eventView); await openContextMenu(1); await userEvent.click(screen.getByRole('menuitemradio', {name: 'Add to filter'})); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'title:"some title"', yAxis: ['count()', 'failure_count()'], }), }); }); it('handles exclude cell action on string value', async function () { renderComponent(initialData, rows, eventView); await openContextMenu(1); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Exclude from filter'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: '!title:"some title"', }), }); }); it('handles exclude cell action on string value replace inclusion', async function () { const view = eventView.clone(); view.query = 'tag:value title:nope'; renderComponent(initialData, rows, view); await openContextMenu(1); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Exclude from filter'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'tag:value !title:"some title"', }), }); }); it('handles exclude cell action on null value', async function () { rows.data[0].title = null; renderComponent(initialData, rows, eventView); await openContextMenu(1); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Exclude from filter'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'has:title', }), }); }); it('handles exclude cell action on null value replace condition', async function () { const view = eventView.clone(); view.query = 'tag:value !has:title'; rows.data[0].title = null; renderComponent(initialData, rows, view); await openContextMenu(1); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Exclude from filter'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'tag:value has:title', }), }); }); it('handles greater than cell action on number value', async function () { renderComponent(initialData, rows, eventView); await openContextMenu(3); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Show values greater than'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'count():>9', }), }); }); it('handles less than cell action on number value', async function () { renderComponent(initialData, rows, eventView); await openContextMenu(3); await userEvent.click( screen.getByRole('menuitemradio', {name: 'Show values less than'}) ); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'count():<9', }), }); }); it('renders transaction summary link', function () { rows.data[0].project = 'project-slug'; renderComponent(initialData, rows, eventView); const firstRow = screen.getAllByRole('row')[1]; const link = within(firstRow).getByTestId('tableView-transaction-link'); expect(link).toHaveAttribute( 'href', expect.stringMatching( RegExp( '/organizations/org-slug/performance/summary/?.*project=2&referrer=performance-transaction-summary.*transaction=%2.*' ) ) ); }); it('renders trace view link', function () { const org = OrganizationFixture({ features: [ 'discover-basic', 'performance-discover-dataset-selector', 'trace-view-v1', ], }); rows = { meta: { trace: 'string', id: 'string', transaction: 'string', timestamp: 'date', project: 'string', 'event.type': 'string', }, data: [ { trace: '7fdf8efed85a4f9092507063ced1995b', id: '509663014077465b8981b65225bdec0f', transaction: '/organizations/', timestamp: '2019-05-23T22:12:48+00:00', project: 'project-slug', 'event.type': '', }, ], }; const loc = LocationFixture({ pathname: '/organizations/org-slug/discover/results/', query: { id: '42', name: 'best query', field: ['id', 'transaction', 'timestamp'], queryDataset: 'transaction-like', sort: ['transaction'], query: '', project: ['123'], statsPeriod: '14d', environment: ['staging'], yAxis: 'p95', }, }); initialData = initializeOrg({ organization: org, router: {location: loc}, }); renderComponent(initialData, rows, EventView.fromLocation(loc)); const firstRow = screen.getAllByRole('row')[1]; const link = within(firstRow).getByTestId('view-event'); expect(link).toHaveAttribute( 'href', expect.stringMatching( RegExp( '/organizations/org-slug/performance/trace/7fdf8efed85a4f9092507063ced1995b/?.*' ) ) ); }); it('handles go to release', async function () { renderComponent(initialData, rows, eventView); await openContextMenu(5); await userEvent.click(screen.getByRole('menuitemradio', {name: 'Go to release'})); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: '/organizations/org-slug/releases/v1.0.2/', query: expect.objectContaining({ environment: eventView.environment, }), }); }); it('has title on integer value greater than 999', function () { rows.data[0]['count()'] = 1000; renderComponent(initialData, rows, eventView); const firstRow = screen.getAllByRole('row')[1]; const emptyValueCell = within(firstRow).getAllByRole('cell')[3]; expect(within(emptyValueCell).getByText('1k')).toHaveAttribute('title', '1,000'); }); it('renders size columns correctly', function () { const orgWithFeature = OrganizationFixture(); render( ); expect(screen.getByText('222.3 KiB')).toBeInTheDocument(); expect(screen.getByText('444.3 KB')).toBeInTheDocument(); }); it('shows events with value less than selected custom performance metric', async function () { const orgWithFeature = OrganizationFixture(); render( ); await userEvent.hover(screen.getByText('444.3 KB')); const buttons = screen.getAllByRole('button'); await userEvent.click(buttons[buttons.length - 1]); await userEvent.click(screen.getByText('Show values less than')); expect(browserHistory.push).toHaveBeenCalledWith({ pathname: location.pathname, query: expect.objectContaining({ query: 'p99(measurements.custom.kilobyte):<444300', }), }); }); });