import {Fragment} from 'react'; import {browserHistory} from 'react-router'; import type {Location} from 'history'; import * as qs from 'query-string'; import type {GridColumnHeader} from 'sentry/components/gridEditable'; import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable'; import Link from 'sentry/components/links/link'; import type {CursorHandler} from 'sentry/components/pagination'; import Pagination from 'sentry/components/pagination'; import {t} from 'sentry/locale'; import type {Organization} from 'sentry/types/organization'; import type {EventsMetaType} from 'sentry/utils/discover/eventView'; import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers'; import type {Sort} from 'sentry/utils/discover/fields'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import {useModuleURL} from 'sentry/views/performance/utils/useModuleURL'; import {renderHeadCell} from 'sentry/views/starfish/components/tableCells/renderHeadCell'; import {OverflowEllipsisTextContainer} from 'sentry/views/starfish/components/textAlign'; import type {SpanMetricsResponse} from 'sentry/views/starfish/types'; import {SpanMetricsField} from 'sentry/views/starfish/types'; import {QueryParameterNames} from 'sentry/views/starfish/views/queryParameters'; import {DataTitles, getThroughputTitle} from 'sentry/views/starfish/views/spans/types'; type Row = Pick< SpanMetricsResponse, | 'transaction' | 'transaction.method' | 'spm()' | 'avg(span.self_time)' | 'sum(span.self_time)' | 'time_spent_percentage()' >; type Column = GridColumnHeader< 'transaction' | 'spm()' | 'avg(span.self_time)' | 'time_spent_percentage()' >; const COLUMN_ORDER: Column[] = [ { key: 'transaction', name: t('Found In'), width: COL_WIDTH_UNDEFINED, }, { key: 'spm()', name: getThroughputTitle('db'), width: COL_WIDTH_UNDEFINED, }, { key: `avg(${SpanMetricsField.SPAN_SELF_TIME})`, name: DataTitles.avg, width: COL_WIDTH_UNDEFINED, }, { key: 'time_spent_percentage()', name: DataTitles.timeSpent, width: COL_WIDTH_UNDEFINED, }, ]; const SORTABLE_FIELDS = [ 'avg(span.self_time)', 'spm()', 'time_spent_percentage()', ] as const; type ValidSort = Sort & { field: (typeof SORTABLE_FIELDS)[number]; }; export function isAValidSort(sort: Sort): sort is ValidSort { return (SORTABLE_FIELDS as unknown as string[]).includes(sort.field); } interface Props { data: Row[]; isLoading: boolean; sort: ValidSort; span: Pick; error?: Error | null; meta?: EventsMetaType; pageLinks?: string; } export function QueryTransactionsTable({ data, isLoading, error, meta, pageLinks, sort, span, }: Props) { const moduleURL = useModuleURL('db'); const location = useLocation(); const organization = useOrganization(); const handleCursor: CursorHandler = (newCursor, pathname, query) => { browserHistory.push({ pathname, query: {...query, [QueryParameterNames.TRANSACTIONS_CURSOR]: newCursor}, }); }; return ( renderHeadCell({ column: col, sort, location, sortParameterName: QueryParameterNames.TRANSACTIONS_SORT, }), renderBodyCell: (column, row) => renderBodyCell(moduleURL, column, row, meta, span, location, organization), }} /> ); } function renderBodyCell( moduleURL: string, column: Column, row: Row, meta: EventsMetaType | undefined, span: Pick, location: Location, organization: Organization ) { if (column.key === 'transaction') { const label = row['transaction.method'] && !row.transaction.startsWith(row['transaction.method']) ? `${row['transaction.method']} ${row.transaction}` : row.transaction; const pathname = `${moduleURL}/spans/span/${encodeURIComponent(span[SpanMetricsField.SPAN_GROUP])}`; const query: {[key: string]: string | undefined} = { ...location.query, transaction: row.transaction, transactionMethod: row['transaction.method'], }; return ( {label} ); } if (!meta || !meta?.fields) { return row[column.key]; } const renderer = getFieldRenderer(column.key, meta.fields, false); const rendered = renderer( {...row, 'span.op': span['span.op']}, { location, organization, unit: meta.units?.[column.key], } ); return rendered; }