content.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. import {Fragment, useCallback, useEffect, useMemo} from 'react';
  2. import {browserHistory} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import {Location} from 'history';
  5. import {Alert} from 'sentry/components/alert';
  6. import {Button} from 'sentry/components/button';
  7. import DatePageFilter from 'sentry/components/datePageFilter';
  8. import EnvironmentPageFilter from 'sentry/components/environmentPageFilter';
  9. import SearchBar from 'sentry/components/events/searchBar';
  10. import FeatureBadge from 'sentry/components/featureBadge';
  11. import * as Layout from 'sentry/components/layouts/thirds';
  12. import PageFilterBar from 'sentry/components/organizations/pageFilterBar';
  13. import PageFiltersContainer from 'sentry/components/organizations/pageFilters/container';
  14. import {PageHeadingQuestionTooltip} from 'sentry/components/pageHeadingQuestionTooltip';
  15. import Pagination from 'sentry/components/pagination';
  16. import {
  17. ProfilingAM1OrMMXUpgrade,
  18. ProfilingBetaAlertBanner,
  19. ProfilingUpgradeButton,
  20. } from 'sentry/components/profiling/billing/alerts';
  21. import {ProfileEventsTable} from 'sentry/components/profiling/profileEventsTable';
  22. import ProjectPageFilter from 'sentry/components/projectPageFilter';
  23. import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
  24. import {SidebarPanelKey} from 'sentry/components/sidebar/types';
  25. import SmartSearchBar, {SmartSearchBarProps} from 'sentry/components/smartSearchBar';
  26. import {MAX_QUERY_LENGTH} from 'sentry/constants';
  27. import {ALL_ACCESS_PROJECTS} from 'sentry/constants/pageFilters';
  28. import {t} from 'sentry/locale';
  29. import SidebarPanelStore from 'sentry/stores/sidebarPanelStore';
  30. import {space} from 'sentry/styles/space';
  31. import {Organization} from 'sentry/types';
  32. import {trackAnalytics} from 'sentry/utils/analytics';
  33. import EventView from 'sentry/utils/discover/eventView';
  34. import {useProfileEvents} from 'sentry/utils/profiling/hooks/useProfileEvents';
  35. import {useProfileFilters} from 'sentry/utils/profiling/hooks/useProfileFilters';
  36. import {formatError, formatSort} from 'sentry/utils/profiling/hooks/utils';
  37. import {decodeScalar} from 'sentry/utils/queryString';
  38. import useOrganization from 'sentry/utils/useOrganization';
  39. import usePageFilters from 'sentry/utils/usePageFilters';
  40. import useProjects from 'sentry/utils/useProjects';
  41. import {DEFAULT_PROFILING_DATETIME_SELECTION} from 'sentry/views/profiling/utils';
  42. import {ProfileCharts} from './landing/profileCharts';
  43. import {ProfilingSlowestTransactionsPanel} from './landing/profilingSlowestTransactionsPanel';
  44. import {ProfilingOnboardingPanel} from './profilingOnboardingPanel';
  45. interface ProfilingContentProps {
  46. location: Location;
  47. }
  48. function ProfilingContent({location}: ProfilingContentProps) {
  49. const organization = useOrganization();
  50. const {selection} = usePageFilters();
  51. const cursor = decodeScalar(location.query.cursor);
  52. const query = decodeScalar(location.query.query, '');
  53. const profilingUsingTransactions = organization.features.includes(
  54. 'profiling-using-transactions'
  55. );
  56. const fields = profilingUsingTransactions ? ALL_FIELDS : BASE_FIELDS;
  57. const sort = formatSort<FieldType>(decodeScalar(location.query.sort), fields, {
  58. key: 'p95()',
  59. order: 'desc',
  60. });
  61. const profileFilters = useProfileFilters({
  62. query: '',
  63. selection,
  64. disabled: profilingUsingTransactions,
  65. });
  66. const {projects} = useProjects();
  67. const transactions = useProfileEvents<FieldType>({
  68. cursor,
  69. fields,
  70. query,
  71. sort,
  72. referrer: 'api.profiling.landing-table',
  73. });
  74. const transactionsError =
  75. transactions.status === 'error' ? formatError(transactions.error) : null;
  76. useEffect(() => {
  77. trackAnalytics('profiling_views.landing', {
  78. organization,
  79. });
  80. }, [organization]);
  81. const handleSearch: SmartSearchBarProps['onSearch'] = useCallback(
  82. (searchQuery: string) => {
  83. browserHistory.push({
  84. ...location,
  85. query: {
  86. ...location.query,
  87. cursor: undefined,
  88. query: searchQuery || undefined,
  89. },
  90. });
  91. },
  92. [location]
  93. );
  94. // Open the modal on demand
  95. const onSetupProfilingClick = useCallback(() => {
  96. trackAnalytics('profiling_views.onboarding', {
  97. organization,
  98. });
  99. SidebarPanelStore.activatePanel(SidebarPanelKey.ProfilingOnboarding);
  100. }, [organization]);
  101. const shouldShowProfilingOnboardingPanel = useMemo((): boolean => {
  102. // if it's My Projects or All projects, only show onboarding if we can't
  103. // find any projects with profiles
  104. if (
  105. selection.projects.length === 0 ||
  106. selection.projects[0] === ALL_ACCESS_PROJECTS
  107. ) {
  108. return projects.every(project => !project.hasProfiles);
  109. }
  110. // otherwise, only show onboarding if we can't find any projects with profiles
  111. // from those that were selected
  112. const projectsWithProfiles = new Set(
  113. projects.filter(project => project.hasProfiles).map(project => project.id)
  114. );
  115. return selection.projects.every(
  116. project => !projectsWithProfiles.has(String(project))
  117. );
  118. }, [selection.projects, projects]);
  119. const eventView = useMemo(() => {
  120. const _eventView = EventView.fromNewQueryWithLocation(
  121. {
  122. id: undefined,
  123. version: 2,
  124. name: t('Profiling'),
  125. fields: [],
  126. query,
  127. projects: selection.projects,
  128. },
  129. location
  130. );
  131. _eventView.additionalConditions.setFilterValues('has', ['profile.id']);
  132. return _eventView;
  133. }, [location, query, selection.projects]);
  134. const isProfilingGA = organization.features.includes('profiling-ga');
  135. return (
  136. <SentryDocumentTitle title={t('Profiling')} orgSlug={organization.slug}>
  137. <PageFiltersContainer
  138. defaultSelection={
  139. profilingUsingTransactions
  140. ? {datetime: DEFAULT_PROFILING_DATETIME_SELECTION}
  141. : undefined
  142. }
  143. >
  144. <Layout.Page>
  145. {isProfilingGA ? (
  146. <ProfilingBetaAlertBanner organization={organization} />
  147. ) : (
  148. <ProfilingBetaEndAlertBanner organization={organization} />
  149. )}
  150. <Layout.Header>
  151. <Layout.HeaderContent>
  152. <Layout.Title>
  153. {t('Profiling')}
  154. <PageHeadingQuestionTooltip
  155. docsUrl="https://docs.sentry.io/product/profiling/"
  156. title={t(
  157. 'Profiling collects detailed information in production about the functions executing in your application and how long they take to run, giving you code-level visibility into your hot paths.'
  158. )}
  159. />
  160. {isProfilingGA ? (
  161. <FeatureBadge type="new" />
  162. ) : (
  163. <FeatureBadge type="beta" />
  164. )}
  165. </Layout.Title>
  166. </Layout.HeaderContent>
  167. </Layout.Header>
  168. <Layout.Body>
  169. <Layout.Main fullWidth>
  170. {transactionsError && (
  171. <Alert type="error" showIcon>
  172. {transactionsError}
  173. </Alert>
  174. )}
  175. <ActionBar>
  176. <PageFilterBar condensed>
  177. <ProjectPageFilter />
  178. <EnvironmentPageFilter />
  179. <DatePageFilter alignDropdown="left" />
  180. </PageFilterBar>
  181. {profilingUsingTransactions ? (
  182. <SearchBar
  183. searchSource="profile_summary"
  184. organization={organization}
  185. projectIds={eventView.project}
  186. query={query}
  187. onSearch={handleSearch}
  188. maxQueryLength={MAX_QUERY_LENGTH}
  189. />
  190. ) : (
  191. <SmartSearchBar
  192. organization={organization}
  193. hasRecentSearches
  194. searchSource="profile_landing"
  195. supportedTags={profileFilters}
  196. query={query}
  197. onSearch={handleSearch}
  198. maxQueryLength={MAX_QUERY_LENGTH}
  199. />
  200. )}
  201. </ActionBar>
  202. {shouldShowProfilingOnboardingPanel ? (
  203. isProfilingGA ? (
  204. // If user is on m2, show default
  205. <ProfilingOnboardingPanel
  206. content={
  207. <ProfilingAM1OrMMXUpgrade
  208. organization={organization}
  209. fallback={
  210. <Fragment>
  211. <h3>{t('Function level insights')}</h3>
  212. <p>
  213. {t(
  214. 'Discover slow-to-execute or resource intensive functions within your application'
  215. )}
  216. </p>
  217. </Fragment>
  218. }
  219. />
  220. }
  221. >
  222. <ProfilingUpgradeButton
  223. organization={organization}
  224. priority="primary"
  225. fallback={
  226. <Button onClick={onSetupProfilingClick} priority="primary">
  227. {t('Set Up Profiling')}
  228. </Button>
  229. }
  230. >
  231. {t('Update plan')}
  232. </ProfilingUpgradeButton>
  233. <Button href="https://docs.sentry.io/product/profiling/" external>
  234. {t('Read Docs')}
  235. </Button>
  236. </ProfilingOnboardingPanel>
  237. ) : (
  238. // show previous state
  239. <ProfilingOnboardingPanel>
  240. <Button onClick={onSetupProfilingClick} priority="primary">
  241. {t('Set Up Profiling')}
  242. </Button>
  243. <Button href="https://docs.sentry.io/product/profiling/" external>
  244. {t('Read Docs')}
  245. </Button>
  246. </ProfilingOnboardingPanel>
  247. )
  248. ) : (
  249. <Fragment>
  250. <PanelsGrid>
  251. <ProfilingSlowestTransactionsPanel />
  252. <ProfileCharts query={query} selection={selection} hideCount />
  253. </PanelsGrid>
  254. <ProfileEventsTable
  255. columns={fields.slice()}
  256. data={transactions.status === 'success' ? transactions.data : null}
  257. error={
  258. transactions.status === 'error'
  259. ? t('Unable to load profiles')
  260. : null
  261. }
  262. isLoading={transactions.status === 'loading'}
  263. sort={sort}
  264. sortableColumns={new Set(fields)}
  265. />
  266. <Pagination
  267. pageLinks={
  268. transactions.status === 'success'
  269. ? transactions.getResponseHeader?.('Link') ?? null
  270. : null
  271. }
  272. />
  273. </Fragment>
  274. )}
  275. </Layout.Main>
  276. </Layout.Body>
  277. </Layout.Page>
  278. </PageFiltersContainer>
  279. </SentryDocumentTitle>
  280. );
  281. }
  282. function ProfilingBetaEndAlertBanner({organization}: {organization: Organization}) {
  283. // beta users will continue to have access
  284. if (organization.features.includes('profiling-beta')) {
  285. return null;
  286. }
  287. return (
  288. <StyledAlert system type="info">
  289. {t(
  290. "The beta program for Profiling is now closed, but Profiling will become generally available soon. If you weren't part of the beta program, any Profiles sent during this time won't appear in your dashboard. Check out the What’s New tab for updates."
  291. )}
  292. </StyledAlert>
  293. );
  294. }
  295. const BASE_FIELDS = [
  296. 'transaction',
  297. 'project.id',
  298. 'last_seen()',
  299. 'p75()',
  300. 'p95()',
  301. 'p99()',
  302. 'count()',
  303. ] as const;
  304. // user misery is only available with the profiling-using-transactions feature
  305. const ALL_FIELDS = [...BASE_FIELDS, 'user_misery()'] as const;
  306. type FieldType = (typeof ALL_FIELDS)[number];
  307. const ActionBar = styled('div')`
  308. display: grid;
  309. gap: ${space(2)};
  310. grid-template-columns: min-content auto;
  311. margin-bottom: ${space(2)};
  312. `;
  313. // TODO: another simple primitive that can easily be <Grid columns={2} />
  314. const PanelsGrid = styled('div')`
  315. display: grid;
  316. grid-template-columns: minmax(0, 1fr) 1fr;
  317. gap: ${space(2)};
  318. @media (max-width: ${p => p.theme.breakpoints.small}) {
  319. grid-template-columns: minmax(0, 1fr);
  320. }
  321. `;
  322. const StyledAlert = styled(Alert)`
  323. margin: 0;
  324. `;
  325. export default ProfilingContent;