traceEventDataSection.tsx 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. import type {AnchorHTMLAttributes} from 'react';
  2. import {cloneElement, createContext, useCallback, useState} from 'react';
  3. import styled from '@emotion/styled';
  4. import {Button} from 'sentry/components/button';
  5. import ButtonBar from 'sentry/components/buttonBar';
  6. import {CompactSelect} from 'sentry/components/compactSelect';
  7. import {SegmentedControl} from 'sentry/components/segmentedControl';
  8. import {Tooltip} from 'sentry/components/tooltip';
  9. import {IconEllipsis, IconLink, IconSort} from 'sentry/icons';
  10. import {t} from 'sentry/locale';
  11. import {space} from 'sentry/styles/space';
  12. import type {PlatformKey, Project} from 'sentry/types';
  13. import type {Event} from 'sentry/types/event';
  14. import {trackAnalytics} from 'sentry/utils/analytics';
  15. import {isMobilePlatform, isNativePlatform} from 'sentry/utils/platform';
  16. import useApi from 'sentry/utils/useApi';
  17. import useOrganization from 'sentry/utils/useOrganization';
  18. import {EventDataSection} from './eventDataSection';
  19. const sortByOptions = {
  20. 'recent-first': t('Newest'),
  21. 'recent-last': t('Oldest'),
  22. };
  23. export const displayOptions = {
  24. 'absolute-addresses': t('Absolute addresses'),
  25. 'absolute-file-paths': t('Absolute file paths'),
  26. minified: t('Unsymbolicated'),
  27. 'raw-stack-trace': t('Raw stack trace'),
  28. 'verbose-function-names': t('Verbose function names'),
  29. };
  30. type State = {
  31. display: Array<keyof typeof displayOptions>;
  32. fullStackTrace: boolean;
  33. sortBy: keyof typeof sortByOptions;
  34. };
  35. type ChildProps = Omit<State, 'sortBy'> & {recentFirst: boolean};
  36. type Props = {
  37. children: (childProps: ChildProps) => React.ReactNode;
  38. eventId: Event['id'];
  39. fullStackTrace: boolean;
  40. hasAbsoluteAddresses: boolean;
  41. hasAbsoluteFilePaths: boolean;
  42. hasAppOnlyFrames: boolean;
  43. hasMinified: boolean;
  44. hasNewestFirst: boolean;
  45. hasVerboseFunctionNames: boolean;
  46. platform: PlatformKey;
  47. projectSlug: Project['slug'];
  48. recentFirst: boolean;
  49. stackTraceNotFound: boolean;
  50. title: React.ReactElement<any, any>;
  51. type: string;
  52. wrapTitle?: boolean;
  53. };
  54. export const TraceEventDataSectionContext = createContext<ChildProps | undefined>(
  55. undefined
  56. );
  57. export function TraceEventDataSection({
  58. type,
  59. title,
  60. wrapTitle,
  61. stackTraceNotFound,
  62. fullStackTrace,
  63. recentFirst,
  64. children,
  65. platform,
  66. projectSlug,
  67. eventId,
  68. hasNewestFirst,
  69. hasMinified,
  70. hasVerboseFunctionNames,
  71. hasAbsoluteFilePaths,
  72. hasAbsoluteAddresses,
  73. hasAppOnlyFrames,
  74. }: Props) {
  75. const api = useApi();
  76. const organization = useOrganization();
  77. const [state, setState] = useState<State>({
  78. sortBy: recentFirst ? 'recent-first' : 'recent-last',
  79. fullStackTrace: !hasAppOnlyFrames ? true : fullStackTrace,
  80. display: [],
  81. });
  82. const isMobile = isMobilePlatform(platform);
  83. const handleFilterFramesChange = useCallback(
  84. (val: 'full' | 'relevant') => {
  85. const isFullOptionClicked = val === 'full';
  86. trackAnalytics(
  87. isFullOptionClicked
  88. ? 'stack-trace.full_stack_trace_clicked'
  89. : 'stack-trace.most_relevant_clicked',
  90. {
  91. organization,
  92. project_slug: projectSlug,
  93. platform,
  94. is_mobile: isMobile,
  95. }
  96. );
  97. setState(currentState => ({...currentState, fullStackTrace: isFullOptionClicked}));
  98. },
  99. [organization, platform, projectSlug, isMobile]
  100. );
  101. const handleSortByChange = useCallback(
  102. (val: keyof typeof sortByOptions) => {
  103. const isRecentFirst = val === 'recent-first';
  104. trackAnalytics(
  105. isRecentFirst
  106. ? 'stack-trace.sort_option_recent_first_clicked'
  107. : 'stack-trace.sort_option_recent_last_clicked',
  108. {
  109. organization,
  110. project_slug: projectSlug,
  111. platform,
  112. is_mobile: isMobile,
  113. }
  114. );
  115. setState(currentState => ({...currentState, sortBy: val}));
  116. },
  117. [organization, platform, projectSlug, isMobile]
  118. );
  119. const handleDisplayChange = useCallback(
  120. (vals: (keyof typeof displayOptions)[]) => {
  121. if (vals.includes('raw-stack-trace')) {
  122. trackAnalytics('stack-trace.display_option_raw_stack_trace_clicked', {
  123. organization,
  124. project_slug: projectSlug,
  125. platform,
  126. is_mobile: isMobile,
  127. checked: true,
  128. });
  129. } else if (state.display.includes('raw-stack-trace')) {
  130. trackAnalytics('stack-trace.display_option_raw_stack_trace_clicked', {
  131. organization,
  132. project_slug: projectSlug,
  133. platform,
  134. is_mobile: isMobile,
  135. checked: false,
  136. });
  137. }
  138. if (vals.includes('absolute-addresses')) {
  139. trackAnalytics('stack-trace.display_option_absolute_addresses_clicked', {
  140. organization,
  141. project_slug: projectSlug,
  142. platform,
  143. is_mobile: isMobile,
  144. checked: true,
  145. });
  146. } else if (state.display.includes('absolute-addresses')) {
  147. trackAnalytics('stack-trace.display_option_absolute_addresses_clicked', {
  148. organization,
  149. project_slug: projectSlug,
  150. platform,
  151. is_mobile: isMobile,
  152. checked: false,
  153. });
  154. }
  155. if (vals.includes('absolute-file-paths')) {
  156. trackAnalytics('stack-trace.display_option_absolute_file_paths_clicked', {
  157. organization,
  158. project_slug: projectSlug,
  159. platform,
  160. is_mobile: isMobile,
  161. checked: true,
  162. });
  163. } else if (state.display.includes('absolute-file-paths')) {
  164. trackAnalytics('stack-trace.display_option_absolute_file_paths_clicked', {
  165. organization,
  166. project_slug: projectSlug,
  167. platform,
  168. is_mobile: isMobile,
  169. checked: false,
  170. });
  171. }
  172. if (vals.includes('minified')) {
  173. trackAnalytics(
  174. platform.startsWith('javascript')
  175. ? 'stack-trace.display_option_minified_clicked'
  176. : 'stack-trace.display_option_unsymbolicated_clicked',
  177. {
  178. organization,
  179. project_slug: projectSlug,
  180. platform,
  181. is_mobile: isMobile,
  182. checked: true,
  183. }
  184. );
  185. } else if (state.display.includes('minified')) {
  186. trackAnalytics(
  187. platform.startsWith('javascript')
  188. ? 'stack-trace.display_option_minified_clicked'
  189. : 'stack-trace.display_option_unsymbolicated_clicked',
  190. {
  191. organization,
  192. project_slug: projectSlug,
  193. platform,
  194. is_mobile: isMobile,
  195. checked: false,
  196. }
  197. );
  198. }
  199. if (vals.includes('verbose-function-names')) {
  200. trackAnalytics('stack-trace.display_option_verbose_function_names_clicked', {
  201. organization,
  202. project_slug: projectSlug,
  203. platform,
  204. is_mobile: isMobile,
  205. checked: true,
  206. });
  207. } else if (state.display.includes('verbose-function-names')) {
  208. trackAnalytics('stack-trace.display_option_verbose_function_names_clicked', {
  209. organization,
  210. project_slug: projectSlug,
  211. platform,
  212. is_mobile: isMobile,
  213. checked: false,
  214. });
  215. }
  216. setState(currentState => ({...currentState, display: vals}));
  217. },
  218. [organization, platform, projectSlug, isMobile, state]
  219. );
  220. function getDisplayOptions(): {
  221. label: string;
  222. value: keyof typeof displayOptions;
  223. disabled?: boolean;
  224. tooltip?: string;
  225. }[] {
  226. if (platform === 'objc' || platform === 'native' || platform === 'cocoa') {
  227. return [
  228. {
  229. label: displayOptions['absolute-addresses'],
  230. value: 'absolute-addresses',
  231. disabled: state.display.includes('raw-stack-trace') || !hasAbsoluteAddresses,
  232. tooltip: state.display.includes('raw-stack-trace')
  233. ? t('Not available on raw stack trace')
  234. : !hasAbsoluteAddresses
  235. ? t('Absolute addresses not available')
  236. : undefined,
  237. },
  238. {
  239. label: displayOptions['absolute-file-paths'],
  240. value: 'absolute-file-paths',
  241. disabled: state.display.includes('raw-stack-trace') || !hasAbsoluteFilePaths,
  242. tooltip: state.display.includes('raw-stack-trace')
  243. ? t('Not available on raw stack trace')
  244. : !hasAbsoluteFilePaths
  245. ? t('Absolute file paths not available')
  246. : undefined,
  247. },
  248. {
  249. label: displayOptions.minified,
  250. value: 'minified',
  251. disabled: !hasMinified,
  252. tooltip: !hasMinified ? t('Unsymbolicated version not available') : undefined,
  253. },
  254. {
  255. label: displayOptions['raw-stack-trace'],
  256. value: 'raw-stack-trace',
  257. },
  258. {
  259. label: displayOptions['verbose-function-names'],
  260. value: 'verbose-function-names',
  261. disabled: state.display.includes('raw-stack-trace') || !hasVerboseFunctionNames,
  262. tooltip: state.display.includes('raw-stack-trace')
  263. ? t('Not available on raw stack trace')
  264. : !hasVerboseFunctionNames
  265. ? t('Verbose function names not available')
  266. : undefined,
  267. },
  268. ];
  269. }
  270. if (platform.startsWith('python')) {
  271. return [
  272. {
  273. label: displayOptions['raw-stack-trace'],
  274. value: 'raw-stack-trace',
  275. },
  276. ];
  277. }
  278. // This logic might be incomplete, but according to the SDK folks, this is 99.9% of the cases
  279. if (platform.startsWith('javascript') || platform.startsWith('node')) {
  280. return [
  281. {
  282. label: t('Minified'),
  283. value: 'minified',
  284. disabled: !hasMinified,
  285. tooltip: !hasMinified ? t('Minified version not available') : undefined,
  286. },
  287. {
  288. label: displayOptions['raw-stack-trace'],
  289. value: 'raw-stack-trace',
  290. },
  291. ];
  292. }
  293. return [
  294. {
  295. label: displayOptions.minified,
  296. value: 'minified',
  297. disabled: !hasMinified,
  298. tooltip: !hasMinified ? t('Minified version not available') : undefined,
  299. },
  300. {
  301. label: displayOptions['raw-stack-trace'],
  302. value: 'raw-stack-trace',
  303. },
  304. ];
  305. }
  306. const nativePlatform = isNativePlatform(platform);
  307. const minified = state.display.includes('minified');
  308. // Apple crash report endpoint
  309. const appleCrashEndpoint = `/projects/${organization.slug}/${projectSlug}/events/${eventId}/apple-crash-report?minified=${minified}`;
  310. const rawStackTraceDownloadLink = `${api.baseUrl}${appleCrashEndpoint}&download=1`;
  311. const sortByTooltip = !hasNewestFirst
  312. ? t('Not available on stack trace with single frame')
  313. : state.display.includes('raw-stack-trace')
  314. ? t('Not available on raw stack trace')
  315. : undefined;
  316. const childProps = {
  317. recentFirst: state.sortBy === 'recent-first',
  318. display: state.display,
  319. fullStackTrace: state.fullStackTrace,
  320. };
  321. return (
  322. <EventDataSection
  323. type={type}
  324. title={cloneElement(title, {type})}
  325. guideTarget="stacktrace"
  326. actions={
  327. !stackTraceNotFound && (
  328. <ButtonBar gap={1}>
  329. {!state.display.includes('raw-stack-trace') && (
  330. <Tooltip
  331. title={t('Only full version available')}
  332. disabled={hasAppOnlyFrames}
  333. >
  334. <SegmentedControl
  335. size="xs"
  336. aria-label={t('Filter frames')}
  337. value={state.fullStackTrace ? 'full' : 'relevant'}
  338. onChange={handleFilterFramesChange}
  339. >
  340. <SegmentedControl.Item key="relevant" disabled={!hasAppOnlyFrames}>
  341. {t('Most Relevant')}
  342. </SegmentedControl.Item>
  343. <SegmentedControl.Item key="full">
  344. {t('Full Stack Trace')}
  345. </SegmentedControl.Item>
  346. </SegmentedControl>
  347. </Tooltip>
  348. )}
  349. {state.display.includes('raw-stack-trace') && nativePlatform && (
  350. <Button
  351. size="xs"
  352. href={rawStackTraceDownloadLink}
  353. title={t('Download raw stack trace file')}
  354. onClick={() => {
  355. trackAnalytics('stack-trace.download_clicked', {
  356. organization,
  357. project_slug: projectSlug,
  358. platform,
  359. is_mobile: isMobile,
  360. });
  361. }}
  362. >
  363. {t('Download')}
  364. </Button>
  365. )}
  366. <CompactSelect
  367. triggerProps={{
  368. icon: <IconSort />,
  369. size: 'xs',
  370. title: sortByTooltip,
  371. }}
  372. disabled={!!sortByTooltip}
  373. position="bottom-end"
  374. onChange={selectedOption => {
  375. handleSortByChange(selectedOption.value);
  376. }}
  377. value={state.sortBy}
  378. options={Object.entries(sortByOptions).map(([value, label]) => ({
  379. label,
  380. value: value as keyof typeof sortByOptions,
  381. }))}
  382. />
  383. <CompactSelect
  384. triggerProps={{
  385. icon: <IconEllipsis />,
  386. size: 'xs',
  387. showChevron: false,
  388. 'aria-label': t('Options'),
  389. }}
  390. multiple
  391. triggerLabel=""
  392. position="bottom-end"
  393. value={state.display}
  394. onChange={opts => handleDisplayChange(opts.map(opt => opt.value))}
  395. options={[{label: t('Display'), options: getDisplayOptions()}]}
  396. />
  397. </ButtonBar>
  398. )
  399. }
  400. showPermalink={false}
  401. wrapTitle={wrapTitle}
  402. >
  403. <TraceEventDataSectionContext.Provider value={childProps}>
  404. {children(childProps)}
  405. </TraceEventDataSectionContext.Provider>
  406. </EventDataSection>
  407. );
  408. }
  409. interface PermalinkTitleProps
  410. extends React.DetailedHTMLProps<
  411. AnchorHTMLAttributes<HTMLAnchorElement>,
  412. HTMLAnchorElement
  413. > {}
  414. export function PermalinkTitle(props: PermalinkTitleProps) {
  415. return (
  416. <Permalink {...props} href={'#' + props.type} className="permalink">
  417. <StyledIconLink size="xs" color="subText" />
  418. <h3>{props.children}</h3>
  419. </Permalink>
  420. );
  421. }
  422. const StyledIconLink = styled(IconLink)`
  423. display: none;
  424. position: absolute;
  425. top: 50%;
  426. left: -${space(2)};
  427. transform: translateY(-50%);
  428. `;
  429. const Permalink = styled('a')`
  430. display: inline-flex;
  431. justify-content: flex-start;
  432. &:hover ${StyledIconLink} {
  433. display: block;
  434. }
  435. `;