123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692 |
- import {Fragment, PureComponent} from 'react';
- import type {WithRouterProps} from 'react-router';
- import styled from '@emotion/styled';
- import color from 'color';
- import type {LineSeriesOption} from 'echarts';
- import moment from 'moment';
- import momentTimezone from 'moment-timezone';
- import type {Client} from 'sentry/api';
- import Feature from 'sentry/components/acl/feature';
- import {OnDemandMetricAlert} from 'sentry/components/alerts/onDemandMetricAlert';
- import {Button} from 'sentry/components/button';
- import type {AreaChartSeries} from 'sentry/components/charts/areaChart';
- import {AreaChart} from 'sentry/components/charts/areaChart';
- import ChartZoom from 'sentry/components/charts/chartZoom';
- import MarkArea from 'sentry/components/charts/components/markArea';
- import MarkLine from 'sentry/components/charts/components/markLine';
- import EventsRequest from 'sentry/components/charts/eventsRequest';
- import LineSeries from 'sentry/components/charts/series/lineSeries';
- import SessionsRequest from 'sentry/components/charts/sessionsRequest';
- import {
- ChartControls,
- HeaderTitleLegend,
- InlineContainer,
- SectionHeading,
- SectionValue,
- } from 'sentry/components/charts/styles';
- import {isEmptySeries} from 'sentry/components/charts/utils';
- import CircleIndicator from 'sentry/components/circleIndicator';
- import type {StatsPeriodType} from 'sentry/components/organizations/pageFilters/parse';
- import {parseStatsPeriod} from 'sentry/components/organizations/pageFilters/parse';
- import Panel from 'sentry/components/panels/panel';
- import PanelBody from 'sentry/components/panels/panelBody';
- import Placeholder from 'sentry/components/placeholder';
- import {Tooltip} from 'sentry/components/tooltip';
- import {IconCheckmark, IconClock, IconFire, IconWarning} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import ConfigStore from 'sentry/stores/configStore';
- import {space} from 'sentry/styles/space';
- import type {DateString, Organization, Project} from 'sentry/types';
- import type {ReactEchartsRef, Series} from 'sentry/types/echarts';
- import toArray from 'sentry/utils/array/toArray';
- import {browserHistory} from 'sentry/utils/browserHistory';
- import {getUtcDateString} from 'sentry/utils/dates';
- import {DiscoverDatasets} from 'sentry/utils/discover/types';
- import getDuration from 'sentry/utils/duration/getDuration';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import {getForceMetricsLayerQueryExtras} from 'sentry/utils/metrics/features';
- import {formatMRIField} from 'sentry/utils/metrics/mri';
- import {shouldShowOnDemandMetricAlertUI} from 'sentry/utils/onDemandMetrics/features';
- import {MINUTES_THRESHOLD_TO_DISPLAY_SECONDS} from 'sentry/utils/sessions';
- import {capitalize} from 'sentry/utils/string/capitalize';
- import theme from 'sentry/utils/theme';
- import {normalizeUrl} from 'sentry/utils/withDomainRequired';
- // eslint-disable-next-line no-restricted-imports
- import withSentryRouter from 'sentry/utils/withSentryRouter';
- import {COMPARISON_DELTA_OPTIONS} from 'sentry/views/alerts/rules/metric/constants';
- import {makeDefaultCta} from 'sentry/views/alerts/rules/metric/metricRulePresets';
- import type {MetricRule} from 'sentry/views/alerts/rules/metric/types';
- import {
- AlertRuleTriggerType,
- Dataset,
- TimePeriod,
- } from 'sentry/views/alerts/rules/metric/types';
- import {getChangeStatus} from 'sentry/views/alerts/utils/getChangeStatus';
- import {AlertWizardAlertNames} from 'sentry/views/alerts/wizard/options';
- import {getAlertTypeFromAggregateDataset} from 'sentry/views/alerts/wizard/utils';
- import type {Incident} from '../../../types';
- import {
- alertDetailsLink,
- alertTooltipValueFormatter,
- isSessionAggregate,
- SESSION_AGGREGATE_TO_FIELD,
- } from '../../../utils';
- import {getMetricDatasetQueryExtras} from '../utils/getMetricDatasetQueryExtras';
- import {isCrashFreeAlert} from '../utils/isCrashFreeAlert';
- import type {TimePeriodType} from './constants';
- import {
- getMetricAlertChartOption,
- transformSessionResponseToSeries,
- } from './metricChartOption';
- type Props = WithRouterProps & {
- api: Client;
- filter: string[] | null;
- interval: string;
- organization: Organization;
- project: Project;
- query: string;
- rule: MetricRule;
- timePeriod: TimePeriodType;
- incidents?: Incident[];
- isOnDemandAlert?: boolean;
- selectedIncident?: Incident | null;
- };
- type State = Record<string, never>;
- function formatTooltipDate(date: moment.MomentInput, format: string): string {
- const {
- options: {timezone},
- } = ConfigStore.get('user');
- return momentTimezone.tz(date, timezone).format(format);
- }
- function getRuleChangeSeries(
- rule: MetricRule,
- data: AreaChartSeries[]
- ): LineSeriesOption[] {
- const {dateModified} = rule;
- if (!data.length || !data[0].data.length || !dateModified) {
- return [];
- }
- const seriesData = data[0].data;
- const seriesStart = new Date(seriesData[0].name).getTime();
- const ruleChanged = new Date(dateModified).getTime();
- if (ruleChanged < seriesStart) {
- return [];
- }
- return [
- {
- type: 'line',
- markLine: MarkLine({
- silent: true,
- animation: false,
- lineStyle: {color: theme.gray200, type: 'solid', width: 1},
- data: [{xAxis: ruleChanged}],
- label: {
- show: false,
- },
- }),
- markArea: MarkArea({
- silent: true,
- itemStyle: {
- color: color(theme.gray100).alpha(0.42).rgb().string(),
- },
- data: [[{xAxis: seriesStart}, {xAxis: ruleChanged}]],
- }),
- data: [],
- },
- ];
- }
- function shouldUseErrorsDataset(
- organization: Organization,
- dataset: Dataset,
- query: string
- ): boolean {
- return (
- dataset === Dataset.ERRORS &&
- /\bis:unresolved\b/.test(query) &&
- organization.features.includes('metric-alert-ignore-archived')
- );
- }
- class MetricChart extends PureComponent<Props, State> {
- ref: null | ReactEchartsRef = null;
- handleZoom = (start: DateString, end: DateString) => {
- const {location} = this.props;
- browserHistory.push({
- pathname: location.pathname,
- query: {
- start,
- end,
- },
- });
- };
- renderChartActions(
- totalDuration: number,
- criticalDuration: number,
- warningDuration: number,
- waitingForDataDuration: number
- ) {
- const {rule, organization, project, timePeriod, query} = this.props;
- let dataset: DiscoverDatasets | undefined = undefined;
- if (shouldUseErrorsDataset(organization, rule.dataset, query)) {
- dataset = DiscoverDatasets.ERRORS;
- }
- const {buttonText, ...props} = makeDefaultCta({
- orgSlug: organization.slug,
- projects: [project],
- rule,
- timePeriod,
- query,
- dataset,
- });
- const resolvedPercent =
- (100 *
- Math.max(
- totalDuration - waitingForDataDuration - criticalDuration - warningDuration,
- 0
- )) /
- totalDuration;
- const criticalPercent = 100 * Math.min(criticalDuration / totalDuration, 1);
- const warningPercent = 100 * Math.min(warningDuration / totalDuration, 1);
- const waitingForDataPercent =
- 100 *
- Math.min(
- (waitingForDataDuration - criticalDuration - warningDuration) / totalDuration,
- 1
- );
- return (
- <StyledChartControls>
- <StyledInlineContainer>
- <Fragment>
- <SectionHeading>{t('Summary')}</SectionHeading>
- <StyledSectionValue>
- <ValueItem>
- <IconCheckmark color="successText" isCircled />
- {resolvedPercent ? resolvedPercent.toFixed(2) : 0}%
- </ValueItem>
- <ValueItem>
- <IconWarning color="warningText" />
- {warningPercent ? warningPercent.toFixed(2) : 0}%
- </ValueItem>
- <ValueItem>
- <IconFire color="errorText" />
- {criticalPercent ? criticalPercent.toFixed(2) : 0}%
- </ValueItem>
- {waitingForDataPercent > 0 && (
- <StyledTooltip
- underlineColor="gray200"
- showUnderline
- title={t(
- 'The time spent waiting for metrics matching the filters used.'
- )}
- >
- <ValueItem>
- <IconClock />
- {waitingForDataPercent.toFixed(2)}%
- </ValueItem>
- </StyledTooltip>
- )}
- </StyledSectionValue>
- </Fragment>
- </StyledInlineContainer>
- {!isSessionAggregate(rule.aggregate) && (
- <Feature features="discover-basic">
- <Button size="sm" {...props}>
- {buttonText}
- </Button>
- </Feature>
- )}
- </StyledChartControls>
- );
- }
- renderChart(
- loading: boolean,
- timeseriesData?: Series[],
- minutesThresholdToDisplaySeconds?: number,
- comparisonTimeseriesData?: Series[]
- ) {
- const {
- router,
- selectedIncident,
- interval,
- filter,
- incidents,
- rule,
- organization,
- timePeriod: {start, end},
- } = this.props;
- const {dateModified, timeWindow} = rule;
- if (loading || !timeseriesData) {
- return this.renderEmpty();
- }
- const handleIncidentClick = (incident: Incident) => {
- router.push(
- normalizeUrl({
- pathname: alertDetailsLink(organization, incident),
- query: {alert: incident.identifier},
- })
- );
- };
- const {
- criticalDuration,
- warningDuration,
- totalDuration,
- waitingForDataDuration,
- chartOption,
- } = getMetricAlertChartOption({
- timeseriesData,
- rule,
- incidents,
- selectedIncident,
- showWaitingForData:
- shouldShowOnDemandMetricAlertUI(organization) && this.props.isOnDemandAlert,
- handleIncidentClick,
- });
- const comparisonSeriesName = capitalize(
- COMPARISON_DELTA_OPTIONS.find(({value}) => value === rule.comparisonDelta)?.label ||
- ''
- );
- const additionalSeries: LineSeriesOption[] = [
- ...(comparisonTimeseriesData || []).map(({data: _data, ...otherSeriesProps}) =>
- LineSeries({
- name: comparisonSeriesName,
- data: _data.map(({name, value}) => [name, value]),
- lineStyle: {color: theme.gray200, type: 'dashed', width: 1},
- itemStyle: {color: theme.gray200},
- animation: false,
- animationThreshold: 1,
- animationDuration: 0,
- ...otherSeriesProps,
- })
- ),
- ...getRuleChangeSeries(rule, timeseriesData),
- ];
- const queryFilter =
- filter?.join(' ') + t(' over ') + getDuration(rule.timeWindow * 60);
- return (
- <ChartPanel>
- <StyledPanelBody withPadding>
- <ChartHeader>
- <HeaderTitleLegend>
- {AlertWizardAlertNames[getAlertTypeFromAggregateDataset(rule)]}
- </HeaderTitleLegend>
- </ChartHeader>
- <ChartFilters>
- <StyledCircleIndicator size={8} />
- <Filters>{formatMRIField(rule.aggregate)}</Filters>
- <Tooltip
- title={queryFilter}
- isHoverable
- skipWrapper
- overlayStyle={{maxWidth: '90vw', lineBreak: 'anywhere', textAlign: 'left'}}
- showOnlyOnOverflow
- >
- <QueryFilters>{queryFilter}</QueryFilters>
- </Tooltip>
- </ChartFilters>
- {getDynamicText({
- value: (
- <ChartZoom
- router={router}
- start={start}
- end={end}
- onZoom={zoomArgs => this.handleZoom(zoomArgs.start, zoomArgs.end)}
- >
- {zoomRenderProps => (
- <AreaChart
- {...zoomRenderProps}
- {...chartOption}
- showTimeInTooltip
- minutesThresholdToDisplaySeconds={minutesThresholdToDisplaySeconds}
- additionalSeries={additionalSeries}
- tooltip={{
- formatter: seriesParams => {
- // seriesParams can be object instead of array
- const pointSeries = toArray(seriesParams);
- const {marker, data: pointData, seriesName} = pointSeries[0];
- const [pointX, pointY] = pointData as [number, number];
- const pointYFormatted = alertTooltipValueFormatter(
- pointY,
- seriesName ?? '',
- rule.aggregate
- );
- const isModified =
- dateModified && pointX <= new Date(dateModified).getTime();
- const startTime = formatTooltipDate(moment(pointX), 'MMM D LT');
- const {period, periodLength} = parseStatsPeriod(interval) ?? {
- periodLength: 'm',
- period: `${timeWindow}`,
- };
- const endTime = formatTooltipDate(
- moment(pointX).add(
- parseInt(period, 10),
- periodLength as StatsPeriodType
- ),
- 'MMM D LT'
- );
- const comparisonSeries =
- pointSeries.length > 1
- ? pointSeries.find(
- ({seriesName: _sn}) => _sn === comparisonSeriesName
- )
- : undefined;
- const comparisonPointY = comparisonSeries?.data[1] as
- | number
- | undefined;
- const comparisonPointYFormatted =
- comparisonPointY !== undefined
- ? alertTooltipValueFormatter(
- comparisonPointY,
- seriesName ?? '',
- rule.aggregate
- )
- : undefined;
- const changePercentage =
- comparisonPointY === undefined
- ? NaN
- : ((pointY - comparisonPointY) * 100) / comparisonPointY;
- const changeStatus = getChangeStatus(
- changePercentage,
- rule.thresholdType,
- rule.triggers
- );
- const changeStatusColor =
- changeStatus === AlertRuleTriggerType.CRITICAL
- ? theme.red300
- : changeStatus === AlertRuleTriggerType.WARNING
- ? theme.yellow300
- : theme.green300;
- return [
- `<div class="tooltip-series">`,
- isModified &&
- `<div><span class="tooltip-label"><strong>${t(
- 'Alert Rule Modified'
- )}</strong></span></div>`,
- `<div><span class="tooltip-label">${marker} <strong>${seriesName}</strong></span>${pointYFormatted}</div>`,
- comparisonSeries &&
- `<div><span class="tooltip-label">${comparisonSeries.marker} <strong>${comparisonSeriesName}</strong></span>${comparisonPointYFormatted}</div>`,
- `</div>`,
- `<div class="tooltip-footer">`,
- `<span>${startTime} — ${endTime}</span>`,
- comparisonPointY !== undefined &&
- Math.abs(changePercentage) !== Infinity &&
- !isNaN(changePercentage) &&
- `<span style="color:${changeStatusColor};margin-left:10px;">${
- Math.sign(changePercentage) === 1 ? '+' : '-'
- }${Math.abs(changePercentage).toFixed(2)}%</span>`,
- `</div>`,
- '<div class="tooltip-arrow"></div>',
- ]
- .filter(e => e)
- .join('');
- },
- }}
- />
- )}
- </ChartZoom>
- ),
- fixed: <Placeholder height="200px" testId="skeleton-ui" />,
- })}
- </StyledPanelBody>
- {this.renderChartActions(
- totalDuration,
- criticalDuration,
- warningDuration,
- waitingForDataDuration
- )}
- </ChartPanel>
- );
- }
- renderEmptyOnDemandAlert(
- organization: Organization,
- timeseriesData: Series[] = [],
- loading?: boolean
- ) {
- if (
- loading ||
- !this.props.isOnDemandAlert ||
- !shouldShowOnDemandMetricAlertUI(organization) ||
- !isEmptySeries(timeseriesData[0])
- ) {
- return null;
- }
- return (
- <OnDemandMetricAlert
- dismissable
- message={t(
- 'This alert lacks historical data due to filters for which we don’t routinely extract metrics.'
- )}
- />
- );
- }
- renderEmpty(placeholderText = '') {
- return (
- <ChartPanel>
- <PanelBody withPadding>
- <TriggerChartPlaceholder>{placeholderText}</TriggerChartPlaceholder>
- </PanelBody>
- </ChartPanel>
- );
- }
- render() {
- const {
- api,
- rule,
- organization,
- timePeriod,
- project,
- interval,
- query,
- location,
- isOnDemandAlert,
- } = this.props;
- const {aggregate, timeWindow, environment, dataset} = rule;
- // Fix for 7 days * 1m interval being over the max number of results from events api
- // 10k events is the current max
- if (
- timePeriod.usingPeriod &&
- timePeriod.period === TimePeriod.SEVEN_DAYS &&
- interval === '1m'
- ) {
- timePeriod.start = getUtcDateString(
- // -5 minutes provides a small cushion for rounding up minutes. This might be able to be smaller
- moment(moment.utc(timePeriod.end).subtract(10000 - 5, 'minutes'))
- );
- }
- // If the chart duration isn't as long as the rollup duration the events-stats
- // endpoint will return an invalid timeseriesData dataset
- const viableStartDate = getUtcDateString(
- moment.min(
- moment.utc(timePeriod.start),
- moment.utc(timePeriod.end).subtract(timeWindow, 'minutes')
- )
- );
- const viableEndDate = getUtcDateString(
- moment.utc(timePeriod.end).add(timeWindow, 'minutes')
- );
- const queryExtras: Record<string, string> = {
- ...getMetricDatasetQueryExtras({
- organization,
- location,
- dataset,
- newAlertOrQuery: false,
- useOnDemandMetrics: isOnDemandAlert,
- }),
- ...getForceMetricsLayerQueryExtras(organization, dataset),
- };
- if (shouldUseErrorsDataset(organization, dataset, query)) {
- queryExtras.dataset = 'errors';
- }
- return isCrashFreeAlert(dataset) ? (
- <SessionsRequest
- api={api}
- organization={organization}
- project={project.id ? [Number(project.id)] : []}
- environment={environment ? [environment] : undefined}
- start={viableStartDate}
- end={viableEndDate}
- query={query}
- interval={interval}
- field={SESSION_AGGREGATE_TO_FIELD[aggregate]}
- groupBy={['session.status']}
- >
- {({loading, response}) =>
- this.renderChart(
- loading,
- transformSessionResponseToSeries(response, rule),
- MINUTES_THRESHOLD_TO_DISPLAY_SECONDS
- )
- }
- </SessionsRequest>
- ) : (
- <EventsRequest
- api={api}
- organization={organization}
- query={query}
- environment={environment ? [environment] : undefined}
- project={project.id ? [Number(project.id)] : []}
- interval={interval}
- comparisonDelta={rule.comparisonDelta ? rule.comparisonDelta * 60 : undefined}
- start={viableStartDate}
- end={viableEndDate}
- yAxis={aggregate}
- includePrevious={false}
- currentSeriesNames={[aggregate]}
- partial={false}
- queryExtras={queryExtras}
- referrer="api.alerts.alert-rule-chart"
- useOnDemandMetrics
- >
- {({loading, timeseriesData, comparisonTimeseriesData}) => (
- <Fragment>
- {this.renderEmptyOnDemandAlert(organization, timeseriesData, loading)}
- {this.renderChart(
- loading,
- timeseriesData,
- undefined,
- comparisonTimeseriesData
- )}
- </Fragment>
- )}
- </EventsRequest>
- );
- }
- }
- export default withSentryRouter(MetricChart);
- const ChartPanel = styled(Panel)`
- margin-top: ${space(2)};
- `;
- const ChartHeader = styled('div')`
- margin-bottom: ${space(3)};
- `;
- const StyledChartControls = styled(ChartControls)`
- display: flex;
- justify-content: space-between;
- flex-wrap: wrap;
- `;
- const StyledInlineContainer = styled(InlineContainer)`
- grid-auto-flow: column;
- grid-column-gap: ${space(1)};
- `;
- const StyledCircleIndicator = styled(CircleIndicator)`
- background: ${p => p.theme.formText};
- height: ${space(1)};
- margin-right: ${space(0.5)};
- `;
- const ChartFilters = styled('div')`
- font-size: ${p => p.theme.fontSizeSmall};
- font-family: ${p => p.theme.text.family};
- color: ${p => p.theme.textColor};
- display: inline-grid;
- grid-template-columns: max-content max-content auto;
- align-items: center;
- `;
- const Filters = styled('span')`
- margin-right: ${space(1)};
- `;
- const QueryFilters = styled('span')`
- min-width: 0px;
- ${p => p.theme.overflowEllipsis}
- `;
- const StyledSectionValue = styled(SectionValue)`
- display: grid;
- grid-template-columns: repeat(4, auto);
- gap: ${space(1.5)};
- margin: 0 0 0 ${space(1.5)};
- `;
- const ValueItem = styled('div')`
- display: grid;
- grid-template-columns: repeat(2, auto);
- gap: ${space(0.5)};
- align-items: center;
- font-variant-numeric: tabular-nums;
- text-underline-offset: ${space(4)};
- `;
- /* Override padding to make chart appear centered */
- const StyledPanelBody = styled(PanelBody)`
- padding-right: 6px;
- `;
- const TriggerChartPlaceholder = styled(Placeholder)`
- height: 200px;
- text-align: center;
- padding: ${space(3)};
- `;
- const StyledTooltip = styled(Tooltip)`
- text-underline-offset: ${space(0.5)} !important;
- `;
|