import {Component, Fragment} from 'react'; import {browserHistory} from 'react-router'; import styled from '@emotion/styled'; import {Location, LocationDescriptorObject} from 'history'; import {addSuccessMessage} from 'sentry/actionCreators/indicator'; import {openModal} from 'sentry/actionCreators/modal'; import GuideAnchor from 'sentry/components/assistant/guideAnchor'; import GridEditable, { COL_WIDTH_UNDEFINED, GridColumn, } from 'sentry/components/gridEditable'; import SortLink from 'sentry/components/gridEditable/sortLink'; import Link from 'sentry/components/links/link'; import Pagination from 'sentry/components/pagination'; import {Tooltip} from 'sentry/components/tooltip'; import {IconStar} from 'sentry/icons'; import {tct} from 'sentry/locale'; import {Organization, Project} from 'sentry/types'; import {trackAnalytics} from 'sentry/utils/analytics'; import DiscoverQuery, { TableData, TableDataRow, } from 'sentry/utils/discover/discoverQuery'; import EventView, {isFieldSortable, MetaType} from 'sentry/utils/discover/eventView'; import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers'; import {fieldAlignment, getAggregateAlias} from 'sentry/utils/discover/fields'; import {MEPConsumer} from 'sentry/utils/performance/contexts/metricsEnhancedSetting'; import {VisuallyCompleteWithData} from 'sentry/utils/performanceForSentry'; import CellAction, {Actions, updateQuery} from 'sentry/views/discover/table/cellAction'; import {TableColumn} from 'sentry/views/discover/table/types'; import {getMEPQueryParams} from './landing/widgets/utils'; import TransactionThresholdModal, { modalCss, TransactionThresholdMetric, } from './transactionSummary/transactionThresholdModal'; import { normalizeSearchConditionsWithTransactionName, transactionSummaryRouteWithQuery, } from './transactionSummary/utils'; import {COLUMN_TITLES} from './data'; import { areMultipleProjectsSelected, createUnnamedTransactionsDiscoverTarget, getProjectID, getSelectedProjectPlatforms, UNPARAMETERIZED_TRANSACTION, } from './utils'; type Props = { eventView: EventView; location: Location; organization: Organization; projects: Project[]; setError: (msg: string | undefined) => void; withStaticFilters: boolean; columnTitles?: string[]; summaryConditions?: string; }; type State = { transaction: string | undefined; transactionThreshold: number | undefined; transactionThresholdMetric: TransactionThresholdMetric | undefined; widths: number[]; }; function getProjectFirstEventGroup(project: Project): '14d' | '30d' | '>30d' { const fourteen_days_ago = new Date(+new Date() - 12096e5); const thirty_days_ago = new Date(+new Date() - 25920e5); const firstEventDate = new Date(project?.firstEvent ?? ''); if (firstEventDate > fourteen_days_ago) { return '14d'; } if (firstEventDate > thirty_days_ago) { return '30d'; } return '>30d'; } class _Table extends Component { state: State = { widths: [], transaction: undefined, transactionThreshold: undefined, transactionThresholdMetric: undefined, }; componentDidMount(): void { const {organization} = this.props; if (!this.tableMetricSet) { this.tableMetricSet = true; trackAnalytics('performance_views.landing.table.seen', { organization, }); } } unparameterizedMetricSet = false; tableMetricSet = false; sendUnparameterizedAnalytic() { const {organization, eventView} = this.props; const statsPeriod = eventView.statsPeriod ?? 'other'; const firstEventData = this.getFirstEventData(); trackAnalytics('performance_views.landing.table.unparameterized', { organization, first_event: firstEventData.firstEventWithin, sent_transaction: firstEventData.singleProject?.firstTransactionEvent ?? false, single_project: firstEventData.isSingleProject, stats_period: statsPeriod, hit_multi_project_cap: firstEventData.isAtMultiCap, }); } /** * Used for cluster warning and analytics. */ getFirstEventData() { const {eventView, projects} = this.props; const isSingleProject = !areMultipleProjectsSelected(eventView); const selectedProjects = eventView.getFullSelectedProjects(projects); const isAtMultiCap = selectedProjects.length > 1000; const singleProject = selectedProjects[0]; let firstEventWithin: 'none' | '14d' | '30d' | '>30d' = '>30d'; if (isSingleProject) { firstEventWithin = getProjectFirstEventGroup(singleProject); } else if (!isAtMultiCap) { const dateGroups = selectedProjects.map(getProjectFirstEventGroup); if (dateGroups.every(g => g === '14d')) { firstEventWithin = '14d'; } if (dateGroups.every(g => g === '14d' || g === '30d')) { firstEventWithin = '30d'; } } return { firstEventWithin, isSingleProject, singleProject, isAtMultiCap, }; } handleCellAction = (column: TableColumn, dataRow: TableDataRow) => { return (action: Actions, value: React.ReactText) => { const {eventView, location, organization, projects} = this.props; trackAnalytics('performance_views.overview.cellaction', { organization, action, }); if (action === Actions.EDIT_THRESHOLD) { const project_threshold = dataRow.project_threshold_config; const transactionName = dataRow.transaction as string; const projectID = getProjectID(dataRow, projects); openModal( modalProps => ( { if ( threshold !== project_threshold[1] || metric !== project_threshold[0] ) { this.setState({ transaction: transactionName, transactionThreshold: threshold, transactionThresholdMetric: metric, }); } addSuccessMessage( tct('[transactionName] updated successfully', { transactionName, }) ); }} /> ), {modalCss, closeEvents: 'escape-key'} ); return; } const searchConditions = normalizeSearchConditionsWithTransactionName( eventView.query ); updateQuery(searchConditions, action, column, value); browserHistory.push({ pathname: location.pathname, query: { ...location.query, cursor: undefined, query: searchConditions.formatString(), }, }); }; }; renderBodyCell( tableData: TableData | null, column: TableColumn, dataRow: TableDataRow ): React.ReactNode { const {eventView, organization, projects, location, withStaticFilters} = this.props; if (!tableData || !tableData.meta) { return dataRow[column.key]; } const tableMeta = tableData.meta; const field = String(column.key); const fieldRenderer = getFieldRenderer(field, tableMeta, false); const rendered = fieldRenderer(dataRow, {organization, location}); const allowActions = [ Actions.ADD, Actions.EXCLUDE, Actions.SHOW_GREATER_THAN, Actions.SHOW_LESS_THAN, Actions.EDIT_THRESHOLD, ]; const cellActions = withStaticFilters ? [] : allowActions; const isUnparameterizedRow = dataRow.transaction === UNPARAMETERIZED_TRANSACTION; if (field === 'transaction') { const projectID = getProjectID(dataRow, projects); const summaryView = eventView.clone(); if (dataRow['http.method']) { summaryView.additionalConditions.setFilterValues('http.method', [ dataRow['http.method'] as string, ]); } summaryView.query = summaryView.getQueryWithAdditionalConditions(); if (isUnparameterizedRow && !this.unparameterizedMetricSet) { this.sendUnparameterizedAnalytic(); this.unparameterizedMetricSet = true; } const target = isUnparameterizedRow ? createUnnamedTransactionsDiscoverTarget({ organization, location, }) : transactionSummaryRouteWithQuery({ orgSlug: organization.slug, transaction: String(dataRow.transaction) || '', query: summaryView.generateQueryStringObject(), projectID, }); return ( {rendered} ); } if (field.startsWith('team_key_transaction')) { // don't display per cell actions for team_key_transaction return rendered; } const fieldName = getAggregateAlias(field); const value = dataRow[fieldName]; if (tableMeta[fieldName] === 'integer' && typeof value === 'number' && value > 999) { return ( {rendered} ); } return ( {rendered} ); } renderBodyCellWithData = (tableData: TableData | null) => { return ( column: TableColumn, dataRow: TableDataRow ): React.ReactNode => this.renderBodyCell(tableData, column, dataRow); }; onSortClick(currentSortKind?: string, currentSortField?: string) { const {organization} = this.props; trackAnalytics('performance_views.landingv2.transactions.sort', { organization, field: currentSortField, direction: currentSortKind, }); } paginationAnalyticsEvent = (direction: string) => { const {organization} = this.props; trackAnalytics('performance_views.landingv3.table_pagination', { organization, direction, }); }; renderHeadCell( tableMeta: TableData['meta'], column: TableColumn, title: React.ReactNode ): React.ReactNode { const {eventView, location} = this.props; const align = fieldAlignment(column.name, column.type, tableMeta); const field = {field: column.name, width: column.width}; const aggregateAliasTableMeta: MetaType = {}; if (tableMeta) { Object.keys(tableMeta).forEach(key => { aggregateAliasTableMeta[getAggregateAlias(key)] = tableMeta[key]; }); } function generateSortLink(): LocationDescriptorObject | undefined { if (!tableMeta) { return undefined; } const nextEventView = eventView.sortOnField(field, aggregateAliasTableMeta); const queryStringObject = nextEventView.generateQueryStringObject(); return { ...location, query: {...location.query, sort: queryStringObject.sort}, }; } const currentSort = eventView.sortForField(field, aggregateAliasTableMeta); const canSort = isFieldSortable(field, aggregateAliasTableMeta); const currentSortKind = currentSort ? currentSort.kind : undefined; const currentSortField = currentSort ? currentSort.field : undefined; const sortLink = ( this.onSortClick(currentSortKind, currentSortField)} /> ); if (field.field.startsWith('user_misery')) { return ( {sortLink} ); } return sortLink; } renderHeadCellWithMeta = (tableMeta: TableData['meta']) => { const columnTitles = this.props.columnTitles ?? COLUMN_TITLES; return (column: TableColumn, index: number): React.ReactNode => this.renderHeadCell(tableMeta, column, columnTitles[index]); }; renderPrependCellWithData = (tableData: TableData | null) => { const {eventView} = this.props; const teamKeyTransactionColumn = eventView .getColumns() .find((col: TableColumn) => col.name === 'team_key_transaction'); return (isHeader: boolean, dataRow?: any) => { if (teamKeyTransactionColumn) { if (isHeader) { const star = ( ); return [this.renderHeadCell(tableData?.meta, teamKeyTransactionColumn, star)]; } return [this.renderBodyCell(tableData, teamKeyTransactionColumn, dataRow)]; } return []; }; }; handleSummaryClick = () => { const {organization, location, projects} = this.props; trackAnalytics('performance_views.overview.navigate.summary', { organization, project_platforms: getSelectedProjectPlatforms(location, projects), }); }; handleResizeColumn = (columnIndex: number, nextColumn: GridColumn) => { const widths: number[] = [...this.state.widths]; widths[columnIndex] = nextColumn.width ? Number(nextColumn.width) : COL_WIDTH_UNDEFINED; this.setState({widths}); }; getSortedEventView() { const {eventView} = this.props; return eventView.withSorts([ { field: 'team_key_transaction', kind: 'desc', }, ...eventView.sorts, ]); } render() { const {eventView, organization, location, setError} = this.props; const {widths, transaction, transactionThreshold} = this.state; const columnOrder = eventView .getColumns() // remove team_key_transactions from the column order as we'll be rendering it // via a prepended column .filter( (col: TableColumn) => col.name !== 'team_key_transaction' && !col.name.startsWith('count_miserable') && col.name !== 'project_threshold_config' ) .map((col: TableColumn, i: number) => { if (typeof widths[i] === 'number') { return {...col, width: widths[i]}; } return col; }); const sortedEventView = this.getSortedEventView(); const columnSortBy = sortedEventView.getSorts(); const prependColumnWidths = ['max-content']; return (
{value => { return ( setError(error?.message)} referrer="api.performance.landing-table" transactionName={transaction} transactionThreshold={transactionThreshold} queryExtras={getMEPQueryParams(value)} > {({pageLinks, isLoading, tableData}) => ( 0 } isLoading={isLoading} > )} ); }}
); } } function Table(props: Omit & {summaryConditions?: string}) { const summaryConditions = props.summaryConditions ?? props.eventView.getQueryWithAdditionalConditions(); return <_Table {...props} summaryConditions={summaryConditions} />; } // Align the contained IconStar with the IconStar buttons in individual table // rows, which have 2px padding + 1px border. const TeamKeyTransactionWrapper = styled('div')` padding: 3px; `; export default Table;