import {Fragment, useCallback, useMemo} from 'react'; import {browserHistory} from 'react-router'; import styled from '@emotion/styled'; import type {Location} from 'history'; import SearchBar from 'sentry/components/events/searchBar'; import type {GridColumnHeader} from 'sentry/components/gridEditable'; import GridEditable, {COL_WIDTH_UNDEFINED} from 'sentry/components/gridEditable'; import Pagination, {type CursorHandler} from 'sentry/components/pagination'; import {SpanSearchQueryBuilder} from 'sentry/components/performance/spanSearchQueryBuilder'; import {ROW_HEIGHT, ROW_PADDING} from 'sentry/components/performance/waterfall/constants'; import PerformanceDuration from 'sentry/components/performanceDuration'; import {Tooltip} from 'sentry/components/tooltip'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Organization} from 'sentry/types/organization'; import type {Project} from 'sentry/types/project'; import {defined} from 'sentry/utils'; import EventView, {type MetaType} from 'sentry/utils/discover/eventView'; import {getFieldRenderer} from 'sentry/utils/discover/fieldRenderers'; import type {ColumnType} from 'sentry/utils/discover/fields'; import { type DiscoverQueryProps, useGenericDiscoverQuery, } from 'sentry/utils/discover/genericDiscoverQuery'; import {DiscoverDatasets} from 'sentry/utils/discover/types'; import {VisuallyCompleteWithData} from 'sentry/utils/performanceForSentry'; import {decodeScalar} from 'sentry/utils/queryString'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import {useLocation} from 'sentry/utils/useLocation'; import {useNavigate} from 'sentry/utils/useNavigate'; import useOrganization from 'sentry/utils/useOrganization'; import {useParams} from 'sentry/utils/useParams'; import {renderHeadCell} from 'sentry/views/insights/common/components/tableCells/renderHeadCell'; import {SpanIdCell} from 'sentry/views/insights/common/components/tableCells/spanIdCell'; import {useSpansIndexed} from 'sentry/views/insights/common/queries/useDiscover'; import {QueryParameterNames} from 'sentry/views/insights/common/views/queryParameters'; import { ModuleName, SpanIndexedField, type SpanIndexedResponse, type SpanMetricsQueryFilters, } from 'sentry/views/insights/types'; import {TraceViewSources} from 'sentry/views/performance/newTraceDetails/traceMetadataHeader'; import {SpanDurationBar} from 'sentry/views/performance/transactionSummary/transactionSpans/spanDetails/spanDetailsTable'; import {SpanSummaryReferrer} from 'sentry/views/performance/transactionSummary/transactionSpans/spanSummary/referrers'; import {useSpanSummarySort} from 'sentry/views/performance/transactionSummary/transactionSpans/spanSummary/useSpanSummarySort'; import {useSpanFieldSupportedTags} from 'sentry/views/performance/utils/useSpanFieldSupportedTags'; import Tab from '../../tabs'; type DataRowKeys = | SpanIndexedField.ID | SpanIndexedField.TIMESTAMP | SpanIndexedField.SPAN_DURATION | SpanIndexedField.TRANSACTION_ID | SpanIndexedField.TRACE | SpanIndexedField.PROJECT; type ColumnKeys = | SpanIndexedField.ID | SpanIndexedField.TIMESTAMP | SpanIndexedField.SPAN_DURATION; type DataRow = Pick & {'transaction.duration': number}; type Column = GridColumnHeader; const COLUMN_ORDER: Column[] = [ { key: SpanIndexedField.ID, name: t('Span ID'), width: COL_WIDTH_UNDEFINED, }, { key: SpanIndexedField.TIMESTAMP, name: t('Timestamp'), width: COL_WIDTH_UNDEFINED, }, { key: SpanIndexedField.SPAN_DURATION, name: t('Span Duration'), width: COL_WIDTH_UNDEFINED, }, ]; const COLUMN_TYPE: Omit< Record, 'spans' | 'transactionDuration' > = { span_id: 'string', timestamp: 'date', 'span.duration': 'duration', }; const LIMIT = 8; type Props = { project: Project | undefined; }; export default function SpanSummaryTable(props: Props) { const {project} = props; const organization = useOrganization(); const supportedTags = useSpanFieldSupportedTags(); const {spanSlug} = useParams(); const navigate = useNavigate(); const [spanOp, groupId] = spanSlug.split(':'); const location = useLocation(); const {transaction} = location.query; const spansCursor = decodeScalar(location.query?.[QueryParameterNames.SPANS_CURSOR]); const spansQuery = decodeScalar(location.query.spansQuery, ''); const filters: SpanMetricsQueryFilters = { 'span.group': groupId, 'span.op': spanOp, transaction: transaction as string, }; const sort = useSpanSummarySort(); const spanSearchString = new MutableSearch(spansQuery).formatString(); const search = MutableSearch.fromQueryObject(filters); search.addStringMultiFilter(spanSearchString); const { data: rowData, pageLinks, isLoading: isRowDataLoading, } = useSpansIndexed( { fields: [ SpanIndexedField.ID, SpanIndexedField.TRANSACTION_ID, SpanIndexedField.TIMESTAMP, SpanIndexedField.SPAN_DURATION, SpanIndexedField.TRACE, SpanIndexedField.PROJECT, ], search, limit: LIMIT, sorts: [sort], cursor: spansCursor, }, SpanSummaryReferrer.SPAN_SUMMARY_TABLE ); const transactionIds = rowData?.map(row => row[SpanIndexedField.TRANSACTION_ID]); const eventView = EventView.fromNewQueryWithLocation( { name: 'Transaction Durations', query: MutableSearch.fromQueryObject({ project: project?.slug, id: `[${transactionIds?.join() ?? ''}]`, }).formatString(), fields: ['id', 'transaction.duration'], version: 2, }, location ); const { isLoading: isTxnDurationDataLoading, data: txnDurationData, isError: isTxnDurationError, } = useGenericDiscoverQuery< { data: any[]; meta: MetaType; }, DiscoverQueryProps >({ route: 'events', eventView, location, orgSlug: organization.slug, getRequestPayload: () => ({ ...eventView.getEventsAPIPayload(location), interval: eventView.interval, }), limit: LIMIT, options: { refetchOnWindowFocus: false, enabled: Boolean(rowData && rowData.length > 0), }, referrer: SpanSummaryReferrer.SPAN_SUMMARY_TABLE, }); // Restructure the transaction durations into a map for faster lookup const transactionDurationMap = {}; txnDurationData?.data.forEach(datum => { transactionDurationMap[datum.id] = datum['transaction.duration']; }); const mergedData: DataRow[] = rowData?.map((row: Pick) => { const transactionId = row[SpanIndexedField.TRANSACTION_ID]; const newRow = { ...row, 'transaction.duration': transactionDurationMap[transactionId], }; return newRow; }) ?? []; const handleCursor: CursorHandler = (cursor, pathname, query) => { browserHistory.push({ pathname, query: {...query, [QueryParameterNames.SPANS_CURSOR]: cursor}, }); }; const handleSearch = useCallback( (searchString: string) => { navigate({ ...location, query: { ...location.query, spansQuery: new MutableSearch(searchString).formatString(), }, }); }, [location, navigate] ); const projectIds = useMemo(() => eventView.project.slice(), [eventView]); return ( {organization.features.includes('search-query-builder-performance') ? ( ) : ( )} renderHeadCell({ column, location, sort, }), renderBodyCell: renderBodyCell( location, organization, spanOp, isTxnDurationDataLoading || isTxnDurationError ), }} /> ); } function renderBodyCell( location: Location, organization: Organization, spanOp: string = '', isTxnDurationDataLoading: boolean ) { return function (column: Column, dataRow: DataRow): React.ReactNode { const {timestamp, span_id, trace, project} = dataRow; const spanDuration = dataRow[SpanIndexedField.SPAN_DURATION]; const transactionId = dataRow[SpanIndexedField.TRANSACTION_ID]; const transactionDuration = dataRow['transaction.duration']; if (column.key === SpanIndexedField.SPAN_DURATION) { if (isTxnDurationDataLoading) { return ; } if (!transactionDuration) { return ( ); } return ( ); } if (column.key === SpanIndexedField.ID) { if (!defined(span_id)) { return null; } return ( ); } const fieldRenderer = getFieldRenderer(column.key, COLUMN_TYPE); const rendered = fieldRenderer(dataRow, {location, organization}); return rendered; }; } const EmptySpanDurationBar = styled('div')` height: ${ROW_HEIGHT - 2 * ROW_PADDING}px; width: 100%; position: relative; display: flex; align-items: center; top: ${space(0.5)}; background-color: ${p => p.theme.gray100}; padding-left: ${space(1)}; color: ${p => p.theme.gray300}; font-size: ${p => p.theme.fontSizeExtraSmall}; font-variant-numeric: tabular-nums; line-height: 1; `; const StyledSearchBarWrapper = styled('div')` margin-bottom: ${space(2)}; `;