profilingSlowestTransactionsPanel.tsx 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. import {useMemo, useState} from 'react';
  2. import styled from '@emotion/styled';
  3. import {PlatformIcon} from 'platformicons';
  4. import {Button} from 'sentry/components/button';
  5. import EmptyStateWarning from 'sentry/components/emptyStateWarning';
  6. import Link from 'sentry/components/links/link';
  7. import LoadingIndicator from 'sentry/components/loadingIndicator';
  8. import {Panel} from 'sentry/components/panels';
  9. import PerformanceDuration from 'sentry/components/performanceDuration';
  10. import {Flex} from 'sentry/components/profiling/flex';
  11. import {
  12. FunctionsMiniGrid,
  13. FunctionsMiniGridEmptyState,
  14. FunctionsMiniGridLoading,
  15. } from 'sentry/components/profiling/functionsMiniGrid';
  16. import {TextTruncateOverflow} from 'sentry/components/profiling/textTruncateOverflow';
  17. import {IconChevron} from 'sentry/icons';
  18. import {t} from 'sentry/locale';
  19. import {space} from 'sentry/styles/space';
  20. import {Organization, Project} from 'sentry/types';
  21. import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent';
  22. import {getAggregateAlias} from 'sentry/utils/discover/fields';
  23. import {
  24. EventsResults,
  25. EventsResultsDataRow,
  26. useProfileEvents,
  27. } from 'sentry/utils/profiling/hooks/useProfileEvents';
  28. import {useProfilingTransactionQuickSummary} from 'sentry/utils/profiling/hooks/useProfilingTransactionQuickSummary';
  29. import {generateProfileSummaryRouteWithQuery} from 'sentry/utils/profiling/routes';
  30. import {makeFormatTo} from 'sentry/utils/profiling/units/units';
  31. import useOrganization from 'sentry/utils/useOrganization';
  32. import useProjects from 'sentry/utils/useProjects';
  33. const fields = ['transaction', 'project.id', 'last_seen()', 'p95()', 'count()'] as const;
  34. type SlowestTransactionsFields = (typeof fields)[number];
  35. export function ProfilingSlowestTransactionsPanel() {
  36. const profilingTransactionsQuery = useProfileEvents({
  37. fields,
  38. sort: {
  39. key: 'p95()',
  40. order: 'desc',
  41. },
  42. limit: 3,
  43. query: 'count():>3',
  44. referrer: 'api.profiling.landing-slowest-transaction-panel',
  45. });
  46. const [openPanel, setOpenPanel] = useState<null | string>(null);
  47. const profilingTransactions = useMemo(
  48. () => profilingTransactionsQuery.data?.[0].data ?? [],
  49. [profilingTransactionsQuery.data]
  50. );
  51. const transactionNames = useMemo(
  52. () => profilingTransactions.map(txn => txn.transaction),
  53. [profilingTransactions]
  54. );
  55. if (transactionNames.length > 0 && !transactionNames.includes(openPanel)) {
  56. const firstTransaction = transactionNames[0];
  57. setOpenPanel(firstTransaction as string);
  58. }
  59. const {isLoading} = profilingTransactionsQuery;
  60. const hasProfilingTransactions =
  61. !isLoading && profilingTransactions && profilingTransactions.length > 0;
  62. return (
  63. <FlexPanel>
  64. <Flex column h="100%">
  65. <Flex column p={space(1.5)}>
  66. <PanelHeading>{t('Slowest Transactions')}</PanelHeading>
  67. <PanelSubheading>
  68. {t('Slowest transactions that could use some optimization.')}
  69. </PanelSubheading>
  70. </Flex>
  71. {(isLoading || !hasProfilingTransactions) && (
  72. <Flex column align="center" justify="center" h="100%">
  73. {isLoading ? (
  74. <LoadingIndicator />
  75. ) : (
  76. !hasProfilingTransactions && (
  77. <Flex.Item>
  78. <EmptyStateWarning>
  79. <p>{t('No results found')}</p>
  80. <EmptyStateDescription>
  81. {t(
  82. 'Transactions may not be listed due to the filters above or a low number of profiles.'
  83. )}
  84. </EmptyStateDescription>
  85. </EmptyStateWarning>
  86. </Flex.Item>
  87. )
  88. )}
  89. </Flex>
  90. )}
  91. {profilingTransactions?.map(transaction => {
  92. return (
  93. <SlowestTransactionPanelItem
  94. key={transaction.transaction}
  95. transaction={transaction}
  96. open={transaction.transaction === openPanel}
  97. onOpen={() => setOpenPanel(transaction.transaction as string)}
  98. units={profilingTransactionsQuery.data?.[0].meta.units}
  99. />
  100. );
  101. })}
  102. </Flex>
  103. </FlexPanel>
  104. );
  105. }
  106. interface SlowestTransactionPanelItemProps {
  107. onOpen: () => void;
  108. open: boolean;
  109. transaction: EventsResultsDataRow<SlowestTransactionsFields>;
  110. units?: EventsResults<SlowestTransactionsFields>['meta']['units'];
  111. }
  112. function SlowestTransactionPanelItem({
  113. transaction,
  114. open,
  115. onOpen,
  116. units,
  117. }: SlowestTransactionPanelItemProps) {
  118. const organization = useOrganization();
  119. const projects = useProjects();
  120. const transactionProject = useMemo(
  121. () => projects.projects.find(p => p.id === String(transaction['project.id'])),
  122. [projects.projects, transaction]
  123. );
  124. if (!transactionProject && !projects.fetching && projects.projects.length > 0) {
  125. return null;
  126. }
  127. const key: SlowestTransactionsFields = 'p95()';
  128. const formatter = makeFormatTo(
  129. units?.[key] ?? units?.[getAggregateAlias(key)] ?? 'nanoseconds',
  130. 'milliseconds'
  131. );
  132. return (
  133. <PanelItem key={transaction.transaction}>
  134. <Flex justify="space-between" gap={space(1)}>
  135. <PlatformIcon platform={transactionProject?.platform ?? 'default'} />
  136. <Flex.Item
  137. grow={1}
  138. onClick={onOpen}
  139. css={{
  140. cursor: 'pointer',
  141. }}
  142. >
  143. <div
  144. css={{
  145. maxWidth: 'fit-content',
  146. }}
  147. >
  148. <Link
  149. to={generateProfileSummaryRouteWithQuery({
  150. orgSlug: organization.slug,
  151. projectSlug: transactionProject?.slug!,
  152. transaction: transaction.transaction as string,
  153. })}
  154. onClick={() => {
  155. trackAdvancedAnalyticsEvent('profiling_views.go_to_transaction', {
  156. source: 'slowest_transaction_panel',
  157. organization,
  158. });
  159. }}
  160. >
  161. <TextTruncateOverflow>
  162. {transaction.transaction as string}
  163. </TextTruncateOverflow>
  164. </Link>
  165. </div>
  166. </Flex.Item>
  167. <PerformanceDuration
  168. milliseconds={formatter(transaction[key] as number)}
  169. abbreviation
  170. />
  171. <Button borderless size="zero" onClick={onOpen}>
  172. <IconChevron direction={open ? 'up' : 'down'} size="xs" />
  173. </Button>
  174. </Flex>
  175. <PanelItemBody
  176. style={{
  177. height: open ? 160 : 0,
  178. }}
  179. >
  180. {open && transactionProject && (
  181. <PanelItemFunctionsMiniGrid
  182. transaction={String(transaction.transaction)}
  183. organization={organization}
  184. project={transactionProject}
  185. />
  186. )}
  187. </PanelItemBody>
  188. </PanelItem>
  189. );
  190. }
  191. interface PanelItemFunctionsMiniGridProps {
  192. organization: Organization;
  193. project: Project;
  194. transaction: string;
  195. }
  196. function PanelItemFunctionsMiniGrid(props: PanelItemFunctionsMiniGridProps) {
  197. const {transaction, project, organization} = props;
  198. const {functionsQuery, functions} = useProfilingTransactionQuickSummary({
  199. transaction,
  200. project,
  201. referrer: 'api.profiling.landing-slowest-transaction-panel',
  202. skipLatestProfile: true,
  203. skipSlowestProfile: true,
  204. });
  205. if (functionsQuery.isLoading) {
  206. return <FunctionsMiniGridLoading />;
  207. }
  208. if (!functions || (functions && functions.length === 0)) {
  209. return <FunctionsMiniGridEmptyState />;
  210. }
  211. return (
  212. <PanelItemBodyInner>
  213. <FunctionsMiniGrid
  214. functions={functions}
  215. organization={organization}
  216. project={project}
  217. onLinkClick={() =>
  218. trackAdvancedAnalyticsEvent('profiling_views.go_to_flamegraph', {
  219. organization,
  220. source: 'slowest_transaction_panel',
  221. })
  222. }
  223. />
  224. </PanelItemBodyInner>
  225. );
  226. }
  227. const FlexPanel = styled(Panel)`
  228. display: flex;
  229. flex-direction: column;
  230. `;
  231. const PanelHeading = styled('span')`
  232. font-size: ${p => p.theme.text.cardTitle.fontSize};
  233. font-weight: ${p => p.theme.text.cardTitle.fontWeight};
  234. line-height: ${p => p.theme.text.cardTitle.lineHeight};
  235. `;
  236. const PanelSubheading = styled('span')`
  237. color: ${p => p.theme.subText};
  238. `;
  239. const PanelItem = styled('div')`
  240. padding: ${space(1)} ${space(1.5)};
  241. border-top: 1px solid ${p => p.theme.border};
  242. `;
  243. const PanelItemBody = styled('div')`
  244. transition: height 0.1s ease;
  245. width: 100%;
  246. overflow: hidden;
  247. `;
  248. // TODO: simple layout stuff like this should come from a primitive component and we should really stop this `styled` nonsense
  249. const PanelItemBodyInner = styled('div')`
  250. padding-top: ${space(1.5)};
  251. `;
  252. const EmptyStateDescription = styled('div')`
  253. font-size: ${p => p.theme.fontSizeMedium};
  254. `;