UNPKG

49.5 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Alarm = exports.TreatMissingData = exports.ComparisonOperator = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const core_1 = require("@aws-cdk/core");
8const alarm_base_1 = require("./alarm-base");
9const cloudwatch_generated_1 = require("./cloudwatch.generated");
10const metric_util_1 = require("./private/metric-util");
11const object_1 = require("./private/object");
12const rendering_1 = require("./private/rendering");
13const statistic_1 = require("./private/statistic");
14/**
15 * Comparison operator for evaluating alarms
16 */
17var ComparisonOperator;
18(function (ComparisonOperator) {
19 /**
20 * Specified statistic is greater than or equal to the threshold
21 */
22 ComparisonOperator["GREATER_THAN_OR_EQUAL_TO_THRESHOLD"] = "GreaterThanOrEqualToThreshold";
23 /**
24 * Specified statistic is strictly greater than the threshold
25 */
26 ComparisonOperator["GREATER_THAN_THRESHOLD"] = "GreaterThanThreshold";
27 /**
28 * Specified statistic is strictly less than the threshold
29 */
30 ComparisonOperator["LESS_THAN_THRESHOLD"] = "LessThanThreshold";
31 /**
32 * Specified statistic is less than or equal to the threshold.
33 */
34 ComparisonOperator["LESS_THAN_OR_EQUAL_TO_THRESHOLD"] = "LessThanOrEqualToThreshold";
35 /**
36 * Specified statistic is lower than or greater than the anomaly model band.
37 * Used only for alarms based on anomaly detection models
38 */
39 ComparisonOperator["LESS_THAN_LOWER_OR_GREATER_THAN_UPPER_THRESHOLD"] = "LessThanLowerOrGreaterThanUpperThreshold";
40 /**
41 * Specified statistic is greater than the anomaly model band.
42 * Used only for alarms based on anomaly detection models
43 */
44 ComparisonOperator["GREATER_THAN_UPPER_THRESHOLD"] = "GreaterThanUpperThreshold";
45 /**
46 * Specified statistic is lower than the anomaly model band.
47 * Used only for alarms based on anomaly detection models
48 */
49 ComparisonOperator["LESS_THAN_LOWER_THRESHOLD"] = "LessThanLowerThreshold";
50})(ComparisonOperator = exports.ComparisonOperator || (exports.ComparisonOperator = {}));
51const OPERATOR_SYMBOLS = {
52 GreaterThanOrEqualToThreshold: '>=',
53 GreaterThanThreshold: '>',
54 LessThanThreshold: '<',
55 LessThanOrEqualToThreshold: '<=',
56};
57/**
58 * Specify how missing data points are treated during alarm evaluation
59 */
60var TreatMissingData;
61(function (TreatMissingData) {
62 /**
63 * Missing data points are treated as breaching the threshold
64 */
65 TreatMissingData["BREACHING"] = "breaching";
66 /**
67 * Missing data points are treated as being within the threshold
68 */
69 TreatMissingData["NOT_BREACHING"] = "notBreaching";
70 /**
71 * The current alarm state is maintained
72 */
73 TreatMissingData["IGNORE"] = "ignore";
74 /**
75 * The alarm does not consider missing data points when evaluating whether to change state
76 */
77 TreatMissingData["MISSING"] = "missing";
78})(TreatMissingData = exports.TreatMissingData || (exports.TreatMissingData = {}));
79/**
80 * An alarm on a CloudWatch metric
81 */
82class Alarm extends alarm_base_1.AlarmBase {
83 constructor(scope, id, props) {
84 var _b;
85 super(scope, id, {
86 physicalName: props.alarmName,
87 });
88 try {
89 jsiiDeprecationWarnings._aws_cdk_aws_cloudwatch_AlarmProps(props);
90 }
91 catch (error) {
92 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
93 Error.captureStackTrace(error, this.constructor);
94 }
95 throw error;
96 }
97 const comparisonOperator = props.comparisonOperator || ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD;
98 // Render metric, process potential overrides from the alarm
99 // (It would be preferable if the statistic etc. was worked into the metric,
100 // but hey we're allowing overrides...)
101 const metricProps = this.renderMetric(props.metric);
102 if (props.period) {
103 metricProps.period = props.period.toSeconds();
104 }
105 if (props.statistic) {
106 // Will overwrite both fields if present
107 Object.assign(metricProps, {
108 statistic: renderIfSimpleStatistic(props.statistic),
109 extendedStatistic: renderIfExtendedStatistic(props.statistic),
110 });
111 }
112 const alarm = new cloudwatch_generated_1.CfnAlarm(this, 'Resource', {
113 // Meta
114 alarmDescription: props.alarmDescription,
115 alarmName: this.physicalName,
116 // Evaluation
117 comparisonOperator,
118 threshold: props.threshold,
119 datapointsToAlarm: props.datapointsToAlarm,
120 evaluateLowSampleCountPercentile: props.evaluateLowSampleCountPercentile,
121 evaluationPeriods: props.evaluationPeriods,
122 treatMissingData: props.treatMissingData,
123 // Actions
124 actionsEnabled: props.actionsEnabled,
125 alarmActions: core_1.Lazy.list({ produce: () => this.alarmActionArns }),
126 insufficientDataActions: core_1.Lazy.list({ produce: (() => this.insufficientDataActionArns) }),
127 okActions: core_1.Lazy.list({ produce: () => this.okActionArns }),
128 // Metric
129 ...metricProps,
130 });
131 this.alarmArn = this.getResourceArnAttribute(alarm.attrArn, {
132 service: 'cloudwatch',
133 resource: 'alarm',
134 resourceName: this.physicalName,
135 arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
136 });
137 this.alarmName = this.getResourceNameAttribute(alarm.ref);
138 this.metric = props.metric;
139 const datapoints = props.datapointsToAlarm || props.evaluationPeriods;
140 this.annotation = {
141 // eslint-disable-next-line max-len
142 label: `${this.metric} ${OPERATOR_SYMBOLS[comparisonOperator]} ${props.threshold} for ${datapoints} datapoints within ${describePeriod(props.evaluationPeriods * metric_util_1.metricPeriod(props.metric).toSeconds())}`,
143 value: props.threshold,
144 };
145 for (const w of (_b = this.metric.warnings) !== null && _b !== void 0 ? _b : []) {
146 core_1.Annotations.of(this).addWarning(w);
147 }
148 }
149 /**
150 * Import an existing CloudWatch alarm provided an ARN
151 *
152 * @param scope The parent creating construct (usually `this`).
153 * @param id The construct's name
154 * @param alarmArn Alarm ARN (i.e. arn:aws:cloudwatch:<region>:<account-id>:alarm:Foo)
155 */
156 static fromAlarmArn(scope, id, alarmArn) {
157 class Import extends alarm_base_1.AlarmBase {
158 constructor() {
159 super(...arguments);
160 this.alarmArn = alarmArn;
161 this.alarmName = core_1.Stack.of(scope).splitArn(alarmArn, core_1.ArnFormat.COLON_RESOURCE_NAME).resourceName;
162 }
163 }
164 return new Import(scope, id);
165 }
166 /**
167 * Turn this alarm into a horizontal annotation
168 *
169 * This is useful if you want to represent an Alarm in a non-AlarmWidget.
170 * An `AlarmWidget` can directly show an alarm, but it can only show a
171 * single alarm and no other metrics. Instead, you can convert the alarm to
172 * a HorizontalAnnotation and add it as an annotation to another graph.
173 *
174 * This might be useful if:
175 *
176 * - You want to show multiple alarms inside a single graph, for example if
177 * you have both a "small margin/long period" alarm as well as a
178 * "large margin/short period" alarm.
179 *
180 * - You want to show an Alarm line in a graph with multiple metrics in it.
181 */
182 toAnnotation() {
183 return this.annotation;
184 }
185 /**
186 * Trigger this action if the alarm fires
187 *
188 * Typically the ARN of an SNS topic or ARN of an AutoScaling policy.
189 */
190 addAlarmAction(...actions) {
191 try {
192 jsiiDeprecationWarnings._aws_cdk_aws_cloudwatch_IAlarmAction(actions);
193 }
194 catch (error) {
195 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
196 Error.captureStackTrace(error, this.addAlarmAction);
197 }
198 throw error;
199 }
200 if (this.alarmActionArns === undefined) {
201 this.alarmActionArns = [];
202 }
203 this.alarmActionArns.push(...actions.map(a => this.validateActionArn(a.bind(this, this).alarmActionArn)));
204 }
205 validateActionArn(actionArn) {
206 var _b, _c, _d;
207 const ec2ActionsRegexp = /arn:aws[a-z0-9-]*:automate:[a-z|\d|-]+:ec2:[a-z]+/;
208 if (ec2ActionsRegexp.test(actionArn)) {
209 // Check per-instance metric
210 const metricConfig = this.metric.toMetricConfig();
211 if (((_c = (_b = metricConfig.metricStat) === null || _b === void 0 ? void 0 : _b.dimensions) === null || _c === void 0 ? void 0 : _c.length) != 1 || ((_d = metricConfig.metricStat) === null || _d === void 0 ? void 0 : _d.dimensions[0].name) != 'InstanceId') {
212 throw new Error(`EC2 alarm actions requires an EC2 Per-Instance Metric. (${JSON.stringify(metricConfig)} does not have an 'InstanceId' dimension)`);
213 }
214 }
215 return actionArn;
216 }
217 renderMetric(metric) {
218 const self = this;
219 return metric_util_1.dispatchMetric(metric, {
220 withStat(stat, conf) {
221 var _b, _c, _d;
222 self.validateMetricStat(stat, metric);
223 const canRenderAsLegacyMetric = ((_b = conf.renderingProperties) === null || _b === void 0 ? void 0 : _b.label) == undefined && !self.requiresAccountId(stat);
224 // Do this to disturb existing templates as little as possible
225 if (canRenderAsLegacyMetric) {
226 return object_1.dropUndefined({
227 dimensions: stat.dimensions,
228 namespace: stat.namespace,
229 metricName: stat.metricName,
230 period: (_c = stat.period) === null || _c === void 0 ? void 0 : _c.toSeconds(),
231 statistic: renderIfSimpleStatistic(stat.statistic),
232 extendedStatistic: renderIfExtendedStatistic(stat.statistic),
233 unit: stat.unitFilter,
234 });
235 }
236 return {
237 metrics: [
238 {
239 metricStat: {
240 metric: {
241 metricName: stat.metricName,
242 namespace: stat.namespace,
243 dimensions: stat.dimensions,
244 },
245 period: stat.period.toSeconds(),
246 stat: stat.statistic,
247 unit: stat.unitFilter,
248 },
249 id: 'm1',
250 accountId: self.requiresAccountId(stat) ? stat.account : undefined,
251 label: (_d = conf.renderingProperties) === null || _d === void 0 ? void 0 : _d.label,
252 returnData: true,
253 },
254 ],
255 };
256 },
257 withExpression() {
258 // Expand the math expression metric into a set
259 const mset = new rendering_1.MetricSet();
260 mset.addTopLevel(true, metric);
261 let eid = 0;
262 function uniqueMetricId() {
263 return `expr_${++eid}`;
264 }
265 return {
266 metrics: mset.entries.map(entry => metric_util_1.dispatchMetric(entry.metric, {
267 withStat(stat, conf) {
268 var _b;
269 self.validateMetricStat(stat, entry.metric);
270 return {
271 metricStat: {
272 metric: {
273 metricName: stat.metricName,
274 namespace: stat.namespace,
275 dimensions: stat.dimensions,
276 },
277 period: stat.period.toSeconds(),
278 stat: stat.statistic,
279 unit: stat.unitFilter,
280 },
281 id: entry.id || uniqueMetricId(),
282 accountId: self.requiresAccountId(stat) ? stat.account : undefined,
283 label: (_b = conf.renderingProperties) === null || _b === void 0 ? void 0 : _b.label,
284 returnData: entry.tag ? undefined : false,
285 };
286 },
287 withExpression(expr, conf) {
288 var _b;
289 const hasSubmetrics = mathExprHasSubmetrics(expr);
290 if (hasSubmetrics) {
291 assertSubmetricsCount(expr);
292 }
293 self.validateMetricExpression(expr);
294 return {
295 expression: expr.expression,
296 id: entry.id || uniqueMetricId(),
297 label: (_b = conf.renderingProperties) === null || _b === void 0 ? void 0 : _b.label,
298 period: hasSubmetrics ? undefined : expr.period,
299 returnData: entry.tag ? undefined : false,
300 };
301 },
302 })),
303 };
304 },
305 });
306 }
307 /**
308 * Validate that if a region is in the given stat config, they match the Alarm
309 */
310 validateMetricStat(stat, metric) {
311 const stack = core_1.Stack.of(this);
312 if (definitelyDifferent(stat.region, stack.region)) {
313 throw new Error(`Cannot create an Alarm in region '${stack.region}' based on metric '${metric}' in '${stat.region}'`);
314 }
315 }
316 /**
317 * Validates that the expression config does not specify searchAccount or searchRegion props
318 * as search expressions are not supported by Alarms.
319 */
320 validateMetricExpression(expr) {
321 if (expr.searchAccount !== undefined || expr.searchRegion !== undefined) {
322 throw new Error('Cannot create an Alarm based on a MathExpression which specifies a searchAccount or searchRegion');
323 }
324 }
325 /**
326 * Determine if the accountId property should be included in the metric.
327 */
328 requiresAccountId(stat) {
329 const stackAccount = core_1.Stack.of(this).account;
330 // if stat.account is undefined, it's by definition in the same account
331 if (stat.account === undefined) {
332 return false;
333 }
334 // Return true if they're different. The ACCOUNT_ID token is interned
335 // so will always have the same string value (and even if we guess wrong
336 // it will still work).
337 return stackAccount !== stat.account;
338 }
339}
340exports.Alarm = Alarm;
341_a = JSII_RTTI_SYMBOL_1;
342Alarm[_a] = { fqn: "@aws-cdk/aws-cloudwatch.Alarm", version: "1.157.0" };
343function definitelyDifferent(x, y) {
344 return x && !core_1.Token.isUnresolved(y) && x !== y;
345}
346/**
347 * Return a human readable string for this period
348 *
349 * We know the seconds are always one of a handful of allowed values.
350 */
351function describePeriod(seconds) {
352 if (seconds === 60) {
353 return '1 minute';
354 }
355 if (seconds === 1) {
356 return '1 second';
357 }
358 if (seconds > 60) {
359 return (seconds / 60) + ' minutes';
360 }
361 return seconds + ' seconds';
362}
363function renderIfSimpleStatistic(statistic) {
364 if (statistic === undefined) {
365 return undefined;
366 }
367 const parsed = statistic_1.parseStatistic(statistic);
368 if (parsed.type === 'simple') {
369 return parsed.statistic;
370 }
371 return undefined;
372}
373function renderIfExtendedStatistic(statistic) {
374 if (statistic === undefined) {
375 return undefined;
376 }
377 const parsed = statistic_1.parseStatistic(statistic);
378 if (parsed.type === 'percentile') {
379 // Already percentile. Avoid parsing because we might get into
380 // floating point rounding issues, return as-is but lowercase the p.
381 return statistic.toLowerCase();
382 }
383 else if (parsed.type === 'generic') {
384 return statistic;
385 }
386 return undefined;
387}
388function mathExprHasSubmetrics(expr) {
389 return Object.keys(expr.usingMetrics).length > 0;
390}
391function assertSubmetricsCount(expr) {
392 if (Object.keys(expr.usingMetrics).length > 10) {
393 // https://docs.aws.amazon.com/AmazonCloudWatch/latest/monitoring/AlarmThatSendsEmail.html#alarms-on-metric-math-expressions
394 throw new Error('Alarms on math expressions cannot contain more than 10 individual metrics');
395 }
396 ;
397}
398//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarm.js","sourceRoot":"","sources":["alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnFormat, Lazy, Stack, Token, Annotations } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { IAlarmAction } from './alarm-action';\nimport { AlarmBase, IAlarm } from './alarm-base';\nimport { CfnAlarm, CfnAlarmProps } from './cloudwatch.generated';\nimport { HorizontalAnnotation } from './graph';\nimport { CreateAlarmOptions } from './metric';\nimport { IMetric, MetricExpressionConfig, MetricStatConfig } from './metric-types';\nimport { dispatchMetric, metricPeriod } from './private/metric-util';\nimport { dropUndefined } from './private/object';\nimport { MetricSet } from './private/rendering';\nimport { parseStatistic } from './private/statistic';\n\n/**\n * Properties for Alarms\n */\nexport interface AlarmProps extends CreateAlarmOptions {\n  /**\n   * The metric to add the alarm on\n   *\n   * Metric objects can be obtained from most resources, or you can construct\n   * custom Metric objects by instantiating one.\n   */\n  readonly metric: IMetric;\n}\n\n/**\n * Comparison operator for evaluating alarms\n */\nexport enum ComparisonOperator {\n  /**\n   * Specified statistic is greater than or equal to the threshold\n   */\n  GREATER_THAN_OR_EQUAL_TO_THRESHOLD = 'GreaterThanOrEqualToThreshold',\n\n  /**\n   * Specified statistic is strictly greater than the threshold\n   */\n  GREATER_THAN_THRESHOLD = 'GreaterThanThreshold',\n\n  /**\n   * Specified statistic is strictly less than the threshold\n   */\n  LESS_THAN_THRESHOLD = 'LessThanThreshold',\n\n  /**\n   * Specified statistic is less than or equal to the threshold.\n   */\n  LESS_THAN_OR_EQUAL_TO_THRESHOLD = 'LessThanOrEqualToThreshold',\n\n  /**\n   * Specified statistic is lower than or greater than the anomaly model band.\n   * Used only for alarms based on anomaly detection models\n   */\n  LESS_THAN_LOWER_OR_GREATER_THAN_UPPER_THRESHOLD = 'LessThanLowerOrGreaterThanUpperThreshold',\n\n  /**\n   * Specified statistic is greater than the anomaly model band.\n   * Used only for alarms based on anomaly detection models\n   */\n  GREATER_THAN_UPPER_THRESHOLD = 'GreaterThanUpperThreshold',\n\n  /**\n   * Specified statistic is lower than the anomaly model band.\n   * Used only for alarms based on anomaly detection models\n   */\n  LESS_THAN_LOWER_THRESHOLD = 'LessThanLowerThreshold',\n}\n\nconst OPERATOR_SYMBOLS: {[key: string]: string} = {\n  GreaterThanOrEqualToThreshold: '>=',\n  GreaterThanThreshold: '>',\n  LessThanThreshold: '<',\n  LessThanOrEqualToThreshold: '<=',\n};\n\n/**\n * Specify how missing data points are treated during alarm evaluation\n */\nexport enum TreatMissingData {\n  /**\n   * Missing data points are treated as breaching the threshold\n   */\n  BREACHING = 'breaching',\n\n  /**\n   * Missing data points are treated as being within the threshold\n   */\n  NOT_BREACHING = 'notBreaching',\n\n  /**\n   * The current alarm state is maintained\n   */\n  IGNORE = 'ignore',\n\n  /**\n   * The alarm does not consider missing data points when evaluating whether to change state\n   */\n  MISSING = 'missing'\n}\n\n/**\n * An alarm on a CloudWatch metric\n */\nexport class Alarm extends AlarmBase {\n\n  /**\n   * Import an existing CloudWatch alarm provided an ARN\n   *\n   * @param scope The parent creating construct (usually `this`).\n   * @param id The construct's name\n   * @param alarmArn Alarm ARN (i.e. arn:aws:cloudwatch:<region>:<account-id>:alarm:Foo)\n   */\n  public static fromAlarmArn(scope: Construct, id: string, alarmArn: string): IAlarm {\n    class Import extends AlarmBase implements IAlarm {\n      public readonly alarmArn = alarmArn;\n      public readonly alarmName = Stack.of(scope).splitArn(alarmArn, ArnFormat.COLON_RESOURCE_NAME).resourceName!;\n    }\n    return new Import(scope, id);\n  }\n\n  /**\n   * ARN of this alarm\n   *\n   * @attribute\n   */\n  public readonly alarmArn: string;\n\n  /**\n   * Name of this alarm.\n   *\n   * @attribute\n   */\n  public readonly alarmName: string;\n\n  /**\n   * The metric object this alarm was based on\n   */\n  public readonly metric: IMetric;\n\n  /**\n   * This metric as an annotation\n   */\n  private readonly annotation: HorizontalAnnotation;\n\n  constructor(scope: Construct, id: string, props: AlarmProps) {\n    super(scope, id, {\n      physicalName: props.alarmName,\n    });\n\n    const comparisonOperator = props.comparisonOperator || ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD;\n\n    // Render metric, process potential overrides from the alarm\n    // (It would be preferable if the statistic etc. was worked into the metric,\n    // but hey we're allowing overrides...)\n    const metricProps: Writeable<Partial<CfnAlarmProps>> = this.renderMetric(props.metric);\n    if (props.period) {\n      metricProps.period = props.period.toSeconds();\n    }\n    if (props.statistic) {\n      // Will overwrite both fields if present\n      Object.assign(metricProps, {\n        statistic: renderIfSimpleStatistic(props.statistic),\n        extendedStatistic: renderIfExtendedStatistic(props.statistic),\n      });\n    }\n\n    const alarm = new CfnAlarm(this, 'Resource', {\n      // Meta\n      alarmDescription: props.alarmDescription,\n      alarmName: this.physicalName,\n\n      // Evaluation\n      comparisonOperator,\n      threshold: props.threshold,\n      datapointsToAlarm: props.datapointsToAlarm,\n      evaluateLowSampleCountPercentile: props.evaluateLowSampleCountPercentile,\n      evaluationPeriods: props.evaluationPeriods,\n      treatMissingData: props.treatMissingData,\n\n      // Actions\n      actionsEnabled: props.actionsEnabled,\n      alarmActions: Lazy.list({ produce: () => this.alarmActionArns }),\n      insufficientDataActions: Lazy.list({ produce: (() => this.insufficientDataActionArns) }),\n      okActions: Lazy.list({ produce: () => this.okActionArns }),\n\n      // Metric\n      ...metricProps,\n    });\n\n    this.alarmArn = this.getResourceArnAttribute(alarm.attrArn, {\n      service: 'cloudwatch',\n      resource: 'alarm',\n      resourceName: this.physicalName,\n      arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n    });\n    this.alarmName = this.getResourceNameAttribute(alarm.ref);\n\n    this.metric = props.metric;\n    const datapoints = props.datapointsToAlarm || props.evaluationPeriods;\n    this.annotation = {\n      // eslint-disable-next-line max-len\n      label: `${this.metric} ${OPERATOR_SYMBOLS[comparisonOperator]} ${props.threshold} for ${datapoints} datapoints within ${describePeriod(props.evaluationPeriods * metricPeriod(props.metric).toSeconds())}`,\n      value: props.threshold,\n    };\n\n    for (const w of this.metric.warnings ?? []) {\n      Annotations.of(this).addWarning(w);\n    }\n  }\n\n  /**\n   * Turn this alarm into a horizontal annotation\n   *\n   * This is useful if you want to represent an Alarm in a non-AlarmWidget.\n   * An `AlarmWidget` can directly show an alarm, but it can only show a\n   * single alarm and no other metrics. Instead, you can convert the alarm to\n   * a HorizontalAnnotation and add it as an annotation to another graph.\n   *\n   * This might be useful if:\n   *\n   * - You want to show multiple alarms inside a single graph, for example if\n   *   you have both a \"small margin/long period\" alarm as well as a\n   *   \"large margin/short period\" alarm.\n   *\n   * - You want to show an Alarm line in a graph with multiple metrics in it.\n   */\n  public toAnnotation(): HorizontalAnnotation {\n    return this.annotation;\n  }\n\n  /**\n   * Trigger this action if the alarm fires\n   *\n   * Typically the ARN of an SNS topic or ARN of an AutoScaling policy.\n   */\n  public addAlarmAction(...actions: IAlarmAction[]) {\n    if (this.alarmActionArns === undefined) {\n      this.alarmActionArns = [];\n    }\n\n    this.alarmActionArns.push(...actions.map(a =>\n      this.validateActionArn(a.bind(this, this).alarmActionArn),\n    ));\n  }\n\n  private validateActionArn(actionArn: string): string {\n    const ec2ActionsRegexp: RegExp = /arn:aws[a-z0-9-]*:automate:[a-z|\\d|-]+:ec2:[a-z]+/;\n    if (ec2ActionsRegexp.test(actionArn)) {\n      // Check per-instance metric\n      const metricConfig = this.metric.toMetricConfig();\n      if (metricConfig.metricStat?.dimensions?.length != 1 || metricConfig.metricStat?.dimensions![0].name != 'InstanceId') {\n        throw new Error(`EC2 alarm actions requires an EC2 Per-Instance Metric. (${JSON.stringify(metricConfig)} does not have an 'InstanceId' dimension)`);\n      }\n    }\n    return actionArn;\n  }\n\n  private renderMetric(metric: IMetric) {\n    const self = this;\n    return dispatchMetric(metric, {\n      withStat(stat, conf) {\n        self.validateMetricStat(stat, metric);\n        const canRenderAsLegacyMetric = conf.renderingProperties?.label == undefined && !self.requiresAccountId(stat);\n        // Do this to disturb existing templates as little as possible\n        if (canRenderAsLegacyMetric) {\n          return dropUndefined({\n            dimensions: stat.dimensions,\n            namespace: stat.namespace,\n            metricName: stat.metricName,\n            period: stat.period?.toSeconds(),\n            statistic: renderIfSimpleStatistic(stat.statistic),\n            extendedStatistic: renderIfExtendedStatistic(stat.statistic),\n            unit: stat.unitFilter,\n          });\n        }\n\n        return {\n          metrics: [\n            {\n              metricStat: {\n                metric: {\n                  metricName: stat.metricName,\n                  namespace: stat.namespace,\n                  dimensions: stat.dimensions,\n                },\n                period: stat.period.toSeconds(),\n                stat: stat.statistic,\n                unit: stat.unitFilter,\n              },\n              id: 'm1',\n              accountId: self.requiresAccountId(stat) ? stat.account : undefined,\n              label: conf.renderingProperties?.label,\n              returnData: true,\n            } as CfnAlarm.MetricDataQueryProperty,\n          ],\n        };\n      },\n\n      withExpression() {\n        // Expand the math expression metric into a set\n        const mset = new MetricSet<boolean>();\n        mset.addTopLevel(true, metric);\n\n        let eid = 0;\n        function uniqueMetricId() {\n          return `expr_${++eid}`;\n        }\n\n        return {\n          metrics: mset.entries.map(entry => dispatchMetric(entry.metric, {\n            withStat(stat, conf) {\n              self.validateMetricStat(stat, entry.metric);\n\n              return {\n                metricStat: {\n                  metric: {\n                    metricName: stat.metricName,\n                    namespace: stat.namespace,\n                    dimensions: stat.dimensions,\n                  },\n                  period: stat.period.toSeconds(),\n                  stat: stat.statistic,\n                  unit: stat.unitFilter,\n                },\n                id: entry.id || uniqueMetricId(),\n                accountId: self.requiresAccountId(stat) ? stat.account : undefined,\n                label: conf.renderingProperties?.label,\n                returnData: entry.tag ? undefined : false, // entry.tag evaluates to true if the metric is the math expression the alarm is based on.\n              };\n            },\n            withExpression(expr, conf) {\n\n              const hasSubmetrics = mathExprHasSubmetrics(expr);\n\n              if (hasSubmetrics) {\n                assertSubmetricsCount(expr);\n              }\n\n              self.validateMetricExpression(expr);\n\n              return {\n                expression: expr.expression,\n                id: entry.id || uniqueMetricId(),\n                label: conf.renderingProperties?.label,\n                period: hasSubmetrics ? undefined : expr.period,\n                returnData: entry.tag ? undefined : false, // entry.tag evaluates to true if the metric is the math expression the alarm is based on.\n              };\n            },\n          }) as CfnAlarm.MetricDataQueryProperty),\n        };\n      },\n    });\n  }\n\n  /**\n   * Validate that if a region is in the given stat config, they match the Alarm\n   */\n  private validateMetricStat(stat: MetricStatConfig, metric: IMetric) {\n    const stack = Stack.of(this);\n\n    if (definitelyDifferent(stat.region, stack.region)) {\n      throw new Error(`Cannot create an Alarm in region '${stack.region}' based on metric '${metric}' in '${stat.region}'`);\n    }\n  }\n\n  /**\n   * Validates that the expression config does not specify searchAccount or searchRegion props\n   * as search expressions are not supported by Alarms.\n   */\n  private validateMetricExpression(expr: MetricExpressionConfig) {\n    if (expr.searchAccount !== undefined || expr.searchRegion !== undefined) {\n      throw new Error('Cannot create an Alarm based on a MathExpression which specifies a searchAccount or searchRegion');\n    }\n  }\n\n  /**\n   * Determine if the accountId property should be included in the metric.\n   */\n  private requiresAccountId(stat: MetricStatConfig): boolean {\n    const stackAccount = Stack.of(this).account;\n\n    // if stat.account is undefined, it's by definition in the same account\n    if (stat.account === undefined) {\n      return false;\n    }\n\n    // Return true if they're different. The ACCOUNT_ID token is interned\n    // so will always have the same string value (and even if we guess wrong\n    // it will still work).\n    return stackAccount !== stat.account;\n  }\n}\n\nfunction definitelyDifferent(x: string | undefined, y: string) {\n  return x && !Token.isUnresolved(y) && x !== y;\n}\n\n/**\n * Return a human readable string for this period\n *\n * We know the seconds are always one of a handful of allowed values.\n */\nfunction describePeriod(seconds: number) {\n  if (seconds === 60) { return '1 minute'; }\n  if (seconds === 1) { return '1 second'; }\n  if (seconds > 60) { return (seconds / 60) + ' minutes'; }\n  return seconds + ' seconds';\n}\n\nfunction renderIfSimpleStatistic(statistic?: string): string | undefined {\n  if (statistic === undefined) { return undefined; }\n\n  const parsed = parseStatistic(statistic);\n  if (parsed.type === 'simple') {\n    return parsed.statistic;\n  }\n  return undefined;\n}\n\nfunction renderIfExtendedStatistic(statistic?: string): string | undefined {\n  if (statistic === undefined) { return undefined; }\n\n  const parsed = parseStatistic(statistic);\n  if (parsed.type === 'percentile') {\n    // Already percentile. Avoid parsing because we might get into\n    // floating point rounding issues, return as-is but lowercase the p.\n    return statistic.toLowerCase();\n  } else if (parsed.type === 'generic') {\n    return statistic;\n  }\n\n  return undefined;\n}\n\nfunction mathExprHasSubmetrics(expr: MetricExpressionConfig) {\n  return Object.keys(expr.usingMetrics).length > 0;\n}\n\nfunction assertSubmetricsCount(expr: MetricExpressionConfig) {\n  if (Object.keys(expr.usingMetrics).length > 10) {\n    // https://docs.aws.amazon.com/AmazonCloudWatch/latest/monitoring/AlarmThatSendsEmail.html#alarms-on-metric-math-expressions\n    throw new Error('Alarms on math expressions cannot contain more than 10 individual metrics');\n  };\n}\n\ntype Writeable<T> = { -readonly [P in keyof T]: T[P] };\n"]}
\No newline at end of file