UNPKG

cdk-monitoring-constructs

Version:

[![NPM version](https://badge.fury.io/js/cdk-monitoring-constructs.svg)](https://badge.fury.io/js/cdk-monitoring-constructs) [![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.github.cdklabs/cdkmonitoringconstructs/badge.svg)](https://m

122 lines 18.4 kB
"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 @message 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"]}