cdk-monitoring-constructs
Version:
[](https://badge.fury.io/js/cdk-monitoring-constructs) [](https://m
122 lines • 18.4 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.LogMonitoring = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_cloudwatch_1 = require("aws-cdk-lib/aws-cloudwatch");
const CloudWatchLogsMetricFactory_1 = require("./CloudWatchLogsMetricFactory");
const common_1 = require("../../common");
const dashboard_1 = require("../../dashboard");
const DefaultLimit = 10;
/**
* Monitors a CloudWatch log group for various patterns.
*/
class LogMonitoring extends common_1.Monitoring {
constructor(scope, props) {
super(scope);
this.logGroupName = props.logGroupName;
this.logGroupUrl = scope
.createAwsConsoleUrlFactory()
.getCloudWatchLogGroupUrl(props.logGroupName);
this.title = props.title;
this.pattern = props.pattern;
this.filterExpressions = props.filterExpressions;
this.limit = props.limit ?? DefaultLimit;
const namingStrategy = new dashboard_1.MonitoringNamingStrategy({
...props,
fallbackConstructName: this.logGroupName,
});
this.alarmFactory = this.createAlarmFactory(namingStrategy.resolveAlarmFriendlyName());
this.usageAlarmFactory = new common_1.UsageAlarmFactory(this.alarmFactory);
this.usageAnnotations = [];
const metricFactory = new CloudWatchLogsMetricFactory_1.CloudWatchLogsMetricFactory(scope.createMetricFactory(), props);
this.incomingLogEventsMetric = metricFactory.metricIncomingLogEvents();
for (const disambiguator in props.addMinIncomingLogsAlarm) {
const alarmProps = props.addMinIncomingLogsAlarm[disambiguator];
const createdAlarm = this.usageAlarmFactory.addMinUsageCountAlarm(this.incomingLogEventsMetric, alarmProps, disambiguator);
this.usageAnnotations.push(createdAlarm.annotation);
this.addAlarm(createdAlarm);
}
for (const disambiguator in props.addMaxIncomingLogsAlarm) {
const alarmProps = props.addMaxIncomingLogsAlarm[disambiguator];
const createdAlarm = this.usageAlarmFactory.addMaxCountAlarm(this.incomingLogEventsMetric, alarmProps, disambiguator);
this.usageAnnotations.push(createdAlarm.annotation);
this.addAlarm(createdAlarm);
}
props.useCreatedAlarms?.consume(this.createdAlarms());
}
summaryWidgets() {
return [
this.createTitleWidget(),
this.createIncomingLogsWidget(common_1.FullWidth, common_1.DefaultSummaryWidgetHeight),
];
}
widgets() {
const filterStatements = [];
if (this.pattern) {
filterStatements.push(`filter like /${this.pattern}/`);
}
if (this.filterExpressions) {
for (const expression of this.filterExpressions) {
if (expression) {
filterStatements.push(`filter ${expression}`);
}
}
}
if (filterStatements.length > 0) {
const height = this.resolveRecommendedHeight(this.limit);
return [
this.createTitleWidget(),
// Log Query Results
new aws_cloudwatch_1.LogQueryWidget({
logGroupNames: [this.logGroupName],
height,
width: common_1.ThreeQuartersWidth,
title: this.title ?? `Find ${this.pattern} (limit = ${this.limit})`,
/**
* https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/CWL_QuerySyntax.html
*/
queryLines: [
"fields @timestamp, @logStream, @message",
...filterStatements,
"sort @timestamp desc",
`limit ${this.limit}`,
],
}),
this.createIncomingLogsWidget(common_1.QuarterWidth, height),
];
}
else {
return [
this.createTitleWidget(),
this.createIncomingLogsWidget(common_1.FullWidth, common_1.DefaultGraphWidgetHeight),
];
}
}
createTitleWidget() {
return new dashboard_1.MonitoringHeaderWidget({
family: "Log Group",
title: this.logGroupName,
goToLinkUrl: this.logGroupUrl,
});
}
createIncomingLogsWidget(width, height) {
return new aws_cloudwatch_1.GraphWidget({
width,
height,
title: "Incoming logs",
left: [this.incomingLogEventsMetric],
leftYAxis: common_1.CountAxisFromZero,
leftAnnotations: this.usageAnnotations,
});
}
resolveRecommendedHeight(numRows) {
const heightPerLine = 1;
const recommendedHeight = heightPerLine * numRows;
return Math.max(recommendedHeight, common_1.DefaultLogWidgetHeight);
}
}
exports.LogMonitoring = LogMonitoring;
_a = JSII_RTTI_SYMBOL_1;
LogMonitoring[_a] = { fqn: "cdk-monitoring-constructs.LogMonitoring", version: "10.0.0" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LogMonitoring.js","sourceRoot":"","sources":["LogMonitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  GraphWidget,\n  HorizontalAnnotation,\n  IWidget,\n  LogQueryWidget,\n} from \"aws-cdk-lib/aws-cloudwatch\";\n\nimport {\n  CloudWatchLogsMetricFactory,\n  CloudWatchLogsMetricFactoryProps,\n} from \"./CloudWatchLogsMetricFactory\";\nimport {\n  AlarmFactory,\n  BaseMonitoringProps,\n  CountAxisFromZero,\n  DefaultGraphWidgetHeight,\n  DefaultLogWidgetHeight,\n  DefaultSummaryWidgetHeight,\n  FullWidth,\n  MaxUsageCountThreshold,\n  MetricWithAlarmSupport,\n  MinUsageCountThreshold,\n  Monitoring,\n  MonitoringScope,\n  QuarterWidth,\n  ThreeQuartersWidth,\n  UsageAlarmFactory,\n} from \"../../common\";\nimport {\n  MonitoringHeaderWidget,\n  MonitoringNamingStrategy,\n} from \"../../dashboard\";\n\nconst DefaultLimit = 10;\n\nexport interface LogMonitoringProps\n  extends BaseMonitoringProps,\n    CloudWatchLogsMetricFactoryProps {\n  /**\n   * Widget title\n   *\n   * @default - auto-generated title based on the pattern and limit\n   */\n  readonly title?: string;\n\n  /**\n   * Pattern to filter `@message` field, e.g. \"ERROR\"\n   */\n  readonly pattern?: string;\n\n  /**\n   * Filter expressions to add.\n   * @example\n   * filterExpressions = [`level = \"ERROR\"`]\n   * // will be appended to the query as\n   * | filter level = \"ERROR\"\n   * @see https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/CWL_QuerySyntax-Filter.html\n   */\n  readonly filterExpressions?: string[];\n\n  /**\n   * Maximum number of log messages to search for.\n   *\n   * @default - 10\n   */\n  readonly limit?: number;\n\n  readonly addMinIncomingLogsAlarm?: Record<string, MinUsageCountThreshold>;\n  readonly addMaxIncomingLogsAlarm?: Record<string, MaxUsageCountThreshold>;\n}\n\n/**\n * Monitors a CloudWatch log group for various patterns.\n */\nexport class LogMonitoring extends Monitoring {\n  readonly logGroupName: string;\n  readonly logGroupUrl?: string;\n\n  readonly title?: string;\n\n  readonly pattern?: string;\n  readonly filterExpressions?: string[];\n  readonly limit: number;\n\n  readonly alarmFactory: AlarmFactory;\n  readonly usageAlarmFactory: UsageAlarmFactory;\n  readonly incomingLogEventsMetric: MetricWithAlarmSupport;\n\n  readonly usageAnnotations: HorizontalAnnotation[];\n\n  constructor(scope: MonitoringScope, props: LogMonitoringProps) {\n    super(scope);\n\n    this.logGroupName = props.logGroupName;\n    this.logGroupUrl = scope\n      .createAwsConsoleUrlFactory()\n      .getCloudWatchLogGroupUrl(props.logGroupName);\n\n    this.title = props.title;\n\n    this.pattern = props.pattern;\n    this.filterExpressions = props.filterExpressions;\n    this.limit = props.limit ?? DefaultLimit;\n\n    const namingStrategy = new MonitoringNamingStrategy({\n      ...props,\n      fallbackConstructName: this.logGroupName,\n    });\n    this.alarmFactory = this.createAlarmFactory(\n      namingStrategy.resolveAlarmFriendlyName(),\n    );\n    this.usageAlarmFactory = new UsageAlarmFactory(this.alarmFactory);\n\n    this.usageAnnotations = [];\n\n    const metricFactory = new CloudWatchLogsMetricFactory(\n      scope.createMetricFactory(),\n      props,\n    );\n    this.incomingLogEventsMetric = metricFactory.metricIncomingLogEvents();\n\n    for (const disambiguator in props.addMinIncomingLogsAlarm) {\n      const alarmProps = props.addMinIncomingLogsAlarm[disambiguator];\n      const createdAlarm = this.usageAlarmFactory.addMinUsageCountAlarm(\n        this.incomingLogEventsMetric,\n        alarmProps,\n        disambiguator,\n      );\n      this.usageAnnotations.push(createdAlarm.annotation);\n      this.addAlarm(createdAlarm);\n    }\n\n    for (const disambiguator in props.addMaxIncomingLogsAlarm) {\n      const alarmProps = props.addMaxIncomingLogsAlarm[disambiguator];\n      const createdAlarm = this.usageAlarmFactory.addMaxCountAlarm(\n        this.incomingLogEventsMetric,\n        alarmProps,\n        disambiguator,\n      );\n      this.usageAnnotations.push(createdAlarm.annotation);\n      this.addAlarm(createdAlarm);\n    }\n\n    props.useCreatedAlarms?.consume(this.createdAlarms());\n  }\n\n  summaryWidgets(): IWidget[] {\n    return [\n      this.createTitleWidget(),\n      this.createIncomingLogsWidget(FullWidth, DefaultSummaryWidgetHeight),\n    ];\n  }\n\n  widgets(): IWidget[] {\n    const filterStatements: string[] = [];\n\n    if (this.pattern) {\n      filterStatements.push(`filter @message like /${this.pattern}/`);\n    }\n\n    if (this.filterExpressions) {\n      for (const expression of this.filterExpressions) {\n        if (expression) {\n          filterStatements.push(`filter ${expression}`);\n        }\n      }\n    }\n\n    if (filterStatements.length > 0) {\n      const height = this.resolveRecommendedHeight(this.limit);\n\n      return [\n        this.createTitleWidget(),\n\n        // Log Query Results\n        new LogQueryWidget({\n          logGroupNames: [this.logGroupName],\n          height,\n          width: ThreeQuartersWidth,\n          title: this.title ?? `Find ${this.pattern} (limit = ${this.limit})`,\n          /**\n           * https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/CWL_QuerySyntax.html\n           */\n          queryLines: [\n            \"fields @timestamp, @logStream, @message\",\n            ...filterStatements,\n            \"sort @timestamp desc\",\n            `limit ${this.limit}`,\n          ],\n        }),\n\n        this.createIncomingLogsWidget(QuarterWidth, height),\n      ];\n    } else {\n      return [\n        this.createTitleWidget(),\n        this.createIncomingLogsWidget(FullWidth, DefaultGraphWidgetHeight),\n      ];\n    }\n  }\n\n  createTitleWidget() {\n    return new MonitoringHeaderWidget({\n      family: \"Log Group\",\n      title: this.logGroupName,\n      goToLinkUrl: this.logGroupUrl,\n    });\n  }\n\n  createIncomingLogsWidget(width: number, height: number) {\n    return new GraphWidget({\n      width,\n      height,\n      title: \"Incoming logs\",\n      left: [this.incomingLogEventsMetric],\n      leftYAxis: CountAxisFromZero,\n      leftAnnotations: this.usageAnnotations,\n    });\n  }\n\n  protected resolveRecommendedHeight(numRows: number) {\n    const heightPerLine = 1;\n    const recommendedHeight = heightPerLine * numRows;\n    return Math.max(recommendedHeight, DefaultLogWidgetHeight);\n  }\n}\n"]}