123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604 |
- import {Component, Fragment} from 'react';
- import {Location, LocationDescriptor} from 'history';
- import DropdownLink from 'sentry/components/dropdownLink';
- import ProjectBadge from 'sentry/components/idBadge/projectBadge';
- import {
- ErrorDestination,
- generateSingleErrorTarget,
- generateSingleTransactionTarget,
- generateTraceTarget,
- isQuickTraceEvent,
- TransactionDestination,
- } from 'sentry/components/quickTrace/utils';
- import Tooltip from 'sentry/components/tooltip';
- import {backend, frontend, mobile, serverless} from 'sentry/data/platformCategories';
- import {IconFire} from 'sentry/icons';
- import {t, tct, tn} from 'sentry/locale';
- import {OrganizationSummary} from 'sentry/types';
- import {Event} from 'sentry/types/event';
- import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent';
- import {getDocsPlatform} from 'sentry/utils/docs';
- import {getDuration} from 'sentry/utils/formatters';
- import localStorage from 'sentry/utils/localStorage';
- import {
- QuickTrace as QuickTraceType,
- QuickTraceEvent,
- TraceError,
- } from 'sentry/utils/performance/quickTrace/types';
- import {parseQuickTrace} from 'sentry/utils/performance/quickTrace/utils';
- import Projects from 'sentry/utils/projects';
- import {Theme} from 'sentry/utils/theme';
- const FRONTEND_PLATFORMS: string[] = [...frontend, ...mobile];
- const BACKEND_PLATFORMS: string[] = [...backend, ...serverless];
- import {
- DropdownContainer,
- DropdownItem,
- DropdownItemSubContainer,
- DropdownMenuHeader,
- ErrorNodeContent,
- EventNode,
- ExternalDropdownLink,
- QuickTraceContainer,
- QuickTraceValue,
- SectionSubtext,
- SingleEventHoverText,
- TraceConnector,
- } from './styles';
- const TOOLTIP_PREFIX = {
- root: 'root',
- ancestors: 'ancestor',
- parent: 'parent',
- current: '',
- children: 'child',
- descendants: 'descendant',
- };
- type QuickTraceProps = Pick<
- EventNodeSelectorProps,
- 'anchor' | 'errorDest' | 'transactionDest'
- > & {
- event: Event;
- location: Location;
- organization: OrganizationSummary;
- quickTrace: QuickTraceType;
- };
- export default function QuickTrace({
- event,
- quickTrace,
- location,
- organization,
- anchor,
- errorDest,
- transactionDest,
- }: QuickTraceProps) {
- let parsedQuickTrace;
- try {
- parsedQuickTrace = parseQuickTrace(quickTrace, event, organization);
- } catch (error) {
- return <Fragment>{'\u2014'}</Fragment>;
- }
- const traceLength = quickTrace.trace && quickTrace.trace.length;
- const {root, ancestors, parent, children, descendants, current} = parsedQuickTrace;
- const nodes: React.ReactNode[] = [];
- if (root) {
- nodes.push(
- <EventNodeSelector
- key="root-node"
- location={location}
- organization={organization}
- events={[root]}
- currentEvent={event}
- text={t('Root')}
- anchor={anchor}
- nodeKey="root"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- nodes.push(<TraceConnector key="root-connector" />);
- }
- if (ancestors?.length) {
- nodes.push(
- <EventNodeSelector
- key="ancestors-node"
- location={location}
- organization={organization}
- events={ancestors}
- currentEvent={event}
- text={tn('%s Ancestor', '%s Ancestors', ancestors.length)}
- anchor={anchor}
- nodeKey="ancestors"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- nodes.push(<TraceConnector key="ancestors-connector" />);
- }
- if (parent) {
- nodes.push(
- <EventNodeSelector
- key="parent-node"
- location={location}
- organization={organization}
- events={[parent]}
- currentEvent={event}
- text={t('Parent')}
- anchor={anchor}
- nodeKey="parent"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- nodes.push(<TraceConnector key="parent-connector" />);
- }
- const currentNode = (
- <EventNodeSelector
- key="current-node"
- location={location}
- organization={organization}
- text={t('This Event')}
- events={[current]}
- currentEvent={event}
- anchor={anchor}
- nodeKey="current"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- if (traceLength === 1) {
- nodes.push(
- <Projects
- key="missing-services"
- orgId={organization.slug}
- slugs={[current.project_slug]}
- >
- {({projects}) => {
- const project = projects.find(p => p.slug === current.project_slug);
- if (project?.platform) {
- if (BACKEND_PLATFORMS.includes(project.platform as string)) {
- return (
- <Fragment>
- <MissingServiceNode
- anchor={anchor}
- organization={organization}
- platform={project.platform}
- connectorSide="right"
- />
- {currentNode}
- </Fragment>
- );
- }
- if (FRONTEND_PLATFORMS.includes(project.platform as string)) {
- return (
- <Fragment>
- {currentNode}
- <MissingServiceNode
- anchor={anchor}
- organization={organization}
- platform={project.platform}
- connectorSide="left"
- />
- </Fragment>
- );
- }
- }
- return currentNode;
- }}
- </Projects>
- );
- } else {
- nodes.push(currentNode);
- }
- if (children.length) {
- nodes.push(<TraceConnector key="children-connector" />);
- nodes.push(
- <EventNodeSelector
- key="children-node"
- location={location}
- organization={organization}
- events={children}
- currentEvent={event}
- text={tn('%s Child', '%s Children', children.length)}
- anchor={anchor}
- nodeKey="children"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- }
- if (descendants?.length) {
- nodes.push(<TraceConnector key="descendants-connector" />);
- nodes.push(
- <EventNodeSelector
- key="descendants-node"
- location={location}
- organization={organization}
- events={descendants}
- currentEvent={event}
- text={tn('%s Descendant', '%s Descendants', descendants.length)}
- anchor={anchor}
- nodeKey="descendants"
- errorDest={errorDest}
- transactionDest={transactionDest}
- />
- );
- }
- return <QuickTraceContainer>{nodes}</QuickTraceContainer>;
- }
- function handleNode(key: string, organization: OrganizationSummary) {
- trackAdvancedAnalyticsEvent('quick_trace.node.clicked', {
- organization: organization.id,
- node_key: key,
- });
- }
- function handleDropdownItem(
- key: string,
- organization: OrganizationSummary,
- extra: boolean
- ) {
- const eventKey = extra
- ? 'quick_trace.dropdown.clicked_extra'
- : 'quick_trace.dropdown.clicked';
- trackAdvancedAnalyticsEvent(eventKey, {
- organization: organization.id,
- node_key: key,
- });
- }
- type EventNodeSelectorProps = {
- anchor: 'left' | 'right';
- currentEvent: Event;
- errorDest: ErrorDestination;
- events: QuickTraceEvent[];
- location: Location;
- nodeKey: keyof typeof TOOLTIP_PREFIX;
- organization: OrganizationSummary;
- text: React.ReactNode;
- transactionDest: TransactionDestination;
- numEvents?: number;
- };
- function EventNodeSelector({
- location,
- organization,
- events = [],
- text,
- currentEvent,
- nodeKey,
- anchor,
- errorDest,
- transactionDest,
- numEvents = 5,
- }: EventNodeSelectorProps) {
- let errors: TraceError[] = events.flatMap(event => event.errors ?? []);
- let type: keyof Theme['tag'] = nodeKey === 'current' ? 'black' : 'white';
- const hasErrors = errors.length > 0;
- if (hasErrors) {
- type = nodeKey === 'current' ? 'error' : 'warning';
- text = (
- <ErrorNodeContent>
- <IconFire size="xs" />
- {text}
- </ErrorNodeContent>
- );
- }
- // make sure to exclude the current event from the dropdown
- events = events.filter(event => event.event_id !== currentEvent.id);
- errors = errors.filter(error => error.event_id !== currentEvent.id);
- if (events.length + errors.length === 0) {
- return (
- <EventNode type={type} data-test-id="event-node">
- {text}
- </EventNode>
- );
- }
- if (events.length + errors.length === 1) {
- /**
- * When there is only 1 event, clicking the node should take the user directly to
- * the event without additional steps.
- */
- const hoverText = errors.length ? (
- t('View the error for this Transaction')
- ) : (
- <SingleEventHoverText event={events[0]} />
- );
- const target = errors.length
- ? generateSingleErrorTarget(errors[0], organization, location, errorDest)
- : generateSingleTransactionTarget(
- events[0],
- organization,
- location,
- transactionDest
- );
- return (
- <StyledEventNode
- text={text}
- hoverText={hoverText}
- to={target}
- onClick={() => handleNode(nodeKey, organization)}
- type={type}
- />
- );
- }
- /**
- * When there is more than 1 event, clicking the node should expand a dropdown to
- * allow the user to select which event to go to.
- */
- const hoverText = tct('View [eventPrefix] [eventType]', {
- eventPrefix: TOOLTIP_PREFIX[nodeKey],
- eventType:
- errors.length && events.length
- ? 'events'
- : events.length
- ? 'transactions'
- : 'errors',
- });
- return (
- <DropdownContainer>
- <DropdownLink
- caret={false}
- title={<StyledEventNode text={text} hoverText={hoverText} type={type} />}
- anchorRight={anchor === 'right'}
- >
- {errors.length > 0 && (
- <DropdownMenuHeader first>
- {tn('Related Error', 'Related Errors', errors.length)}
- </DropdownMenuHeader>
- )}
- {errors.slice(0, numEvents).map(error => {
- const target = generateSingleErrorTarget(
- error,
- organization,
- location,
- errorDest
- );
- return (
- <DropdownNodeItem
- key={error.event_id}
- event={error}
- to={target}
- allowDefaultEvent
- onSelect={() => handleDropdownItem(nodeKey, organization, false)}
- organization={organization}
- anchor={anchor}
- />
- );
- })}
- {events.length > 0 && (
- <DropdownMenuHeader first={errors.length === 0}>
- {tn('Transaction', 'Transactions', events.length)}
- </DropdownMenuHeader>
- )}
- {events.slice(0, numEvents).map(event => {
- const target = generateSingleTransactionTarget(
- event,
- organization,
- location,
- transactionDest
- );
- return (
- <DropdownNodeItem
- key={event.event_id}
- event={event}
- to={target}
- onSelect={() => handleDropdownItem(nodeKey, organization, false)}
- allowDefaultEvent
- organization={organization}
- subtext={getDuration(
- event['transaction.duration'] / 1000,
- event['transaction.duration'] < 1000 ? 0 : 2,
- true
- )}
- anchor={anchor}
- />
- );
- })}
- {(errors.length > numEvents || events.length > numEvents) && (
- <DropdownItem
- to={generateTraceTarget(currentEvent, organization)}
- allowDefaultEvent
- onSelect={() => handleDropdownItem(nodeKey, organization, true)}
- >
- {t('View all events')}
- </DropdownItem>
- )}
- </DropdownLink>
- </DropdownContainer>
- );
- }
- type DropdownNodeProps = {
- anchor: 'left' | 'right';
- event: TraceError | QuickTraceEvent;
- organization: OrganizationSummary;
- allowDefaultEvent?: boolean;
- onSelect?: (eventKey: any) => void;
- subtext?: string;
- to?: LocationDescriptor;
- };
- function DropdownNodeItem({
- event,
- onSelect,
- to,
- allowDefaultEvent,
- organization,
- subtext,
- anchor,
- }: DropdownNodeProps) {
- return (
- <DropdownItem to={to} onSelect={onSelect} allowDefaultEvent={allowDefaultEvent}>
- <DropdownItemSubContainer>
- <Projects orgId={organization.slug} slugs={[event.project_slug]}>
- {({projects}) => {
- const project = projects.find(p => p.slug === event.project_slug);
- return (
- <ProjectBadge
- disableLink
- hideName
- project={project ? project : {slug: event.project_slug}}
- avatarSize={16}
- />
- );
- }}
- </Projects>
- {isQuickTraceEvent(event) ? (
- <QuickTraceValue
- value={event.transaction}
- // expand in the opposite direction of the anchor
- expandDirection={anchor === 'left' ? 'right' : 'left'}
- maxLength={35}
- leftTrim
- trimRegex={/\.|\//g}
- />
- ) : (
- <QuickTraceValue
- value={event.title}
- // expand in the opposite direction of the anchor
- expandDirection={anchor === 'left' ? 'right' : 'left'}
- maxLength={45}
- />
- )}
- </DropdownItemSubContainer>
- {subtext && <SectionSubtext>{subtext}</SectionSubtext>}
- </DropdownItem>
- );
- }
- type EventNodeProps = {
- hoverText: React.ReactNode;
- text: React.ReactNode;
- onClick?: (eventKey: any) => void;
- to?: LocationDescriptor;
- type?: keyof Theme['tag'];
- };
- function StyledEventNode({text, hoverText, to, onClick, type = 'white'}: EventNodeProps) {
- return (
- <Tooltip position="top" containerDisplayMode="inline-flex" title={hoverText}>
- <EventNode
- data-test-id="event-node"
- type={type}
- icon={null}
- to={to}
- onClick={onClick}
- >
- {text}
- </EventNode>
- </Tooltip>
- );
- }
- type MissingServiceProps = Pick<QuickTraceProps, 'anchor' | 'organization'> & {
- connectorSide: 'left' | 'right';
- platform: string;
- };
- type MissingServiceState = {
- hideMissing: boolean;
- };
- const HIDE_MISSING_SERVICE_KEY = 'quick-trace:hide-missing-services';
- // 30 days
- const HIDE_MISSING_EXPIRES = 1000 * 60 * 60 * 24 * 30;
- function readHideMissingServiceState() {
- const value = localStorage.getItem(HIDE_MISSING_SERVICE_KEY);
- if (value === null) {
- return false;
- }
- const expires = parseInt(value, 10);
- const now = new Date().getTime();
- return expires > now;
- }
- class MissingServiceNode extends Component<MissingServiceProps, MissingServiceState> {
- state: MissingServiceState = {
- hideMissing: readHideMissingServiceState(),
- };
- dismissMissingService = () => {
- const {organization, platform} = this.props;
- const now = new Date().getTime();
- localStorage.setItem(
- HIDE_MISSING_SERVICE_KEY,
- (now + HIDE_MISSING_EXPIRES).toString()
- );
- this.setState({hideMissing: true});
- trackAdvancedAnalyticsEvent('quick_trace.missing_service.dismiss', {
- organization: organization.id,
- platform,
- });
- };
- trackExternalLink = () => {
- const {organization, platform} = this.props;
- trackAdvancedAnalyticsEvent('quick_trace.missing_service.docs', {
- organization: organization.id,
- platform,
- });
- };
- render() {
- const {hideMissing} = this.state;
- const {anchor, connectorSide, platform} = this.props;
- if (hideMissing) {
- return null;
- }
- const docPlatform = getDocsPlatform(platform, true);
- const docsHref =
- docPlatform === null || docPlatform === 'javascript'
- ? 'https://docs.sentry.io/platforms/javascript/performance/connect-services/'
- : `https://docs.sentry.io/platforms/${docPlatform}/performance/connect-services`;
- return (
- <Fragment>
- {connectorSide === 'left' && <TraceConnector />}
- <DropdownContainer>
- <DropdownLink
- caret={false}
- title={
- <StyledEventNode
- type="white"
- hoverText={t('No services connected')}
- text="???"
- />
- }
- anchorRight={anchor === 'right'}
- >
- <DropdownItem width="small">
- <ExternalDropdownLink href={docsHref} onClick={this.trackExternalLink}>
- {t('Connect to a service')}
- </ExternalDropdownLink>
- </DropdownItem>
- <DropdownItem onSelect={this.dismissMissingService} width="small">
- {t('Dismiss')}
- </DropdownItem>
- </DropdownLink>
- </DropdownContainer>
- {connectorSide === 'right' && <TraceConnector />}
- </Fragment>
- );
- }
- }
|