123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526 |
- import {Component, Fragment} from 'react';
- import {browserHistory, InjectedRouter} from 'react-router';
- import {Theme, withTheme} from '@emotion/react';
- import {Location} from 'history';
- import {Client} from 'sentry/api';
- import {BarChart} from 'sentry/components/charts/barChart';
- import LoadingPanel from 'sentry/components/charts/loadingPanel';
- import OptionSelector from 'sentry/components/charts/optionSelector';
- import {
- ChartContainer,
- ChartControls,
- InlineContainer,
- SectionHeading,
- SectionValue,
- } from 'sentry/components/charts/styles';
- import {
- getDiffInMinutes,
- ONE_HOUR,
- ONE_WEEK,
- TWENTY_FOUR_HOURS,
- TWO_WEEKS,
- } from 'sentry/components/charts/utils';
- import {Panel} from 'sentry/components/panels';
- import Placeholder from 'sentry/components/placeholder';
- import {CHART_PALETTE} from 'sentry/constants/chartPalette';
- import {NOT_AVAILABLE_MESSAGES} from 'sentry/constants/notAvailableMessages';
- import {t} from 'sentry/locale';
- import {Organization, Project, SelectValue} from 'sentry/types';
- import {defined} from 'sentry/utils';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {decodeScalar} from 'sentry/utils/queryString';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import withApi from 'sentry/utils/withApi';
- import {
- getSessionTermDescription,
- SessionTerm,
- } from 'sentry/views/releases/utils/sessionTerm';
- import {getTermHelp, PerformanceTerm} from '../performance/data';
- import ProjectBaseEventsChart from './charts/projectBaseEventsChart';
- import ProjectBaseSessionsChart from './charts/projectBaseSessionsChart';
- import ProjectErrorsBasicChart from './charts/projectErrorsBasicChart';
- export enum DisplayModes {
- APDEX = 'apdex',
- FAILURE_RATE = 'failure_rate',
- TPM = 'tpm',
- ERRORS = 'errors',
- TRANSACTIONS = 'transactions',
- STABILITY = 'crash_free',
- STABILITY_USERS = 'crash_free_users',
- ANR_RATE = 'anr_rate',
- FOREGROUND_ANR_RATE = 'foreground_anr_rate',
- SESSIONS = 'sessions',
- }
- type Props = {
- api: Client;
- chartId: string;
- chartIndex: number;
- hasSessions: boolean | null;
- hasTransactions: boolean;
- location: Location;
- organization: Organization;
- router: InjectedRouter;
- theme: Theme;
- visibleCharts: string[];
- project?: Project;
- projectId?: string;
- query?: string;
- };
- type State = {
- totalValues: number | null;
- };
- class ProjectCharts extends Component<Props, State> {
- state: State = {
- totalValues: null,
- };
- get defaultDisplayModes() {
- const {hasSessions, hasTransactions, organization, project} = this.props;
- if (!hasSessions && !hasTransactions) {
- return [DisplayModes.ERRORS];
- }
- if (hasSessions && !hasTransactions) {
- if (organization.features.includes('anr-rate') && project?.platform === 'android') {
- return [DisplayModes.STABILITY, DisplayModes.ANR_RATE];
- }
- return [DisplayModes.STABILITY, DisplayModes.ERRORS];
- }
- if (!hasSessions && hasTransactions) {
- return [DisplayModes.FAILURE_RATE, DisplayModes.APDEX];
- }
- if (organization.features.includes('anr-rate') && project?.platform === 'android') {
- return [DisplayModes.STABILITY, DisplayModes.ANR_RATE];
- }
- return [DisplayModes.STABILITY, DisplayModes.APDEX];
- }
- get otherActiveDisplayModes() {
- const {location, visibleCharts, chartId} = this.props;
- return visibleCharts
- .filter(visibleChartId => visibleChartId !== chartId)
- .map(urlKey => {
- return decodeScalar(
- location.query[urlKey],
- this.defaultDisplayModes[visibleCharts.findIndex(value => value === urlKey)]
- );
- });
- }
- get displayMode() {
- const {location, chartId, chartIndex} = this.props;
- const displayMode =
- decodeScalar(location.query[chartId]) || this.defaultDisplayModes[chartIndex];
- if (!Object.values(DisplayModes).includes(displayMode as DisplayModes)) {
- return this.defaultDisplayModes[chartIndex];
- }
- return displayMode;
- }
- get displayModes(): SelectValue<string>[] {
- const {organization, hasSessions, hasTransactions, project} = this.props;
- const hasPerformance = organization.features.includes('performance-view');
- const noPerformanceTooltip = NOT_AVAILABLE_MESSAGES.performance;
- const noHealthTooltip = NOT_AVAILABLE_MESSAGES.releaseHealth;
- const options = [
- {
- value: DisplayModes.STABILITY,
- label: t('Crash Free Sessions'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.STABILITY) || !hasSessions,
- tooltip: !hasSessions ? noHealthTooltip : undefined,
- },
- {
- value: DisplayModes.STABILITY_USERS,
- label: t('Crash Free Users'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.STABILITY_USERS) ||
- !hasSessions,
- tooltip: !hasSessions ? noHealthTooltip : undefined,
- },
- {
- value: DisplayModes.APDEX,
- label: t('Apdex'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.APDEX) ||
- !hasPerformance ||
- !hasTransactions,
- tooltip:
- hasPerformance && hasTransactions
- ? getTermHelp(organization, PerformanceTerm.APDEX)
- : noPerformanceTooltip,
- },
- {
- value: DisplayModes.FAILURE_RATE,
- label: t('Failure Rate'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.FAILURE_RATE) ||
- !hasPerformance ||
- !hasTransactions,
- tooltip:
- hasPerformance && hasTransactions
- ? getTermHelp(organization, PerformanceTerm.FAILURE_RATE)
- : noPerformanceTooltip,
- },
- {
- value: DisplayModes.TPM,
- label: t('Transactions Per Minute'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.TPM) ||
- !hasPerformance ||
- !hasTransactions,
- tooltip:
- hasPerformance && hasTransactions
- ? getTermHelp(organization, PerformanceTerm.TPM)
- : noPerformanceTooltip,
- },
- {
- value: DisplayModes.ERRORS,
- label: t('Number of Errors'),
- disabled: this.otherActiveDisplayModes.includes(DisplayModes.ERRORS),
- },
- {
- value: DisplayModes.SESSIONS,
- label: t('Number of Sessions'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.SESSIONS) || !hasSessions,
- tooltip: !hasSessions ? noHealthTooltip : undefined,
- },
- {
- value: DisplayModes.TRANSACTIONS,
- label: t('Number of Transactions'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.TRANSACTIONS) ||
- !hasPerformance ||
- !hasTransactions,
- tooltip: hasPerformance && hasTransactions ? undefined : noPerformanceTooltip,
- },
- ];
- if (organization.features.includes('anr-rate') && project?.platform === 'android') {
- return [
- {
- value: DisplayModes.ANR_RATE,
- label: t('ANR Rate'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.ANR_RATE) || !hasSessions,
- tooltip: !hasSessions ? noHealthTooltip : undefined,
- },
- {
- value: DisplayModes.FOREGROUND_ANR_RATE,
- label: t('Foreground ANR Rate'),
- disabled:
- this.otherActiveDisplayModes.includes(DisplayModes.FOREGROUND_ANR_RATE) ||
- !hasSessions,
- tooltip: !hasSessions ? noHealthTooltip : undefined,
- },
- ...options,
- ];
- }
- return options;
- }
- get summaryHeading() {
- switch (this.displayMode) {
- case DisplayModes.ERRORS:
- return t('Total Errors');
- case DisplayModes.STABILITY:
- case DisplayModes.SESSIONS:
- return t('Total Sessions');
- case DisplayModes.STABILITY_USERS:
- case DisplayModes.ANR_RATE:
- case DisplayModes.FOREGROUND_ANR_RATE:
- return t('Total Users');
- case DisplayModes.APDEX:
- case DisplayModes.FAILURE_RATE:
- case DisplayModes.TPM:
- case DisplayModes.TRANSACTIONS:
- default:
- return t('Total Transactions');
- }
- }
- get barChartInterval() {
- const {query} = this.props.location;
- const diffInMinutes = getDiffInMinutes({
- ...query,
- period: decodeScalar(query.statsPeriod),
- });
- if (diffInMinutes >= TWO_WEEKS) {
- return '1d';
- }
- if (diffInMinutes >= ONE_WEEK) {
- return '12h';
- }
- if (diffInMinutes > TWENTY_FOUR_HOURS) {
- return '6h';
- }
- if (diffInMinutes === TWENTY_FOUR_HOURS) {
- return '1h';
- }
- if (diffInMinutes <= ONE_HOUR) {
- return '1m';
- }
- return '15m';
- }
- handleDisplayModeChange = (value: string) => {
- const {location, chartId, chartIndex, organization} = this.props;
- trackAnalytics('project_detail.change_chart', {
- organization,
- metric: value,
- chart_index: chartIndex,
- });
- browserHistory.push({
- pathname: location.pathname,
- query: {...location.query, [chartId]: value},
- });
- };
- handleTotalValuesChange = (value: number | null) => {
- if (value !== this.state.totalValues) {
- this.setState({totalValues: value});
- }
- };
- render() {
- const {
- api,
- router,
- location,
- organization,
- theme,
- projectId,
- hasSessions,
- query,
- project,
- } = this.props;
- const {totalValues} = this.state;
- const hasDiscover = organization.features.includes('discover-basic');
- const displayMode = this.displayMode;
- const hasAnrRateFeature =
- organization.features.includes('anr-rate') && project?.platform === 'android';
- return (
- <Panel>
- <ChartContainer>
- {!defined(hasSessions) ? (
- <LoadingPanel />
- ) : (
- <Fragment>
- {displayMode === DisplayModes.APDEX && (
- <ProjectBaseEventsChart
- title={t('Apdex')}
- help={getTermHelp(organization, PerformanceTerm.APDEX)}
- query={new MutableSearch([
- 'event.type:transaction',
- query ?? '',
- ]).formatString()}
- yAxis="apdex()"
- field={['apdex()']}
- api={api}
- router={router}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- colors={[CHART_PALETTE[0][0], theme.purple200]}
- />
- )}
- {displayMode === DisplayModes.FAILURE_RATE && (
- <ProjectBaseEventsChart
- title={t('Failure Rate')}
- help={getTermHelp(organization, PerformanceTerm.FAILURE_RATE)}
- query={new MutableSearch([
- 'event.type:transaction',
- query ?? '',
- ]).formatString()}
- yAxis="failure_rate()"
- field={[`failure_rate()`]}
- api={api}
- router={router}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- colors={[theme.red300, theme.purple200]}
- />
- )}
- {displayMode === DisplayModes.TPM && (
- <ProjectBaseEventsChart
- title={t('Transactions Per Minute')}
- help={getTermHelp(organization, PerformanceTerm.TPM)}
- query={new MutableSearch([
- 'event.type:transaction',
- query ?? '',
- ]).formatString()}
- yAxis="tpm()"
- field={[`tpm()`]}
- api={api}
- router={router}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- colors={[theme.yellow300, theme.purple200]}
- disablePrevious
- />
- )}
- {displayMode === DisplayModes.ERRORS &&
- (hasDiscover ? (
- <ProjectBaseEventsChart
- title={t('Number of Errors')}
- query={new MutableSearch([
- '!event.type:transaction',
- query ?? '',
- ]).formatString()}
- yAxis="count()"
- field={[`count()`]}
- api={api}
- router={router}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- colors={[theme.purple300, theme.purple200]}
- interval={this.barChartInterval}
- chartComponent={BarChart}
- disableReleases
- />
- ) : (
- <ProjectErrorsBasicChart
- organization={organization}
- projectId={projectId}
- location={location}
- onTotalValuesChange={this.handleTotalValuesChange}
- />
- ))}
- {displayMode === DisplayModes.TRANSACTIONS && (
- <ProjectBaseEventsChart
- title={t('Number of Transactions')}
- query={new MutableSearch([
- 'event.type:transaction',
- query ?? '',
- ]).formatString()}
- yAxis="count()"
- field={[`count()`]}
- api={api}
- router={router}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- colors={[theme.gray200, theme.purple200]}
- interval={this.barChartInterval}
- chartComponent={BarChart}
- disableReleases
- />
- )}
- {displayMode === DisplayModes.STABILITY && (
- <ProjectBaseSessionsChart
- title={t('Crash Free Sessions')}
- help={getSessionTermDescription(SessionTerm.STABILITY, null)}
- router={router}
- api={api}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- displayMode={displayMode}
- query={query}
- />
- )}
- {hasAnrRateFeature && displayMode === DisplayModes.ANR_RATE && (
- <ProjectBaseSessionsChart
- title={t('ANR Rate')}
- help={getSessionTermDescription(SessionTerm.ANR_RATE, null)}
- router={router}
- api={api}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- displayMode={displayMode}
- query={query}
- />
- )}
- {hasAnrRateFeature && displayMode === DisplayModes.FOREGROUND_ANR_RATE && (
- <ProjectBaseSessionsChart
- title={t('Foreground ANR Rate')}
- help={getSessionTermDescription(SessionTerm.FOREGROUND_ANR_RATE, null)}
- router={router}
- api={api}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- displayMode={displayMode}
- query={query}
- />
- )}
- {displayMode === DisplayModes.STABILITY_USERS && (
- <ProjectBaseSessionsChart
- title={t('Crash Free Users')}
- help={getSessionTermDescription(SessionTerm.CRASH_FREE_USERS, null)}
- router={router}
- api={api}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- displayMode={displayMode}
- query={query}
- />
- )}
- {displayMode === DisplayModes.SESSIONS && (
- <ProjectBaseSessionsChart
- title={t('Number of Sessions')}
- router={router}
- api={api}
- organization={organization}
- onTotalValuesChange={this.handleTotalValuesChange}
- displayMode={displayMode}
- disablePrevious
- query={query}
- />
- )}
- </Fragment>
- )}
- </ChartContainer>
- <ChartControls>
- {/* if hasSessions is not yet defined, it means that request is still in progress and we can't decide what default chart to show */}
- {defined(hasSessions) ? (
- <Fragment>
- <InlineContainer>
- <SectionHeading>{this.summaryHeading}</SectionHeading>
- <SectionValue>
- {typeof totalValues === 'number'
- ? totalValues.toLocaleString()
- : '\u2014'}
- </SectionValue>
- </InlineContainer>
- <InlineContainer>
- <OptionSelector
- title={t('Display')}
- selected={displayMode}
- options={this.displayModes}
- onChange={this.handleDisplayModeChange}
- />
- </InlineContainer>
- </Fragment>
- ) : (
- <Placeholder height="34px" />
- )}
- </ChartControls>
- </Panel>
- );
- }
- }
- export default withApi(withTheme(ProjectCharts));
|