import {Component} from 'react'; // eslint-disable-next-line no-restricted-imports import {withRouter, WithRouterProps} from 'react-router'; import {withTheme} from '@emotion/react'; import round from 'lodash/round'; import {AreaChart, AreaChartProps} from 'sentry/components/charts/areaChart'; import ChartZoom from 'sentry/components/charts/chartZoom'; import StackedAreaChart from 'sentry/components/charts/stackedAreaChart'; import {HeaderTitleLegend, HeaderValue} from 'sentry/components/charts/styles'; import TransitionChart from 'sentry/components/charts/transitionChart'; import TransparentLoadingMask from 'sentry/components/charts/transparentLoadingMask'; import QuestionTooltip from 'sentry/components/questionTooltip'; import {PlatformKey} from 'sentry/data/platformCategories'; import {t} from 'sentry/locale'; import { ReleaseComparisonChartType, ReleaseProject, ReleaseWithHealth, SessionApiResponse, SessionFieldWithOperation, SessionStatus, } from 'sentry/types'; import {defined} from 'sentry/utils'; import {getDuration, getExactDuration} from 'sentry/utils/formatters'; import { getCountSeries, getCrashFreeRateSeries, getSessionP50Series, getSessionStatusRateSeries, initSessionsChart, MINUTES_THRESHOLD_TO_DISPLAY_SECONDS, } from 'sentry/utils/sessions'; import {Theme} from 'sentry/utils/theme'; import {displayCrashFreePercent, roundDuration} from 'sentry/views/releases/utils'; import { generateReleaseMarkLines, releaseComparisonChartHelp, releaseComparisonChartTitles, releaseMarkLinesLabels, } from '../../utils'; type Props = { allSessions: SessionApiResponse | null; chartType: ReleaseComparisonChartType; diff: React.ReactNode; loading: boolean; platform: PlatformKey; project: ReleaseProject; release: ReleaseWithHealth; releaseSessions: SessionApiResponse | null; reloading: boolean; theme: Theme; value: React.ReactNode; end?: string; period?: string | null; start?: string; utc?: boolean; } & WithRouterProps; class ReleaseSessionsChart extends Component { formatTooltipValue = (value: string | number | null, label?: string) => { if (label && Object.values(releaseMarkLinesLabels).includes(label)) { return ''; } const {chartType} = this.props; if (value === null) { return '\u2015'; } switch (chartType) { case ReleaseComparisonChartType.CRASH_FREE_SESSIONS: case ReleaseComparisonChartType.HEALTHY_SESSIONS: case ReleaseComparisonChartType.ABNORMAL_SESSIONS: case ReleaseComparisonChartType.ERRORED_SESSIONS: case ReleaseComparisonChartType.CRASHED_SESSIONS: case ReleaseComparisonChartType.CRASH_FREE_USERS: case ReleaseComparisonChartType.HEALTHY_USERS: case ReleaseComparisonChartType.ABNORMAL_USERS: case ReleaseComparisonChartType.ERRORED_USERS: case ReleaseComparisonChartType.CRASHED_USERS: return defined(value) ? `${value}%` : '\u2015'; case ReleaseComparisonChartType.SESSION_DURATION: return defined(value) && typeof value === 'number' ? getExactDuration(value, true) : '\u2015'; case ReleaseComparisonChartType.SESSION_COUNT: case ReleaseComparisonChartType.USER_COUNT: default: return typeof value === 'number' ? value.toLocaleString() : value; } }; getYAxis() { const {theme, chartType} = this.props; switch (chartType) { case ReleaseComparisonChartType.CRASH_FREE_SESSIONS: case ReleaseComparisonChartType.CRASH_FREE_USERS: return { max: 100, scale: true, axisLabel: { formatter: (value: number) => displayCrashFreePercent(value), color: theme.chartLabel, }, }; case ReleaseComparisonChartType.HEALTHY_SESSIONS: case ReleaseComparisonChartType.ABNORMAL_SESSIONS: case ReleaseComparisonChartType.ERRORED_SESSIONS: case ReleaseComparisonChartType.CRASHED_SESSIONS: case ReleaseComparisonChartType.HEALTHY_USERS: case ReleaseComparisonChartType.ABNORMAL_USERS: case ReleaseComparisonChartType.ERRORED_USERS: case ReleaseComparisonChartType.CRASHED_USERS: return { scale: true, axisLabel: { formatter: (value: number) => `${round(value, 2)}%`, color: theme.chartLabel, }, }; case ReleaseComparisonChartType.SESSION_DURATION: return { scale: true, axisLabel: { formatter: (value: number) => getDuration(value, undefined, true), color: theme.chartLabel, }, }; case ReleaseComparisonChartType.SESSION_COUNT: case ReleaseComparisonChartType.USER_COUNT: default: return undefined; } } getChart(): | React.ComponentType | React.ComponentType { const {chartType} = this.props; switch (chartType) { case ReleaseComparisonChartType.CRASH_FREE_SESSIONS: case ReleaseComparisonChartType.HEALTHY_SESSIONS: case ReleaseComparisonChartType.ABNORMAL_SESSIONS: case ReleaseComparisonChartType.ERRORED_SESSIONS: case ReleaseComparisonChartType.CRASHED_SESSIONS: case ReleaseComparisonChartType.CRASH_FREE_USERS: case ReleaseComparisonChartType.HEALTHY_USERS: case ReleaseComparisonChartType.ABNORMAL_USERS: case ReleaseComparisonChartType.ERRORED_USERS: case ReleaseComparisonChartType.CRASHED_USERS: default: return AreaChart; case ReleaseComparisonChartType.SESSION_COUNT: case ReleaseComparisonChartType.SESSION_DURATION: case ReleaseComparisonChartType.USER_COUNT: return StackedAreaChart; } } getColors() { const {theme, chartType} = this.props; const colors = theme.charts.getColorPalette(14); switch (chartType) { case ReleaseComparisonChartType.CRASH_FREE_SESSIONS: return [colors[0]]; case ReleaseComparisonChartType.HEALTHY_SESSIONS: return [theme.green300]; case ReleaseComparisonChartType.ABNORMAL_SESSIONS: return [colors[15]]; case ReleaseComparisonChartType.ERRORED_SESSIONS: return [colors[12]]; case ReleaseComparisonChartType.CRASHED_SESSIONS: return [theme.red300]; case ReleaseComparisonChartType.CRASH_FREE_USERS: return [colors[6]]; case ReleaseComparisonChartType.HEALTHY_USERS: return [theme.green300]; case ReleaseComparisonChartType.ABNORMAL_USERS: return [colors[15]]; case ReleaseComparisonChartType.ERRORED_USERS: return [colors[12]]; case ReleaseComparisonChartType.CRASHED_USERS: return [theme.red300]; case ReleaseComparisonChartType.SESSION_COUNT: case ReleaseComparisonChartType.SESSION_DURATION: case ReleaseComparisonChartType.USER_COUNT: default: return undefined; } } getSeries(chartType: ReleaseComparisonChartType) { const {releaseSessions, allSessions, release, location, project, theme} = this.props; const countCharts = initSessionsChart(theme); if (!releaseSessions) { return {}; } const markLines = generateReleaseMarkLines(release, project, theme, location); switch (chartType) { case ReleaseComparisonChartType.CRASH_FREE_SESSIONS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getCrashFreeRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.SESSIONS ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getCrashFreeRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.SESSIONS ), }, ], markLines, }; case ReleaseComparisonChartType.HEALTHY_SESSIONS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.HEALTHY ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.HEALTHY ), }, ], markLines, }; case ReleaseComparisonChartType.ABNORMAL_SESSIONS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.ABNORMAL ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.ABNORMAL ), }, ], markLines, }; case ReleaseComparisonChartType.ERRORED_SESSIONS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.ERRORED ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.ERRORED ), }, ], markLines, }; case ReleaseComparisonChartType.CRASHED_SESSIONS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.CRASHED ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.SESSIONS, SessionStatus.CRASHED ), }, ], markLines, }; case ReleaseComparisonChartType.CRASH_FREE_USERS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getCrashFreeRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.USERS ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getCrashFreeRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.USERS ), }, ], markLines, }; case ReleaseComparisonChartType.HEALTHY_USERS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.HEALTHY ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.HEALTHY ), }, ], markLines, }; case ReleaseComparisonChartType.ABNORMAL_USERS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.ABNORMAL ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.ABNORMAL ), }, ], markLines, }; case ReleaseComparisonChartType.ERRORED_USERS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.ERRORED ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.ERRORED ), }, ], markLines, }; case ReleaseComparisonChartType.CRASHED_USERS: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionStatusRateSeries( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.CRASHED ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionStatusRateSeries( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.USERS, SessionStatus.CRASHED ), }, ], markLines, }; case ReleaseComparisonChartType.SESSION_COUNT: return { series: [ { ...countCharts[SessionStatus.HEALTHY], data: getCountSeries( SessionFieldWithOperation.SESSIONS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.HEALTHY ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.ERRORED], data: getCountSeries( SessionFieldWithOperation.SESSIONS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.ERRORED ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.ABNORMAL], data: getCountSeries( SessionFieldWithOperation.SESSIONS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.ABNORMAL ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.CRASHED], data: getCountSeries( SessionFieldWithOperation.SESSIONS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.CRASHED ), releaseSessions.intervals ), }, ], markLines, }; case ReleaseComparisonChartType.SESSION_DURATION: return { series: [ { seriesName: t('This Release'), connectNulls: true, data: getSessionP50Series( releaseSessions?.groups, releaseSessions?.intervals, SessionFieldWithOperation.DURATION, duration => roundDuration(duration / 1000) ), }, ], previousSeries: [ { seriesName: t('All Releases'), data: getSessionP50Series( allSessions?.groups, allSessions?.intervals, SessionFieldWithOperation.DURATION, duration => roundDuration(duration / 1000) ), }, ], markLines, }; case ReleaseComparisonChartType.USER_COUNT: return { series: [ { ...countCharts[SessionStatus.HEALTHY], data: getCountSeries( SessionFieldWithOperation.USERS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.HEALTHY ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.ERRORED], data: getCountSeries( SessionFieldWithOperation.USERS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.ERRORED ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.ABNORMAL], data: getCountSeries( SessionFieldWithOperation.USERS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.ABNORMAL ), releaseSessions.intervals ), }, { ...countCharts[SessionStatus.CRASHED], data: getCountSeries( SessionFieldWithOperation.USERS, releaseSessions.groups.find( g => g.by['session.status'] === SessionStatus.CRASHED ), releaseSessions.intervals ), }, ], markLines, }; default: return {}; } } render() { const {chartType, router, period, start, end, utc, value, diff, loading, reloading} = this.props; const Chart = this.getChart(); const {series, previousSeries, markLines} = this.getSeries(chartType); const legend = { right: 10, top: 0, textStyle: { padding: [2, 0, 0, 0], }, data: [...(series ?? []), ...(previousSeries ?? [])].map(s => s.seriesName), }; return ( {releaseComparisonChartTitles[chartType]} {releaseComparisonChartHelp[chartType] && ( )} {value} {diff} {zoomRenderProps => ( )} ); } } export default withTheme(withRouter(ReleaseSessionsChart));