import type {MouseEvent as ReactMouseEvent} from 'react'; import {Fragment} from 'react'; import type {WithRouterProps} from 'react-router'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import isEqual from 'lodash/isEqual'; import moment from 'moment'; import {navigateTo} from 'sentry/actionCreators/navigation'; import OptionSelector from 'sentry/components/charts/optionSelector'; import {InlineContainer, SectionHeading} from 'sentry/components/charts/styles'; import type {DateTimeObject} from 'sentry/components/charts/utils'; import {getSeriesApiInterval} from 'sentry/components/charts/utils'; import DeprecatedAsyncComponent from 'sentry/components/deprecatedAsyncComponent'; import ErrorBoundary from 'sentry/components/errorBoundary'; import NotAvailable from 'sentry/components/notAvailable'; import type {ScoreCardProps} from 'sentry/components/scoreCard'; import ScoreCard from 'sentry/components/scoreCard'; import {DATA_CATEGORY_INFO, DEFAULT_STATS_PERIOD} from 'sentry/constants'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {DataCategoryInfo, IntervalPeriod, Organization} from 'sentry/types'; import {Outcome} from 'sentry/types'; import {parsePeriodToHours} from 'sentry/utils/duration/parsePeriodToHours'; import {hasCustomMetrics} from 'sentry/utils/metrics/features'; import { FORMAT_DATETIME_DAILY, FORMAT_DATETIME_HOURLY, getDateFromMoment, } from './usageChart/utils'; import type {UsageSeries, UsageStat} from './types'; import type {ChartStats, UsageChartProps} from './usageChart'; import UsageChart, {CHART_OPTIONS_DATA_TRANSFORM, ChartDataTransform} from './usageChart'; import UsageStatsPerMin from './usageStatsPerMin'; import {formatUsageWithUnits, getFormatUsageOptions, isDisplayUtc} from './utils'; export interface UsageStatsOrganizationProps extends WithRouterProps { dataCategory: DataCategoryInfo['plural']; dataCategoryName: string; dataDatetime: DateTimeObject; handleChangeState: (state: { dataCategory?: DataCategoryInfo['plural']; pagePeriod?: string | null; transform?: ChartDataTransform; }) => void; isSingleProject: boolean; organization: Organization; projectIds: number[]; chartTransform?: string; } type UsageStatsOrganizationState = { orgStats: UsageSeries | undefined; metricOrgStats?: UsageSeries | undefined; } & DeprecatedAsyncComponent['state']; /** * This component is replaced by EnhancedUsageStatsOrganization in getsentry, which inherits * heavily from this one. Take care if changing any existing function signatures to ensure backwards * compatibility. */ class UsageStatsOrganization< P extends UsageStatsOrganizationProps = UsageStatsOrganizationProps, S extends UsageStatsOrganizationState = UsageStatsOrganizationState, > extends DeprecatedAsyncComponent { componentDidUpdate(prevProps: UsageStatsOrganizationProps) { const {dataDatetime: prevDateTime, projectIds: prevProjectIds} = prevProps; const {dataDatetime: currDateTime, projectIds: currProjectIds} = this.props; if ( prevDateTime.start !== currDateTime.start || prevDateTime.end !== currDateTime.end || prevDateTime.period !== currDateTime.period || prevDateTime.utc !== currDateTime.utc || !isEqual(prevProjectIds, currProjectIds) ) { this.reloadData(); } } getEndpoints(): ReturnType { return [ ['orgStats', this.endpointPath, {query: this.endpointQuery}], ...this.metricsEndpoint, ]; } /** List of components to render on single-project view */ get projectDetails(): JSX.Element[] { return []; } get endpointPath() { const {organization} = this.props; return `/organizations/${organization.slug}/stats_v2/`; } get endpointQueryDatetime() { const {dataDatetime} = this.props; const queryDatetime = dataDatetime.start && dataDatetime.end ? { start: dataDatetime.start, end: dataDatetime.end, utc: dataDatetime.utc, } : { statsPeriod: dataDatetime.period || DEFAULT_STATS_PERIOD, }; return queryDatetime; } get endpointQuery() { const {dataDatetime, projectIds} = this.props; const queryDatetime = this.endpointQueryDatetime; return { ...queryDatetime, interval: getSeriesApiInterval(dataDatetime), groupBy: ['category', 'outcome'], project: projectIds, field: ['sum(quantity)'], }; } // Metric stats are not reported when grouping by category, so we make a separate request // and combine the results get metricsEndpoint(): ReturnType { if (hasCustomMetrics(this.props.organization)) { return [ [ 'metricOrgStats', this.endpointPath, { query: { ...this.endpointQuery, category: DATA_CATEGORY_INFO.metrics.apiName, groupBy: ['outcome'], }, }, ], ]; } return []; } // Combines non-metric and metric stats get orgStats() { const {orgStats, metricOrgStats} = this.state; if (!orgStats || !metricOrgStats) { return orgStats; } const metricsGroups = metricOrgStats.groups.map(group => { return { ...group, by: { ...group.by, category: DATA_CATEGORY_INFO.metrics.apiName, }, }; }); return { ...orgStats, groups: [...orgStats.groups, ...metricsGroups], }; } get chartData(): { cardStats: { accepted?: string; dropped?: string; filtered?: string; total?: string; }; chartDateEnd: string; chartDateEndDisplay: string; chartDateInterval: IntervalPeriod; chartDateStart: string; chartDateStartDisplay: string; chartDateTimezoneDisplay: string; chartDateUtc: boolean; chartStats: ChartStats; chartTransform: ChartDataTransform; dataError?: Error; } { return { ...this.mapSeriesToChart(this.orgStats), ...this.chartDateRange, ...this.chartTransform, }; } get chartTransform(): {chartTransform: ChartDataTransform} { const {chartTransform} = this.props; switch (chartTransform) { case ChartDataTransform.CUMULATIVE: case ChartDataTransform.PERIODIC: return {chartTransform}; default: return {chartTransform: ChartDataTransform.PERIODIC}; } } get chartDateRange(): { chartDateEnd: string; chartDateEndDisplay: string; chartDateInterval: IntervalPeriod; chartDateStart: string; chartDateStartDisplay: string; chartDateTimezoneDisplay: string; chartDateUtc: boolean; } { const {orgStats} = this.state; const {dataDatetime} = this.props; const interval = getSeriesApiInterval(dataDatetime); // Use fillers as loading/error states will not display datetime at all if (!orgStats || !orgStats.intervals) { return { chartDateInterval: interval, chartDateStart: '', chartDateEnd: '', chartDateUtc: true, chartDateStartDisplay: '', chartDateEndDisplay: '', chartDateTimezoneDisplay: '', }; } const {intervals} = orgStats; const intervalHours = parsePeriodToHours(interval); // Keep datetime in UTC until we want to display it to users const startTime = moment(intervals[0]).utc(); const endTime = intervals.length < 2 ? moment(startTime) // when statsPeriod and interval is the same value : moment(intervals[intervals.length - 1]).utc(); const useUtc = isDisplayUtc(dataDatetime); // If interval is a day or more, use UTC to format date. Otherwise, the date // may shift ahead/behind when converting to the user's local time. const FORMAT_DATETIME = intervalHours >= 24 ? FORMAT_DATETIME_DAILY : FORMAT_DATETIME_HOURLY; const xAxisStart = moment(startTime); const xAxisEnd = moment(endTime); const displayStart = useUtc ? moment(startTime).utc() : moment(startTime).local(); const displayEnd = useUtc ? moment(endTime).utc() : moment(endTime).local(); if (intervalHours < 24) { displayEnd.add(intervalHours, 'h'); } return { chartDateInterval: interval, chartDateStart: xAxisStart.format(), chartDateEnd: xAxisEnd.format(), chartDateUtc: useUtc, chartDateStartDisplay: displayStart.format(FORMAT_DATETIME), chartDateEndDisplay: displayEnd.format(FORMAT_DATETIME), chartDateTimezoneDisplay: displayStart.format('Z'), }; } get chartProps(): UsageChartProps { const {dataCategory} = this.props; const {error, errors, loading} = this.state; const { chartStats, dataError, chartDateInterval, chartDateStart, chartDateEnd, chartDateUtc, chartTransform, } = this.chartData; const hasError = error || !!dataError; const chartErrors: any = dataError ? {...errors, data: dataError} : errors; // TODO(ts): AsyncComponent const chartProps = { isLoading: loading, isError: hasError, errors: chartErrors, title: ' ', // Force the title to be blank footer: this.renderChartFooter(), dataCategory, dataTransform: chartTransform, usageDateStart: chartDateStart, usageDateEnd: chartDateEnd, usageDateShowUtc: chartDateUtc, usageDateInterval: chartDateInterval, usageStats: chartStats, } as UsageChartProps; return chartProps; } get cardMetadata() { const {dataCategory, dataCategoryName, organization, projectIds, router} = this.props; const {total, accepted, dropped, filtered} = this.chartData.cardStats; const navigateToInboundFilterSettings = (event: ReactMouseEvent) => { event.preventDefault(); const url = `/settings/${organization.slug}/projects/:projectId/filters/data-filters/`; if (router) { navigateTo(url, router); } }; const navigateToMetricsSettings = (event: ReactMouseEvent) => { event.preventDefault(); const url = `/settings/${organization.slug}/projects/:projectId/metrics/`; if (router) { navigateTo(url, router); } }; const cardMetadata: Record = { total: { title: tct('Total [dataCategory]', {dataCategory: dataCategoryName}), score: total, }, accepted: { title: tct('Accepted [dataCategory]', {dataCategory: dataCategoryName}), help: tct('Accepted [dataCategory] were successfully processed by Sentry', { dataCategory, }), score: accepted, trend: ( ), }, filtered: { title: tct('Filtered [dataCategory]', {dataCategory: dataCategoryName}), help: dataCategory === DATA_CATEGORY_INFO.metrics.plural ? tct( 'Filtered metrics were blocked due to your disabled metrics [settings: settings]', { dataCategory, settings: ( navigateToMetricsSettings(event)} /> ), } ) : tct( 'Filtered [dataCategory] were blocked due to your [filterSettings: inbound data filter] rules', { dataCategory, filterSettings: ( navigateToInboundFilterSettings(event)} /> ), } ), score: filtered, }, dropped: { title: tct('Dropped [dataCategory]', {dataCategory: dataCategoryName}), help: tct( 'Dropped [dataCategory] were discarded due to invalid data, rate-limits, quota limits, or spike protection', {dataCategory} ), score: dropped, }, }; return cardMetadata; } mapSeriesToChart(orgStats?: UsageSeries): { cardStats: { accepted?: string; dropped?: string; filtered?: string; total?: string; }; chartStats: ChartStats; dataError?: Error; } { const cardStats = { total: undefined, accepted: undefined, dropped: undefined, filtered: undefined, }; const chartStats: ChartStats = { accepted: [], dropped: [], projected: [], filtered: [], }; if (!orgStats) { return {cardStats, chartStats}; } try { const {dataCategory} = this.props; const {chartDateInterval, chartDateUtc} = this.chartDateRange; const usageStats: UsageStat[] = orgStats.intervals.map(interval => { const dateTime = moment(interval); return { date: getDateFromMoment(dateTime, chartDateInterval, chartDateUtc), total: 0, accepted: 0, filtered: 0, dropped: {total: 0}, }; }); // Tally totals for card data const count: Record<'total' | Outcome, number> = { total: 0, [Outcome.ACCEPTED]: 0, [Outcome.FILTERED]: 0, [Outcome.DROPPED]: 0, [Outcome.INVALID]: 0, // Combined with dropped later [Outcome.RATE_LIMITED]: 0, // Combined with dropped later [Outcome.CLIENT_DISCARD]: 0, // Not exposed yet [Outcome.CARDINALITY_LIMITED]: 0, // Combined with dropped later }; orgStats.groups.forEach(group => { const {outcome} = group.by; // TODO(metrics): remove this when metrics category name is updated const category = group.by.category === DATA_CATEGORY_INFO.metrics.apiName ? DATA_CATEGORY_INFO.metrics.plural : group.by.category; // HACK: The backend enum are singular, but the frontend enums are plural const fullDataCategory = Object.values(DATA_CATEGORY_INFO).find( data => data.plural === dataCategory ); if (fullDataCategory?.apiName !== category) { return; } if (outcome !== Outcome.CLIENT_DISCARD) { count.total += group.totals['sum(quantity)']; } count[outcome] += group.totals['sum(quantity)']; group.series['sum(quantity)'].forEach((stat, i) => { switch (outcome) { case Outcome.ACCEPTED: case Outcome.FILTERED: usageStats[i][outcome] += stat; return; case Outcome.DROPPED: case Outcome.RATE_LIMITED: case Outcome.CARDINALITY_LIMITED: case Outcome.INVALID: usageStats[i].dropped.total += stat; // TODO: add client discards to dropped? return; default: return; } }); }); // Invalid and rate_limited data is combined with dropped count[Outcome.DROPPED] += count[Outcome.INVALID]; count[Outcome.DROPPED] += count[Outcome.RATE_LIMITED]; count[Outcome.DROPPED] += count[Outcome.CARDINALITY_LIMITED]; usageStats.forEach(stat => { stat.total = stat.accepted + stat.filtered + stat.dropped.total; // Chart Data (chartStats.accepted as any[]).push({value: [stat.date, stat.accepted]}); (chartStats.dropped as any[]).push({ value: [stat.date, stat.dropped.total], } as any); (chartStats.filtered as any[])?.push({value: [stat.date, stat.filtered]}); }); return { cardStats: { total: formatUsageWithUnits( count.total, dataCategory, getFormatUsageOptions(dataCategory) ), accepted: formatUsageWithUnits( count[Outcome.ACCEPTED], dataCategory, getFormatUsageOptions(dataCategory) ), filtered: formatUsageWithUnits( count[Outcome.FILTERED], dataCategory, getFormatUsageOptions(dataCategory) ), dropped: formatUsageWithUnits( count[Outcome.DROPPED], dataCategory, getFormatUsageOptions(dataCategory) ), }, chartStats, }; } catch (err) { Sentry.withScope(scope => { scope.setContext('query', this.endpointQuery); scope.setContext('body', {...orgStats}); Sentry.captureException(err); }); return { cardStats, chartStats, dataError: new Error('Failed to parse stats data'), }; } } renderCards() { const {loading} = this.state; const cardMetadata = Object.values(this.cardMetadata); return cardMetadata.map((card, i) => ( )); } renderChart() { const {loading} = this.state; return ; } renderChartFooter = () => { const {handleChangeState} = this.props; const {loading, error} = this.state; const { chartDateInterval, chartTransform, chartDateStartDisplay, chartDateEndDisplay, chartDateTimezoneDisplay, } = this.chartData; return (
{t('Date Range:')} {loading || error ? ( ) : ( tct('[start] — [end] ([timezone] UTC, [interval] interval)', { start: chartDateStartDisplay, end: chartDateEndDisplay, timezone: chartDateTimezoneDisplay, interval: chartDateInterval, }) )} handleChangeState({transform: val as ChartDataTransform}) } />
); }; renderProjectDetails() { const {isSingleProject} = this.props; const projectDetails = this.projectDetails.map((projectDetailComponent, i) => ( {projectDetailComponent} )); return isSingleProject ? projectDetails : null; } renderComponent() { return ( {this.renderCards()} {this.renderChart()} {this.renderProjectDetails()} ); } } export default UsageStatsOrganization; const PageGrid = styled('div')` display: grid; grid-template-columns: 1fr; gap: ${space(2)}; @media (min-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: repeat(2, 1fr); } @media (min-width: ${p => p.theme.breakpoints.large}) { grid-template-columns: repeat(4, 1fr); } `; const StyledScoreCard = styled(ScoreCard)` grid-column: auto / span 1; margin: 0; `; const ChartWrapper = styled('div')` grid-column: 1 / -1; `; const Footer = styled('div')` display: flex; flex-direction: row; justify-content: space-between; padding: ${space(1)} ${space(3)}; border-top: 1px solid ${p => p.theme.border}; `; const FooterDate = styled('div')` display: flex; flex-direction: row; align-items: center; > ${SectionHeading} { margin-right: ${space(1.5)}; } > span:last-child { font-weight: ${p => p.theme.fontWeightNormal}; font-size: ${p => p.theme.fontSizeMedium}; } `;