import {useState} from 'react'; import {css, type Theme, useTheme} from '@emotion/react'; import styled from '@emotion/styled'; import type {Location} from 'history'; 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, tn} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {generateLinkToEventInTraceView} from 'sentry/utils/discover/urls'; import {getShortEventId} from 'sentry/utils/events'; import Projects from 'sentry/utils/projects'; import normalizeUrl from 'sentry/utils/url/normalizeUrl'; 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 type {SpanIndexedField, SpanIndexedResponse} from 'sentry/views/insights/types'; import {getTraceDetailsUrl} from 'sentry/views/performance/traceDetails/utils'; import {transactionSummaryRouteWithQuery} from 'sentry/views/performance/transactionSummary/utils'; import {TraceViewSources} from '../performance/newTraceDetails/traceHeader/breadcrumbs'; import type {TraceResult} from './hooks/useTraces'; import {BREAKDOWN_SLICES} from './hooks/useTraces'; import type {SpanResult} from './hooks/useTraceSpans'; import type {Field} from './data'; import {getShortenedSdkName, getStylingSliceName} from './utils'; export const ProjectBadgeWrapper = styled('span')` /** * Max of 2 visible projects, 16px each, 2px border, 8px overlap. */ width: 32px; min-width: 32px; `; export function SpanDescriptionRenderer({span}: {span: SpanResult}) { return ( {span['span.op']} {'\u2014'} {span['span.description']} {} ); } interface ProjectsRendererProps { projectSlugs: string[]; maxVisibleProjects?: number; } export function ProjectsRenderer({ projectSlugs, maxVisibleProjects = 2, }: ProjectsRendererProps) { const organization = useOrganization(); return ( {({projects}) => { // ensure that projectAvatars is in the same order as the projectSlugs prop const projectAvatars = projectSlugs.map(slug => { return projects.find(project => project.slug === slug) ?? {slug}; }); const numProjects = projectAvatars.length; const numVisibleProjects = maxVisibleProjects - numProjects >= 0 ? numProjects : maxVisibleProjects - 1; const visibleProjectAvatars = projectAvatars .slice(0, numVisibleProjects) .reverse(); const collapsedProjectAvatars = projectAvatars.slice(numVisibleProjects); const numCollapsedProjects = collapsedProjectAvatars.length; return ( {numCollapsedProjects > 0 && ( {tn( 'This trace contains %s more project.', 'This trace contains %s more projects.', numCollapsedProjects )} {collapsedProjectAvatars.map(project => ( ))} } > +{numCollapsedProjects} )} {visibleProjectAvatars.map(project => ( ))} ); }} ); } const ProjectList = styled('div')` display: flex; align-items: center; flex-direction: row-reverse; justify-content: flex-end; padding-right: 8px; `; const CollapsedProjects = styled('div')` width: 200px; display: flex; flex-direction: column; gap: ${space(0.5)}; `; const AvatarStyle = p => css` border: 2px solid ${p.theme.background}; margin-right: -8px; cursor: default; &:hover { z-index: 1; } `; const StyledProjectBadge = styled(ProjectBadge)` overflow: hidden; z-index: 0; ${AvatarStyle} `; const CollapsedBadge = styled('div')<{fontSize: number; size: number}>` display: flex; align-items: center; justify-content: center; position: relative; text-align: center; font-weight: ${p => p.theme.fontWeightBold}; background-color: ${p => p.theme.gray200}; color: ${p => p.theme.gray300}; font-size: ${p => p.fontSize}px; width: ${p => p.size}px; height: ${p => p.size}px; border-radius: ${p => p.theme.borderRadius}; ${AvatarStyle} `; interface ProjectRendererProps { projectSlug: string; hideName?: boolean; } 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 => css` filter: var(--highlightedSlice-${p.sliceName}-saturate, var(--defaultSlice-saturate)); `} ${p => css` opacity: var( --highlightedSlice-${p.sliceName ?? ''}-opacity, var(--defaultSlice-opacity, 1) ); `} ${p => css` transform: var( --hoveredSlice-${p.offset}-translateY, var( --highlightedSlice-${p.sliceName ?? ''}-transform, var(--defaultSlice-transform, 1) ) ); `} 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; /** * 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; onClick?: () => void; } export function SpanIdRenderer({ projectSlug, spanId, timestamp, traceId, transactionId, onClick, }: SpanIdRendererProps) { const location = useLocation(); const organization = useOrganization(); const target = generateLinkToEventInTraceView({ projectSlug, traceSlug: traceId, timestamp, eventId: transactionId, organization, location, spanId, source: TraceViewSources.TRACES, }); return ( {getShortEventId(spanId)} ); } interface TraceIdRendererProps { location: Location; timestamp: number; // in milliseconds traceId: string; onClick?: () => void; transactionId?: string; } export function TraceIdRenderer({ traceId, timestamp, transactionId, location, onClick, }: TraceIdRendererProps) { const organization = useOrganization(); const {selection} = usePageFilters(); const target = getTraceDetailsUrl({ organization, traceSlug: traceId, dateSelection: { start: selection.datetime.start, end: selection.datetime.end, statsPeriod: selection.datetime.period, }, timestamp: timestamp / 1000, eventId: transactionId, location, source: TraceViewSources.TRACES, }); 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, onClick, }: { trace: TraceResult; onClick?: () => void; }) { 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')}; `; export const Description = styled('div')` ${p => p.theme.overflowEllipsis}; display: flex; flex-direction: row; align-items: center; gap: ${space(1)}; `;