123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606 |
- import {Component, createRef, Fragment} from 'react';
- import type {RouteComponentProps} from 'react-router';
- import styled from '@emotion/styled';
- import connectDotsImg from 'sentry-images/spot/performance-connect-dots.svg';
- import {Alert} from 'sentry/components/alert';
- import GuideAnchor from 'sentry/components/assistant/guideAnchor';
- import {Button} from 'sentry/components/button';
- import ButtonBar from 'sentry/components/buttonBar';
- import DiscoverButton from 'sentry/components/discoverButton';
- import {DropdownMenu} from 'sentry/components/dropdownMenu';
- import * as Layout from 'sentry/components/layouts/thirds';
- import ExternalLink from 'sentry/components/links/externalLink';
- import LoadingError from 'sentry/components/loadingError';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import {SidebarPanelKey} from 'sentry/components/sidebar/types';
- import TimeSince from 'sentry/components/timeSince';
- import {withPerformanceOnboarding} from 'sentry/data/platformCategories';
- import {IconClose} from 'sentry/icons';
- import {t, tct, tn} from 'sentry/locale';
- import SidebarPanelStore from 'sentry/stores/sidebarPanelStore';
- import {space} from 'sentry/styles/space';
- import type {Organization} from 'sentry/types';
- import {defined} from 'sentry/utils';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import type EventView from 'sentry/utils/discover/eventView';
- import type {QueryError} from 'sentry/utils/discover/genericDiscoverQuery';
- import {getDuration} from 'sentry/utils/formatters';
- import type {Fuse} from 'sentry/utils/fuzzySearch';
- import {createFuzzySearch} from 'sentry/utils/fuzzySearch';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import type {
- TraceError,
- TraceFullDetailed,
- TraceMeta,
- } from 'sentry/utils/performance/quickTrace/types';
- import {filterTrace, reduceTrace} from 'sentry/utils/performance/quickTrace/utils';
- import {VisuallyCompleteWithData} from 'sentry/utils/performanceForSentry';
- import useDismissAlert from 'sentry/utils/useDismissAlert';
- import useProjects from 'sentry/utils/useProjects';
- import Breadcrumb from 'sentry/views/performance/breadcrumb';
- import {MetaData} from 'sentry/views/performance/transactionDetails/styles';
- import {TraceDetailHeader, TraceSearchBar, TraceSearchContainer} from './styles';
- import TraceNotFound from './traceNotFound';
- import TraceView from './traceView';
- import type {TraceInfo} from './types';
- import {getTraceInfo, hasTraceData, isRootTransaction} from './utils';
- type IndexedFusedTransaction = {
- event: TraceFullDetailed | TraceError;
- indexed: string[];
- };
- type Props = Pick<RouteComponentProps<{traceSlug: string}, {}>, 'params' | 'location'> & {
- dateSelected: boolean;
- error: QueryError | null;
- isLoading: boolean;
- meta: TraceMeta | null;
- organization: Organization;
- traceEventView: EventView;
- traceSlug: string;
- traces: TraceFullDetailed[] | null;
- handleLimitChange?: (newLimit: number) => void;
- orphanErrors?: TraceError[];
- };
- type State = {
- filteredEventIds: Set<string> | undefined;
- searchQuery: string | undefined;
- };
- class TraceDetailsContent extends Component<Props, State> {
- state: State = {
- searchQuery: undefined,
- filteredEventIds: undefined,
- };
- componentDidMount() {
- this.initFuse();
- }
- componentDidUpdate(prevProps: Props) {
- if (
- this.props.traces !== prevProps.traces ||
- this.props.orphanErrors !== prevProps.orphanErrors
- ) {
- this.initFuse();
- }
- }
- fuse: Fuse<IndexedFusedTransaction> | null = null;
- traceViewRef = createRef<HTMLDivElement>();
- virtualScrollbarContainerRef = createRef<HTMLDivElement>();
- async initFuse() {
- const {traces, orphanErrors} = this.props;
- if (!hasTraceData(traces, orphanErrors)) {
- return;
- }
- const transformedEvents: IndexedFusedTransaction[] =
- traces?.flatMap(trace =>
- reduceTrace<IndexedFusedTransaction[]>(
- trace,
- (acc, transaction) => {
- const indexed: string[] = [
- transaction['transaction.op'],
- transaction.transaction,
- transaction.project_slug,
- ];
- acc.push({
- event: transaction,
- indexed,
- });
- return acc;
- },
- []
- )
- ) ?? [];
- // Include orphan error titles and project slugs during fuzzy search
- orphanErrors?.forEach(orphanError => {
- const indexed: string[] = [orphanError.title, orphanError.project_slug, 'Unknown'];
- transformedEvents.push({
- indexed,
- event: orphanError,
- });
- });
- this.fuse = await createFuzzySearch(transformedEvents, {
- keys: ['indexed'],
- includeMatches: true,
- threshold: 0.6,
- location: 0,
- distance: 100,
- maxPatternLength: 32,
- });
- }
- renderTraceLoading() {
- return (
- <LoadingContainer>
- <StyledLoadingIndicator />
- {t('Hang in there, as we build your trace view!')}
- </LoadingContainer>
- );
- }
- renderTraceRequiresDateRangeSelection() {
- return <LoadingError message={t('Trace view requires a date range selection.')} />;
- }
- handleTransactionFilter = (searchQuery: string) => {
- this.setState({searchQuery: searchQuery || undefined}, this.filterTransactions);
- };
- filterTransactions = () => {
- const {traces, orphanErrors} = this.props;
- const {filteredEventIds, searchQuery} = this.state;
- if (!searchQuery || !hasTraceData(traces, orphanErrors) || !defined(this.fuse)) {
- if (filteredEventIds !== undefined) {
- this.setState({
- filteredEventIds: undefined,
- });
- }
- return;
- }
- const fuseMatches = this.fuse
- .search<IndexedFusedTransaction>(searchQuery)
- /**
- * Sometimes, there can be matches that don't include any
- * indices. These matches are often noise, so exclude them.
- */
- .filter(({matches}) => matches?.length)
- .map(({item}) => item.event.event_id);
- /**
- * Fuzzy search on ids result in seemingly random results. So switch to
- * doing substring matches on ids to provide more meaningful results.
- */
- const idMatches: string[] = [];
- traces
- ?.flatMap(trace =>
- filterTrace(
- trace,
- ({event_id, span_id}) =>
- event_id.includes(searchQuery) || span_id.includes(searchQuery)
- )
- )
- .forEach(transaction => idMatches.push(transaction.event_id));
- // Include orphan error event_ids and span_ids during substring search
- orphanErrors?.forEach(orphanError => {
- const {event_id, span} = orphanError;
- if (event_id.includes(searchQuery) || span.includes(searchQuery)) {
- idMatches.push(event_id);
- }
- });
- this.setState({
- filteredEventIds: new Set([...fuseMatches, ...idMatches]),
- });
- };
- renderSearchBar() {
- return (
- <TraceSearchContainer>
- <TraceSearchBar
- defaultQuery=""
- query={this.state.searchQuery || ''}
- placeholder={t('Search for events')}
- onSearch={this.handleTransactionFilter}
- />
- </TraceSearchContainer>
- );
- }
- renderTraceHeader(traceInfo: TraceInfo) {
- const {meta} = this.props;
- const errors = meta?.errors ?? traceInfo.errors.size;
- const performanceIssues =
- meta?.performance_issues ?? traceInfo.performanceIssues.size;
- return (
- <TraceDetailHeader>
- <GuideAnchor target="trace_view_guide_breakdown">
- <MetaData
- headingText={t('Event Breakdown')}
- tooltipText={t(
- 'The number of transactions and issues there are in this trace.'
- )}
- bodyText={tct('[transactions] | [errors]', {
- transactions: tn(
- '%s Transaction',
- '%s Transactions',
- meta?.transactions ?? traceInfo.transactions.size
- ),
- errors: tn('%s Issue', '%s Issues', errors + performanceIssues),
- })}
- subtext={tn(
- 'Across %s project',
- 'Across %s projects',
- meta?.projects ?? traceInfo.projects.size
- )}
- />
- </GuideAnchor>
- <MetaData
- headingText={t('Total Duration')}
- tooltipText={t('The time elapsed between the start and end of this trace.')}
- bodyText={getDuration(
- traceInfo.endTimestamp - traceInfo.startTimestamp,
- 2,
- true
- )}
- subtext={getDynamicText({
- value: <TimeSince date={(traceInfo.endTimestamp || 0) * 1000} />,
- fixed: '5 days ago',
- })}
- />
- </TraceDetailHeader>
- );
- }
- renderTraceWarnings() {
- const {traces, orphanErrors} = this.props;
- const {roots, orphans} = (traces ?? []).reduce(
- (counts, trace) => {
- if (isRootTransaction(trace)) {
- counts.roots++;
- } else {
- counts.orphans++;
- }
- return counts;
- },
- {roots: 0, orphans: 0}
- );
- let warning: React.ReactNode = null;
- if (roots === 0 && orphans > 0) {
- warning = (
- <Alert type="info" showIcon>
- <ExternalLink href="https://docs.sentry.io/product/performance/trace-view/#orphan-traces-and-broken-subtraces">
- {t(
- 'A root transaction is missing. Transactions linked by a dashed line have been orphaned and cannot be directly linked to the root.'
- )}
- </ExternalLink>
- </Alert>
- );
- } else if (roots === 1 && orphans > 0) {
- warning = (
- <Alert type="info" showIcon>
- <ExternalLink href="https://docs.sentry.io/product/performance/trace-view/#orphan-traces-and-broken-subtraces">
- {t(
- 'This trace has broken subtraces. Transactions linked by a dashed line have been orphaned and cannot be directly linked to the root.'
- )}
- </ExternalLink>
- </Alert>
- );
- } else if (roots > 1) {
- warning = (
- <Alert type="info" showIcon>
- <ExternalLink href="https://docs.sentry.io/product/sentry-basics/tracing/trace-view/#multiple-roots">
- {t('Multiple root transactions have been found with this trace ID.')}
- </ExternalLink>
- </Alert>
- );
- } else if (orphanErrors && orphanErrors.length > 0) {
- warning = <OnlyOrphanErrorWarnings orphanErrors={orphanErrors} />;
- }
- return warning;
- }
- renderContent() {
- const {
- dateSelected,
- isLoading,
- error,
- organization,
- location,
- traceEventView,
- traceSlug,
- traces,
- meta,
- orphanErrors,
- } = this.props;
- if (!dateSelected) {
- return this.renderTraceRequiresDateRangeSelection();
- }
- if (isLoading) {
- return this.renderTraceLoading();
- }
- const hasData = hasTraceData(traces, orphanErrors);
- if (error !== null || !hasData) {
- return (
- <TraceNotFound
- meta={meta}
- traceEventView={traceEventView}
- traceSlug={traceSlug}
- location={location}
- organization={organization}
- />
- );
- }
- const traceInfo = traces ? getTraceInfo(traces, orphanErrors) : undefined;
- return (
- <Fragment>
- {this.renderTraceWarnings()}
- {traceInfo && this.renderTraceHeader(traceInfo)}
- {this.renderSearchBar()}
- <Margin>
- <VisuallyCompleteWithData id="PerformanceDetails-TraceView" hasData={hasData}>
- <TraceView
- filteredEventIds={this.state.filteredEventIds}
- traceInfo={traceInfo}
- location={location}
- organization={organization}
- traceEventView={traceEventView}
- traceSlug={traceSlug}
- traces={traces || []}
- meta={meta}
- orphanErrors={orphanErrors || []}
- handleLimitChange={this.props.handleLimitChange}
- />
- </VisuallyCompleteWithData>
- </Margin>
- </Fragment>
- );
- }
- render() {
- const {organization, location, traceEventView, traceSlug} = this.props;
- return (
- <Fragment>
- <Layout.Header>
- <Layout.HeaderContent>
- <Breadcrumb
- organization={organization}
- location={location}
- traceSlug={traceSlug}
- />
- <Layout.Title data-test-id="trace-header">
- {t('Trace ID: %s', traceSlug)}
- </Layout.Title>
- </Layout.HeaderContent>
- <Layout.HeaderActions>
- <ButtonBar gap={1}>
- <DiscoverButton
- size="sm"
- to={traceEventView.getResultsViewUrlTarget(organization.slug)}
- onClick={() => {
- trackAnalytics('performance_views.trace_view.open_in_discover', {
- organization,
- });
- }}
- >
- {t('Open in Discover')}
- </DiscoverButton>
- </ButtonBar>
- </Layout.HeaderActions>
- </Layout.Header>
- <Layout.Body>
- <Layout.Main fullWidth>{this.renderContent()}</Layout.Main>
- </Layout.Body>
- </Fragment>
- );
- }
- }
- type OnlyOrphanErrorWarningsProps = {
- orphanErrors: TraceError[];
- };
- function OnlyOrphanErrorWarnings({orphanErrors}: OnlyOrphanErrorWarningsProps) {
- const {projects} = useProjects();
- const projectSlug = orphanErrors[0] ? orphanErrors[0].project_slug : '';
- const project = projects.find(p => p.slug === projectSlug);
- const LOCAL_STORAGE_KEY = `${project?.id}:performance-orphan-error-onboarding-banner-hide`;
- const currentPlatform = project?.platform;
- const hasPerformanceOnboarding = currentPlatform
- ? withPerformanceOnboarding.has(currentPlatform)
- : false;
- const {dismiss: snooze, isDismissed: isSnoozed} = useDismissAlert({
- key: LOCAL_STORAGE_KEY,
- expirationDays: 7,
- });
- const {dismiss, isDismissed} = useDismissAlert({
- key: LOCAL_STORAGE_KEY,
- expirationDays: 365,
- });
- if (!orphanErrors.length) {
- return null;
- }
- if (!hasPerformanceOnboarding) {
- return (
- <Alert type="info" showIcon>
- {tct(
- "The good news is we know these errors are related to each other. The bad news is that we can't tell you more than that. If you haven't already, [tracingLink: configure performance monitoring for your SDKs] to learn more about service interactions.",
- {
- tracingLink: (
- <ExternalLink href="https://docs.sentry.io/product/performance/getting-started/" />
- ),
- }
- )}
- </Alert>
- );
- }
- if (isDismissed || isSnoozed) {
- return null;
- }
- return (
- <BannerWrapper>
- <ActionsWrapper>
- <BannerTitle>{t('Connect with Dots')}</BannerTitle>
- <BannerDescription>
- {t(
- "Want to know why this string of errors happened? If you haven't already, update your SDK with this snippet to figure out what operations are running between your errors."
- )}
- </BannerDescription>
- <ButtonsWrapper>
- <ActionButton>
- <Button
- priority="primary"
- onClick={event => {
- event.preventDefault();
- SidebarPanelStore.activatePanel(SidebarPanelKey.PERFORMANCE_ONBOARDING);
- }}
- >
- {t('Update SDKs')}
- </Button>
- </ActionButton>
- <ActionButton>
- <Button href="https://docs.sentry.io/product/performance/" external>
- {t('Learn More')}
- </Button>
- </ActionButton>
- </ButtonsWrapper>
- </ActionsWrapper>
- {<Background image={connectDotsImg} />}
- <CloseDropdownMenu
- position="bottom-end"
- triggerProps={{
- showChevron: false,
- borderless: true,
- icon: <IconClose color="subText" />,
- }}
- size="xs"
- items={[
- {
- key: 'dismiss',
- label: t('Dismiss'),
- onAction: () => {
- dismiss();
- },
- },
- {
- key: 'snooze',
- label: t('Snooze'),
- onAction: () => {
- snooze();
- },
- },
- ]}
- />
- </BannerWrapper>
- );
- }
- const BannerWrapper = styled('div')`
- position: relative;
- border: 1px solid ${p => p.theme.border};
- border-radius: ${p => p.theme.borderRadius};
- padding: 26px;
- margin: ${space(1)} 0;
- background: linear-gradient(
- 90deg,
- ${p => p.theme.backgroundSecondary}00 0%,
- ${p => p.theme.backgroundSecondary}FF 70%,
- ${p => p.theme.backgroundSecondary}FF 100%
- );
- min-width: 850px;
- `;
- const ActionsWrapper = styled('div')`
- max-width: 50%;
- `;
- const ButtonsWrapper = styled('div')`
- display: flex;
- align-items: center;
- gap: ${space(0.5)};
- `;
- const BannerTitle = styled('div')`
- font-size: ${p => p.theme.fontSizeExtraLarge};
- margin-bottom: ${space(1)};
- font-weight: 600;
- `;
- const BannerDescription = styled('div')`
- margin-bottom: ${space(1.5)};
- `;
- const CloseDropdownMenu = styled(DropdownMenu)`
- position: absolute;
- display: block;
- top: ${space(1)};
- right: ${space(1)};
- color: ${p => p.theme.white};
- cursor: pointer;
- z-index: 1;
- `;
- const Background = styled('div')<{image: any}>`
- display: flex;
- justify-self: flex-end;
- position: absolute;
- top: 14px;
- right: 15px;
- height: 81%;
- width: 100%;
- max-width: 413px;
- background-image: url(${p => p.image});
- background-repeat: no-repeat;
- background-size: contain;
- `;
- const ActionButton = styled('div')`
- display: flex;
- gap: ${space(1)};
- `;
- const StyledLoadingIndicator = styled(LoadingIndicator)`
- margin-bottom: 0;
- `;
- const LoadingContainer = styled('div')`
- font-size: ${p => p.theme.fontSizeLarge};
- color: ${p => p.theme.subText};
- text-align: center;
- `;
- const Margin = styled('div')`
- margin-top: ${space(2)};
- `;
- export default TraceDetailsContent;
|