import {Component, createRef, Fragment, useEffect} 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/organization'; 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/duration/getDuration'; 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, '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 | undefined; searchQuery: string | undefined; }; class TraceDetailsContent extends Component { 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 | null = null; traceViewRef = createRef(); virtualScrollbarContainerRef = createRef(); async initFuse() { const {traces, orphanErrors} = this.props; if (!hasTraceData(traces, orphanErrors)) { return; } const transformedEvents: IndexedFusedTransaction[] = traces?.flatMap(trace => reduceTrace( 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 ( {t('Hang in there, as we build your trace view!')} ); } renderTraceRequiresDateRangeSelection() { return ; } 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(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 ( ); } renderTraceHeader(traceInfo: TraceInfo) { const {meta} = this.props; const errors = meta?.errors ?? traceInfo.errors.size; const performanceIssues = meta?.performance_issues ?? traceInfo.performanceIssues.size; return ( , fixed: '5 days ago', })} /> ); } 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 = ( {t( 'A root transaction is missing. Transactions linked by a dashed line have been orphaned and cannot be directly linked to the root.' )} ); } else if (roots === 1 && orphans > 0) { warning = ( {t( 'This trace has broken subtraces. Transactions linked by a dashed line have been orphaned and cannot be directly linked to the root.' )} ); } else if (roots > 1) { warning = ( {t('Multiple root transactions have been found with this trace ID.')} ); } else if (orphanErrors && orphanErrors.length > 0) { warning = ; } 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 ( ); } const traceInfo = traces ? getTraceInfo(traces, orphanErrors) : undefined; return ( {this.renderTraceWarnings()} {traceInfo && this.renderTraceHeader(traceInfo)} {this.renderSearchBar()} ); } render() { const {organization, location, traceEventView, traceSlug} = this.props; return ( {t('Trace ID: %s', traceSlug)} { trackAnalytics('performance_views.trace_view.open_in_discover', { organization, }); }} > {t('Open in Discover')} {this.renderContent()} ); } } 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; useEffect(() => { if (hasPerformanceOnboarding && location.hash === '#performance-sidequest') { SidebarPanelStore.activatePanel(SidebarPanelKey.PERFORMANCE_ONBOARDING); } }, [hasPerformanceOnboarding]); 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 ( {t( "The good news is we know these errors are related to each other in the same trace. The bad news is that we can't tell you more than that due to limited sampling." )} ); } if (isDismissed || isSnoozed) { return null; } return ( {t('Connect the Dots')} {t( "If you haven't already, configure performance monitoring to learn more about how your services are interacting with each other. This will provide more clarity about how your errors are linked." )} {} , }} size="xs" items={[ { key: 'dismiss', label: t('Dismiss'), onAction: () => { dismiss(); }, }, { key: 'snooze', label: t('Snooze'), onAction: () => { snooze(); }, }, ]} /> ); } const BannerWrapper = styled('div')` position: relative; border: 1px solid ${p => p.theme.border}; border-radius: ${p => p.theme.borderRadius}; padding: ${space(2)} ${space(3)}; margin-bottom: ${space(2)}; 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;