usageStatsOrg.tsx 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. import type {MouseEvent as ReactMouseEvent} from 'react';
  2. import {Fragment} from 'react';
  3. import styled from '@emotion/styled';
  4. import isEqual from 'lodash/isEqual';
  5. import moment from 'moment-timezone';
  6. import {navigateTo} from 'sentry/actionCreators/navigation';
  7. import type {TooltipSubLabel} from 'sentry/components/charts/components/tooltip';
  8. import OptionSelector from 'sentry/components/charts/optionSelector';
  9. import {InlineContainer, SectionHeading} from 'sentry/components/charts/styles';
  10. import type {DateTimeObject} from 'sentry/components/charts/utils';
  11. import {getSeriesApiInterval} from 'sentry/components/charts/utils';
  12. import {Flex} from 'sentry/components/container/flex';
  13. import DeprecatedAsyncComponent from 'sentry/components/deprecatedAsyncComponent';
  14. import ErrorBoundary from 'sentry/components/errorBoundary';
  15. import ExternalLink from 'sentry/components/links/externalLink';
  16. import NotAvailable from 'sentry/components/notAvailable';
  17. import QuestionTooltip from 'sentry/components/questionTooltip';
  18. import type {ScoreCardProps} from 'sentry/components/scoreCard';
  19. import ScoreCard from 'sentry/components/scoreCard';
  20. import SwitchButton from 'sentry/components/switchButton';
  21. import {DEFAULT_STATS_PERIOD} from 'sentry/constants';
  22. import {t, tct} from 'sentry/locale';
  23. import {space} from 'sentry/styles/space';
  24. import type {DataCategoryInfo, IntervalPeriod} from 'sentry/types/core';
  25. import type {WithRouterProps} from 'sentry/types/legacyReactRouter';
  26. import type {Organization} from 'sentry/types/organization';
  27. import {trackAnalytics} from 'sentry/utils/analytics';
  28. import {parsePeriodToHours} from 'sentry/utils/duration/parsePeriodToHours';
  29. import {FORMAT_DATETIME_DAILY, FORMAT_DATETIME_HOURLY} from './usageChart/utils';
  30. import {mapSeriesToChart} from './mapSeriesToChart';
  31. import type {UsageSeries} from './types';
  32. import type {ChartStats, UsageChartProps} from './usageChart';
  33. import UsageChart, {
  34. CHART_OPTIONS_DATA_TRANSFORM,
  35. ChartDataTransform,
  36. SeriesTypes,
  37. } from './usageChart';
  38. import UsageStatsPerMin from './usageStatsPerMin';
  39. import {isDisplayUtc} from './utils';
  40. export interface UsageStatsOrganizationProps extends WithRouterProps {
  41. dataCategory: DataCategoryInfo['plural'];
  42. dataCategoryApiName: DataCategoryInfo['apiName'];
  43. dataCategoryName: string;
  44. dataDatetime: DateTimeObject;
  45. handleChangeState: (state: {
  46. clientDiscard?: boolean;
  47. dataCategory?: DataCategoryInfo['plural'];
  48. pagePeriod?: string | null;
  49. transform?: ChartDataTransform;
  50. }) => void;
  51. isSingleProject: boolean;
  52. organization: Organization;
  53. projectIds: number[];
  54. chartTransform?: string;
  55. clientDiscard?: boolean;
  56. }
  57. type UsageStatsOrganizationState = {
  58. orgStats: UsageSeries | undefined;
  59. metricOrgStats?: UsageSeries | undefined;
  60. } & DeprecatedAsyncComponent['state'];
  61. /**
  62. * This component is replaced by EnhancedUsageStatsOrganization in getsentry, which inherits
  63. * heavily from this one. Take care if changing any existing function signatures to ensure backwards
  64. * compatibility.
  65. */
  66. class UsageStatsOrganization<
  67. P extends UsageStatsOrganizationProps = UsageStatsOrganizationProps,
  68. S extends UsageStatsOrganizationState = UsageStatsOrganizationState,
  69. > extends DeprecatedAsyncComponent<P, S> {
  70. componentDidUpdate(prevProps: UsageStatsOrganizationProps) {
  71. const {
  72. dataDatetime: prevDateTime,
  73. projectIds: prevProjectIds,
  74. dataCategoryApiName: prevDataCategoryApiName,
  75. } = prevProps;
  76. const {
  77. dataDatetime: currDateTime,
  78. projectIds: currProjectIds,
  79. dataCategoryApiName: currentDataCategoryApiName,
  80. } = this.props;
  81. if (
  82. prevDateTime.start !== currDateTime.start ||
  83. prevDateTime.end !== currDateTime.end ||
  84. prevDateTime.period !== currDateTime.period ||
  85. prevDateTime.utc !== currDateTime.utc ||
  86. prevDataCategoryApiName !== currentDataCategoryApiName ||
  87. !isEqual(prevProjectIds, currProjectIds)
  88. ) {
  89. this.reloadData();
  90. }
  91. }
  92. getEndpoints(): ReturnType<DeprecatedAsyncComponent['getEndpoints']> {
  93. return [['orgStats', this.endpointPath, {query: this.endpointQuery}]];
  94. }
  95. /** List of components to render on single-project view */
  96. get projectDetails(): JSX.Element[] {
  97. return [];
  98. }
  99. get endpointPath() {
  100. const {organization} = this.props;
  101. return `/organizations/${organization.slug}/stats_v2/`;
  102. }
  103. get endpointQueryDatetime() {
  104. const {dataDatetime} = this.props;
  105. const queryDatetime =
  106. dataDatetime.start && dataDatetime.end
  107. ? {
  108. start: dataDatetime.start,
  109. end: dataDatetime.end,
  110. utc: dataDatetime.utc,
  111. }
  112. : {
  113. statsPeriod: dataDatetime.period || DEFAULT_STATS_PERIOD,
  114. };
  115. return queryDatetime;
  116. }
  117. get endpointQuery() {
  118. const {dataDatetime, projectIds, dataCategoryApiName} = this.props;
  119. const queryDatetime = this.endpointQueryDatetime;
  120. return {
  121. ...queryDatetime,
  122. interval: getSeriesApiInterval(dataDatetime),
  123. groupBy: ['outcome', 'reason'],
  124. project: projectIds,
  125. field: ['sum(quantity)'],
  126. category: dataCategoryApiName,
  127. };
  128. }
  129. get chartData(): {
  130. cardStats: {
  131. accepted?: string;
  132. filtered?: string;
  133. invalid?: string;
  134. rateLimited?: string;
  135. total?: string;
  136. };
  137. chartDateEnd: string;
  138. chartDateEndDisplay: string;
  139. chartDateInterval: IntervalPeriod;
  140. chartDateStart: string;
  141. chartDateStartDisplay: string;
  142. chartDateTimezoneDisplay: string;
  143. chartDateUtc: boolean;
  144. chartStats: ChartStats;
  145. chartSubLabels: TooltipSubLabel[];
  146. chartTransform: ChartDataTransform;
  147. dataError?: Error;
  148. } {
  149. return {
  150. ...mapSeriesToChart({
  151. orgStats: this.state.orgStats,
  152. chartDateInterval: this.chartDateRange.chartDateInterval,
  153. chartDateUtc: this.chartDateRange.chartDateUtc,
  154. dataCategory: this.props.dataCategory,
  155. endpointQuery: this.endpointQuery,
  156. }),
  157. ...this.chartDateRange,
  158. ...this.chartTransform,
  159. };
  160. }
  161. get chartTransform(): {chartTransform: ChartDataTransform} {
  162. const {chartTransform} = this.props;
  163. switch (chartTransform) {
  164. case ChartDataTransform.CUMULATIVE:
  165. case ChartDataTransform.PERIODIC:
  166. return {chartTransform};
  167. default:
  168. return {chartTransform: ChartDataTransform.PERIODIC};
  169. }
  170. }
  171. get chartDateRange(): {
  172. chartDateEnd: string;
  173. chartDateEndDisplay: string;
  174. chartDateInterval: IntervalPeriod;
  175. chartDateStart: string;
  176. chartDateStartDisplay: string;
  177. chartDateTimezoneDisplay: string;
  178. chartDateUtc: boolean;
  179. } {
  180. const {orgStats} = this.state;
  181. const {dataDatetime} = this.props;
  182. const interval = getSeriesApiInterval(dataDatetime);
  183. // Use fillers as loading/error states will not display datetime at all
  184. if (!orgStats || !orgStats.intervals) {
  185. return {
  186. chartDateInterval: interval,
  187. chartDateStart: '',
  188. chartDateEnd: '',
  189. chartDateUtc: true,
  190. chartDateStartDisplay: '',
  191. chartDateEndDisplay: '',
  192. chartDateTimezoneDisplay: '',
  193. };
  194. }
  195. const {intervals} = orgStats;
  196. const intervalHours = parsePeriodToHours(interval);
  197. // Keep datetime in UTC until we want to display it to users
  198. const startTime = moment(intervals[0]).utc();
  199. const endTime =
  200. intervals.length < 2
  201. ? moment(startTime) // when statsPeriod and interval is the same value
  202. : moment(intervals[intervals.length - 1]).utc();
  203. const useUtc = isDisplayUtc(dataDatetime);
  204. // If interval is a day or more, use UTC to format date. Otherwise, the date
  205. // may shift ahead/behind when converting to the user's local time.
  206. const FORMAT_DATETIME =
  207. intervalHours >= 24 ? FORMAT_DATETIME_DAILY : FORMAT_DATETIME_HOURLY;
  208. const xAxisStart = moment(startTime);
  209. const xAxisEnd = moment(endTime);
  210. const displayStart = useUtc ? moment(startTime).utc() : moment(startTime).local();
  211. const displayEnd = useUtc ? moment(endTime).utc() : moment(endTime).local();
  212. if (intervalHours < 24) {
  213. displayEnd.add(intervalHours, 'h');
  214. }
  215. return {
  216. chartDateInterval: interval,
  217. chartDateStart: xAxisStart.format(),
  218. chartDateEnd: xAxisEnd.format(),
  219. chartDateUtc: useUtc,
  220. chartDateStartDisplay: displayStart.format(FORMAT_DATETIME),
  221. chartDateEndDisplay: displayEnd.format(FORMAT_DATETIME),
  222. chartDateTimezoneDisplay: displayStart.format('Z'),
  223. };
  224. }
  225. get chartProps(): UsageChartProps {
  226. const {dataCategory, clientDiscard, handleChangeState} = this.props;
  227. const {error, errors, loading} = this.state;
  228. const {
  229. chartStats,
  230. dataError,
  231. chartDateInterval,
  232. chartDateStart,
  233. chartDateEnd,
  234. chartDateUtc,
  235. chartTransform,
  236. chartSubLabels,
  237. } = this.chartData;
  238. const hasError = error || !!dataError;
  239. const chartErrors: any = dataError ? {...errors, data: dataError} : errors; // TODO(ts): AsyncComponent
  240. const chartProps = {
  241. isLoading: loading,
  242. isError: hasError,
  243. errors: chartErrors,
  244. title: (
  245. <Fragment>
  246. {t('Project(s) Stats')}
  247. <QuestionTooltip
  248. size="xs"
  249. title={tct(
  250. 'You can find more information about each category in our [link:docs]',
  251. {
  252. link: (
  253. <ExternalLink
  254. href="https://docs.sentry.io/product/stats/#usage-stats"
  255. onClick={() => this.handleOnDocsClick('chart-title')}
  256. />
  257. ),
  258. }
  259. )}
  260. isHoverable
  261. />
  262. </Fragment>
  263. ),
  264. footer: this.renderChartFooter(),
  265. dataCategory,
  266. dataTransform: chartTransform,
  267. usageDateStart: chartDateStart,
  268. usageDateEnd: chartDateEnd,
  269. usageDateShowUtc: chartDateUtc,
  270. usageDateInterval: chartDateInterval,
  271. usageStats: chartStats,
  272. chartTooltip: {
  273. subLabels: chartSubLabels,
  274. skipZeroValuedSubLabels: true,
  275. },
  276. legendSelected: {[SeriesTypes.CLIENT_DISCARD]: !!clientDiscard},
  277. onLegendSelectChanged: ({name, selected}) => {
  278. if (name === SeriesTypes.CLIENT_DISCARD) {
  279. handleChangeState({clientDiscard: selected[name]});
  280. }
  281. },
  282. } as UsageChartProps;
  283. return chartProps;
  284. }
  285. handleOnDocsClick = (
  286. source:
  287. | 'card-accepted'
  288. | 'card-filtered'
  289. | 'card-rate-limited'
  290. | 'card-invalid'
  291. | 'chart-title'
  292. ) => {
  293. const {organization} = this.props;
  294. trackAnalytics('stats.docs_clicked', {
  295. organization,
  296. source,
  297. });
  298. };
  299. get cardMetadata() {
  300. const {
  301. dataCategory,
  302. dataCategoryName,
  303. organization,
  304. projectIds,
  305. router,
  306. dataCategoryApiName,
  307. } = this.props;
  308. const {total, accepted, invalid, rateLimited, filtered} = this.chartData.cardStats;
  309. const navigateToInboundFilterSettings = (event: ReactMouseEvent) => {
  310. event.preventDefault();
  311. const url = `/settings/${organization.slug}/projects/:projectId/filters/data-filters/`;
  312. if (router) {
  313. navigateTo(url, router);
  314. }
  315. };
  316. const cardMetadata: Record<string, ScoreCardProps> = {
  317. total: {
  318. title: tct('Total [dataCategory]', {dataCategory: dataCategoryName}),
  319. score: total,
  320. },
  321. accepted: {
  322. title: tct('Accepted [dataCategory]', {dataCategory: dataCategoryName}),
  323. help: tct(
  324. 'Accepted [dataCategory] were successfully processed by Sentry. For more information, read our [docsLink:docs].',
  325. {
  326. dataCategory,
  327. docsLink: (
  328. <ExternalLink
  329. href="https://docs.sentry.io/product/stats/#accepted"
  330. onClick={() => this.handleOnDocsClick('card-accepted')}
  331. />
  332. ),
  333. }
  334. ),
  335. score: accepted,
  336. trend: (
  337. <UsageStatsPerMin
  338. dataCategoryApiName={dataCategoryApiName}
  339. dataCategory={dataCategory}
  340. organization={organization}
  341. projectIds={projectIds}
  342. />
  343. ),
  344. },
  345. filtered: {
  346. title: tct('Filtered [dataCategory]', {dataCategory: dataCategoryName}),
  347. help: tct(
  348. 'Filtered [dataCategory] were blocked due to your [filterSettings: inbound data filter] rules. For more information, read our [docsLink:docs].',
  349. {
  350. dataCategory,
  351. filterSettings: (
  352. <a href="#" onClick={event => navigateToInboundFilterSettings(event)} />
  353. ),
  354. docsLink: (
  355. <ExternalLink
  356. href="https://docs.sentry.io/product/stats/#filtered"
  357. onClick={() => this.handleOnDocsClick('card-filtered')}
  358. />
  359. ),
  360. }
  361. ),
  362. score: filtered,
  363. },
  364. rateLimited: {
  365. title: tct('Rate Limited [dataCategory]', {dataCategory: dataCategoryName}),
  366. help: tct(
  367. 'Rate Limited [dataCategory] were discarded due to rate limits or quota. For more information, read our [docsLink:docs].',
  368. {
  369. dataCategory,
  370. docsLink: (
  371. <ExternalLink
  372. href="https://docs.sentry.io/product/stats/#rate-limited"
  373. onClick={() => this.handleOnDocsClick('card-rate-limited')}
  374. />
  375. ),
  376. }
  377. ),
  378. score: rateLimited,
  379. },
  380. invalid: {
  381. title: tct('Invalid [dataCategory]', {dataCategory: dataCategoryName}),
  382. help: tct(
  383. 'Invalid [dataCategory] were sent by the SDK and were discarded because the data did not meet the basic schema requirements. For more information, read our [docsLink:docs].',
  384. {
  385. dataCategory,
  386. docsLink: (
  387. <ExternalLink
  388. href="https://docs.sentry.io/product/stats/#invalid"
  389. onClick={() => this.handleOnDocsClick('card-invalid')}
  390. />
  391. ),
  392. }
  393. ),
  394. score: invalid,
  395. },
  396. };
  397. return cardMetadata;
  398. }
  399. renderCards() {
  400. const {loading} = this.state;
  401. const cardMetadata = Object.values(this.cardMetadata);
  402. return cardMetadata.map((card, i) => (
  403. <StyledScoreCard
  404. key={i}
  405. title={card.title}
  406. score={loading ? undefined : card.score}
  407. help={card.help}
  408. trend={card.trend}
  409. isTooltipHoverable
  410. />
  411. ));
  412. }
  413. renderChart() {
  414. const {loading} = this.state;
  415. return <UsageChart {...this.chartProps} isLoading={loading} />;
  416. }
  417. renderChartFooter = () => {
  418. const {handleChangeState, clientDiscard} = this.props;
  419. const {loading, error} = this.state;
  420. const {
  421. chartDateInterval,
  422. chartTransform,
  423. chartDateStartDisplay,
  424. chartDateEndDisplay,
  425. chartDateTimezoneDisplay,
  426. } = this.chartData;
  427. return (
  428. <Footer>
  429. <InlineContainer>
  430. <FooterDate>
  431. <SectionHeading>{t('Date Range:')}</SectionHeading>
  432. <span>
  433. {loading || error ? (
  434. <NotAvailable />
  435. ) : (
  436. tct('[start] — [end] ([timezone] UTC, [interval] interval)', {
  437. start: chartDateStartDisplay,
  438. end: chartDateEndDisplay,
  439. timezone: chartDateTimezoneDisplay,
  440. interval: chartDateInterval,
  441. })
  442. )}
  443. </span>
  444. </FooterDate>
  445. </InlineContainer>
  446. <InlineContainer>
  447. {(this.chartData.chartStats.clientDiscard ?? []).length > 0 && (
  448. <Flex align="center" gap={space(1)}>
  449. <strong>{t('Show client-discarded data:')}</strong>
  450. <SwitchButton
  451. toggle={() => {
  452. handleChangeState({clientDiscard: !clientDiscard});
  453. }}
  454. isActive={clientDiscard}
  455. />
  456. </Flex>
  457. )}
  458. </InlineContainer>
  459. <InlineContainer>
  460. <OptionSelector
  461. title={t('Type')}
  462. selected={chartTransform}
  463. options={CHART_OPTIONS_DATA_TRANSFORM}
  464. onChange={(val: string) =>
  465. handleChangeState({transform: val as ChartDataTransform})
  466. }
  467. />
  468. </InlineContainer>
  469. </Footer>
  470. );
  471. };
  472. renderProjectDetails() {
  473. const {isSingleProject} = this.props;
  474. const projectDetails = this.projectDetails.map((projectDetailComponent, i) => (
  475. <ErrorBoundary mini key={i}>
  476. {projectDetailComponent}
  477. </ErrorBoundary>
  478. ));
  479. return isSingleProject ? projectDetails : null;
  480. }
  481. renderComponent() {
  482. return (
  483. <Fragment>
  484. <PageGrid>
  485. {this.renderCards()}
  486. <ChartWrapper data-test-id="usage-stats-chart">
  487. {this.renderChart()}
  488. </ChartWrapper>
  489. </PageGrid>
  490. {this.renderProjectDetails()}
  491. </Fragment>
  492. );
  493. }
  494. }
  495. export default UsageStatsOrganization;
  496. const PageGrid = styled('div')`
  497. display: grid;
  498. grid-template-columns: 1fr;
  499. gap: ${space(2)};
  500. @media (min-width: ${p => p.theme.breakpoints.small}) {
  501. grid-template-columns: repeat(2, 1fr);
  502. }
  503. @media (min-width: ${p => p.theme.breakpoints.large}) {
  504. grid-template-columns: repeat(5, 1fr);
  505. }
  506. `;
  507. const StyledScoreCard = styled(ScoreCard)`
  508. grid-column: auto / span 1;
  509. margin: 0;
  510. `;
  511. const ChartWrapper = styled('div')`
  512. grid-column: 1 / -1;
  513. `;
  514. const Footer = styled('div')`
  515. display: flex;
  516. flex-direction: row;
  517. flex-wrap: wrap;
  518. align-items: center;
  519. gap: ${space(1.5)};
  520. padding: ${space(1)} ${space(3)};
  521. border-top: 1px solid ${p => p.theme.border};
  522. > *:first-child {
  523. flex-grow: 1;
  524. }
  525. `;
  526. const FooterDate = styled('div')`
  527. display: flex;
  528. flex-direction: row;
  529. align-items: center;
  530. > ${SectionHeading} {
  531. margin-right: ${space(1.5)};
  532. }
  533. > span:last-child {
  534. font-weight: ${p => p.theme.fontWeightNormal};
  535. font-size: ${p => p.theme.fontSizeMedium};
  536. }
  537. `;