import {Fragment, useCallback, useMemo, useState} from 'react'; import {useTheme} from '@emotion/react'; import styled from '@emotion/styled'; import debounce from 'lodash/debounce'; import omit from 'lodash/omit'; import {Alert} from 'sentry/components/alert'; import {Button} from 'sentry/components/button'; import Count from 'sentry/components/count'; import EmptyStateWarning from 'sentry/components/emptyStateWarning'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import {DatePageFilter} from 'sentry/components/organizations/datePageFilter'; import {EnvironmentPageFilter} from 'sentry/components/organizations/environmentPageFilter'; import PageFilterBar from 'sentry/components/organizations/pageFilterBar'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import Panel from 'sentry/components/panels/panel'; import PanelHeader from 'sentry/components/panels/panelHeader'; import PanelItem from 'sentry/components/panels/panelItem'; import PerformanceDuration from 'sentry/components/performanceDuration'; import {IconChevron} from 'sentry/icons/iconChevron'; import {IconClose} from 'sentry/icons/iconClose'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {PageFilters} from 'sentry/types/core'; import type {MRI} from 'sentry/types/metrics'; import {browserHistory} from 'sentry/utils/browserHistory'; import {getFormattedMQL} from 'sentry/utils/metrics'; import {useApiQuery} from 'sentry/utils/queryClient'; import {decodeInteger, decodeList, decodeScalar} from 'sentry/utils/queryString'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import usePageFilters from 'sentry/utils/usePageFilters'; import * as ModuleLayout from 'sentry/views/performance/moduleLayout'; import {type Field, FIELDS, SORTS} from './data'; import { BREAKDOWN_SLICES, ProjectRenderer, SpanBreakdownSliceRenderer, SpanDescriptionRenderer, SpanIdRenderer, SpanTimeRenderer, TraceBreakdownContainer, TraceBreakdownRenderer, TraceIdRenderer, TraceIssuesRenderer, } from './fieldRenderers'; import {TracesChart} from './tracesChart'; import {TracesSearchBar} from './tracesSearchBar'; import { areQueriesEmpty, getSecondaryNameFromSpan, getStylingSliceName, normalizeTraces, } from './utils'; const DEFAULT_PER_PAGE = 50; export function Content() { const location = useLocation(); const queries = useMemo(() => { return decodeList(location.query.query); }, [location.query.query]); const limit = useMemo(() => { return decodeInteger(location.query.perPage, DEFAULT_PER_PAGE); }, [location.query.perPage]); const removeMetric = useCallback(() => { browserHistory.push({ ...location, query: omit(location.query, [ 'mri', 'metricsOp', 'metricsQuery', 'metricsMax', 'metricsMin', ]), }); }, [location]); const metricsMax = decodeScalar(location.query.metricsMax); const metricsMin = decodeScalar(location.query.metricsMin); const metricsOp = decodeScalar(location.query.metricsOp); const metricsQuery = decodeScalar(location.query.metricsQuery); const mri = decodeScalar(location.query.mri); const handleSearch = useCallback( (searchIndex: number, searchQuery: string) => { const newQueries = [...queries]; if (newQueries.length === 0) { // In the odd case someone wants to add search bars before any query has been made, we add both the default one shown and a new one. newQueries[0] = ''; } newQueries[searchIndex] = searchQuery; browserHistory.push({ ...location, query: { ...location.query, cursor: undefined, query: typeof searchQuery === 'string' ? newQueries : queries, }, }); }, [location, queries] ); const handleClearSearch = useCallback( (searchIndex: number) => { const newQueries = [...queries]; if (typeof newQueries[searchIndex] !== undefined) { delete newQueries[searchIndex]; browserHistory.push({ ...location, query: { ...location.query, cursor: undefined, query: newQueries, }, }); return true; } return false; }, [location, queries] ); const hasMetric = metricsOp && mri; const traces = useTraces({ fields: [ ...FIELDS, ...SORTS.map(field => field.startsWith('-') ? (field.substring(1) as Field) : (field as Field) ), ], limit, query: queries, sort: SORTS, mri: hasMetric ? mri : undefined, metricsMax: hasMetric ? metricsMax : undefined, metricsMin: hasMetric ? metricsMin : undefined, metricsOp: hasMetric ? metricsOp : undefined, metricsQuery: hasMetric ? metricsQuery : undefined, }); const isLoading = traces.isFetching; const isError = !isLoading && traces.isError; const isEmpty = !isLoading && !isError && (traces?.data?.data?.length ?? 0) === 0; const data = normalizeTraces(!isLoading && !isError ? traces?.data?.data : undefined); return ( {hasMetric && ( } > {tct('The metric query [metricQuery] is filtering the results below.', { metricQuery: ( {getFormattedMQL({mri: mri as MRI, op: metricsOp, query: metricsQuery})} ), })} )} {isError && typeof traces.error?.responseJSON?.detail === 'string' ? ( {traces.error?.responseJSON?.detail} ) : null} {t('Trace ID')} {t('Trace Root')} {areQueriesEmpty(queries) ? t('Total Spans') : t('Matching Spans')} {t('Timeline')} {t('Duration')} {t('Timestamp')} {t('Issues')} {isLoading && ( )} {isError && ( // TODO: need an error state )} {isEmpty && ( {t('No results found')} {t('There are no traces that match the conditions above.')}
{t('Try adjusting your filters starting with your time range.')}
)} {data?.map((trace, i) => ( ))}
); } function TraceRow({ defaultExpanded, trace, }: { defaultExpanded; trace: TraceResult; }) { const [expanded, setExpanded] = useState(defaultExpanded); const [highlightedSliceName, _setHighlightedSliceName] = useState(''); const location = useLocation(); const queries = useMemo(() => { return decodeList(location.query.query); }, [location.query.query]); const setHighlightedSliceName = useMemo( () => debounce(sliceName => _setHighlightedSliceName(sliceName), 100, { leading: true, }), [_setHighlightedSliceName] ); const onClickExpand = useCallback(() => setExpanded(e => !e), [setExpanded]); return (