import {Fragment, type PropsWithChildren, useMemo} from 'react'; import styled from '@emotion/styled'; import type {LocationDescriptor} from 'history'; import {Button, LinkButton} from 'sentry/components/button'; import {CopyToClipboardButton} from 'sentry/components/copyToClipboardButton'; import { DropdownMenu, type DropdownMenuProps, type MenuItemProps, } from 'sentry/components/dropdownMenu'; import EventTagsDataSection from 'sentry/components/events/eventTagsAndScreenshot/tags'; import {DataSection} from 'sentry/components/events/styles'; import FileSize from 'sentry/components/fileSize'; import KeyValueData, { CardPanel, type KeyValueDataContentProps, Subject, } from 'sentry/components/keyValueData'; import {LazyRender, type LazyRenderProps} from 'sentry/components/lazyRender'; import Link from 'sentry/components/links/link'; import QuestionTooltip from 'sentry/components/questionTooltip'; import {Tooltip} from 'sentry/components/tooltip'; import {IconChevron, IconOpen} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Event, EventTransaction} from 'sentry/types/event'; import type {KeyValueListData} from 'sentry/types/group'; import type {Organization} from 'sentry/types/organization'; import {formatBytesBase10} from 'sentry/utils/bytes/formatBytesBase10'; import getDuration from 'sentry/utils/duration/getDuration'; import type {ColorOrAlias} from 'sentry/utils/theme'; import {useNavigate} from 'sentry/utils/useNavigate'; import useOrganization from 'sentry/utils/useOrganization'; import {useParams} from 'sentry/utils/useParams'; import {traceAnalytics} from '../../traceAnalytics'; import {useTransaction} from '../../traceApi/useTransaction'; import {useDrawerContainerRef} from '../../traceDrawer/details/drawerContainerRefContext'; import {makeTraceContinuousProfilingLink} from '../../traceDrawer/traceProfilingLink'; import { isAutogroupedNode, isMissingInstrumentationNode, isRootNode, isSpanNode, isTraceErrorNode, isTransactionNode, } from '../../traceGuards'; import type {MissingInstrumentationNode} from '../../traceModels/missingInstrumentationNode'; import type {ParentAutogroupNode} from '../../traceModels/parentAutogroupNode'; import type {SiblingAutogroupNode} from '../../traceModels/siblingAutogroupNode'; import {TraceTree} from '../../traceModels/traceTree'; import type {TraceTreeNode} from '../../traceModels/traceTreeNode'; const DetailContainer = styled('div')` display: flex; flex-direction: column; gap: ${space(2)}; padding: ${space(1)}; ${DataSection} { padding: 0; } `; const FlexBox = styled('div')` display: flex; align-items: center; `; const Actions = styled(FlexBox)` gap: ${space(0.5)}; justify-content: end; width: 100%; `; const Title = styled(FlexBox)` gap: ${space(1)}; flex-grow: 1; overflow: hidden; > span { min-width: 30px; } `; const TitleText = styled('div')` ${p => p.theme.overflowEllipsis} `; function TitleWithTestId(props: PropsWithChildren<{}>) { return {props.children}; } function TitleOp({text}: {text: string}) { return ( {text} } showOnlyOnOverflow isHoverable > {text} ); } const Type = styled('div')` font-size: ${p => p.theme.fontSizeSmall}; `; const TitleOpText = styled('div')` font-size: 15px; font-weight: ${p => p.theme.fontWeightBold}; ${p => p.theme.overflowEllipsis} `; const Table = styled('table')` margin-bottom: 0 !important; td { overflow: hidden; } `; const IconTitleWrapper = styled(FlexBox)` gap: ${space(1)}; min-width: 30px; `; const IconBorder = styled('div')<{backgroundColor: string; errored?: boolean}>` background-color: ${p => p.backgroundColor}; border-radius: ${p => p.theme.borderRadius}; padding: 0; display: flex; align-items: center; justify-content: center; width: 30px; height: 30px; min-width: 30px; svg { fill: ${p => p.theme.white}; width: 14px; height: 14px; } `; const HeaderContainer = styled(FlexBox)` justify-content: space-between; gap: ${space(3)}; container-type: inline-size; @container (max-width: 780px) { .DropdownMenu { display: block; } .Actions { display: none; } } @container (min-width: 781px) { .DropdownMenu { display: none; } } `; const DURATION_COMPARISON_STATUS_COLORS: { equal: {light: ColorOrAlias; normal: ColorOrAlias}; faster: {light: ColorOrAlias; normal: ColorOrAlias}; slower: {light: ColorOrAlias; normal: ColorOrAlias}; } = { faster: { light: 'green100', normal: 'green300', }, slower: { light: 'red100', normal: 'red300', }, equal: { light: 'gray100', normal: 'gray300', }, }; const MIN_PCT_DURATION_DIFFERENCE = 10; type DurationProps = { baseline: number | undefined; duration: number; node: TraceTreeNode; baseDescription?: string; ratio?: number; }; function Duration(props: DurationProps) { if (typeof props.duration !== 'number' || Number.isNaN(props.duration)) { return {t('unknown')}; } // Since transactions have ms precision, we show 2 decimal places only if the duration is greater than 1 second. const precision = isTransactionNode(props.node) ? (props.duration > 1 ? 2 : 0) : 2; if (props.baseline === undefined || props.baseline === 0) { return ( {getDuration(props.duration, precision, true)} ); } const delta = props.duration - props.baseline; const deltaPct = Math.round(Math.abs((delta / props.baseline) * 100)); const status = delta > 0 ? 'slower' : delta < 0 ? 'faster' : 'equal'; const formattedBaseDuration = ( {getDuration(props.baseline, 2, true)} ); const deltaText = status === 'equal' ? tct(`equal to the avg of [formattedBaseDuration]`, { formattedBaseDuration, }) : status === 'faster' ? tct(`[deltaPct] faster than the avg of [formattedBaseDuration]`, { formattedBaseDuration, deltaPct: `${deltaPct}%`, }) : tct(`[deltaPct] slower than the avg of [formattedBaseDuration]`, { formattedBaseDuration, deltaPct: `${deltaPct}%`, }); return ( {getDuration(props.duration, precision, true)}{' '} {props.ratio ? `(${(props.ratio * 100).toFixed()}%)` : null} {deltaPct >= MIN_PCT_DURATION_DIFFERENCE ? ( {deltaText} ) : null} ); } function TableRow({ title, keep, children, prefix, extra = null, toolTipText, }: { children: React.ReactNode; title: JSX.Element | string | null; extra?: React.ReactNode; keep?: boolean; prefix?: JSX.Element; toolTipText?: string; }) { if (!keep && !children) { return null; } return ( {prefix} {title} {toolTipText ? : null} {children} {extra} ); } function IssuesLink({ node, children, }: { children: React.ReactNode; node: TraceTreeNode; }) { const organization = useOrganization(); const params = useParams<{traceSlug?: string}>(); const traceSlug = params.traceSlug?.trim() ?? ''; // Adding a buffer of 15mins for errors only traces, where there is no concept of // trace duration and start equals end timestamps. const buffer = node.space[1] > 0 ? 0 : 15 * 60 * 1000; return ( {children} ); } const LAZY_RENDER_PROPS: Partial = { observerOptions: {rootMargin: '50px'}, }; const DurationContainer = styled('span')` font-weight: ${p => p.theme.fontWeightBold}; margin-right: ${space(1)}; `; const Comparison = styled('span')<{status: 'faster' | 'slower' | 'equal'}>` color: ${p => p.theme[DURATION_COMPARISON_STATUS_COLORS[p.status].normal]}; `; const Flex = styled('div')` display: flex; align-items: center; `; const TableValueRow = styled('div')` display: grid; grid-template-columns: auto min-content; gap: ${space(1)}; border-radius: 4px; background-color: ${p => p.theme.surface200}; margin: 2px; `; const StyledQuestionTooltip = styled(QuestionTooltip)` margin-left: ${space(0.5)}; `; const StyledPre = styled('pre')` margin: 0 !important; background-color: transparent !important; `; const TableRowButtonContainer = styled('div')` padding: 8px 10px; `; const ValueTd = styled('td')` position: relative; `; function getThreadIdFromNode( node: TraceTreeNode, transaction: EventTransaction | undefined ): string | undefined { if (isSpanNode(node) && node.value.data?.['thread.id']) { return node.value.data['thread.id']; } if (transaction) { return transaction.contexts?.trace?.data?.['thread.id']; } return undefined; } // Renders the dropdown menu list at the root trace drawer content container level, to prevent // being stacked under other content. function DropdownMenuWithPortal(props: DropdownMenuProps) { const drawerContainerRef = useDrawerContainerRef(); return ( ); } function TypeSafeBoolean(value: T | null | undefined): value is NonNullable { return value !== null && value !== undefined; } function NodeActions(props: { node: TraceTreeNode; onTabScrollToNode: ( node: | TraceTreeNode | ParentAutogroupNode | SiblingAutogroupNode | MissingInstrumentationNode ) => void; organization: Organization; eventSize?: number | undefined; }) { const navigate = useNavigate(); const organization = useOrganization(); const params = useParams<{traceSlug?: string}>(); const {data: transaction} = useTransaction({ node: isTransactionNode(props.node) ? props.node : null, organization, }); const profilerId = useMemo(() => { if (isTransactionNode(props.node)) { return props.node.value.profiler_id; } if (isSpanNode(props.node)) { return props.node.value.sentry_tags?.profiler_id ?? ''; } return ''; }, [props]); const profileLink = makeTraceContinuousProfilingLink(props.node, profilerId, { orgSlug: props.organization.slug, projectSlug: props.node.metadata.project_slug ?? '', traceId: params.traceSlug ?? '', threadId: getThreadIdFromNode(props.node, transaction), }); const items = useMemo((): MenuItemProps[] => { const showInView: MenuItemProps = { key: 'show-in-view', label: t('Show in View'), onAction: () => { traceAnalytics.trackShowInView(props.organization); props.onTabScrollToNode(props.node); }, }; const eventId = props.node.metadata.event_id ?? TraceTree.ParentTransaction(props.node)?.metadata.event_id; const projectSlug = props.node.metadata.project_slug ?? TraceTree.ParentTransaction(props.node)?.metadata.project_slug; const eventSize = props.eventSize; const jsonDetails: MenuItemProps = { key: 'json-details', onAction: () => { traceAnalytics.trackViewEventJSON(props.organization); window.open( `/api/0/projects/${props.organization.slug}/${projectSlug}/events/${eventId}/json/`, '_blank' ); }, label: t('JSON') + (typeof eventSize === 'number' ? ` (${formatBytesBase10(eventSize, 0)})` : ''), }; const continuousProfileLink: MenuItemProps | null = organization.features.includes('continuous-profiling-ui') && !!profileLink ? { key: 'continuous-profile', onAction: () => { traceAnalytics.trackViewContinuousProfile(props.organization); navigate(profileLink!); }, label: t('Continuous Profile'), } : null; if (isTransactionNode(props.node)) { return [showInView, jsonDetails, continuousProfileLink].filter(TypeSafeBoolean); } if (isSpanNode(props.node)) { return [showInView, continuousProfileLink].filter(TypeSafeBoolean); } if (isMissingInstrumentationNode(props.node)) { return [showInView, continuousProfileLink].filter(TypeSafeBoolean); } if (isTraceErrorNode(props.node)) { return [showInView, continuousProfileLink].filter(TypeSafeBoolean); } if (isRootNode(props.node)) { return [showInView]; } if (isAutogroupedNode(props.node)) { return [showInView]; } return [showInView]; }, [props, profileLink, navigate, organization.features]); return ( {organization.features.includes('continuous-profiling-ui') && !!profileLink ? ( {t('Continuous Profile')} ) : null} {isTransactionNode(props.node) ? ( } onClick={() => traceAnalytics.trackViewEventJSON(props.organization)} href={`/api/0/projects/${props.organization.slug}/${props.node.value.project_slug}/events/${props.node.value.event_id}/json/`} external > {t('JSON')} () ) : null} ( {t('Actions')} )} /> ); } const ActionsButtonTrigger = styled(Button)` svg { margin-left: ${space(0.5)}; width: 10px; height: 10px; } `; const ActionsContainer = styled('div')` display: flex; justify-content: end; align-items: center; gap: ${space(1)}; `; function EventTags({projectSlug, event}: {event: Event; projectSlug: string}) { return ( ); } const TagsWrapper = styled('div')` h3 { color: ${p => p.theme.textColor}; } `; export type SectionCardKeyValueList = KeyValueListData; function SectionCard({ items, title, disableTruncate, sortAlphabetically = false, itemProps = {}, }: { items: SectionCardKeyValueList; title: React.ReactNode; disableTruncate?: boolean; itemProps?: Partial; sortAlphabetically?: boolean; }) { const contentItems = items.map(item => ({item, ...itemProps})); return ( ); } // This is trace-view specific styling. The card is rendered in a number of different places // with tests failing otherwise, since @container queries are not supported by the version of // jsdom currently used by jest. const CardWrapper = styled('div')` ${CardPanel} { container-type: inline-size; } ${Subject} { @container (width < 350px) { max-width: 200px; } } `; function SectionCardGroup({children}: {children: React.ReactNode}) { return {children}; } function CopyableCardValueWithLink({ value, linkTarget, linkText, onClick, }: { value: React.ReactNode; linkTarget?: LocationDescriptor; linkText?: string; onClick?: () => void; }) { return ( {value} {typeof value === 'string' ? ( ) : null} {linkTarget && linkTarget ? ( {linkText} ) : null} ); } function TraceDataSection({event}: {event: EventTransaction}) { const traceData = event.contexts.trace?.data; if (!traceData) { return null; } return ( ({ key, subject: key, value, }))} title={t('Trace Data')} /> ); } const StyledCopyToClipboardButton = styled(CopyToClipboardButton)` transform: translateY(2px); `; const CardValueContainer = styled(FlexBox)` justify-content: space-between; gap: ${space(1)}; flex-wrap: wrap; `; const CardValueText = styled('span')` overflow-wrap: anywhere; `; export const CardContentSubject = styled('div')` grid-column: span 1; font-family: ${p => p.theme.text.familyMono}; word-wrap: break-word; `; const TraceDrawerComponents = { DetailContainer, FlexBox, Title: TitleWithTestId, Type, TitleOp, HeaderContainer, Actions, NodeActions, Table, IconTitleWrapper, IconBorder, TitleText, Duration, TableRow, LAZY_RENDER_PROPS, TableRowButtonContainer, TableValueRow, IssuesLink, SectionCard, CopyableCardValueWithLink, EventTags, TraceDataSection, SectionCardGroup, DropdownMenuWithPortal, }; export {TraceDrawerComponents};