import {useState} from 'react'; import {type Theme, useTheme} from '@emotion/react'; import styled from '@emotion/styled'; import Tag from 'sentry/components/badge/tag'; import {LinkButton} from 'sentry/components/button'; import ProjectBadge from 'sentry/components/idBadge/projectBadge'; import Link from 'sentry/components/links/link'; import {RowRectangle} from 'sentry/components/performance/waterfall/rowBar'; import {pickBarColor} from 'sentry/components/performance/waterfall/utils'; import PerformanceDuration from 'sentry/components/performanceDuration'; import TimeSince from 'sentry/components/timeSince'; import {Tooltip} from 'sentry/components/tooltip'; import {IconIssues} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {DateString} from 'sentry/types/core'; import {generateLinkToEventInTraceView} from 'sentry/utils/discover/urls'; import {getShortEventId} from 'sentry/utils/events'; import Projects from 'sentry/utils/projects'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import usePageFilters from 'sentry/utils/usePageFilters'; import useProjects from 'sentry/utils/useProjects'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import {getTraceDetailsUrl} from 'sentry/views/performance/traceDetails/utils'; import {transactionSummaryRouteWithQuery} from 'sentry/views/performance/transactionSummary/utils'; import type {SpanIndexedField, SpanIndexedResponse} from 'sentry/views/starfish/types'; import type {SpanResult, TraceResult} from './content'; import type {Field} from './data'; import {getShortenedSdkName, getStylingSliceName} from './utils'; interface ProjectRendererProps { projectSlug: string; hideName?: boolean; } export function SpanDescriptionRenderer({span}: {span: SpanResult}) { return ( {span['span.op']} {'\u2014'} {span['span.description']} {} ); } export function ProjectRenderer({projectSlug, hideName}: ProjectRendererProps) { const organization = useOrganization(); return ( {({projects}) => { const project = projects.find(p => p.slug === projectSlug); return ( ); }} ); } const WrappingText = styled('div')` ${p => p.theme.overflowEllipsis}; width: auto; `; export const TraceBreakdownContainer = styled('div')<{hoveredIndex?: number}>` position: relative; display: flex; min-width: 200px; height: 15px; background-color: ${p => p.theme.gray100}; ${p => `--hoveredSlice-${p.hoveredIndex ?? -1}-translateY: translateY(-3px)`}; `; const RectangleTraceBreakdown = styled(RowRectangle)<{ sliceColor: string; sliceName: string | null; offset?: number; }>` background-color: ${p => p.sliceColor}; position: relative; width: 100%; height: 15px; ${p => ` filter: var(--highlightedSlice-${p.sliceName}-saturate, var(--defaultSlice-saturate)); `} ${p => ` opacity: var(--highlightedSlice-${p.sliceName ?? ''}-opacity, var(--defaultSlice-opacity, 1.0)); `} ${p => ` transform: var(--hoveredSlice-${p.offset}-translateY, var(--highlightedSlice-${p.sliceName ?? ''}-transform, var(--defaultSlice-transform, 1.0))); `} transition: filter,opacity,transform 0.2s cubic-bezier(0.4, 0, 0.2, 1); `; export function TraceBreakdownRenderer({ trace, setHighlightedSliceName, }: { setHighlightedSliceName: (sliceName: string) => void; trace: TraceResult; }) { const theme = useTheme(); const [hoveredIndex, setHoveredIndex] = useState(-1); return ( setHoveredIndex(-1)} > {trace.breakdowns.map((breakdown, index) => { return ( { setHoveredIndex(index); breakdown.project ? setHighlightedSliceName( getStylingSliceName(breakdown.project, breakdown.sdkName) ?? '' ) : null; }} /> ); })} ); } const BREAKDOWN_SIZE_PX = 200; export const BREAKDOWN_SLICES = 40; /** * This renders slices in two different ways; * - Slices in the breakdown for the trace. These have slice numbers returned for quantization from the backend. * - Slices derived from span timings. Spans aren't quantized into slices. */ export function SpanBreakdownSliceRenderer({ trace, theme, sliceName, sliceStart, sliceEnd, sliceNumberStart, sliceNumberWidth, sliceDurationReal, sliceSecondaryName, onMouseEnter, offset, }: { onMouseEnter: () => void; sliceEnd: number; sliceName: string | null; sliceSecondaryName: string | null; sliceStart: number; theme: Theme; trace: TraceResult; offset?: number; sliceDurationReal?: number; sliceNumberStart?: number; sliceNumberWidth?: number; }) { const traceDuration = trace.end - trace.start; const sliceDuration = sliceEnd - sliceStart; const pixelsPerSlice = BREAKDOWN_SIZE_PX / BREAKDOWN_SLICES; const relativeSliceStart = sliceStart - trace.start; const stylingSliceName = getStylingSliceName(sliceName, sliceSecondaryName); const sliceColor = stylingSliceName ? pickBarColor(stylingSliceName) : theme.gray100; const sliceWidth = sliceNumberWidth !== undefined ? pixelsPerSlice * sliceNumberWidth : pixelsPerSlice * Math.ceil(BREAKDOWN_SLICES * (sliceDuration / traceDuration)); const sliceOffset = sliceNumberStart !== undefined ? pixelsPerSlice * sliceNumberStart : pixelsPerSlice * Math.floor((BREAKDOWN_SLICES * relativeSliceStart) / traceDuration); return ( {sliceName ? : null} {sliceName} ({getShortenedSdkName(sliceSecondaryName)})
} containerDisplayMode="block" >
); } const Subtext = styled('span')` font-weight: ${p => p.theme.fontWeightNormal}; color: ${p => p.theme.gray300}; `; const FlexContainer = styled('div')` display: flex; flex-direction: row; align-items: center; gap: ${space(0.5)}; padding-bottom: ${space(0.5)}; `; const BreakdownSlice = styled('div')<{ sliceName: string | null; sliceOffset: number; sliceWidth: number; }>` position: absolute; width: max(3px, ${p => p.sliceWidth}px); left: ${p => p.sliceOffset}px; ${p => (p.sliceName ? null : 'z-index: -1;')} `; interface SpanIdRendererProps { projectSlug: string; spanId: string; timestamp: string; traceId: string; transactionId: string; } export function SpanIdRenderer({ projectSlug, spanId, timestamp, traceId, transactionId, }: SpanIdRendererProps) { const location = useLocation(); const organization = useOrganization(); const target = generateLinkToEventInTraceView({ projectSlug, traceSlug: traceId, timestamp, eventId: transactionId, organization, location, spanId, }); return {getShortEventId(spanId)}; } interface TraceIdRendererProps { traceId: string; timestamp?: DateString; transactionId?: string; } export function TraceIdRenderer({ traceId, timestamp, transactionId, }: TraceIdRendererProps) { const organization = useOrganization(); const {selection} = usePageFilters(); const stringOrNumberTimestamp = timestamp instanceof Date ? timestamp.toISOString() : timestamp ?? ''; const target = getTraceDetailsUrl( organization, traceId, { start: selection.datetime.start, end: selection.datetime.end, statsPeriod: selection.datetime.period, }, stringOrNumberTimestamp, transactionId ); return ( {getShortEventId(traceId)} ); } interface TransactionRendererProps { projectSlug: string; transaction: string; } export function TransactionRenderer({ projectSlug, transaction, }: TransactionRendererProps) { const location = useLocation(); const organization = useOrganization(); const {projects} = useProjects({slugs: [projectSlug]}); const target = transactionSummaryRouteWithQuery({ orgSlug: organization.slug, transaction, query: { ...location.query, query: undefined, }, projectID: String(projects[0]?.id ?? ''), }); return {transaction}; } export function TraceIssuesRenderer({trace}: {trace: TraceResult}) { const organization = useOrganization(); const issueCount = trace.numErrors + trace.numOccurrences; const issueText = issueCount >= 100 ? '99+' : issueCount === 0 ? '\u2014' : issueCount; return ( } disabled={issueCount === 0} style={{minHeight: '24px', height: '24px', minWidth: '44px'}} > {issueText} ); } export function SpanTimeRenderer({ timestamp, tooltipShowSeconds, }: { timestamp: number; tooltipShowSeconds?: boolean; }) { const date = new Date(timestamp); return ( ); } type SpanStatus = SpanIndexedResponse[SpanIndexedField.SPAN_STATUS]; const STATUS_TO_TAG_TYPE: Record = { ok: 'success', cancelled: 'warning', unknown: 'info', invalid_argument: 'warning', deadline_exceeded: 'error', not_found: 'warning', already_exists: 'warning', permission_denied: 'warning', resource_exhausted: 'warning', failed_precondition: 'warning', aborted: 'warning', out_of_range: 'warning', unimplemented: 'error', internal_error: 'error', unavailable: 'error', data_loss: 'error', unauthenticated: 'warning', }; function statusToTagType(status: string) { return STATUS_TO_TAG_TYPE[status]; } const OMITTED_SPAN_STATUS = ['unknown']; /** * This display a tag for the status (not to be confused with 'status_code' which has values like '200', '429'). */ export function StatusTag({status, onClick}: {status: string; onClick?: () => void}) { const tagType = statusToTagType(status); if (!tagType) { return null; } if (OMITTED_SPAN_STATUS.includes(status)) { return null; } return ( {status} ); } const StyledTag = styled(Tag)` cursor: ${p => (p.onClick ? 'pointer' : 'default')}; `; const Description = styled('div')` ${p => p.theme.overflowEllipsis}; display: flex; flex-direction: row; align-items: center; gap: ${space(1)}; `;