diff --git a/x-pack/plugins/infra/public/alerting/metric_threshold/components/expression.tsx b/x-pack/plugins/infra/public/alerting/metric_threshold/components/expression.tsx index 8835a7cd55ce8..be0ecbb1dab65 100644 --- a/x-pack/plugins/infra/public/alerting/metric_threshold/components/expression.tsx +++ b/x-pack/plugins/infra/public/alerting/metric_threshold/components/expression.tsx @@ -17,6 +17,8 @@ import { EuiToolTip, EuiIcon, EuiFieldSearch, + EuiAccordion, + EuiPanel, } from '@elastic/eui'; import { FormattedMessage } from '@kbn/i18n/react'; import { i18n } from '@kbn/i18n'; @@ -259,6 +261,11 @@ export const Expressions: React.FC = (props) => { return alertParams.groupBy; }, [alertParams.groupBy]); + const areAllAggsRate = useMemo( + () => alertParams.criteria?.every((c) => c.aggType === Aggregators.RATE), + [alertParams.criteria] + ); + return ( <> @@ -323,27 +330,60 @@ export const Expressions: React.FC = (props) => { - - {i18n.translate('xpack.infra.metrics.alertFlyout.alertOnNoData', { - defaultMessage: "Alert me if there's no data", - })}{' '} - - - - - } - checked={alertParams.alertOnNoData} - onChange={(e) => setAlertParams('alertOnNoData', e.target.checked)} - /> - + + + + {i18n.translate('xpack.infra.metrics.alertFlyout.alertOnNoData', { + defaultMessage: "Alert me if there's no data", + })}{' '} + + + + + } + checked={alertParams.alertOnNoData} + onChange={(e) => setAlertParams('alertOnNoData', e.target.checked)} + /> + + {i18n.translate('xpack.infra.metrics.alertFlyout.shouldDropPartialBuckets', { + defaultMessage: 'Drop partial buckets when evaluating data', + })}{' '} + + + + + } + checked={areAllAggsRate || alertParams.shouldDropPartialBuckets} + disabled={areAllAggsRate} + onChange={(e) => setAlertParams('shouldDropPartialBuckets', e.target.checked)} + /> + + = (props) => { alertThrottle={alertThrottle} alertNotifyWhen={alertNotifyWhen} alertType={METRIC_THRESHOLD_ALERT_TYPE_ID} - alertParams={pick(alertParams, 'criteria', 'groupBy', 'filterQuery', 'sourceId')} + alertParams={pick( + alertParams, + 'criteria', + 'groupBy', + 'filterQuery', + 'sourceId', + 'shouldDropPartialBuckets' + )} showNoDataResults={alertParams.alertOnNoData} validate={validateMetricThreshold} groupByDisplayName={groupByPreviewDisplayName} diff --git a/x-pack/plugins/infra/public/alerting/metric_threshold/types.ts b/x-pack/plugins/infra/public/alerting/metric_threshold/types.ts index fca4160199030..a679579e57235 100644 --- a/x-pack/plugins/infra/public/alerting/metric_threshold/types.ts +++ b/x-pack/plugins/infra/public/alerting/metric_threshold/types.ts @@ -61,4 +61,5 @@ export interface AlertParams { sourceId: string; filterQueryText?: string; alertOnNoData?: boolean; + shouldDropPartialBuckets?: boolean; } diff --git a/x-pack/plugins/infra/server/lib/alerting/metric_threshold/lib/evaluate_alert.ts b/x-pack/plugins/infra/server/lib/alerting/metric_threshold/lib/evaluate_alert.ts index aeeb705f9b25a..6d99b6e2f5fe3 100644 --- a/x-pack/plugins/infra/server/lib/alerting/metric_threshold/lib/evaluate_alert.ts +++ b/x-pack/plugins/infra/server/lib/alerting/metric_threshold/lib/evaluate_alert.ts @@ -6,6 +6,7 @@ */ import { mapValues, first, last, isNaN } from 'lodash'; +import moment from 'moment'; import { ElasticsearchClient } from 'kibana/server'; import { isTooManyBucketsPreviewException, @@ -44,6 +45,7 @@ export interface EvaluatedAlertParams { criteria: MetricExpressionParams[]; groupBy: string | undefined | string[]; filterQuery: string | undefined; + shouldDropPartialBuckets?: boolean; } export const evaluateAlert = ( @@ -52,7 +54,7 @@ export const evaluateAlert = { - const { criteria, groupBy, filterQuery } = params; + const { criteria, groupBy, filterQuery, shouldDropPartialBuckets } = params; return Promise.all( criteria.map(async (criterion) => { const currentValues = await getMetric( @@ -62,7 +64,8 @@ export const evaluateAlert = Promise> = async function ( esClient, params, @@ -112,7 +114,8 @@ const getMetric: ( timefield, groupBy, filterQuery, - timeframe + timeframe, + shouldDropPartialBuckets ) { const { aggType, timeSize, timeUnit } = params; const hasGroupBy = groupBy && groupBy.length; @@ -121,10 +124,15 @@ const getMetric: ( const intervalAsSeconds = getIntervalInSeconds(interval); const intervalAsMS = intervalAsSeconds * 1000; - const to = roundTimestamp(timeframe ? timeframe.end : Date.now(), timeUnit); - // We need enough data for 5 buckets worth of data. We also need - // to convert the intervalAsSeconds to milliseconds. - const minimumFrom = to - intervalAsMS * MINIMUM_BUCKETS; + const to = moment(timeframe ? timeframe.end : Date.now()) + .add(1, timeUnit) + .startOf(timeUnit) + .valueOf(); + + // Rate aggregations need 5 buckets worth of data + const minimumBuckets = aggType === Aggregators.RATE ? 5 : 1; + + const minimumFrom = to - intervalAsMS * minimumBuckets; const from = roundTimestamp( timeframe && timeframe.start <= minimumFrom ? timeframe.start : minimumFrom, @@ -139,6 +147,16 @@ const getMetric: ( filterQuery ); + const dropPartialBucketsOptions = + // Rate aggs always drop partial buckets; guard against this boolean being passed as false + shouldDropPartialBuckets || aggType === Aggregators.RATE + ? { + from, + to, + bucketSizeInMillis: intervalAsMS, + } + : null; + try { if (hasGroupBy) { const bucketSelector = ( @@ -160,11 +178,7 @@ const getMetric: ( ...result, [Object.values(bucket.key) .map((value) => value) - .join(', ')]: getValuesFromAggregations(bucket, aggType, { - from, - to, - bucketSizeInMillis: intervalAsMS, - }), + .join(', ')]: getValuesFromAggregations(bucket, aggType, dropPartialBucketsOptions), }), {} ); @@ -178,7 +192,7 @@ const getMetric: ( [UNGROUPED_FACTORY_KEY]: getValuesFromAggregations( (result.aggregations! as unknown) as Aggregation, aggType, - { from, to, bucketSizeInMillis: intervalAsMS } + dropPartialBucketsOptions ), }; } catch (e) { @@ -218,35 +232,46 @@ const dropPartialBuckets = ({ from, to, bucketSizeInMillis }: DropPartialBucketO const getValuesFromAggregations = ( aggregations: Aggregation, aggType: MetricExpressionParams['aggType'], - dropPartialBucketsOptions: DropPartialBucketOptions + dropPartialBucketsOptions: DropPartialBucketOptions | null ) => { try { const { buckets } = aggregations.aggregatedIntervals; if (!buckets.length) return null; // No Data state + + let mappedBuckets; + if (aggType === Aggregators.COUNT) { - return buckets - .map((bucket) => ({ - key: bucket.from_as_string, - value: bucket.doc_count, - })) - .filter(dropPartialBuckets(dropPartialBucketsOptions)); - } - if (aggType === Aggregators.P95 || aggType === Aggregators.P99) { - return buckets - .map((bucket) => { - const values = bucket.aggregatedValue?.values || []; - const firstValue = first(values); - if (!firstValue) return null; - return { key: bucket.from_as_string, value: firstValue.value }; - }) - .filter(dropPartialBuckets(dropPartialBucketsOptions)); - } - return buckets - .map((bucket) => ({ + mappedBuckets = buckets.map((bucket) => ({ + key: bucket.from_as_string, + value: bucket.doc_count, + })); + } else if (aggType === Aggregators.P95 || aggType === Aggregators.P99) { + mappedBuckets = buckets.map((bucket) => { + const values = bucket.aggregatedValue?.values || []; + const firstValue = first(values); + if (!firstValue) return null; + return { key: bucket.from_as_string, value: firstValue.value }; + }); + } else if (aggType === Aggregators.AVERAGE) { + mappedBuckets = buckets.map((bucket) => ({ + key: bucket.key_as_string ?? bucket.from_as_string, + value: bucket.aggregatedValue?.value ?? null, + })); + } else if (aggType === Aggregators.RATE) { + mappedBuckets = buckets.map((bucket) => ({ key: bucket.key_as_string ?? bucket.from_as_string, value: bucket.aggregatedValue?.value ?? null, - })) - .filter(dropPartialBuckets(dropPartialBucketsOptions)); + })); + } else { + mappedBuckets = buckets.map((bucket) => ({ + key: bucket.key_as_string ?? bucket.from_as_string, + value: bucket.aggregatedValue?.value ?? null, + })); + } + if (dropPartialBucketsOptions) { + return mappedBuckets.filter(dropPartialBuckets(dropPartialBucketsOptions)); + } + return mappedBuckets; } catch (e) { return NaN; // Error state } diff --git a/x-pack/plugins/infra/server/lib/alerting/metric_threshold/preview_metric_threshold_alert.ts b/x-pack/plugins/infra/server/lib/alerting/metric_threshold/preview_metric_threshold_alert.ts index a4c207f4006d5..931b830875cdf 100644 --- a/x-pack/plugins/infra/server/lib/alerting/metric_threshold/preview_metric_threshold_alert.ts +++ b/x-pack/plugins/infra/server/lib/alerting/metric_threshold/preview_metric_threshold_alert.ts @@ -26,6 +26,7 @@ interface PreviewMetricThresholdAlertParams { criteria: MetricExpressionParams[]; groupBy: string | undefined | string[]; filterQuery: string | undefined; + shouldDropPartialBuckets?: boolean; }; config: InfraSource['configuration']; lookback: Unit;