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