123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404 |
- import {Fragment, useCallback, useMemo, useState} from 'react';
- import {browserHistory} from 'react-router';
- import {useTheme} from '@emotion/react';
- import styled from '@emotion/styled';
- 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 {ProjectPageFilter} from 'sentry/components/organizations/projectPageFilter';
- 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 type {SmartSearchBarProps} from 'sentry/components/smartSearchBar';
- import {IconChevron} from 'sentry/icons/iconChevron';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import type {PageFilters} from 'sentry/types/core';
- import {useApiQuery} from 'sentry/utils/queryClient';
- import {decodeInteger, decodeScalar} from 'sentry/utils/queryString';
- import {useLocation} from 'sentry/utils/useLocation';
- import useOrganization from 'sentry/utils/useOrganization';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import {type Field, FIELDS, SORTS} from './data';
- import {
- ProjectRenderer,
- SpanBreakdownSliceRenderer,
- SpanIdRenderer,
- TraceBreakdownContainer,
- TraceBreakdownRenderer,
- TraceIdRenderer,
- TraceIssuesRenderer,
- } from './fieldRenderers';
- import {TracesSearchBar} from './tracesSearchBar';
- import {normalizeTraces} from './utils';
- const DEFAULT_PER_PAGE = 20;
- export function Content() {
- const location = useLocation();
- const query = useMemo(() => {
- return decodeScalar(location.query.query, '');
- }, [location.query.query]);
- const limit = useMemo(() => {
- return decodeInteger(location.query.perPage, DEFAULT_PER_PAGE);
- }, [location.query.perPage]);
- const handleSearch: SmartSearchBarProps['onSearch'] = useCallback(
- (searchQuery: string) => {
- browserHistory.push({
- ...location,
- query: {
- ...location.query,
- cursor: undefined,
- query: searchQuery || undefined,
- },
- });
- },
- [location]
- );
- const traces = useTraces<Field>({
- fields: [
- ...FIELDS,
- ...SORTS.map(field =>
- field.startsWith('-') ? (field.substring(1) as Field) : (field as Field)
- ),
- ],
- limit,
- query,
- sort: SORTS,
- });
- 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 (
- <LayoutMain fullWidth>
- <PageFilterBar condensed>
- <ProjectPageFilter />
- <EnvironmentPageFilter />
- <DatePageFilter />
- </PageFilterBar>
- <TracesSearchBar query={query} handleSearch={handleSearch} />
- <StyledPanel>
- <TracePanelContent>
- <StyledPanelHeader align="right" lightText>
- {t('Trace ID')}
- </StyledPanelHeader>
- <StyledPanelHeader align="left" lightText>
- {t('Trace Root Name')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText>
- {t('Spans')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText>
- {t('Breakdown')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText>
- {t('Trace Duration')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText>
- {t('Issues')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText style={{padding: '5px'}} />
- {isLoading && (
- <StyledPanelItem span={7}>
- <LoadingIndicator />
- </StyledPanelItem>
- )}
- {isError && ( // TODO: need an error state
- <StyledPanelItem span={7}>
- <EmptyStateWarning withIcon />
- </StyledPanelItem>
- )}
- {isEmpty && (
- <StyledPanelItem span={7}>
- <EmptyStateWarning withIcon />
- </StyledPanelItem>
- )}
- {data?.map(trace => <TraceRow key={trace.trace} trace={trace} />)}
- </TracePanelContent>
- </StyledPanel>
- </LayoutMain>
- );
- }
- function TraceRow({trace}: {trace: TraceResult<Field>}) {
- const [expanded, setExpanded] = useState<boolean>(false);
- return (
- <Fragment>
- <StyledPanelItem align="center" center>
- <Button
- icon={<IconChevron size="xs" direction={expanded ? 'down' : 'right'} />}
- aria-label={t('Toggle trace details')}
- aria-expanded={expanded}
- size="zero"
- borderless
- onClick={() => setExpanded(e => !e)}
- />
- <TraceIdRenderer traceId={trace.trace} timestamp={trace.spans[0].timestamp} />
- </StyledPanelItem>
- <StyledPanelItem align="left">
- {trace.name ? (
- trace.name
- ) : (
- <EmptyValueContainer>{t('No Name Available')}</EmptyValueContainer>
- )}
- </StyledPanelItem>
- <StyledPanelItem align="right">
- <Count value={trace.numSpans} />
- </StyledPanelItem>
- <StyledPanelItem align="right">
- <TraceBreakdownRenderer trace={trace} />
- </StyledPanelItem>
- <StyledPanelItem align="right">
- <PerformanceDuration milliseconds={trace.duration} abbreviation />
- </StyledPanelItem>
- <StyledPanelItem align="right">
- <TraceIssuesRenderer trace={trace} />
- </StyledPanelItem>
- <StyledPanelItem style={{padding: '5px'}} />
- {expanded && <SpanTable spans={trace.spans} trace={trace} />}
- </Fragment>
- );
- }
- function SpanTable({
- spans,
- trace,
- }: {
- spans: SpanResult<Field>[];
- trace: TraceResult<Field>;
- }) {
- return (
- <SpanTablePanelItem span={7}>
- <StyledPanel>
- <SpanPanelContent>
- <StyledPanelHeader align="left" lightText>
- {t('Span ID')}
- </StyledPanelHeader>
- <StyledPanelHeader align="left" lightText>
- {t('Span Description')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText />
- <StyledPanelHeader align="right" lightText>
- {t('Span Duration')}
- </StyledPanelHeader>
- <StyledPanelHeader align="right" lightText>
- {t('Issues')}
- </StyledPanelHeader>
- {spans.map(span => (
- <SpanRow key={span.id} span={span} trace={trace} />
- ))}
- </SpanPanelContent>
- </StyledPanel>
- </SpanTablePanelItem>
- );
- }
- function SpanRow({span, trace}: {span: SpanResult<Field>; trace: TraceResult<Field>}) {
- const theme = useTheme();
- return (
- <Fragment>
- <StyledSpanPanelItem align="right">
- <SpanIdRenderer
- projectSlug={span.project}
- transactionId={span['transaction.id']}
- spanId={span.id}
- traceId={trace.trace}
- timestamp={span.timestamp}
- />
- </StyledSpanPanelItem>
- <StyledSpanPanelItem align="left">
- <Description>
- <ProjectRenderer projectSlug={span.project} hideName />
- <strong>{span['span.op']}</strong>
- <em>{'\u2014'}</em>
- {span['span.description']}
- </Description>
- </StyledSpanPanelItem>
- <StyledSpanPanelItem align="right">
- <TraceBreakdownContainer>
- <SpanBreakdownSliceRenderer
- sliceName={span.project}
- sliceStart={Math.ceil(span['precise.start_ts'] * 1000)}
- sliceEnd={Math.floor(span['precise.finish_ts'] * 1000)}
- trace={trace}
- theme={theme}
- />
- </TraceBreakdownContainer>
- </StyledSpanPanelItem>
- <StyledSpanPanelItem align="right">
- <PerformanceDuration milliseconds={span['span.duration']} abbreviation />
- </StyledSpanPanelItem>
- <StyledSpanPanelItem align="right">
- <EmptyValueContainer>{'\u2014'}</EmptyValueContainer>
- </StyledSpanPanelItem>
- </Fragment>
- );
- }
- type SpanResult<F extends string> = Record<F, any>;
- export interface TraceResult<F extends string> {
- breakdowns: TraceBreakdownResult[];
- duration: number;
- end: number;
- name: string | null;
- numSpans: number;
- spans: SpanResult<F>[];
- start: number;
- trace: string;
- }
- interface TraceBreakdownBase {
- end: number;
- start: number;
- }
- type TraceBreakdownProject = TraceBreakdownBase & {
- kind: 'project';
- project: string;
- };
- type TraceBreakdownMissing = TraceBreakdownBase & {
- kind: 'missing';
- project: null;
- };
- export type TraceBreakdownResult = TraceBreakdownProject | TraceBreakdownMissing;
- interface TraceResults<F extends string> {
- data: TraceResult<F>[];
- meta: any;
- }
- interface UseTracesOptions<F extends string> {
- fields: F[];
- datetime?: PageFilters['datetime'];
- enabled?: boolean;
- limit?: number;
- query?: string;
- sort?: string[];
- suggestedQuery?: string;
- }
- function useTraces<F extends string>({
- fields,
- datetime,
- enabled,
- limit,
- query,
- suggestedQuery,
- sort,
- }: UseTracesOptions<F>) {
- const organization = useOrganization();
- const {selection} = usePageFilters();
- const path = `/organizations/${organization.slug}/traces/`;
- const endpointOptions = {
- query: {
- project: selection.projects,
- environment: selection.environments,
- ...(datetime ?? normalizeDateTimeParams(selection.datetime)),
- field: fields,
- query,
- suggestedQuery,
- sort,
- per_page: limit,
- maxSpansPerTrace: 10,
- },
- };
- return useApiQuery<TraceResults<F>>([path, endpointOptions], {
- staleTime: 0,
- refetchOnWindowFocus: false,
- retry: false,
- enabled,
- });
- }
- const LayoutMain = styled(Layout.Main)`
- display: flex;
- flex-direction: column;
- gap: ${space(2)};
- `;
- const StyledPanel = styled(Panel)`
- margin-bottom: 0px;
- `;
- const TracePanelContent = styled('div')`
- width: 100%;
- display: grid;
- grid-template-columns: repeat(1, min-content) auto repeat(2, min-content) 120px 66px 10px;
- `;
- const SpanPanelContent = styled('div')`
- width: 100%;
- display: grid;
- grid-template-columns: repeat(1, min-content) auto repeat(1, min-content) 120px 66px;
- `;
- const StyledPanelHeader = styled(PanelHeader)<{align: 'left' | 'right'}>`
- white-space: nowrap;
- justify-content: ${p => (p.align === 'left' ? 'flex-start' : 'flex-end')};
- padding: ${space(2)} ${space(1)};
- `;
- const Description = styled('div')`
- ${p => p.theme.overflowEllipsis};
- display: flex;
- flex-direction: row;
- align-items: center;
- gap: ${space(1)};
- `;
- const StyledPanelItem = styled(PanelItem)<{
- align?: 'left' | 'center' | 'right';
- span?: number;
- }>`
- padding: ${space(1)};
- ${p => p.theme.overflowEllipsis};
- ${p =>
- p.align === 'center'
- ? `
- justify-content: space-around;`
- : p.align === 'left' || p.align === 'right'
- ? `text-align: ${p.align};`
- : undefined}
- ${p => p.span && `grid-column: auto / span ${p.span}`}
- `;
- const StyledSpanPanelItem = styled(StyledPanelItem)`
- &:nth-child(10n + 1),
- &:nth-child(10n + 2),
- &:nth-child(10n + 3),
- &:nth-child(10n + 4),
- &:nth-child(10n + 5) {
- background-color: ${p => p.theme.backgroundSecondary};
- }
- `;
- const SpanTablePanelItem = styled(StyledPanelItem)`
- background-color: ${p => p.theme.gray100};
- `;
- const EmptyValueContainer = styled('span')`
- color: ${p => p.theme.gray300};
- `;
|