@rio-cloud/cdk-v2-constructs 6.0.0 → 6.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -107,4 +107,4 @@ const mapProperties = (scope, props) => {
107
107
  },
108
108
  };
109
109
  };
110
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafka-topic.js","sourceRoot":"","sources":["../../src/kafka/kafka-topic.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, CfnResource, Annotations } from 'aws-cdk-lib';\nimport * as cdk from 'aws-cdk-lib';\nimport { Construct, IConstruct } from 'constructs';\nimport { KafkaEventSpec, KafkaEventSpecSource } from './kafka-event-spec';\nimport { RioLandingZone } from '../rio-landing-zone';\n\n/**\n * Wrapper for all meta data of a topic.\n */\nexport interface KafkaTopicMetaData {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n}\n\n/**\n * Wrapper for all meta data of a v4 topic spec.\n */\nexport interface KafkaTopicMetaDataV4 {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n\n  /**\n   * Indicates if a topic is meant for component internal usage only or if other services could\n   * use it too.\n   */\n  readonly audience: TopicAudience;\n\n  /**\n   * Indicates if the producer will restore the data in case of a disaster or not.\n   */\n  readonly recoveryPolicy: TopicRecoveryPolicy;\n}\n\n/**\n * Read and write permissions for the topic.\n *\n * strings are matched against the CNAME of the certificate of the Kafka clients.\n */\nexport interface KafkaAclStatement {\n  /**\n  * List of clients that should get write permissions\n  */\n  readonly write: string[];\n\n  /**\n   * List of clients that should get read permissions\n   */\n  readonly read: string[];\n\n  /**\n   * List of clients that should get delete permissions.\n   *\n   * Attention: Only use Deletion policy if you know what you are doing!\n   * This is only necessary for KStream and allows the application to delete messages and the topic.<br>\n   * If you just want to \"delete\" a message on a log compacted topic, you should not set this permission and send a tombstone message instead.\n   */\n  readonly delete?: string[];\n}\n\n/**\n * Wrapper for all log compaction related properties.\n */\nexport interface LogCompactionProperties {\n  /**\n   * The amount of time to retain delete tombstone markers for log compacted topics;\n   * The soft limit is 10 days but can be increased upon requests.\n   * @defaultValue 1 day\n   */\n  readonly deleteRetention?: Duration;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicProps {\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @defaultValue 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @defaultValue 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @defaultValue false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaData;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @defaultValue undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicV4Props {\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @defaultValue 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @defaultValue 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @defaultValue false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaDataV4;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @defaultValue undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * The Topics target audience\n */\nexport type TopicAudience = 'component-internal' | 'company-internal';\n\n/**\n * The Topics recovery policy guaranteed by the producer\n */\nexport type TopicRecoveryPolicy = 'none' | 'recoverable';\n\n/**\n * Construct to create a kafka topic.\n *\n * @deprecated use {@link KafkaTopicV4} instead\n */\nexport class KafkaTopic extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicProps) {\n    super(scope, id);\n\n\n    Annotations.of(this).addDeprecation('KafkaTopic', \"Use 'KafkaTopicV4'!\");\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ...mapProperties(this, props),\n        Version: '3',\n      },\n    });\n\n    this.node.addValidation(validator(props));\n  }\n}\n\n/**\n * Construct to create a kafka topic.\n */\nexport class KafkaTopicV4 extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicV4Props) {\n    super(scope, id);\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ...mapProperties(this, props),\n        Version: '4',\n        Audience: props.metadata.audience,\n        RecoveryPolicy: props.metadata.recoveryPolicy,\n      },\n    });\n\n    this.node.addValidation(validator(props));\n  }\n}\n\nconst validator = (props: KafkaTopicProps) => {\n  return {\n    validate: (): string[] => {\n      const result = [];\n\n      if (!props.name.match(/^[A-Za-z0-9.-]+$/)) {\n        result.push('Invalid [name]: expecting topic name to match the reg exp `[A-Za-z0-9.-]+`.');\n      }\n\n      if (!Number.isInteger(props.numberOfPartitions) || props.numberOfPartitions < 1) {\n        result.push(`Invalid [numberOfPartitions]: expecting number of partitions to be a positive integer but got ${props.numberOfPartitions}.`);\n      }\n\n      if (props.replicationFactor !== undefined &&\n          (!Number.isInteger(props.replicationFactor) || props.replicationFactor < 1 || props.replicationFactor > 3)) {\n        result.push('Invalid [replicationFactor]: expecting an integer between 1 and 3.');\n      }\n\n      if (props.metadata.events.length === 0) {\n        result.push('Invalid [metadata]: expecting [events] to have at least on element.');\n      }\n\n      if (!props.isLogCompacted && props.logCompactionProperties) {\n        result.push('Invalid [logCompactionProperties]: log compaction is disabled so properties must be undefined.');\n      }\n\n      return result;\n    },\n  };\n};\n\nconst mapProperties = (scope: IConstruct, props: KafkaTopicProps | KafkaTopicV4Props) => {\n  const teamEmail = RioLandingZone.getTeamEmailParameter(scope).stringValue;\n\n  const events = props.metadata.events.map(source => {\n    new KafkaEventSpec(scope, `${source.eventName.replace(/\\./g, '_')}-EventSpec`, { source });\n    return source.eventName;\n  });\n\n  const deleteRetentionDuration = props.logCompactionProperties?.deleteRetention ?? Duration.days(7);\n  if (props.acl.delete != undefined && props.acl.delete.length > 0) {\n    const annotations = cdk.Annotations.of(cdk.Stack.of(scope));\n    annotations.addWarningV2(\n      `@rio-cloud/cdk-v2-constructs/kafka/kafka-topic/${props.name}/DeletionPolicy`,\n      `\n ❌ You have set \"delete\" permission for kafka-topic ${props.name}. You probably do not need this. Please use this only if you know what you are doing.\n`,\n    );\n  }\n  return {\n    ServiceToken: 'arn:aws:sns:eu-west-1:186993757734:dp-topic-manager-sns-topic',\n    TopicName: props.name,\n    Description: props.metadata.description,\n    Events: events,\n    NumberOfPartitions: props.numberOfPartitions,\n    ReplicationFactor: props.replicationFactor ?? 3,\n    RetentionMs: (props.retention ?? Duration.days(7)).toMilliseconds(),\n    DeleteRetentionMs: props.isLogCompacted ? deleteRetentionDuration.toMilliseconds() : undefined,\n    InstantDeletionEnabled: props.instantDeletionEnabled ?? false,\n    CleanupPolicy: props.isLogCompacted ? 'compact' : 'delete',\n    TeamEmail: teamEmail,\n    ACL: {\n      Read: props.acl.read,\n      Write: props.acl.write,\n      Delete: props.acl.delete,\n    },\n  };\n};"]}
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafka-topic.js","sourceRoot":"","sources":["../../src/kafka/kafka-topic.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, CfnResource, Annotations } from 'aws-cdk-lib';\nimport * as cdk from 'aws-cdk-lib';\nimport { Construct, IConstruct } from 'constructs';\nimport { KafkaEventSpec, KafkaEventSpecSource } from './kafka-event-spec';\nimport { RioLandingZone } from '../rio-landing-zone';\n\n/**\n * Wrapper for all meta data of a topic.\n */\nexport interface KafkaTopicMetaData {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n}\n\n/**\n * Wrapper for all meta data of a v4 topic spec.\n */\nexport interface KafkaTopicMetaDataV4 {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n\n  /**\n   * Indicates if a topic is meant for component internal usage only or if other services could\n   * use it too.\n   */\n  readonly audience: TopicAudience;\n\n  /**\n   * Indicates if the producer will restore the data in case of a disaster or not.\n   */\n  readonly recoveryPolicy: TopicRecoveryPolicy;\n}\n\n/**\n * Read and write permissions for the topic.\n *\n * strings are matched against the CNAME of the certificate of the Kafka clients.\n */\nexport interface KafkaAclStatement {\n  /**\n  * List of clients that should get write permissions\n  */\n  readonly write: string[];\n\n  /**\n   * List of clients that should get read permissions\n   */\n  readonly read: string[];\n\n  /**\n   * List of clients that should get delete permissions.\n   *\n   * Attention: Only use Deletion policy if you know what you are doing!\n   * This is only necessary for KStream and allows the application to delete messages and the topic.<br>\n   * If you just want to \"delete\" a message on a log compacted topic, you should not set this permission and send a tombstone message instead.\n   */\n  readonly delete?: string[];\n}\n\n/**\n * Wrapper for all log compaction related properties.\n */\nexport interface LogCompactionProperties {\n  /**\n   * The amount of time to retain delete tombstone markers for log compacted topics;\n   * The soft limit is 10 days but can be increased upon requests.\n   * @defaultValue 1 day\n   */\n  readonly deleteRetention?: Duration;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicProps {\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @defaultValue 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @defaultValue 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @defaultValue false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaData;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @defaultValue undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicV4Props {\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @defaultValue 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @defaultValue 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @defaultValue false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaDataV4;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @defaultValue undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * The Topics target audience\n */\nexport type TopicAudience = 'component-internal' | 'company-internal';\n\n/**\n * The Topics recovery policy guaranteed by the producer\n */\nexport type TopicRecoveryPolicy = 'none' | 'recoverable';\n\n/**\n * Construct to create a kafka topic.\n *\n * @deprecated use {@link KafkaTopicV4} instead\n */\nexport class KafkaTopic extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicProps) {\n    super(scope, id);\n\n\n    Annotations.of(this).addDeprecation('KafkaTopic', \"Use 'KafkaTopicV4'!\");\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ...mapProperties(this, props),\n        Version: '3',\n      },\n    });\n\n    this.node.addValidation(validator(props));\n  }\n}\n\n/**\n * Construct to create a kafka topic.\n */\nexport class KafkaTopicV4 extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicV4Props) {\n    super(scope, id);\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ...mapProperties(this, props),\n        Version: '4',\n        Audience: props.metadata.audience,\n        RecoveryPolicy: props.metadata.recoveryPolicy,\n      },\n    });\n\n    this.node.addValidation(validator(props));\n  }\n}\n\nconst validator = (props: KafkaTopicProps) => {\n  return {\n    validate: (): string[] => {\n      const result = [];\n\n      if (!props.name.match(/^[A-Za-z0-9.-]+$/)) {\n        result.push('Invalid [name]: expecting topic name to match the reg exp `[A-Za-z0-9.-]+`.');\n      }\n\n      if (!Number.isInteger(props.numberOfPartitions) || props.numberOfPartitions < 1) {\n        result.push(`Invalid [numberOfPartitions]: expecting number of partitions to be a positive integer but got ${props.numberOfPartitions}.`);\n      }\n\n      if (props.replicationFactor !== undefined &&\n          (!Number.isInteger(props.replicationFactor) || props.replicationFactor < 1 || props.replicationFactor > 3)) {\n        result.push('Invalid [replicationFactor]: expecting an integer between 1 and 3.');\n      }\n\n      if (props.metadata.events.length === 0) {\n        result.push('Invalid [metadata]: expecting [events] to have at least on element.');\n      }\n\n      if (!props.isLogCompacted && props.logCompactionProperties) {\n        result.push('Invalid [logCompactionProperties]: log compaction is disabled so properties must be undefined.');\n      }\n\n      return result;\n    },\n  };\n};\n\nconst mapProperties = (scope: IConstruct, props: KafkaTopicProps | KafkaTopicV4Props) => {\n  const teamEmail = RioLandingZone.getTeamEmailParameter(scope).stringValue;\n\n  const events = props.metadata.events.map(source => {\n    new KafkaEventSpec(scope, `${source.eventName.replace(/\\./g, '_')}-EventSpec`, { source });\n    return source.eventName;\n  });\n\n  const deleteRetentionDuration = props.logCompactionProperties?.deleteRetention ?? Duration.days(7);\n  if (props.acl.delete != undefined && props.acl.delete.length > 0) {\n    const annotations = cdk.Annotations.of(cdk.Stack.of(scope));\n    annotations.addWarningV2(\n      `@rio-cloud/cdk-v2-constructs/kafka/kafka-topic/${props.name}/DeletionPolicy`,\n      `\n ❌ You have set \"delete\" permission for kafka-topic ${props.name}. You probably do not need this. Please use this only if you know what you are doing.\n`,\n    );\n  }\n  return {\n    ServiceToken: 'arn:aws:sns:eu-west-1:186993757734:dp-topic-manager-sns-topic',\n    TopicName: props.name,\n    Description: props.metadata.description,\n    Events: events,\n    NumberOfPartitions: props.numberOfPartitions,\n    ReplicationFactor: props.replicationFactor ?? 3,\n    RetentionMs: (props.retention ?? Duration.days(7)).toMilliseconds(),\n    DeleteRetentionMs: props.isLogCompacted ? deleteRetentionDuration.toMilliseconds() : undefined,\n    InstantDeletionEnabled: props.instantDeletionEnabled ?? false,\n    CleanupPolicy: props.isLogCompacted ? 'compact' : 'delete',\n    TeamEmail: teamEmail,\n    ACL: {\n      Read: props.acl.read,\n      Write: props.acl.write,\n      Delete: props.acl.delete,\n    },\n  };\n};"]}
@@ -102,4 +102,4 @@ function determineSecretsFilePathCommandsForCodeBuild(userDefinedSecretsFilePath
102
102
  ];
103
103
  }
104
104
  }
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rio-backup-secrets-restore-stage.js","sourceRoot":"","sources":["../../src/pipeline/rio-backup-secrets-restore-stage.ts"],"names":[],"mappings":";;;;;AACA,mCAAmC;AACnC,2CAA2C;AAC3C,mDAAmD;AAiBnD,MAAa,oBAAqB,SAAQ,SAAS,CAAC,aAAa;IAC/D;;;;;;;;;;;;;;;;;;;;;OAqBG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAgC;QACxE,KAAK,CAAC,EAAE,EAAE;YACR,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,GAAG,4BAA4B,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,eAAe,CAAC;SAChH,CAAC,CAAC;IACL,CAAC;;AA5BH,oDA6BC;;;AAED,SAAS,4BAA4B,CAAC,OAAe,EAAE,MAAc,EAAE,0BAA8C;IACnH,OAAO;QACL,QAAQ,EAAE,oBAAoB,CAAC,MAAM,EAAE,0BAA0B,CAAC;QAClE,oBAAoB,EAAE,gCAAgC,CAAC,OAAO,CAAC;KAChE,CAAC;AACJ,CAAC;AAED,MAAM,sCAAsC,GAAG,mBAAmB,CAAC;AAEnE,SAAS,oBAAoB,CAAC,MAAc,EAAE,0BAA8C;IAC1F,OAAO;QACL,wEAAwE;QACxE,wBAAwB;QACxB,4FAA4F;QAC5F,gDAAgD;QAChD,GAAG,4CAA4C,CAAC,0BAA0B,CAAC;QAC3E,gEAAgE;QAChE,kDAAkD,sCAAsC,IAAI;QAC5F,qJAAqJ,MAAM,sCAAsC;QACjM,kCAAkC;QAClC,6MAA6M;KAC9M,CAAC;AACJ,CAAC;AAED,SAAS,gCAAgC,CAAC,OAAe;IACvD,MAAM,eAAe,GAAG,cAAc,CAAC;IACvC,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;QAC3C,OAAO,EAAE,CAAC,aAAa,CAAC;QACxB,SAAS,EAAE;YACT,iBAAiB,eAAe,QAAQ;YACxC,iBAAiB,OAAO,QAAQ;SACjC;QACD,UAAU,EAAE;YACV,wBAAwB,EAAE;gBACxB,qBAAqB,EAAE,0BAA0B;aAClD;SACF;KACF,CAAC,CAAC;IACH,MAAM,wBAAwB,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;QACvD,OAAO,EAAE,CAAC,uBAAuB,CAAC;QAClC,SAAS,EAAE;YACT,4BAA4B,OAAO,iCAAiC;SACrE;KACF,CAAC,CAAC;IACH,OAAO;QACL,YAAY;QACZ,wBAAwB;KACzB,CAAC;AACJ,CAAC;AAED,SAAS,4CAA4C,CAAC,0BAA8C;IAClG,MAAM,sBAAsB,GAAG,cAAc,CAAC;IAC9C,MAAM,wBAAwB,GAAG,gCAAgC,sBAAsB,EAAE,CAAC;IAC1F,MAAM,wBAAwB,GAAG,iBAAiB,sBAAsB,EAAE,CAAC;IAC3E,IAAI,0BAA0B,EAAE;QAC9B,OAAO;YACL,yCAAyC,0BAA0B,GAAG;YACtE,UAAU,sCAAsC,KAAK,0BAA0B,GAAG;SACnF,CAAC;KACH;SAAM;QACL,OAAO;YACL,yEAAyE,CAAC,wBAAwB,EAAE,wBAAwB,CAAC,GAAG;YAChI,YAAY,wBAAwB,oBAAoB,sCAAsC,KAAK,wBAAwB,kBAAkB,sCAAsC,KAAK,wBAAwB,OAAO;SACxN,CAAC;KACH;AACH,CAAC","sourcesContent":["\nimport * as cdk from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as pipelines from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\n\nexport interface RioSecretsDeployStepProps {\n  /**\n   * Points to CodePipeline source step\n   */\n  readonly input: pipelines.CodePipelineSource;\n  /**\n   * Path to secrets file containing encrypted secrets\n   * @defaultValue Looks for file under 2 paths with the following priority:\n   * 1. ./infrastructure/src/secrets/secrets.yaml followed by path\n   * 2. ./src/secrets/secrets.yaml\n   */\n  readonly secretsFilePath?: string;\n}\n\nexport class RioSecretsDeployStep extends pipelines.CodeBuildStep {\n  /**\n   * Creates and attaches a codebuild project to the pipeline which looks for secrets.yaml under first ./infrastructure/src/secrets and then ./src/secrets.\n   * The secrets are then uploaded to SSM and Secrets Manager according to the data in secrets.yaml\n   * The Format of secrets file should be like the following example in yaml:\n   *\n   * ---\n   * SSM:\n   *\n   *         /config/application/secret: password\n   *\n   *         /config/application2/secret: password\n   *\n   * SecretsManager:\n   *\n   *         plain_text_secret: value\n   *\n   *         json_secret:\n   *\n   *             key1: value1\n   *\n   *             key2: value2\n   */\n  constructor(scope: Construct, id: string, props: RioSecretsDeployStepProps ) {\n    super(id, {\n      input: props.input,\n      ...getCommonCodeBuildProperties(cdk.Stack.of(scope).account, cdk.Stack.of(scope).region, props.secretsFilePath),\n    });\n  }\n}\n\nfunction getCommonCodeBuildProperties(account: string, region: string, userDefinedSecretsFilePath: string | undefined): Pick<pipelines.CodeBuildStepProps, 'commands' | 'rolePolicyStatements'> {\n  return {\n    commands: getCodeBuildCommands(region, userDefinedSecretsFilePath),\n    rolePolicyStatements: getCodeBuildRolePolicyStatements(account),\n  };\n}\n\nconst SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE = 'SECRETS_FILE_PATH';\n\nfunction getCodeBuildCommands(region: string, userDefinedSecretsFilePath: string | undefined): string[] {\n  return [\n    'curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add -',\n    'apt-get --quiet update',\n    'wget --quiet https://github.com/mozilla/sops/releases/download/v3.7.1/sops_3.7.1_amd64.deb',\n    'apt-get --quiet install ./sops_3.7.1_amd64.deb',\n    ...determineSecretsFilePathCommandsForCodeBuild(userDefinedSecretsFilePath),\n    'echo \"Using the following secrets file: \\${SECRETS_FILE_PATH}\"',\n    `PAYLOAD=$(sops --output-type json --decrypt \\${${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}})`,\n    `LAMBDA_ERROR=$(aws lambda invoke --function-name \"SecretsRestoreHandler\" --payload \\\"\\${PAYLOAD}\\\" --cli-binary-format raw-in-base64-out --region ${region} output | jq 'has(\"FunctionError\")')`,\n    // eslint-disable-next-line quotes\n    `if [ \"\\${LAMBDA_ERROR}\" = 'true' ]; then echo 'The \\'SecretsRestoreHandler\\' lambda failed with an error. Check the Cloudwatch logs \\'/aws/lambda/SecretsRestoreHandler\\' for more information'; exit 1; fi`,\n  ];\n}\n\nfunction getCodeBuildRolePolicyStatements(account: string): iam.PolicyStatement[] {\n  const backupAccountId = '903404386550';\n  const iamPolicyKMS = new iam.PolicyStatement({\n    actions: ['kms:Decrypt'],\n    resources: [\n      `arn:aws:kms:*:${backupAccountId}:key/*`,\n      `arn:aws:kms:*:${account}:key/*`,\n    ],\n    conditions: {\n      'ForAnyValue:StringLike': {\n        'kms:ResourceAliases': 'alias/rio-lz-backup-key*',\n      },\n    },\n  });\n  const iamPolicyLambdaExecution = new iam.PolicyStatement({\n    actions: ['lambda:InvokeFunction'],\n    resources: [\n      `arn:aws:lambda:eu-west-1:${account}:function:SecretsRestoreHandler`,\n    ],\n  });\n  return [\n    iamPolicyKMS,\n    iamPolicyLambdaExecution,\n  ];\n}\n\nfunction determineSecretsFilePathCommandsForCodeBuild(userDefinedSecretsFilePath: string | undefined): string[] {\n  const defaultSecretsFileName = 'secrets.yaml';\n  const defaultSecretsFilePaths1 = `./infrastructure/src/secrets/${defaultSecretsFileName}`;\n  const defaultSecretsFilePaths2 = `./src/secrets/${defaultSecretsFileName}`;\n  if (userDefinedSecretsFilePath) {\n    return [\n      `echo \"Secrets file explicitly set to: ${userDefinedSecretsFilePath}\"`,\n      `export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${userDefinedSecretsFilePath}'`,\n    ];\n  } else {\n    return [\n      `echo 'No explicit secrets file set, looking at the default locations: ${[defaultSecretsFilePaths1, defaultSecretsFilePaths2]}'`,\n      `if [ -f '${defaultSecretsFilePaths1}' ]; then export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${defaultSecretsFilePaths1}'; else export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${defaultSecretsFilePaths2}'; fi`,\n    ];\n  }\n}\n"]}
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rio-backup-secrets-restore-stage.js","sourceRoot":"","sources":["../../src/pipeline/rio-backup-secrets-restore-stage.ts"],"names":[],"mappings":";;;;;AACA,mCAAmC;AACnC,2CAA2C;AAC3C,mDAAmD;AAiBnD,MAAa,oBAAqB,SAAQ,SAAS,CAAC,aAAa;IAC/D;;;;;;;;;;;;;;;;;;;;;OAqBG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAgC;QACxE,KAAK,CAAC,EAAE,EAAE;YACR,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,GAAG,4BAA4B,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,eAAe,CAAC;SAChH,CAAC,CAAC;IACL,CAAC;;AA5BH,oDA6BC;;;AAED,SAAS,4BAA4B,CAAC,OAAe,EAAE,MAAc,EAAE,0BAA8C;IACnH,OAAO;QACL,QAAQ,EAAE,oBAAoB,CAAC,MAAM,EAAE,0BAA0B,CAAC;QAClE,oBAAoB,EAAE,gCAAgC,CAAC,OAAO,CAAC;KAChE,CAAC;AACJ,CAAC;AAED,MAAM,sCAAsC,GAAG,mBAAmB,CAAC;AAEnE,SAAS,oBAAoB,CAAC,MAAc,EAAE,0BAA8C;IAC1F,OAAO;QACL,wEAAwE;QACxE,wBAAwB;QACxB,4FAA4F;QAC5F,gDAAgD;QAChD,GAAG,4CAA4C,CAAC,0BAA0B,CAAC;QAC3E,gEAAgE;QAChE,kDAAkD,sCAAsC,IAAI;QAC5F,qJAAqJ,MAAM,sCAAsC;QACjM,kCAAkC;QAClC,6MAA6M;KAC9M,CAAC;AACJ,CAAC;AAED,SAAS,gCAAgC,CAAC,OAAe;IACvD,MAAM,eAAe,GAAG,cAAc,CAAC;IACvC,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;QAC3C,OAAO,EAAE,CAAC,aAAa,CAAC;QACxB,SAAS,EAAE;YACT,iBAAiB,eAAe,QAAQ;YACxC,iBAAiB,OAAO,QAAQ;SACjC;QACD,UAAU,EAAE;YACV,wBAAwB,EAAE;gBACxB,qBAAqB,EAAE,0BAA0B;aAClD;SACF;KACF,CAAC,CAAC;IACH,MAAM,wBAAwB,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;QACvD,OAAO,EAAE,CAAC,uBAAuB,CAAC;QAClC,SAAS,EAAE;YACT,4BAA4B,OAAO,iCAAiC;SACrE;KACF,CAAC,CAAC;IACH,OAAO;QACL,YAAY;QACZ,wBAAwB;KACzB,CAAC;AACJ,CAAC;AAED,SAAS,4CAA4C,CAAC,0BAA8C;IAClG,MAAM,sBAAsB,GAAG,cAAc,CAAC;IAC9C,MAAM,wBAAwB,GAAG,gCAAgC,sBAAsB,EAAE,CAAC;IAC1F,MAAM,wBAAwB,GAAG,iBAAiB,sBAAsB,EAAE,CAAC;IAC3E,IAAI,0BAA0B,EAAE,CAAC;QAC/B,OAAO;YACL,yCAAyC,0BAA0B,GAAG;YACtE,UAAU,sCAAsC,KAAK,0BAA0B,GAAG;SACnF,CAAC;IACJ,CAAC;SAAM,CAAC;QACN,OAAO;YACL,yEAAyE,CAAC,wBAAwB,EAAE,wBAAwB,CAAC,GAAG;YAChI,YAAY,wBAAwB,oBAAoB,sCAAsC,KAAK,wBAAwB,kBAAkB,sCAAsC,KAAK,wBAAwB,OAAO;SACxN,CAAC;IACJ,CAAC;AACH,CAAC","sourcesContent":["\nimport * as cdk from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as pipelines from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\n\nexport interface RioSecretsDeployStepProps {\n  /**\n   * Points to CodePipeline source step\n   */\n  readonly input: pipelines.CodePipelineSource;\n  /**\n   * Path to secrets file containing encrypted secrets\n   * @defaultValue Looks for file under 2 paths with the following priority:\n   * 1. ./infrastructure/src/secrets/secrets.yaml followed by path\n   * 2. ./src/secrets/secrets.yaml\n   */\n  readonly secretsFilePath?: string;\n}\n\nexport class RioSecretsDeployStep extends pipelines.CodeBuildStep {\n  /**\n   * Creates and attaches a codebuild project to the pipeline which looks for secrets.yaml under first ./infrastructure/src/secrets and then ./src/secrets.\n   * The secrets are then uploaded to SSM and Secrets Manager according to the data in secrets.yaml\n   * The Format of secrets file should be like the following example in yaml:\n   *\n   * ---\n   * SSM:\n   *\n   *         /config/application/secret: password\n   *\n   *         /config/application2/secret: password\n   *\n   * SecretsManager:\n   *\n   *         plain_text_secret: value\n   *\n   *         json_secret:\n   *\n   *             key1: value1\n   *\n   *             key2: value2\n   */\n  constructor(scope: Construct, id: string, props: RioSecretsDeployStepProps ) {\n    super(id, {\n      input: props.input,\n      ...getCommonCodeBuildProperties(cdk.Stack.of(scope).account, cdk.Stack.of(scope).region, props.secretsFilePath),\n    });\n  }\n}\n\nfunction getCommonCodeBuildProperties(account: string, region: string, userDefinedSecretsFilePath: string | undefined): Pick<pipelines.CodeBuildStepProps, 'commands' | 'rolePolicyStatements'> {\n  return {\n    commands: getCodeBuildCommands(region, userDefinedSecretsFilePath),\n    rolePolicyStatements: getCodeBuildRolePolicyStatements(account),\n  };\n}\n\nconst SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE = 'SECRETS_FILE_PATH';\n\nfunction getCodeBuildCommands(region: string, userDefinedSecretsFilePath: string | undefined): string[] {\n  return [\n    'curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add -',\n    'apt-get --quiet update',\n    'wget --quiet https://github.com/mozilla/sops/releases/download/v3.7.1/sops_3.7.1_amd64.deb',\n    'apt-get --quiet install ./sops_3.7.1_amd64.deb',\n    ...determineSecretsFilePathCommandsForCodeBuild(userDefinedSecretsFilePath),\n    'echo \"Using the following secrets file: \\${SECRETS_FILE_PATH}\"',\n    `PAYLOAD=$(sops --output-type json --decrypt \\${${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}})`,\n    `LAMBDA_ERROR=$(aws lambda invoke --function-name \"SecretsRestoreHandler\" --payload \\\"\\${PAYLOAD}\\\" --cli-binary-format raw-in-base64-out --region ${region} output | jq 'has(\"FunctionError\")')`,\n    // eslint-disable-next-line quotes\n    `if [ \"\\${LAMBDA_ERROR}\" = 'true' ]; then echo 'The \\'SecretsRestoreHandler\\' lambda failed with an error. Check the Cloudwatch logs \\'/aws/lambda/SecretsRestoreHandler\\' for more information'; exit 1; fi`,\n  ];\n}\n\nfunction getCodeBuildRolePolicyStatements(account: string): iam.PolicyStatement[] {\n  const backupAccountId = '903404386550';\n  const iamPolicyKMS = new iam.PolicyStatement({\n    actions: ['kms:Decrypt'],\n    resources: [\n      `arn:aws:kms:*:${backupAccountId}:key/*`,\n      `arn:aws:kms:*:${account}:key/*`,\n    ],\n    conditions: {\n      'ForAnyValue:StringLike': {\n        'kms:ResourceAliases': 'alias/rio-lz-backup-key*',\n      },\n    },\n  });\n  const iamPolicyLambdaExecution = new iam.PolicyStatement({\n    actions: ['lambda:InvokeFunction'],\n    resources: [\n      `arn:aws:lambda:eu-west-1:${account}:function:SecretsRestoreHandler`,\n    ],\n  });\n  return [\n    iamPolicyKMS,\n    iamPolicyLambdaExecution,\n  ];\n}\n\nfunction determineSecretsFilePathCommandsForCodeBuild(userDefinedSecretsFilePath: string | undefined): string[] {\n  const defaultSecretsFileName = 'secrets.yaml';\n  const defaultSecretsFilePaths1 = `./infrastructure/src/secrets/${defaultSecretsFileName}`;\n  const defaultSecretsFilePaths2 = `./src/secrets/${defaultSecretsFileName}`;\n  if (userDefinedSecretsFilePath) {\n    return [\n      `echo \"Secrets file explicitly set to: ${userDefinedSecretsFilePath}\"`,\n      `export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${userDefinedSecretsFilePath}'`,\n    ];\n  } else {\n    return [\n      `echo 'No explicit secrets file set, looking at the default locations: ${[defaultSecretsFilePaths1, defaultSecretsFilePaths2]}'`,\n      `if [ -f '${defaultSecretsFilePaths1}' ]; then export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${defaultSecretsFilePaths1}'; else export ${SECRETS_FILE_PATH_ENVIRONMENT_VARIABLE}='${defaultSecretsFilePaths2}'; fi`,\n    ];\n  }\n}\n"]}
package/lib/toggle.js CHANGED
@@ -41,4 +41,4 @@ function isToggleEnabled(context, toggleName, fallback = false) {
41
41
  return fallback;
42
42
  }
43
43
  exports.isToggleEnabled = isToggleEnabled;
44
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9nZ2xlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3RvZ2dsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwyQ0FBdUM7QUFFdkM7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWtCRztBQUNILFNBQWdCLGVBQWUsQ0FBQyxPQUFrQixFQUFFLFVBQWtCLEVBQUUsV0FBb0IsS0FBSztJQUUvRixJQUFJLFlBQVksQ0FBQztJQUNqQixJQUFJLE9BQU8sWUFBWSxzQkFBUyxFQUFFO1FBQ2hDLFlBQVksR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxVQUFVLENBQUMsQ0FBQztLQUN2RDtJQUVELElBQUksT0FBTyxZQUFZLEtBQUssU0FBUyxFQUFFO1FBQ3JDLE9BQU8sWUFBWSxDQUFDO0tBQ3JCO0lBQ0QsSUFBSSxPQUFPLFlBQVksS0FBSyxRQUFRLEVBQUU7UUFDcEMsT0FBTyxZQUFZLEtBQUssTUFBTSxDQUFDO0tBQ2hDO0lBQ0QsSUFBSSxPQUFPLFlBQVksS0FBSyxTQUFTLEVBQUU7UUFDckMsT0FBTyxZQUFZLENBQUM7S0FDckI7SUFDRCxJQUFJLE9BQU8sWUFBWSxLQUFLLFFBQVEsRUFBRTtRQUNwQyxPQUFPLFlBQVksS0FBSyxNQUFNLENBQUM7S0FDaEM7SUFFRCxPQUFPLFFBQVEsQ0FBQztBQUNsQixDQUFDO0FBckJELDBDQXFCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKiogQ2hlY2tzIGlmIHRoZSBnaXZlbiBmZWF0dXJlIGlzIGVuYWJsZWQgb3Igbm90LlxuICpcbiAqIFRoZSB0b2dnbGUgY29uZmlndXJhdGlvbiBpcyBzdG9yZWQgaW4gY29udGV4dCBzZWN0aW9uIG9mIGpzb24gZmlsZS5cbiAqIEl0IHJldHVybnMgdHJ1ZSBpZiB0aGUgdmFsdWUgaXMgJ3RydWUnLiBBbGwgb3RoZXIgdmFsdWVzIGFyZSBpbnRlcnByZXRlZCBhcyAnZmFsc2UnLlxuICogSWYgbm8gdG9nZ2xlIGlzIGZvdW5kLCBpdCByZXR1cm5zIHRoZSBnaXZlbiBmYWxsYmFjayAoZGVmYXVsdCBpcyBmYWxzZSk7XG4gKlxuICogVGhlIGtleSBvZiBqc29uIGNvbnRleHQgc2hvdWxkIGluY2x1ZGUgdGhlIHRpY2tldCBudW1iZXIuXG4gKlxuICogRXhhbXBsZSBqc29uXG4gKiB7XG4gKiAgICBjb250ZXh0OiB7XG4gKiAgICAgIFwicmlvY2xhaWQtMTIzNC1hd2Vzb21lRmVhdHVyZVRvZ2dsZVwiOiBcInRydWVcIlxuICogICAgfVxuICogfVxuICpcbiAqIEBwYXJhbSBjb250ZXh0IEVpdGhlciB0aGUgcHJvamVuIHByb2plY3Qgb3IgYSBjZGsgY29uc3RydWN0LiBVc2VkIHRvIGxvb2sgdXAgdGhlIGpzb24gY29uZmlnLlxuICogQHBhcmFtIHRvZ2dsZU5hbWUgTmFtZSBvZiB0aGUgZmVhdHVyZSB0b2dnbGUuIChleGFtcGxlOiByaW9jbGFpZC0xMjM0LWF3ZXNvbWVGZWF0dXJlVG9nZ2xlKVxuICogQHBhcmFtIGZhbGxiYWNrIERlZmF1bHQgcmV0dXJuIHZhbHVlIGlmIG5vIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSB0b2dnbGUgZXhpc3QuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1RvZ2dsZUVuYWJsZWQoY29udGV4dDogQ29uc3RydWN0LCB0b2dnbGVOYW1lOiBzdHJpbmcsIGZhbGxiYWNrOiBib29sZWFuID0gZmFsc2UpOiBib29sZWFuIHtcblxuICBsZXQgY29udGV4dFZhbHVlO1xuICBpZiAoY29udGV4dCBpbnN0YW5jZW9mIENvbnN0cnVjdCkge1xuICAgIGNvbnRleHRWYWx1ZSA9IGNvbnRleHQubm9kZS50cnlHZXRDb250ZXh0KHRvZ2dsZU5hbWUpO1xuICB9XG5cbiAgaWYgKHR5cGVvZiBjb250ZXh0VmFsdWUgPT09ICdib29sZWFuJykge1xuICAgIHJldHVybiBjb250ZXh0VmFsdWU7XG4gIH1cbiAgaWYgKHR5cGVvZiBjb250ZXh0VmFsdWUgPT09ICdzdHJpbmcnKSB7XG4gICAgcmV0dXJuIGNvbnRleHRWYWx1ZSA9PT0gJ3RydWUnO1xuICB9XG4gIGlmICh0eXBlb2YgY29udGV4dFZhbHVlID09PSAnYm9vbGVhbicpIHtcbiAgICByZXR1cm4gY29udGV4dFZhbHVlO1xuICB9XG4gIGlmICh0eXBlb2YgY29udGV4dFZhbHVlID09PSAnc3RyaW5nJykge1xuICAgIHJldHVybiBjb250ZXh0VmFsdWUgPT09ICd0cnVlJztcbiAgfVxuXG4gIHJldHVybiBmYWxsYmFjaztcbn0iXX0=
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9nZ2xlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3RvZ2dsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwyQ0FBdUM7QUFFdkM7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWtCRztBQUNILFNBQWdCLGVBQWUsQ0FBQyxPQUFrQixFQUFFLFVBQWtCLEVBQUUsV0FBb0IsS0FBSztJQUUvRixJQUFJLFlBQVksQ0FBQztJQUNqQixJQUFJLE9BQU8sWUFBWSxzQkFBUyxFQUFFLENBQUM7UUFDakMsWUFBWSxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0lBQ3hELENBQUM7SUFFRCxJQUFJLE9BQU8sWUFBWSxLQUFLLFNBQVMsRUFBRSxDQUFDO1FBQ3RDLE9BQU8sWUFBWSxDQUFDO0lBQ3RCLENBQUM7SUFDRCxJQUFJLE9BQU8sWUFBWSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3JDLE9BQU8sWUFBWSxLQUFLLE1BQU0sQ0FBQztJQUNqQyxDQUFDO0lBQ0QsSUFBSSxPQUFPLFlBQVksS0FBSyxTQUFTLEVBQUUsQ0FBQztRQUN0QyxPQUFPLFlBQVksQ0FBQztJQUN0QixDQUFDO0lBQ0QsSUFBSSxPQUFPLFlBQVksS0FBSyxRQUFRLEVBQUUsQ0FBQztRQUNyQyxPQUFPLFlBQVksS0FBSyxNQUFNLENBQUM7SUFDakMsQ0FBQztJQUVELE9BQU8sUUFBUSxDQUFDO0FBQ2xCLENBQUM7QUFyQkQsMENBcUJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbi8qKiBDaGVja3MgaWYgdGhlIGdpdmVuIGZlYXR1cmUgaXMgZW5hYmxlZCBvciBub3QuXG4gKlxuICogVGhlIHRvZ2dsZSBjb25maWd1cmF0aW9uIGlzIHN0b3JlZCBpbiBjb250ZXh0IHNlY3Rpb24gb2YganNvbiBmaWxlLlxuICogSXQgcmV0dXJucyB0cnVlIGlmIHRoZSB2YWx1ZSBpcyAndHJ1ZScuIEFsbCBvdGhlciB2YWx1ZXMgYXJlIGludGVycHJldGVkIGFzICdmYWxzZScuXG4gKiBJZiBubyB0b2dnbGUgaXMgZm91bmQsIGl0IHJldHVybnMgdGhlIGdpdmVuIGZhbGxiYWNrIChkZWZhdWx0IGlzIGZhbHNlKTtcbiAqXG4gKiBUaGUga2V5IG9mIGpzb24gY29udGV4dCBzaG91bGQgaW5jbHVkZSB0aGUgdGlja2V0IG51bWJlci5cbiAqXG4gKiBFeGFtcGxlIGpzb25cbiAqIHtcbiAqICAgIGNvbnRleHQ6IHtcbiAqICAgICAgXCJyaW9jbGFpZC0xMjM0LWF3ZXNvbWVGZWF0dXJlVG9nZ2xlXCI6IFwidHJ1ZVwiXG4gKiAgICB9XG4gKiB9XG4gKlxuICogQHBhcmFtIGNvbnRleHQgRWl0aGVyIHRoZSBwcm9qZW4gcHJvamVjdCBvciBhIGNkayBjb25zdHJ1Y3QuIFVzZWQgdG8gbG9vayB1cCB0aGUganNvbiBjb25maWcuXG4gKiBAcGFyYW0gdG9nZ2xlTmFtZSBOYW1lIG9mIHRoZSBmZWF0dXJlIHRvZ2dsZS4gKGV4YW1wbGU6IHJpb2NsYWlkLTEyMzQtYXdlc29tZUZlYXR1cmVUb2dnbGUpXG4gKiBAcGFyYW0gZmFsbGJhY2sgRGVmYXVsdCByZXR1cm4gdmFsdWUgaWYgbm8gY29uZmlndXJhdGlvbiBmb3IgdGhlIHRvZ2dsZSBleGlzdC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzVG9nZ2xlRW5hYmxlZChjb250ZXh0OiBDb25zdHJ1Y3QsIHRvZ2dsZU5hbWU6IHN0cmluZywgZmFsbGJhY2s6IGJvb2xlYW4gPSBmYWxzZSk6IGJvb2xlYW4ge1xuXG4gIGxldCBjb250ZXh0VmFsdWU7XG4gIGlmIChjb250ZXh0IGluc3RhbmNlb2YgQ29uc3RydWN0KSB7XG4gICAgY29udGV4dFZhbHVlID0gY29udGV4dC5ub2RlLnRyeUdldENvbnRleHQodG9nZ2xlTmFtZSk7XG4gIH1cblxuICBpZiAodHlwZW9mIGNvbnRleHRWYWx1ZSA9PT0gJ2Jvb2xlYW4nKSB7XG4gICAgcmV0dXJuIGNvbnRleHRWYWx1ZTtcbiAgfVxuICBpZiAodHlwZW9mIGNvbnRleHRWYWx1ZSA9PT0gJ3N0cmluZycpIHtcbiAgICByZXR1cm4gY29udGV4dFZhbHVlID09PSAndHJ1ZSc7XG4gIH1cbiAgaWYgKHR5cGVvZiBjb250ZXh0VmFsdWUgPT09ICdib29sZWFuJykge1xuICAgIHJldHVybiBjb250ZXh0VmFsdWU7XG4gIH1cbiAgaWYgKHR5cGVvZiBjb250ZXh0VmFsdWUgPT09ICdzdHJpbmcnKSB7XG4gICAgcmV0dXJuIGNvbnRleHRWYWx1ZSA9PT0gJ3RydWUnO1xuICB9XG5cbiAgcmV0dXJuIGZhbGxiYWNrO1xufSJdfQ==
@@ -59,4 +59,4 @@ class WatchfulAspect {
59
59
  exports.WatchfulAspect = WatchfulAspect;
60
60
  _a = JSII_RTTI_SYMBOL_1;
61
61
  WatchfulAspect[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.WatchfulAspect", version: "0.0.0" };
62
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNwZWN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3dhdGNoZnVsL2FzcGVjdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUFtRDtBQUNuRCx5REFBeUQ7QUFDekQsK0NBQStDO0FBQy9DLHFEQUFxRDtBQUNyRCwyQ0FBMkM7QUFDM0MsZ0VBQWdFO0FBQ2hFLGlEQUFpRDtBQUNqRCwyQ0FBMkM7QUFJM0M7O0dBRUc7QUFDSCxNQUFhLGNBQWM7SUFDekI7Ozs7T0FJRztJQUNILFlBQTZCLFFBQTBCO1FBQTFCLGFBQVEsR0FBUixRQUFRLENBQWtCO0lBQUcsQ0FBQztJQUVwRCxLQUFLLENBQUMsSUFBZ0I7UUFDM0IsSUFBSSxJQUFJLFlBQVksTUFBTSxDQUFDLFFBQVEsRUFBRTtZQUNuQyxNQUFNLFdBQVcsR0FBd0IsSUFBSSxDQUFDLElBQUksQ0FBQyxZQUFtQyxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQyxRQUFRLENBQUM7WUFDakgsSUFBSSxXQUFXLElBQUksT0FBTyxFQUFFO2dCQUMxQix5QkFBVyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxPQUFPLENBQUMsbUNBQW1DLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQzthQUNqRjtpQkFBTTtnQkFDTCxJQUFJLENBQUMsUUFBUSxDQUFDLG1CQUFtQixDQUFDLElBQUksQ0FBQyxDQUFDO2FBQ3pDO1NBQ0Y7UUFDRCxJQUFJLElBQUksWUFBWSxHQUFHLENBQUMsV0FBVyxFQUFFO1lBQ25DLElBQUksQ0FBQyxRQUFRLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQ3JDO1FBQ0QsSUFBSSxJQUFJLFlBQVksS0FBSyxDQUFDLHVCQUF1QixFQUFFO1lBQ2pELElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQzlCO1FBQ0QsSUFBSSxJQUFJLFlBQVksS0FBSyxDQUFDLHNCQUFzQixFQUFFO1lBQ2hELElBQUksQ0FBQyxRQUFRLENBQUMsMkJBQTJCLENBQUMsSUFBSSxDQUFDLENBQUM7U0FDakQ7UUFDRCxJQUFJLElBQUksWUFBWSxLQUFLLENBQUMsZUFBZSxFQUFFO1lBQ3pDLElBQUksQ0FBQyxRQUFRLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQ2hDO1FBQ0QsSUFBSSxJQUFJLFlBQVksR0FBRyxDQUFDLGVBQWUsRUFBRTtZQUN2QyxJQUFJLENBQUMsUUFBUSxDQUFDLG9CQUFvQixDQUFDLElBQUksQ0FBQyxDQUFDO1NBQzFDO1FBQ0QsSUFBSSxJQUFJLFlBQVksVUFBVSxDQUFDLFlBQVksRUFBRTtZQUMzQyxJQUFJLENBQUMsUUFBUSxDQUFDLDJCQUEyQixDQUFDLElBQUksQ0FBQyxDQUFDO1NBQ2pEO1FBQ0QsSUFBSSxJQUFJLFlBQVksUUFBUSxDQUFDLEtBQUssRUFBRTtZQUNsQyxJQUFJLENBQUMsUUFBUSxDQUFDLGtCQUFrQixDQUFDLElBQUksQ0FBQyxDQUFDO1NBQ3hDO0lBQ0gsQ0FBQzs7QUF0Q0gsd0NBdUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQW5ub3RhdGlvbnMsIElBc3BlY3QgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgKiBhcyBjbG91ZGZyb250IGZyb20gJ2F3cy1jZGstbGliL2F3cy1jbG91ZGZyb250JztcbmltcG9ydCAqIGFzIGRvY2RiIGZyb20gJ2F3cy1jZGstbGliL2F3cy1kb2NkYic7XG5pbXBvcnQgKiBhcyBkeW5hbW9kYiBmcm9tICdhd3MtY2RrLWxpYi9hd3MtZHluYW1vZGInO1xuaW1wb3J0ICogYXMgZWNzIGZyb20gJ2F3cy1jZGstbGliL2F3cy1lY3MnO1xuaW1wb3J0ICogYXMgZWxidjIgZnJvbSAnYXdzLWNkay1saWIvYXdzLWVsYXN0aWNsb2FkYmFsYW5jaW5ndjInO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0ICogYXMgcmRzIGZyb20gJ2F3cy1jZGstbGliL2F3cy1yZHMnO1xuaW1wb3J0IHsgSUNvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgQWJzdHJhY3RXYXRjaGZ1bCB9IGZyb20gJy4vd2F0Y2hmdWwnO1xuXG4vKipcbiAqIEEgQ0RLIGFzcGVjdCB0aGF0IGNhbiBhdXRvbWF0aWNhbGx5IHdhdGNoIGFsbCByZXNvdXJjZXMgd2l0aGluIGEgc2NvcGUuXG4gKi9cbmV4cG9ydCBjbGFzcyBXYXRjaGZ1bEFzcGVjdCBpbXBsZW1lbnRzIElBc3BlY3Qge1xuICAvKipcbiAgICogRGVmaW5lcyBhIHdhdGNoZnVsIGFzcGVjdFxuICAgKlxuICAgKiBAcGFyYW0gd2F0Y2hmdWwgLSBUaGUgd2F0Y2hmdWwgdG8gYWRkIHRob3NlIHJlc291cmNlcyB0b1xuICAgKi9cbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSB3YXRjaGZ1bDogQWJzdHJhY3RXYXRjaGZ1bCkge31cblxuICBwdWJsaWMgdmlzaXQobm9kZTogSUNvbnN0cnVjdCk6IHZvaWQge1xuICAgIGlmIChub2RlIGluc3RhbmNlb2YgbGFtYmRhLkZ1bmN0aW9uKSB7XG4gICAgICBjb25zdCB3YXRjaGZ1bFRhZzogc3RyaW5nIHwgdW5kZWZpbmVkID0gKG5vZGUubm9kZS5kZWZhdWx0Q2hpbGQgYXMgbGFtYmRhLkNmbkZ1bmN0aW9uKS50YWdzLnRhZ1ZhbHVlcygpLldhdGNoZnVsO1xuICAgICAgaWYgKHdhdGNoZnVsVGFnID09ICdGYWxzZScpIHtcbiAgICAgICAgQW5ub3RhdGlvbnMub2Yobm9kZSkuYWRkSW5mbyhgV2F0Y2hmdWwgd2lsbCBpZ25vcmUgdGhlIGxhbWJkYSAke25vZGUubm9kZS5pZH1gKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hMYW1iZGFGdW5jdGlvbihub2RlKTtcbiAgICAgIH1cbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiBlY3MuQmFzZVNlcnZpY2UpIHtcbiAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hFY3NTZXJ2aWNlKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGVsYnYyLkFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoQWxiKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGVsYnYyLkFwcGxpY2F0aW9uVGFyZ2V0R3JvdXApIHtcbiAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hBcHBsaWNhdGlvblRhcmdldEdyb3VwKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGRvY2RiLkRhdGFiYXNlQ2x1c3Rlcikge1xuICAgICAgdGhpcy53YXRjaGZ1bC53YXRjaERvY0RiKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIHJkcy5EYXRhYmFzZUNsdXN0ZXIpIHtcbiAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hEYXRhYmFzZUNsdXN0ZXIobm9kZSk7XG4gICAgfVxuICAgIGlmIChub2RlIGluc3RhbmNlb2YgY2xvdWRmcm9udC5EaXN0cmlidXRpb24pIHtcbiAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hDbG91ZGZyb250RGlzdHJpYnV0aW9uKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGR5bmFtb2RiLlRhYmxlKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoRHluYW1vRGJUYWJsZShub2RlKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNwZWN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3dhdGNoZnVsL2FzcGVjdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUFtRDtBQUNuRCx5REFBeUQ7QUFDekQsK0NBQStDO0FBQy9DLHFEQUFxRDtBQUNyRCwyQ0FBMkM7QUFDM0MsZ0VBQWdFO0FBQ2hFLGlEQUFpRDtBQUNqRCwyQ0FBMkM7QUFJM0M7O0dBRUc7QUFDSCxNQUFhLGNBQWM7SUFDekI7Ozs7T0FJRztJQUNILFlBQTZCLFFBQTBCO1FBQTFCLGFBQVEsR0FBUixRQUFRLENBQWtCO0lBQUcsQ0FBQztJQUVwRCxLQUFLLENBQUMsSUFBZ0I7UUFDM0IsSUFBSSxJQUFJLFlBQVksTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO1lBQ3BDLE1BQU0sV0FBVyxHQUF3QixJQUFJLENBQUMsSUFBSSxDQUFDLFlBQW1DLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDLFFBQVEsQ0FBQztZQUNqSCxJQUFJLFdBQVcsSUFBSSxPQUFPLEVBQUUsQ0FBQztnQkFDM0IseUJBQVcsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLENBQUMsT0FBTyxDQUFDLG1DQUFtQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDbEYsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLElBQUksQ0FBQyxRQUFRLENBQUMsbUJBQW1CLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDMUMsQ0FBQztRQUNILENBQUM7UUFDRCxJQUFJLElBQUksWUFBWSxHQUFHLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDcEMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEMsQ0FBQztRQUNELElBQUksSUFBSSxZQUFZLEtBQUssQ0FBQyx1QkFBdUIsRUFBRSxDQUFDO1lBQ2xELElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQy9CLENBQUM7UUFDRCxJQUFJLElBQUksWUFBWSxLQUFLLENBQUMsc0JBQXNCLEVBQUUsQ0FBQztZQUNqRCxJQUFJLENBQUMsUUFBUSxDQUFDLDJCQUEyQixDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ2xELENBQUM7UUFDRCxJQUFJLElBQUksWUFBWSxLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7WUFDMUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDakMsQ0FBQztRQUNELElBQUksSUFBSSxZQUFZLEdBQUcsQ0FBQyxlQUFlLEVBQUUsQ0FBQztZQUN4QyxJQUFJLENBQUMsUUFBUSxDQUFDLG9CQUFvQixDQUFDLElBQUksQ0FBQyxDQUFDO1FBQzNDLENBQUM7UUFDRCxJQUFJLElBQUksWUFBWSxVQUFVLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDNUMsSUFBSSxDQUFDLFFBQVEsQ0FBQywyQkFBMkIsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNsRCxDQUFDO1FBQ0QsSUFBSSxJQUFJLFlBQVksUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ25DLElBQUksQ0FBQyxRQUFRLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDekMsQ0FBQztJQUNILENBQUM7O0FBdENILHdDQXVDQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFubm90YXRpb25zLCBJQXNwZWN0IH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0ICogYXMgY2xvdWRmcm9udCBmcm9tICdhd3MtY2RrLWxpYi9hd3MtY2xvdWRmcm9udCc7XG5pbXBvcnQgKiBhcyBkb2NkYiBmcm9tICdhd3MtY2RrLWxpYi9hd3MtZG9jZGInO1xuaW1wb3J0ICogYXMgZHluYW1vZGIgZnJvbSAnYXdzLWNkay1saWIvYXdzLWR5bmFtb2RiJztcbmltcG9ydCAqIGFzIGVjcyBmcm9tICdhd3MtY2RrLWxpYi9hd3MtZWNzJztcbmltcG9ydCAqIGFzIGVsYnYyIGZyb20gJ2F3cy1jZGstbGliL2F3cy1lbGFzdGljbG9hZGJhbGFuY2luZ3YyJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCAqIGFzIHJkcyBmcm9tICdhd3MtY2RrLWxpYi9hd3MtcmRzJztcbmltcG9ydCB7IElDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEFic3RyYWN0V2F0Y2hmdWwgfSBmcm9tICcuL3dhdGNoZnVsJztcblxuLyoqXG4gKiBBIENESyBhc3BlY3QgdGhhdCBjYW4gYXV0b21hdGljYWxseSB3YXRjaCBhbGwgcmVzb3VyY2VzIHdpdGhpbiBhIHNjb3BlLlxuICovXG5leHBvcnQgY2xhc3MgV2F0Y2hmdWxBc3BlY3QgaW1wbGVtZW50cyBJQXNwZWN0IHtcbiAgLyoqXG4gICAqIERlZmluZXMgYSB3YXRjaGZ1bCBhc3BlY3RcbiAgICpcbiAgICogQHBhcmFtIHdhdGNoZnVsIC0gVGhlIHdhdGNoZnVsIHRvIGFkZCB0aG9zZSByZXNvdXJjZXMgdG9cbiAgICovXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVhZG9ubHkgd2F0Y2hmdWw6IEFic3RyYWN0V2F0Y2hmdWwpIHt9XG5cbiAgcHVibGljIHZpc2l0KG5vZGU6IElDb25zdHJ1Y3QpOiB2b2lkIHtcbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGxhbWJkYS5GdW5jdGlvbikge1xuICAgICAgY29uc3Qgd2F0Y2hmdWxUYWc6IHN0cmluZyB8IHVuZGVmaW5lZCA9IChub2RlLm5vZGUuZGVmYXVsdENoaWxkIGFzIGxhbWJkYS5DZm5GdW5jdGlvbikudGFncy50YWdWYWx1ZXMoKS5XYXRjaGZ1bDtcbiAgICAgIGlmICh3YXRjaGZ1bFRhZyA9PSAnRmFsc2UnKSB7XG4gICAgICAgIEFubm90YXRpb25zLm9mKG5vZGUpLmFkZEluZm8oYFdhdGNoZnVsIHdpbGwgaWdub3JlIHRoZSBsYW1iZGEgJHtub2RlLm5vZGUuaWR9YCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aGlzLndhdGNoZnVsLndhdGNoTGFtYmRhRnVuY3Rpb24obm9kZSk7XG4gICAgICB9XG4gICAgfVxuICAgIGlmIChub2RlIGluc3RhbmNlb2YgZWNzLkJhc2VTZXJ2aWNlKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoRWNzU2VydmljZShub2RlKTtcbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiBlbGJ2Mi5BcHBsaWNhdGlvbkxvYWRCYWxhbmNlcikge1xuICAgICAgdGhpcy53YXRjaGZ1bC53YXRjaEFsYihub2RlKTtcbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiBlbGJ2Mi5BcHBsaWNhdGlvblRhcmdldEdyb3VwKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoQXBwbGljYXRpb25UYXJnZXRHcm91cChub2RlKTtcbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiBkb2NkYi5EYXRhYmFzZUNsdXN0ZXIpIHtcbiAgICAgIHRoaXMud2F0Y2hmdWwud2F0Y2hEb2NEYihub2RlKTtcbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiByZHMuRGF0YWJhc2VDbHVzdGVyKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoRGF0YWJhc2VDbHVzdGVyKG5vZGUpO1xuICAgIH1cbiAgICBpZiAobm9kZSBpbnN0YW5jZW9mIGNsb3VkZnJvbnQuRGlzdHJpYnV0aW9uKSB7XG4gICAgICB0aGlzLndhdGNoZnVsLndhdGNoQ2xvdWRmcm9udERpc3RyaWJ1dGlvbihub2RlKTtcbiAgICB9XG4gICAgaWYgKG5vZGUgaW5zdGFuY2VvZiBkeW5hbW9kYi5UYWJsZSkge1xuICAgICAgdGhpcy53YXRjaGZ1bC53YXRjaER5bmFtb0RiVGFibGUobm9kZSk7XG4gICAgfVxuICB9XG59XG4iXX0=
@@ -28,7 +28,7 @@ class DataDogLogAlarm extends constructs_1.Construct {
28
28
  pascalCaseProperties: true,
29
29
  resourceType: 'Custom::DataDogMonitor',
30
30
  properties: {
31
- alertTypes: ['opsgenie'],
31
+ alertTypes: ['opsgenie'], // alert Type is needed to integrate with OpsGenie and to set the autoCloseOpsGenieAlerts option
32
32
  autoCloseOpsGenieAlerts: props.autoCloseOpsGenieAlerts,
33
33
  ServiceName: props.serviceName,
34
34
  version: '1.0.0',
@@ -36,7 +36,7 @@ class DataDogLogAlarm extends constructs_1.Construct {
36
36
  type: 'log alert',
37
37
  priority: props.priority,
38
38
  query: this.alarmToDataDogQuery(props),
39
- name: `${this.node.scope?.node.id} ${id}`,
39
+ name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName
40
40
  message: `${props.alarmDescription}: {{log.link}}`,
41
41
  options: {
42
42
  notify_audit: false,
@@ -84,4 +84,4 @@ class DataDogLogAlarm extends constructs_1.Construct {
84
84
  exports.DataDogLogAlarm = DataDogLogAlarm;
85
85
  _a = JSII_RTTI_SYMBOL_1;
86
86
  DataDogLogAlarm[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.DataDogLogAlarm", version: "0.0.0" };
87
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-alarm.js","sourceRoot":"","sources":["../../src/watchful/datadog-log-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Stack, CustomResource } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface DataDogLogQueryAlarmProps extends DataDogProps, LogAlarmProps {\n}\n\nexport enum UnitOfPeriod {\n  SECONDS = 's',\n  MINUTES = 'm',\n  HOURS = 'h',\n  DAYS = 'd',\n}\n\nconst DEFAULT_UNIT_OF_PERIOD = UnitOfPeriod.SECONDS;\n\nexport enum ComparisonOperator {\n  GREATER_THAN_OR_EQUAL_TO_THRESHOLD = '>=',\n  GREATER_THAN_THRESHOLD = '>',\n  LESS_THAN_OR_EQUAL_TO_THRESHOLD = '<=',\n  LESS_THAN_THRESHOLD = '<',\n}\n\nexport interface DataDogProps {\n  readonly serviceToken: string; // points to DD monitor custom resource lambda or sns topic\n  readonly serviceName: string; // needs to match the service name used in datadog-integration\n}\n\nexport interface LogAlarmProps {\n  readonly autoCloseOpsGenieAlerts: boolean;\n  readonly period: Duration;\n  readonly unitOfPeriod?: UnitOfPeriod; // unit which will be used to create the alert. if none is provided the periods unit will default to seconds\n  readonly periodOperator: ComparisonOperator;\n  readonly statistic: 'count'; // aggregation function\n  readonly queryFilters: 'error'|'warn'; // it should either say error or warn\n  readonly alarmDescription: string;\n  readonly threshold: number;\n  readonly priority: number;\n  readonly index: string;\n  readonly lambdaCloudwatchGroupName?: string; // Should only be set in case of lambda function. The cloudwatch log group name is same the lambda function name\n}\n\nexport class DataDogLogAlarm extends Construct {\n  constructor(scope: Construct, id: string, props: DataDogLogQueryAlarmProps) {\n    super(scope, id);\n\n    new CustomResource(this, 'DDMonitor', {\n      serviceToken: props.serviceToken,\n      pascalCaseProperties: true,\n      resourceType: 'Custom::DataDogMonitor',\n      properties: {\n        alertTypes: ['opsgenie'], // alert Type is needed to integrate with OpsGenie and to set the autoCloseOpsGenieAlerts option\n        autoCloseOpsGenieAlerts: props.autoCloseOpsGenieAlerts,\n        ServiceName: props.serviceName,\n        version: '1.0.0',\n        monitor: {\n          type: 'log alert',\n          priority: props.priority,\n          query: this.alarmToDataDogQuery(props),\n          name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName\n          message: `${props.alarmDescription}: {{log.link}}`,\n          options: {\n            notify_audit: false,\n            locked: false,\n            silenced: {},\n            include_tags: true,\n            require_full_window: false,\n            notify_no_data: false,\n            escalation_message: '',\n            enable_logs_sample: true,\n            renotify_interval: 10,\n            renotify_statuses: ['alert'],\n          },\n        },\n      },\n    });\n  }\n\n  private alarmToDataDogQuery(alarm: DataDogLogQueryAlarmProps): string {\n\n    const [periodValue, periodUnit] = this.convertToDatadogPeriod(alarm.period, alarm.unitOfPeriod);\n\n    let comparisonOperator = alarm.periodOperator;\n\n    const filters = new Map([\n      ['service', alarm.serviceName],\n      ['account_id', Stack.of(this).account],\n      ['status', alarm.queryFilters],\n    ]);\n\n    const concatenatedFilters = Array.from(filters.entries())\n      .map(([filterKey, filterValue]) => `${filterKey}:${filterValue}`)\n      .join(' ');\n\n    return `logs(\"${concatenatedFilters}\").index(\"${alarm.index}\").rollup(\"${alarm.statistic}\").last(\"${periodValue}${periodUnit}\") ${comparisonOperator} ${alarm.threshold}`;\n  }\n\n  private convertToDatadogPeriod(period: Duration, unitOfPeriod?: UnitOfPeriod): [number, string] {\n    const unit = unitOfPeriod ?? DEFAULT_UNIT_OF_PERIOD;\n    const mapPeriodToDatadogTime = {\n      [UnitOfPeriod.SECONDS]: () => period.toSeconds(),\n      [UnitOfPeriod.MINUTES]: () => period.toMinutes(),\n      [UnitOfPeriod.HOURS]: () => period.toHours(),\n      [UnitOfPeriod.DAYS]: () => period.toDays(),\n    };\n    return [\n      mapPeriodToDatadogTime[unit](),\n      unit.toString(),\n    ];\n  }\n}"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-alarm.js","sourceRoot":"","sources":["../../src/watchful/datadog-log-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Stack, CustomResource } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface DataDogLogQueryAlarmProps extends DataDogProps, LogAlarmProps {\n}\n\nexport enum UnitOfPeriod {\n  SECONDS = 's',\n  MINUTES = 'm',\n  HOURS = 'h',\n  DAYS = 'd',\n}\n\nconst DEFAULT_UNIT_OF_PERIOD = UnitOfPeriod.SECONDS;\n\nexport enum ComparisonOperator {\n  GREATER_THAN_OR_EQUAL_TO_THRESHOLD = '>=',\n  GREATER_THAN_THRESHOLD = '>',\n  LESS_THAN_OR_EQUAL_TO_THRESHOLD = '<=',\n  LESS_THAN_THRESHOLD = '<',\n}\n\nexport interface DataDogProps {\n  readonly serviceToken: string; // points to DD monitor custom resource lambda or sns topic\n  readonly serviceName: string; // needs to match the service name used in datadog-integration\n}\n\nexport interface LogAlarmProps {\n  readonly autoCloseOpsGenieAlerts: boolean;\n  readonly period: Duration;\n  readonly unitOfPeriod?: UnitOfPeriod; // unit which will be used to create the alert. if none is provided the periods unit will default to seconds\n  readonly periodOperator: ComparisonOperator;\n  readonly statistic: 'count'; // aggregation function\n  readonly queryFilters: 'error'|'warn'; // it should either say error or warn\n  readonly alarmDescription: string;\n  readonly threshold: number;\n  readonly priority: number;\n  readonly index: string;\n  readonly lambdaCloudwatchGroupName?: string; // Should only be set in case of lambda function. The cloudwatch log group name is same the lambda function name\n}\n\nexport class DataDogLogAlarm extends Construct {\n  constructor(scope: Construct, id: string, props: DataDogLogQueryAlarmProps) {\n    super(scope, id);\n\n    new CustomResource(this, 'DDMonitor', {\n      serviceToken: props.serviceToken,\n      pascalCaseProperties: true,\n      resourceType: 'Custom::DataDogMonitor',\n      properties: {\n        alertTypes: ['opsgenie'], // alert Type is needed to integrate with OpsGenie and to set the autoCloseOpsGenieAlerts option\n        autoCloseOpsGenieAlerts: props.autoCloseOpsGenieAlerts,\n        ServiceName: props.serviceName,\n        version: '1.0.0',\n        monitor: {\n          type: 'log alert',\n          priority: props.priority,\n          query: this.alarmToDataDogQuery(props),\n          name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName\n          message: `${props.alarmDescription}: {{log.link}}`,\n          options: {\n            notify_audit: false,\n            locked: false,\n            silenced: {},\n            include_tags: true,\n            require_full_window: false,\n            notify_no_data: false,\n            escalation_message: '',\n            enable_logs_sample: true,\n            renotify_interval: 10,\n            renotify_statuses: ['alert'],\n          },\n        },\n      },\n    });\n  }\n\n  private alarmToDataDogQuery(alarm: DataDogLogQueryAlarmProps): string {\n\n    const [periodValue, periodUnit] = this.convertToDatadogPeriod(alarm.period, alarm.unitOfPeriod);\n\n    let comparisonOperator = alarm.periodOperator;\n\n    const filters = new Map([\n      ['service', alarm.serviceName],\n      ['account_id', Stack.of(this).account],\n      ['status', alarm.queryFilters],\n    ]);\n\n    const concatenatedFilters = Array.from(filters.entries())\n      .map(([filterKey, filterValue]) => `${filterKey}:${filterValue}`)\n      .join(' ');\n\n    return `logs(\"${concatenatedFilters}\").index(\"${alarm.index}\").rollup(\"${alarm.statistic}\").last(\"${periodValue}${periodUnit}\") ${comparisonOperator} ${alarm.threshold}`;\n  }\n\n  private convertToDatadogPeriod(period: Duration, unitOfPeriod?: UnitOfPeriod): [number, string] {\n    const unit = unitOfPeriod ?? DEFAULT_UNIT_OF_PERIOD;\n    const mapPeriodToDatadogTime = {\n      [UnitOfPeriod.SECONDS]: () => period.toSeconds(),\n      [UnitOfPeriod.MINUTES]: () => period.toMinutes(),\n      [UnitOfPeriod.HOURS]: () => period.toHours(),\n      [UnitOfPeriod.DAYS]: () => period.toDays(),\n    };\n    return [\n      mapPeriodToDatadogTime[unit](),\n      unit.toString(),\n    ];\n  }\n}"]}
@@ -26,7 +26,7 @@ class DataDogMetricAlarm extends constructs_1.Construct {
26
26
  monitor: {
27
27
  type: 'query alert',
28
28
  query: this.alarmToDataDogQuery(props),
29
- name: `${this.node.scope?.node.id} ${id}`,
29
+ name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName
30
30
  // TeamName
31
31
  message: `
32
32
  {{#is_alert}}
@@ -40,7 +40,7 @@ ${props.alarmDescription} is back to normal.
40
40
  @opsgenie-${props.dataDogOpsGenieIntegrationName}`,
41
41
  options: {
42
42
  notify_audit: false,
43
- locked: false,
43
+ locked: false, // double-check!
44
44
  timeout_h: 0,
45
45
  silenced: {},
46
46
  include_tags: false,
@@ -154,4 +154,4 @@ ${props.alarmDescription} is back to normal.
154
154
  exports.DataDogMetricAlarm = DataDogMetricAlarm;
155
155
  _a = JSII_RTTI_SYMBOL_1;
156
156
  DataDogMetricAlarm[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.DataDogMetricAlarm", version: "0.0.0" };
157
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-metric-alarm.js","sourceRoot":"","sources":["../../src/watchful/datadog-metric-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, CustomResource } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport { Construct } from 'constructs';\nimport { MetricAlarmProps } from './metric-alarm';\nimport { UpperToLower } from './upperToLower';\n\n/**\n * Interface for DataDogMonitor implementation.\n */\nexport interface DataDogMetricAlarmProps extends MetricAlarmProps {\n  /**\n     * Service token which points to the DataDog Monitor custom resource lambda or sns topic.\n     */\n  readonly serviceToken: string;\n\n  /**\n   * The service name used to name the montors (in datadog-integration).\n   */\n  readonly serviceName: string;\n\n  /**\n   * The name of DataDog's OpsGenie integration. Used to automatically create OpsGenie alarms for the created DataDog Monitor.\n   */\n  readonly dataDogOpsGenieIntegrationName: string;\n}\n\n/**\n * A wrapper for a custom resource that creates a DataDog Monitor.\n *\n * To use this construct, the DataDog integration needs to be installed.\n */\nexport class DataDogMetricAlarm extends Construct {\n  private serviceToken: string;\n  constructor(scope: Construct, id: string, props: DataDogMetricAlarmProps) {\n    super(scope, id);\n\n    this.serviceToken = UpperToLower.getOrCreate(this).provider.serviceToken;\n\n    new CustomResource(this, 'DDMonitor', {\n      serviceToken: props.serviceToken,\n      pascalCaseProperties: true,\n      resourceType: 'Custom::DataDogMonitor',\n      properties: {\n        ServiceName: props.serviceName,\n        version: '1.0.0',\n        monitor: {\n          type: 'query alert',\n          query: this.alarmToDataDogQuery(props),\n          name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName\n          // TeamName\n          message: `\n{{#is_alert}}\n${props.alarmDescription}\n{{/is_alert}} \n\n{{#is_alert_recovery}}\n${props.alarmDescription} is back to normal.\n{{/is_alert_recovery}} \n\n@opsgenie-${props.dataDogOpsGenieIntegrationName}`,\n          options: {\n            notify_audit: false,\n            locked: false, // double-check!\n            timeout_h: 0,\n            silenced: {},\n            include_tags: false,\n            new_host_delay: 300,\n            require_full_window: false,\n            notify_no_data: props.treatMissingData === cw.TreatMissingData.BREACHING,\n            renotify_interval: 0,\n            escalation_message: '',\n            evaluation_delay: 900,\n          },\n        },\n      },\n    });\n  }\n\n  /**\n     * Helper function to convert a generic alarm into a DataDog query string.\n     */\n  alarmToDataDogQuery(alarm: MetricAlarmProps): string {\n    let unit = 'not supported';\n    let totalPeriod = 0;\n    if (alarm.metric.period.toSeconds() < 60) {\n      totalPeriod = alarm.metric.period.toSeconds() * alarm.evaluationPeriods;\n      unit = 's';\n    }\n    if (alarm.metric.period.toMinutes() < 1440) {\n      totalPeriod = alarm.metric.period.toMinutes() * alarm.evaluationPeriods;\n      unit = 'm';\n    } else {\n      totalPeriod = alarm.metric.period.toDays() * alarm.evaluationPeriods;\n      unit = 'd';\n    }\n\n    // avg(last_15m):max:aws.fargate.cpu.percent{ecs_container_name:$ServiceName$} > 95\n    const namespace = alarm.metric.namespace.toLocaleLowerCase().replace('/', '.');\n\n    let metricName = alarm.metric.metricName;\n\n    // Special case for ALB / TargetGroup metrics\n    switch (metricName) {\n      case 'HTTPCode_ELB_5XX_Count':\n        metricName = 'HTTPCode_ELB_5XX';\n        break;\n      case 'HTTPCode_Target_5XX_Count':\n        metricName = 'HTTPCode_Target_5XX';\n        break;\n    }\n    if (!metricName.includes('_')) {\n      metricName = metricName.replace(/([a-z])([A-Z])/g, '$1_$2');\n    }\n    metricName = metricName.toLocaleLowerCase();\n\n    const metric = `${namespace}.${metricName}`;\n\n    // statistics must be lowerCase and probably also converted (see allowed values in cw.Metric)\n    let comparisonOperator = 'not supported';\n    switch (alarm.comparisonOperator) {\n      case cw.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '>=';\n        break;\n      case cw.ComparisonOperator.GREATER_THAN_THRESHOLD:\n        comparisonOperator = '>';\n        break;\n      case cw.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '<=';\n        break;\n      case cw.ComparisonOperator.LESS_THAN_THRESHOLD:\n        comparisonOperator = '<';\n        break;\n    }\n\n    //const timeStatistic = 'sum'; // For count it should be `sum`, for gauges maybe `avg`\n\n    let alarmStatistic = 'not supported';\n    switch (alarm.metric.statistic) {\n      case cw.Statistic.AVERAGE:\n        alarmStatistic = 'avg';\n        break;\n      case cw.Statistic.MAXIMUM:\n        alarmStatistic = 'max';\n        break;\n      case cw.Statistic.MINIMUM:\n        alarmStatistic = 'min';\n        break;\n      case cw.Statistic.SAMPLE_COUNT:\n        break;\n      case cw.Statistic.SUM:\n        alarmStatistic = 'sum';\n        break;\n      default:\n        alarmStatistic = 'avg';\n        break;\n    }\n\n    let dimensions = `account_id:${Stack.of(this).account}`;\n\n    if (alarm.metric.dimensions) {\n      for (const key in alarm.metric.dimensions) {\n        const upperToLowerCr = new CustomResource(this, `${key}UpperToLower`, {\n          resourceType: 'Custom::UpperToLower',\n          serviceToken: this.serviceToken,\n          properties: {\n            Upper: alarm.metric.dimensions[key],\n          },\n        });\n        const value = upperToLowerCr.getAtt('Lower');\n        dimensions += `,${key.toLocaleLowerCase()}:${value}`;\n      }\n    }\n\n    switch (alarm.treatMissingData) {\n      case cw.TreatMissingData.NOT_BREACHING:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}) ${comparisonOperator} ${alarm.threshold}`;\n      case cw.TreatMissingData.BREACHING:\n      case cw.TreatMissingData.IGNORE:\n      default:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}} ${comparisonOperator} ${alarm.threshold}`;\n    }\n  }\n}\n"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-metric-alarm.js","sourceRoot":"","sources":["../../src/watchful/datadog-metric-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, CustomResource } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport { Construct } from 'constructs';\nimport { MetricAlarmProps } from './metric-alarm';\nimport { UpperToLower } from './upperToLower';\n\n/**\n * Interface for DataDogMonitor implementation.\n */\nexport interface DataDogMetricAlarmProps extends MetricAlarmProps {\n  /**\n     * Service token which points to the DataDog Monitor custom resource lambda or sns topic.\n     */\n  readonly serviceToken: string;\n\n  /**\n   * The service name used to name the montors (in datadog-integration).\n   */\n  readonly serviceName: string;\n\n  /**\n   * The name of DataDog's OpsGenie integration. Used to automatically create OpsGenie alarms for the created DataDog Monitor.\n   */\n  readonly dataDogOpsGenieIntegrationName: string;\n}\n\n/**\n * A wrapper for a custom resource that creates a DataDog Monitor.\n *\n * To use this construct, the DataDog integration needs to be installed.\n */\nexport class DataDogMetricAlarm extends Construct {\n  private serviceToken: string;\n  constructor(scope: Construct, id: string, props: DataDogMetricAlarmProps) {\n    super(scope, id);\n\n    this.serviceToken = UpperToLower.getOrCreate(this).provider.serviceToken;\n\n    new CustomResource(this, 'DDMonitor', {\n      serviceToken: props.serviceToken,\n      pascalCaseProperties: true,\n      resourceType: 'Custom::DataDogMonitor',\n      properties: {\n        ServiceName: props.serviceName,\n        version: '1.0.0',\n        monitor: {\n          type: 'query alert',\n          query: this.alarmToDataDogQuery(props),\n          name: `${this.node.scope?.node.id} ${id}`, // datadog-integration custom resource prepend it with Team and ServiceName\n          // TeamName\n          message: `\n{{#is_alert}}\n${props.alarmDescription}\n{{/is_alert}} \n\n{{#is_alert_recovery}}\n${props.alarmDescription} is back to normal.\n{{/is_alert_recovery}} \n\n@opsgenie-${props.dataDogOpsGenieIntegrationName}`,\n          options: {\n            notify_audit: false,\n            locked: false, // double-check!\n            timeout_h: 0,\n            silenced: {},\n            include_tags: false,\n            new_host_delay: 300,\n            require_full_window: false,\n            notify_no_data: props.treatMissingData === cw.TreatMissingData.BREACHING,\n            renotify_interval: 0,\n            escalation_message: '',\n            evaluation_delay: 900,\n          },\n        },\n      },\n    });\n  }\n\n  /**\n     * Helper function to convert a generic alarm into a DataDog query string.\n     */\n  alarmToDataDogQuery(alarm: MetricAlarmProps): string {\n    let unit = 'not supported';\n    let totalPeriod = 0;\n    if (alarm.metric.period.toSeconds() < 60) {\n      totalPeriod = alarm.metric.period.toSeconds() * alarm.evaluationPeriods;\n      unit = 's';\n    }\n    if (alarm.metric.period.toMinutes() < 1440) {\n      totalPeriod = alarm.metric.period.toMinutes() * alarm.evaluationPeriods;\n      unit = 'm';\n    } else {\n      totalPeriod = alarm.metric.period.toDays() * alarm.evaluationPeriods;\n      unit = 'd';\n    }\n\n    // avg(last_15m):max:aws.fargate.cpu.percent{ecs_container_name:$ServiceName$} > 95\n    const namespace = alarm.metric.namespace.toLocaleLowerCase().replace('/', '.');\n\n    let metricName = alarm.metric.metricName;\n\n    // Special case for ALB / TargetGroup metrics\n    switch (metricName) {\n      case 'HTTPCode_ELB_5XX_Count':\n        metricName = 'HTTPCode_ELB_5XX';\n        break;\n      case 'HTTPCode_Target_5XX_Count':\n        metricName = 'HTTPCode_Target_5XX';\n        break;\n    }\n    if (!metricName.includes('_')) {\n      metricName = metricName.replace(/([a-z])([A-Z])/g, '$1_$2');\n    }\n    metricName = metricName.toLocaleLowerCase();\n\n    const metric = `${namespace}.${metricName}`;\n\n    // statistics must be lowerCase and probably also converted (see allowed values in cw.Metric)\n    let comparisonOperator = 'not supported';\n    switch (alarm.comparisonOperator) {\n      case cw.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '>=';\n        break;\n      case cw.ComparisonOperator.GREATER_THAN_THRESHOLD:\n        comparisonOperator = '>';\n        break;\n      case cw.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '<=';\n        break;\n      case cw.ComparisonOperator.LESS_THAN_THRESHOLD:\n        comparisonOperator = '<';\n        break;\n    }\n\n    //const timeStatistic = 'sum'; // For count it should be `sum`, for gauges maybe `avg`\n\n    let alarmStatistic = 'not supported';\n    switch (alarm.metric.statistic) {\n      case cw.Statistic.AVERAGE:\n        alarmStatistic = 'avg';\n        break;\n      case cw.Statistic.MAXIMUM:\n        alarmStatistic = 'max';\n        break;\n      case cw.Statistic.MINIMUM:\n        alarmStatistic = 'min';\n        break;\n      case cw.Statistic.SAMPLE_COUNT:\n        break;\n      case cw.Statistic.SUM:\n        alarmStatistic = 'sum';\n        break;\n      default:\n        alarmStatistic = 'avg';\n        break;\n    }\n\n    let dimensions = `account_id:${Stack.of(this).account}`;\n\n    if (alarm.metric.dimensions) {\n      for (const key in alarm.metric.dimensions) {\n        const upperToLowerCr = new CustomResource(this, `${key}UpperToLower`, {\n          resourceType: 'Custom::UpperToLower',\n          serviceToken: this.serviceToken,\n          properties: {\n            Upper: alarm.metric.dimensions[key],\n          },\n        });\n        const value = upperToLowerCr.getAtt('Lower');\n        dimensions += `,${key.toLocaleLowerCase()}:${value}`;\n      }\n    }\n\n    switch (alarm.treatMissingData) {\n      case cw.TreatMissingData.NOT_BREACHING:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}) ${comparisonOperator} ${alarm.threshold}`;\n      case cw.TreatMissingData.BREACHING:\n      case cw.TreatMissingData.IGNORE:\n      default:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}} ${comparisonOperator} ${alarm.threshold}`;\n    }\n  }\n}\n"]}
@@ -30,7 +30,7 @@ class WatchEcsService extends constructs_1.Construct {
30
30
  return {
31
31
  autoCloseOpsGenieAlerts: true,
32
32
  period: aws_cdk_lib_1.Duration.minutes(5),
33
- unitOfPeriod: __1.UnitOfPeriod.MINUTES,
33
+ unitOfPeriod: __1.UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?
34
34
  periodOperator: __1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
35
35
  statistic: 'count',
36
36
  queryFilters: 'error',
@@ -72,4 +72,4 @@ class WatchEcsService extends constructs_1.Construct {
72
72
  exports.WatchEcsService = WatchEcsService;
73
73
  _a = JSII_RTTI_SYMBOL_1;
74
74
  WatchEcsService[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.WatchEcsService", version: "0.0.0" };
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs.js","sourceRoot":"","sources":["../../src/watchful/ecs.ts"],"names":[],"mappings":";;;;;AAAA,6CAAuC;AACvC,iDAAiD;AAGjD,2CAAuC;AAEvC,0BAAoG;AAEpG,MAAM,yCAAyC,GAAG,EAAE,CAAC;AACrD,MAAM,4CAA4C,GAAG,EAAE,CAAC;AAqCxD;;;;;;;GAOG;AACH,MAAa,eAAgB,SAAQ,sBAAS;IAI5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA2B;QACnE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;QAC/B,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC;QAE7B,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,EAC5B,eAAW,CAAC,eAAe,EAC3B,IAAI,CAAC,2BAA2B,CAAC,KAAK,CAAC,8BAA8B,CAAC,CACvE,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,EAC5B,eAAW,CAAC,kBAAkB,EAC9B,IAAI,CAAC,8BAA8B,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAC7E,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,EAC/B,eAAW,CAAC,iBAAiB,EAC7B,IAAI,CAAC,qBAAqB,EAAE,CAC7B,CAAC;IACJ,CAAC;IAEO,qBAAqB;QAC3B,OAAO;YACL,uBAAuB,EAAE,IAAI;YAC7B,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC3B,YAAY,EAAE,gBAAY,CAAC,OAAO;YAClC,cAAc,EAAE,sBAAkB,CAAC,kCAAkC;YACrE,SAAS,EAAE,OAAO;YAClB,YAAY,EAAE,OAAO;YACrB,gBAAgB,EAAE,qCAAqC;YACvD,SAAS,EAAE,CAAC;YACZ,QAAQ,EAAE,CAAC;YACX,KAAK,EAAE,GAAG,EAAE,sGAAsG;SACnH,CAAC;IACJ,CAAC;IAEO,2BAA2B,CACjC,iCAAyC,yCAAyC;QAElF,MAAM,oBAAoB,GAAG,IAAI,CAAC,OAAO,CAAC,oBAAoB,EAAE,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QACxG,OAAO;YACL,MAAM,EAAE,oBAAoB;YAC5B,gBAAgB,EAAE,4DAA4D,8BAA8B,GAAG;YAC/G,kBAAkB,EAAE,EAAE,CAAC,kBAAkB,CAAC,sBAAsB;YAChE,SAAS,EAAE,8BAA8B;YACzC,iBAAiB,EAAE,CAAC;YACpB,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;IAEO,8BAA8B,CACpC,oCAA4C,4CAA4C;QAExF,MAAM,uBAAuB,GAAG,IAAI,CAAC,OAAO;aACzC,uBAAuB,EAAE;aACzB,IAAI,CAAC,EAAE,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAE1C,MAAM,YAAY,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC;YACjC,GAAG,uBAAsC;YACzC,UAAU,EAAE,GAAG,uBAAuB,CAAC,UAAU,UAAU;SAC5D,CAAC,CAAC;QACH,OAAO;YACL,MAAM,EAAE,YAAY;YACpB,gBAAgB,EAAE,+DAA+D,iCAAiC,GAAG;YACrH,kBAAkB,EAAE,EAAE,CAAC,kBAAkB,CAAC,sBAAsB;YAChE,SAAS,EAAE,iCAAiC;YAC5C,iBAAiB,EAAE,CAAC;YACpB,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;;AAxEH,0CA0EC","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport { MetricProps } from 'aws-cdk-lib/aws-cloudwatch';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { Construct } from 'constructs';\nimport { IWatchful } from './watchful';\nimport { ComparisonOperator, LogAlarmProps, MetricAlarmProps, MonitorType, UnitOfPeriod } from '..';\n\nconst DEFAULT_CPU_UTILIZATION_THRESHOLD_PERCENT = 80;\nconst DEFAULT_MEMORY_UTILIZATION_THRESHOLD_PERCENT = 85;\n\n/**\n * Options for defining alarms.\n */\nexport interface WatchEcsServiceOptions {\n  /**\n     * Threshold for the cpu utilization alarm as percentage.\n     *\n     * @defaultValue 80\n     */\n  readonly cpuUtilizationThresholdPercent?: number;\n\n  /**\n     * Threshold for the memory utilization alarm as percentage.\n     *\n     * @defaultValue 95\n     */\n  readonly memoryUtilizationThresholdPercent?: number;\n}\n\n/**\n * Properties for defining a WatchEcsService\n */\nexport interface WatchEcsServiceProps extends WatchEcsServiceOptions {\n  /**\n     * The reference to IWatchful class. Used internally.\n     */\n  readonly watchful: IWatchful;\n\n  /**\n     * The ECS service that should be watched.\n     * [disable-awslint:ref-via-interface]\n     */\n  readonly service: ecs.BaseService;\n}\n\n/**\n * A Construct which creates default alarms for ECS services.\n * A Construct which creates the following alarms for ECS services:\n *\n * - cpu utilization\n * - memory utilization\n * - error log alert\n */\nexport class WatchEcsService extends Construct {\n  private readonly watchful: IWatchful;\n  private readonly service: ecs.BaseService;\n\n  constructor(scope: Construct, id: string, props: WatchEcsServiceProps) {\n    super(scope, id);\n\n    this.watchful = props.watchful;\n    this.service = props.service;\n\n    this.watchful.createAlarm(this,\n      MonitorType.CPU_UTILIZATION,\n      this.createCpuUtilizationMonitor(props.cpuUtilizationThresholdPercent),\n    );\n    this.watchful.createAlarm(this,\n      MonitorType.MEMORY_UTILIZATION,\n      this.createMemoryUtilizationMonitor(props.memoryUtilizationThresholdPercent),\n    );\n    this.watchful.createLogAlarm(this,\n      MonitorType.LOG_ERROR_MONITOR,\n      this.createLogErrorMonitor(),\n    );\n  }\n\n  private createLogErrorMonitor(): LogAlarmProps {\n    return {\n      autoCloseOpsGenieAlerts: true,\n      period: Duration.minutes(5),\n      unitOfPeriod: UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?\n      periodOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      statistic: 'count',\n      queryFilters: 'error',\n      alarmDescription: 'Auto-generated alert for error logs',\n      threshold: 1,\n      priority: 3,\n      index: '*', // toDo: we should take here the team name but for older setups to work we keep that for compatibility\n    };\n  }\n\n  private createCpuUtilizationMonitor(\n    cpuUtilizationThresholdPercent: number = DEFAULT_CPU_UTILIZATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const cpuUtilizationMetric = this.service.metricCpuUtilization().with({ period: Duration.minutes(10) });\n    return {\n      metric: cpuUtilizationMetric,\n      alarmDescription: `Average CPU utilization over last 10 minutes higher than ${cpuUtilizationThresholdPercent}%`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: cpuUtilizationThresholdPercent,\n      evaluationPeriods: 1,\n      priority: 3,\n    };\n  }\n\n  private createMemoryUtilizationMonitor(\n    memoryUtilizationThresholdPercent: number = DEFAULT_MEMORY_UTILIZATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const memoryUtilizationMetric = this.service\n      .metricMemoryUtilization()\n      .with({ period: Duration.minutes(10) });\n\n    const mappedMetric = new cw.Metric({\n      ...memoryUtilizationMetric as MetricProps,\n      metricName: `${memoryUtilizationMetric.metricName}.maximum`,\n    });\n    return {\n      metric: mappedMetric,\n      alarmDescription: `Maximum memory utilization over last 10 minutes higher than ${memoryUtilizationThresholdPercent}%`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: memoryUtilizationThresholdPercent,\n      evaluationPeriods: 1,\n      priority: 3,\n    };\n  }\n\n}\n"]}
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs.js","sourceRoot":"","sources":["../../src/watchful/ecs.ts"],"names":[],"mappings":";;;;;AAAA,6CAAuC;AACvC,iDAAiD;AAGjD,2CAAuC;AAEvC,0BAAoG;AAEpG,MAAM,yCAAyC,GAAG,EAAE,CAAC;AACrD,MAAM,4CAA4C,GAAG,EAAE,CAAC;AAqCxD;;;;;;;GAOG;AACH,MAAa,eAAgB,SAAQ,sBAAS;IAI5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA2B;QACnE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;QAC/B,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC;QAE7B,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,EAC5B,eAAW,CAAC,eAAe,EAC3B,IAAI,CAAC,2BAA2B,CAAC,KAAK,CAAC,8BAA8B,CAAC,CACvE,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,EAC5B,eAAW,CAAC,kBAAkB,EAC9B,IAAI,CAAC,8BAA8B,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAC7E,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,EAC/B,eAAW,CAAC,iBAAiB,EAC7B,IAAI,CAAC,qBAAqB,EAAE,CAC7B,CAAC;IACJ,CAAC;IAEO,qBAAqB;QAC3B,OAAO;YACL,uBAAuB,EAAE,IAAI;YAC7B,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC3B,YAAY,EAAE,gBAAY,CAAC,OAAO,EAAE,yDAAyD;YAC7F,cAAc,EAAE,sBAAkB,CAAC,kCAAkC;YACrE,SAAS,EAAE,OAAO;YAClB,YAAY,EAAE,OAAO;YACrB,gBAAgB,EAAE,qCAAqC;YACvD,SAAS,EAAE,CAAC;YACZ,QAAQ,EAAE,CAAC;YACX,KAAK,EAAE,GAAG,EAAE,sGAAsG;SACnH,CAAC;IACJ,CAAC;IAEO,2BAA2B,CACjC,iCAAyC,yCAAyC;QAElF,MAAM,oBAAoB,GAAG,IAAI,CAAC,OAAO,CAAC,oBAAoB,EAAE,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QACxG,OAAO;YACL,MAAM,EAAE,oBAAoB;YAC5B,gBAAgB,EAAE,4DAA4D,8BAA8B,GAAG;YAC/G,kBAAkB,EAAE,EAAE,CAAC,kBAAkB,CAAC,sBAAsB;YAChE,SAAS,EAAE,8BAA8B;YACzC,iBAAiB,EAAE,CAAC;YACpB,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;IAEO,8BAA8B,CACpC,oCAA4C,4CAA4C;QAExF,MAAM,uBAAuB,GAAG,IAAI,CAAC,OAAO;aACzC,uBAAuB,EAAE;aACzB,IAAI,CAAC,EAAE,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAE1C,MAAM,YAAY,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC;YACjC,GAAG,uBAAsC;YACzC,UAAU,EAAE,GAAG,uBAAuB,CAAC,UAAU,UAAU;SAC5D,CAAC,CAAC;QACH,OAAO;YACL,MAAM,EAAE,YAAY;YACpB,gBAAgB,EAAE,+DAA+D,iCAAiC,GAAG;YACrH,kBAAkB,EAAE,EAAE,CAAC,kBAAkB,CAAC,sBAAsB;YAChE,SAAS,EAAE,iCAAiC;YAC5C,iBAAiB,EAAE,CAAC;YACpB,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;;AAxEH,0CA0EC","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport { MetricProps } from 'aws-cdk-lib/aws-cloudwatch';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { Construct } from 'constructs';\nimport { IWatchful } from './watchful';\nimport { ComparisonOperator, LogAlarmProps, MetricAlarmProps, MonitorType, UnitOfPeriod } from '..';\n\nconst DEFAULT_CPU_UTILIZATION_THRESHOLD_PERCENT = 80;\nconst DEFAULT_MEMORY_UTILIZATION_THRESHOLD_PERCENT = 85;\n\n/**\n * Options for defining alarms.\n */\nexport interface WatchEcsServiceOptions {\n  /**\n     * Threshold for the cpu utilization alarm as percentage.\n     *\n     * @defaultValue 80\n     */\n  readonly cpuUtilizationThresholdPercent?: number;\n\n  /**\n     * Threshold for the memory utilization alarm as percentage.\n     *\n     * @defaultValue 95\n     */\n  readonly memoryUtilizationThresholdPercent?: number;\n}\n\n/**\n * Properties for defining a WatchEcsService\n */\nexport interface WatchEcsServiceProps extends WatchEcsServiceOptions {\n  /**\n     * The reference to IWatchful class. Used internally.\n     */\n  readonly watchful: IWatchful;\n\n  /**\n     * The ECS service that should be watched.\n     * [disable-awslint:ref-via-interface]\n     */\n  readonly service: ecs.BaseService;\n}\n\n/**\n * A Construct which creates default alarms for ECS services.\n * A Construct which creates the following alarms for ECS services:\n *\n * - cpu utilization\n * - memory utilization\n * - error log alert\n */\nexport class WatchEcsService extends Construct {\n  private readonly watchful: IWatchful;\n  private readonly service: ecs.BaseService;\n\n  constructor(scope: Construct, id: string, props: WatchEcsServiceProps) {\n    super(scope, id);\n\n    this.watchful = props.watchful;\n    this.service = props.service;\n\n    this.watchful.createAlarm(this,\n      MonitorType.CPU_UTILIZATION,\n      this.createCpuUtilizationMonitor(props.cpuUtilizationThresholdPercent),\n    );\n    this.watchful.createAlarm(this,\n      MonitorType.MEMORY_UTILIZATION,\n      this.createMemoryUtilizationMonitor(props.memoryUtilizationThresholdPercent),\n    );\n    this.watchful.createLogAlarm(this,\n      MonitorType.LOG_ERROR_MONITOR,\n      this.createLogErrorMonitor(),\n    );\n  }\n\n  private createLogErrorMonitor(): LogAlarmProps {\n    return {\n      autoCloseOpsGenieAlerts: true,\n      period: Duration.minutes(5),\n      unitOfPeriod: UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?\n      periodOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      statistic: 'count',\n      queryFilters: 'error',\n      alarmDescription: 'Auto-generated alert for error logs',\n      threshold: 1,\n      priority: 3,\n      index: '*', // toDo: we should take here the team name but for older setups to work we keep that for compatibility\n    };\n  }\n\n  private createCpuUtilizationMonitor(\n    cpuUtilizationThresholdPercent: number = DEFAULT_CPU_UTILIZATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const cpuUtilizationMetric = this.service.metricCpuUtilization().with({ period: Duration.minutes(10) });\n    return {\n      metric: cpuUtilizationMetric,\n      alarmDescription: `Average CPU utilization over last 10 minutes higher than ${cpuUtilizationThresholdPercent}%`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: cpuUtilizationThresholdPercent,\n      evaluationPeriods: 1,\n      priority: 3,\n    };\n  }\n\n  private createMemoryUtilizationMonitor(\n    memoryUtilizationThresholdPercent: number = DEFAULT_MEMORY_UTILIZATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const memoryUtilizationMetric = this.service\n      .metricMemoryUtilization()\n      .with({ period: Duration.minutes(10) });\n\n    const mappedMetric = new cw.Metric({\n      ...memoryUtilizationMetric as MetricProps,\n      metricName: `${memoryUtilizationMetric.metricName}.maximum`,\n    });\n    return {\n      metric: mappedMetric,\n      alarmDescription: `Maximum memory utilization over last 10 minutes higher than ${memoryUtilizationThresholdPercent}%`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: memoryUtilizationThresholdPercent,\n      evaluationPeriods: 1,\n      priority: 3,\n    };\n  }\n\n}\n"]}
@@ -32,14 +32,14 @@ class WatchLambdaFunction extends constructs_1.Construct {
32
32
  return {
33
33
  autoCloseOpsGenieAlerts: true,
34
34
  period: aws_cdk_lib_1.Duration.minutes(5),
35
- unitOfPeriod: datadog_log_alarm_1.UnitOfPeriod.MINUTES,
35
+ unitOfPeriod: datadog_log_alarm_1.UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?
36
36
  periodOperator: datadog_log_alarm_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
37
37
  statistic: 'count',
38
38
  queryFilters: 'error',
39
39
  alarmDescription: 'Auto-generated alert for Lambda error logs',
40
40
  threshold: 1,
41
41
  priority: 3,
42
- index: '*',
42
+ index: '*', // toDo: we should take here the team name but for older setups to work we keep that for compatibility
43
43
  lambdaCloudwatchGroupName: this.fn.functionName,
44
44
  };
45
45
  }
@@ -75,7 +75,7 @@ class WatchLambdaFunction extends constructs_1.Construct {
75
75
  metric: durationMetric,
76
76
  alarmDescription: `p99 latency >= ${durationThresholdSec}s (${durationPercentThreshold}%)`,
77
77
  comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,
78
- threshold: durationThresholdSec * 1000,
78
+ threshold: durationThresholdSec * 1000, // milliseconds
79
79
  evaluationPeriods: 3,
80
80
  priority: 3,
81
81
  };
@@ -84,4 +84,4 @@ class WatchLambdaFunction extends constructs_1.Construct {
84
84
  exports.WatchLambdaFunction = WatchLambdaFunction;
85
85
  _a = JSII_RTTI_SYMBOL_1;
86
86
  WatchLambdaFunction[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.WatchLambdaFunction", version: "0.0.0" };
87
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda.js","sourceRoot":"","sources":["../../src/watchful/lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { Construct } from 'constructs';\nimport { ComparisonOperator, LogAlarmProps, UnitOfPeriod } from './datadog-log-alarm';\nimport { MetricAlarmProps } from './metric-alarm';\nimport { MonitorType } from './monitor-type';\nimport { IWatchful } from './watchful';\n\nconst DEFAULT_DURATION_THRESHOLD_PERCENT = 80;\n\n/**\n * Options for defining alarms.\n */\nexport interface WatchLambdaFunctionOptions {\n  /**\n     * Number of allowed errors per minute. If there are more errors than that, an alarm will trigger.\n     *\n     * @defaultValue 0\n     */\n  readonly errorsPerMinuteThreshold?: number;\n\n  /**\n     * Number of allowed throttles per minute.\n     *\n     * @defaultValue 0\n     */\n  readonly throttlesPerMinuteThreshold?: number;\n\n  /**\n     * Threshold for the duration alarm as percentage of the function's timeout\n     * value.\n     *\n     * If this is set to 50%, the alarm will be set when p99 latency of the\n     * function exceeds 50% of the function's timeout setting.\n     *\n     * @defaultValue 80\n     */\n  readonly durationThresholdPercent?: number;\n}\n\n/**\n * Properties for defining a WatchLambdaFunction\n */\nexport interface WatchLambdaFunctionProps extends WatchLambdaFunctionOptions {\n  /**\n     * The reference to IWatchful class. Used internally.\n     */\n  readonly watchful: IWatchful;\n\n  /**\n     * The lambda function that should be watched.\n     * [disable-awslint:ref-via-interface]\n     */\n  readonly fn: lambda.Function;\n}\n\n/**\n * A Construct which creates the following alarms for lambda functions:\n *\n * - errors\n * - throttles\n * - duration\n */\nexport class WatchLambdaFunction extends Construct {\n  private readonly watchful: IWatchful;\n  private readonly fn: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: WatchLambdaFunctionProps) {\n    super(scope, id);\n\n    const cfnFunction = props.fn.node.defaultChild as lambda.CfnFunction;\n    const timeoutSec = cfnFunction.timeout || 3;\n\n    this.watchful = props.watchful;\n    this.fn = props.fn;\n\n    this.watchful.createAlarm(this, MonitorType.ERRORS, this.createErrorsMonitor(props.errorsPerMinuteThreshold));\n    this.watchful.createAlarm(this, MonitorType.THROTTLES, this.createThrottlesMonitor(props.throttlesPerMinuteThreshold));\n    this.watchful.createAlarm(this,\n      MonitorType.DURATION,\n      this.createDurationMonitor(timeoutSec, props.durationThresholdPercent),\n    );\n    this.watchful.createLogAlarm(\n      this,\n      MonitorType.LOG_ERROR_MONITOR,\n      this.createLogErrorMonitor(),\n    );\n  }\n\n  private createLogErrorMonitor(): LogAlarmProps {\n    return {\n      autoCloseOpsGenieAlerts: true,\n      period: Duration.minutes(5),\n      unitOfPeriod: UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?\n      periodOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      statistic: 'count',\n      queryFilters: 'error',\n      alarmDescription: 'Auto-generated alert for Lambda error logs',\n      threshold: 1,\n      priority: 3,\n      index: '*', // toDo: we should take here the team name but for older setups to work we keep that for compatibility\n      lambdaCloudwatchGroupName: this.fn.functionName,\n    };\n  }\n  private createErrorsMonitor(errorsPerMinuteThreshold = 0): MetricAlarmProps {\n    const fn = this.fn;\n    const errorsMetric = fn.metricErrors();\n    return {\n      metric: errorsMetric,\n      alarmDescription: `Over ${errorsPerMinuteThreshold} errors per minute`,\n      threshold: errorsPerMinuteThreshold,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n\n  private createThrottlesMonitor(throttlesPerMinuteThreshold = 0): MetricAlarmProps {\n    const fn = this.fn;\n    const throttlesMetric = fn.metricThrottles();\n    return {\n      metric: throttlesMetric,\n      alarmDescription: `Over ${throttlesPerMinuteThreshold} throttles per minute`,\n      threshold: throttlesPerMinuteThreshold,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n\n  private createDurationMonitor(\n    timeoutSec: number,\n    durationPercentThreshold: number = DEFAULT_DURATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const fn = this.fn;\n    const durationMetric = fn.metricDuration();\n    const durationThresholdSec = Math.floor((durationPercentThreshold / 100) * timeoutSec);\n    return {\n      metric: durationMetric,\n      alarmDescription: `p99 latency >= ${durationThresholdSec}s (${durationPercentThreshold}%)`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: durationThresholdSec * 1000, // milliseconds\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n}\n"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda.js","sourceRoot":"","sources":["../../src/watchful/lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport * as cw from 'aws-cdk-lib/aws-cloudwatch';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { Construct } from 'constructs';\nimport { ComparisonOperator, LogAlarmProps, UnitOfPeriod } from './datadog-log-alarm';\nimport { MetricAlarmProps } from './metric-alarm';\nimport { MonitorType } from './monitor-type';\nimport { IWatchful } from './watchful';\n\nconst DEFAULT_DURATION_THRESHOLD_PERCENT = 80;\n\n/**\n * Options for defining alarms.\n */\nexport interface WatchLambdaFunctionOptions {\n  /**\n     * Number of allowed errors per minute. If there are more errors than that, an alarm will trigger.\n     *\n     * @defaultValue 0\n     */\n  readonly errorsPerMinuteThreshold?: number;\n\n  /**\n     * Number of allowed throttles per minute.\n     *\n     * @defaultValue 0\n     */\n  readonly throttlesPerMinuteThreshold?: number;\n\n  /**\n     * Threshold for the duration alarm as percentage of the function's timeout\n     * value.\n     *\n     * If this is set to 50%, the alarm will be set when p99 latency of the\n     * function exceeds 50% of the function's timeout setting.\n     *\n     * @defaultValue 80\n     */\n  readonly durationThresholdPercent?: number;\n}\n\n/**\n * Properties for defining a WatchLambdaFunction\n */\nexport interface WatchLambdaFunctionProps extends WatchLambdaFunctionOptions {\n  /**\n     * The reference to IWatchful class. Used internally.\n     */\n  readonly watchful: IWatchful;\n\n  /**\n     * The lambda function that should be watched.\n     * [disable-awslint:ref-via-interface]\n     */\n  readonly fn: lambda.Function;\n}\n\n/**\n * A Construct which creates the following alarms for lambda functions:\n *\n * - errors\n * - throttles\n * - duration\n */\nexport class WatchLambdaFunction extends Construct {\n  private readonly watchful: IWatchful;\n  private readonly fn: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: WatchLambdaFunctionProps) {\n    super(scope, id);\n\n    const cfnFunction = props.fn.node.defaultChild as lambda.CfnFunction;\n    const timeoutSec = cfnFunction.timeout || 3;\n\n    this.watchful = props.watchful;\n    this.fn = props.fn;\n\n    this.watchful.createAlarm(this, MonitorType.ERRORS, this.createErrorsMonitor(props.errorsPerMinuteThreshold));\n    this.watchful.createAlarm(this, MonitorType.THROTTLES, this.createThrottlesMonitor(props.throttlesPerMinuteThreshold));\n    this.watchful.createAlarm(this,\n      MonitorType.DURATION,\n      this.createDurationMonitor(timeoutSec, props.durationThresholdPercent),\n    );\n    this.watchful.createLogAlarm(\n      this,\n      MonitorType.LOG_ERROR_MONITOR,\n      this.createLogErrorMonitor(),\n    );\n  }\n\n  private createLogErrorMonitor(): LogAlarmProps {\n    return {\n      autoCloseOpsGenieAlerts: true,\n      period: Duration.minutes(5),\n      unitOfPeriod: UnitOfPeriod.MINUTES, // toDo: move that somewhere else and set defaults there?\n      periodOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      statistic: 'count',\n      queryFilters: 'error',\n      alarmDescription: 'Auto-generated alert for Lambda error logs',\n      threshold: 1,\n      priority: 3,\n      index: '*', // toDo: we should take here the team name but for older setups to work we keep that for compatibility\n      lambdaCloudwatchGroupName: this.fn.functionName,\n    };\n  }\n  private createErrorsMonitor(errorsPerMinuteThreshold = 0): MetricAlarmProps {\n    const fn = this.fn;\n    const errorsMetric = fn.metricErrors();\n    return {\n      metric: errorsMetric,\n      alarmDescription: `Over ${errorsPerMinuteThreshold} errors per minute`,\n      threshold: errorsPerMinuteThreshold,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n\n  private createThrottlesMonitor(throttlesPerMinuteThreshold = 0): MetricAlarmProps {\n    const fn = this.fn;\n    const throttlesMetric = fn.metricThrottles();\n    return {\n      metric: throttlesMetric,\n      alarmDescription: `Over ${throttlesPerMinuteThreshold} throttles per minute`,\n      threshold: throttlesPerMinuteThreshold,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n\n  private createDurationMonitor(\n    timeoutSec: number,\n    durationPercentThreshold: number = DEFAULT_DURATION_THRESHOLD_PERCENT,\n  ): MetricAlarmProps {\n    const fn = this.fn;\n    const durationMetric = fn.metricDuration();\n    const durationThresholdSec = Math.floor((durationPercentThreshold / 100) * timeoutSec);\n    return {\n      metric: durationMetric,\n      alarmDescription: `p99 latency >= ${durationThresholdSec}s (${durationPercentThreshold}%)`,\n      comparisonOperator: cw.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: durationThresholdSec * 1000, // milliseconds\n      evaluationPeriods: 3,\n      priority: 3,\n    };\n  }\n}\n"]}
@@ -188,10 +188,10 @@ class Watchful extends AbstractWatchful {
188
188
  serviceName: alarm.lambdaCloudwatchGroupName ?? this.dataDogMonitorServiceName,
189
189
  autoCloseOpsGenieAlerts: alarm.autoCloseOpsGenieAlerts,
190
190
  period: alarm.period,
191
- unitOfPeriod: alarm.unitOfPeriod,
191
+ unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?
192
192
  periodOperator: alarm.periodOperator,
193
193
  statistic: alarm.statistic,
194
- queryFilters: alarm.queryFilters,
194
+ queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?
195
195
  alarmDescription: alarm.alarmDescription,
196
196
  threshold: shouldOverrideThreshold ? threshold : alarm.threshold,
197
197
  priority: alarm.priority,
@@ -211,4 +211,4 @@ class Watchful extends AbstractWatchful {
211
211
  exports.Watchful = Watchful;
212
212
  _b = JSII_RTTI_SYMBOL_1;
213
213
  Watchful[_b] = { fqn: "@rio-cloud/cdk-v2-constructs.Watchful", version: "0.0.0" };
214
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watchful.js","sourceRoot":"","sources":["../../src/watchful/watchful.ts"],"names":[],"mappings":"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","sourcesContent":["import { Annotations, Aspects, Stack } from 'aws-cdk-lib';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as cw_actions from 'aws-cdk-lib/aws-cloudwatch-actions';\nimport * as docdb from 'aws-cdk-lib/aws-docdb';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport * as elbv2 from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as rds from 'aws-cdk-lib/aws-rds';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport { Construct, IConstruct } from 'constructs';\nimport {\n  WatchfulAspect,\n  WatchLambdaFunction,\n  WatchLambdaFunctionOptions,\n  WatchCloudfrontDistribution,\n  WatchCloudfrontDistributionOptions,\n  WatchApplicationLoadBalancer,\n  WatchApplicationLoadBalancerOptions,\n  WatchEcsService,\n  WatchEcsServiceOptions,\n  WatchApplicationTargetGroup,\n  WatchApplicationTargetGroupOptions,\n  DataDogMetricAlarm,\n  DataDogLogAlarm,\n  MetricAlarmProps,\n  WatchDocDbCluster,\n  WatchDocDbClusterOptions,\n  WatchDatabaseCluster,\n  WatchDatabaseClusterOptions,\n  WatchDynamoDbTable,\n  WatchDynamoDbTableOptions,\n  LogAlarmProps,\n  MonitorType,\n} from './';\n\n/**\n * Interface for Watchful implementation.\n */\nexport interface IWatchful extends IConstruct {\n  /**\n   * Adds and alert for error logs\n   * @param id - The name of the service\n   */\n  createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps): void;\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   * @param alarm - The watchful to add those resources to\n   *\n   */\n  createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void;\n}\n\nexport interface OverrideAlarmThresholdProps {\n  /**\n   * The scope of the resource for which the threshold needs to be overriden\n   */\n  readonly monitoredResourceScope: Construct;\n\n  /**\n   * Id of the monitor that needs to be overriden.\n   * Eg - Errors, Thresholds, LogErrorMonitor\n   */\n  readonly monitorType: MonitorType;\n\n  /**\n   * The threshold that needs to be set\n  */\n  readonly threshold: number;\n}\n\nexport interface ShouldOverrideThresholdProps {\n  readonly watchfulMonitorScope: Construct;\n  readonly watchfulNodeId: string;\n}\n\nexport interface ShouldOverrideThresholdReturnProps {\n  readonly shouldOverrideThreshold: boolean;\n  readonly threshold?: number;\n}\n\nexport abstract class AbstractWatchful extends Construct implements IWatchful {\n  protected readonly metricsAlarmOverrides: Array<OverrideAlarmThresholdProps> = [];\n  abstract createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps): void\n  abstract createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void;\n\n  /**\n   * Override the default thresholds for the watchful monitors\n   */\n  public overrideAlarmThreshold(props: OverrideAlarmThresholdProps) {\n    this.metricsAlarmOverrides.push({\n      monitoredResourceScope: props.monitoredResourceScope,\n      monitorType: props.monitorType,\n      threshold: props.threshold,\n    });\n  }\n\n  protected shouldOverrideThreshold(props: ShouldOverrideThresholdProps): ShouldOverrideThresholdReturnProps {\n    for (const metricsAlarmOverride of this.metricsAlarmOverrides) {\n      if (\n        props.watchfulMonitorScope.node.id === metricsAlarmOverride.monitoredResourceScope.node.id &&\n        props.watchfulNodeId === metricsAlarmOverride.monitorType) {\n        return { shouldOverrideThreshold: true, threshold: metricsAlarmOverride.threshold };\n      }\n    }\n    return { shouldOverrideThreshold: false };\n  };\n\n  /**\n   * Adds alarms for the given lambda function.\n   * @param fn - A lambda function that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchLambdaFunction(fn: lambda.Function, options: WatchLambdaFunctionOptions = {}): void {\n\n    new WatchLambdaFunction(this, fn.node.id, {\n      watchful: this,\n      fn,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given Cloudfront Distribution.\n   * @param cf - A Cloudfront Distribution that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchCloudfrontDistribution(\n    cf: cloudfront.Distribution,\n    options: WatchCloudfrontDistributionOptions = {},\n  ): void {\n    new WatchCloudfrontDistribution(this, cf.node.id, {\n      watchful: this,\n      cf,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given ECS service.\n   * @param service - An ECS service that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchEcsService(service: ecs.BaseService, options: WatchEcsServiceOptions = {}): void {\n    new WatchEcsService(this, service.node.id, {\n      watchful: this,\n      service,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given application load balancer.\n   * @param alb - An application load balancer that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchAlb(alb: elbv2.ApplicationLoadBalancer, options: WatchApplicationLoadBalancerOptions = {}): void {\n    new WatchApplicationLoadBalancer(this, alb.node.id, {\n      watchful: this,\n      alb,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given application target gorup.\n   * @param atg - An application target group that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchApplicationTargetGroup(\n    atg: elbv2.ApplicationTargetGroup,\n    options: WatchApplicationTargetGroupOptions = {},\n  ): void {\n    new WatchApplicationTargetGroup(this, atg.node.id, {\n      watchful: this,\n      atg,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given DocDb cluster.\n   * @param docDb - An DocDb cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDocDb(docDb: docdb.DatabaseCluster, options: WatchDocDbClusterOptions = {}): void {\n    new WatchDocDbCluster(this, docDb.node.id, {\n      watchful: this,\n      docDb,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given database cluster.\n   * @param databaseCluster - An database cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDatabaseCluster(databaseCluster: rds.DatabaseCluster, options: WatchDatabaseClusterOptions = {}): void {\n    new WatchDatabaseCluster(this, databaseCluster.node.id, {\n      watchful: this,\n      databaseCluster,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given DynamoDb table.\n   * @param dynamoDbTable - An database cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDynamoDbTable(dynamoDbTable: dynamodb.Table, options: WatchDynamoDbTableOptions = {}): void {\n    new WatchDynamoDbTable(this, dynamoDbTable.node.id, {\n      watchful: this,\n      dynamoDbTable: dynamoDbTable,\n      ...options,\n    });\n  }\n}\n\n/**\n * Properties for defining Watchful\n */\nexport interface WatchfulProps {\n  /**\n   * An SNS topic as target for alarm actions.\n   *\n   * @defaultValue  The SNS topic provided by OpsGenie Integration account module.\n   */\n  readonly alarmSns?: sns.ITopic;\n\n  /**\n   * The service token for the DataDog Monitor custom resource.\n   *\n   * @defaultValue Empty.\n   */\n  readonly dataDogMonitorServiceToken?: string;\n\n  /**\n   * The service name.\n   *\n   * @defaultValue The stack name.\n   */\n  readonly dataDogMonitorServiceName?: string;\n\n  /**\n   * The OpsGenie handle used in DataDog to forward the DD monitor to OpsGenie.\n   *\n   * @defaultValue Empty if dataDogMonitorServiceToken is not set. Otherwise, team SSM parameter from LandingZone is used.\n   */\n  readonly dataDogOpsGenieIntegrationName?: string;\n}\n\n/**\n * A construct to watch given scope or resources. Opinionated DataDog alarms are automatically created for watched resources.\n * @deprecated Please use watchfulv2 instead\n */\nexport class Watchful extends AbstractWatchful {\n  private readonly alarmTopic?: sns.ITopic;\n  private readonly dataDogMonitorServiceToken?: string;\n  private readonly dataDogOpsGenieIntegrationName?: string;\n  private readonly dataDogMonitorServiceName: string;\n\n  constructor(scope: Construct, id: string, props: WatchfulProps = {}) {\n    super(scope, id);\n\n    // const opsGenieAlarmTopicArn = Fn.importValue('opsgenie-cloudwatch-alarm-sns-topic-arn');\n    // const opsGenieAlarmTopic = sns.Topic.fromTopicArn(this, 'OpsGenieTopic', opsGenieAlarmTopicArn);\n    this.alarmTopic = props.alarmSns;\n    this.dataDogMonitorServiceToken = props.dataDogMonitorServiceToken;\n\n    this.dataDogMonitorServiceName = props.dataDogMonitorServiceName ?? Stack.of(this).stackName;\n\n    if (props.dataDogMonitorServiceToken !== undefined) {\n\n      const teamNameParam = ssm.StringParameter.fromStringParameterName(\n        this,\n        'TeamNameParam',\n        '/config/team/name',\n      );\n      this.dataDogOpsGenieIntegrationName = props.dataDogOpsGenieIntegrationName ?? teamNameParam.stringValue;\n    }\n    Annotations.of(scope).addDeprecation('@rio-cloud/cdk-v2-constructs/watchful', 'watchful deprecated! Please use watchfulv2 instead');\n  }\n\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   * @param alarm The watchful to add those resources to\n   */\n  public createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    if (this.alarmTopic !== undefined) {\n      const cwAlarm = alarm.metric.createAlarm(scope, id, {\n        alarmDescription: alarm.alarmDescription,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        comparisonOperator: alarm.comparisonOperator,\n        evaluationPeriods: alarm.evaluationPeriods,\n      });\n      if (this.alarmTopic) {\n        cwAlarm.addAlarmAction(new cw_actions.SnsAction(this.alarmTopic));\n        cwAlarm.addOkAction(new cw_actions.SnsAction(this.alarmTopic));\n      }\n    }\n\n    if (this.dataDogMonitorServiceToken !== undefined) {\n      new DataDogMetricAlarm(scope, id, {\n        serviceToken: this.dataDogMonitorServiceToken ?? '',\n        serviceName: this.dataDogMonitorServiceName,\n        dataDogOpsGenieIntegrationName: this.dataDogOpsGenieIntegrationName ?? '',\n        alarmDescription: alarm.alarmDescription,\n        comparisonOperator: alarm.comparisonOperator,\n        evaluationPeriods: alarm.evaluationPeriods,\n        metric: alarm.metric,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        treatMissingData: alarm.treatMissingData,\n        priority: alarm.priority,\n      });\n    }\n  }\n\n  /**\n   * Adds an alert for error logs\n   * @param scope\n   */\n  public createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps):void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    if (this.dataDogMonitorServiceToken !== undefined) {\n      new DataDogLogAlarm(scope, id, {\n        serviceToken: this.dataDogMonitorServiceToken ?? '',\n        serviceName: alarm.lambdaCloudwatchGroupName ?? this.dataDogMonitorServiceName,\n        autoCloseOpsGenieAlerts: alarm.autoCloseOpsGenieAlerts,\n        period: alarm.period,\n        unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?\n        periodOperator: alarm.periodOperator,\n        statistic: alarm.statistic,\n        queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?\n        alarmDescription: alarm.alarmDescription,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        priority: alarm.priority,\n        index: alarm.index,\n        lambdaCloudwatchGroupName: alarm.lambdaCloudwatchGroupName,\n      });\n    }\n  }\n\n  /**\n   * Watches the given scope and adds alarms for known resources.\n   */\n  public watchScope(scope: Construct): void {\n    const aspect = new WatchfulAspect(this);\n    Aspects.of(scope).add(aspect);\n  }\n\n}\n"]}
214
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watchful.js","sourceRoot":"","sources":["../../src/watchful/watchful.ts"],"names":[],"mappings":";;;;;AAAA,6CAA0D;AAE1D,iEAAiE;AAQjE,2CAA2C;AAC3C,2CAAmD;AACnD,yBAuBY;AA+CZ,MAAsB,gBAAiB,SAAQ,sBAAS;IAAxD;;QACqB,0BAAqB,GAAuC,EAAE,CAAC;KAgInF;IA5HC;;OAEG;IACI,sBAAsB,CAAC,KAAkC;QAC9D,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;YAC9B,sBAAsB,EAAE,KAAK,CAAC,sBAAsB;YACpD,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,SAAS,EAAE,KAAK,CAAC,SAAS;SAC3B,CAAC,CAAC;IACL,CAAC;IAES,uBAAuB,CAAC,KAAmC;QACnE,KAAK,MAAM,oBAAoB,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC9D,IACE,KAAK,CAAC,oBAAoB,CAAC,IAAI,CAAC,EAAE,KAAK,oBAAoB,CAAC,sBAAsB,CAAC,IAAI,CAAC,EAAE;gBAC1F,KAAK,CAAC,cAAc,KAAK,oBAAoB,CAAC,WAAW,EAAE,CAAC;gBAC5D,OAAO,EAAE,uBAAuB,EAAE,IAAI,EAAE,SAAS,EAAE,oBAAoB,CAAC,SAAS,EAAE,CAAC;YACtF,CAAC;QACH,CAAC;QACD,OAAO,EAAE,uBAAuB,EAAE,KAAK,EAAE,CAAC;IAC5C,CAAC;IAAA,CAAC;IAEF;;;OAGG;IACI,mBAAmB,CAAC,EAAmB,EAAE,UAAsC,EAAE;QAEtF,IAAI,sBAAmB,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE;YACxC,QAAQ,EAAE,IAAI;YACd,EAAE;YACF,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,2BAA2B,CAChC,EAA2B,EAC3B,UAA8C,EAAE;QAEhD,IAAI,8BAA2B,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE;YAChD,QAAQ,EAAE,IAAI;YACd,EAAE;YACF,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,eAAe,CAAC,OAAwB,EAAE,UAAkC,EAAE;QACnF,IAAI,kBAAe,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE;YACzC,QAAQ,EAAE,IAAI;YACd,OAAO;YACP,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,QAAQ,CAAC,GAAkC,EAAE,UAA+C,EAAE;QACnG,IAAI,+BAA4B,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE;YAClD,QAAQ,EAAE,IAAI;YACd,GAAG;YACH,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,2BAA2B,CAChC,GAAiC,EACjC,UAA8C,EAAE;QAEhD,IAAI,8BAA2B,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE;YACjD,QAAQ,EAAE,IAAI;YACd,GAAG;YACH,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,UAAU,CAAC,KAA4B,EAAE,UAAoC,EAAE;QACpF,IAAI,oBAAiB,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,EAAE,EAAE;YACzC,QAAQ,EAAE,IAAI;YACd,KAAK;YACL,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,oBAAoB,CAAC,eAAoC,EAAE,UAAuC,EAAE;QACzG,IAAI,uBAAoB,CAAC,IAAI,EAAE,eAAe,CAAC,IAAI,CAAC,EAAE,EAAE;YACtD,QAAQ,EAAE,IAAI;YACd,eAAe;YACf,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,kBAAkB,CAAC,aAA6B,EAAE,UAAqC,EAAE;QAC9F,IAAI,qBAAkB,CAAC,IAAI,EAAE,aAAa,CAAC,IAAI,CAAC,EAAE,EAAE;YAClD,QAAQ,EAAE,IAAI;YACd,aAAa,EAAE,aAAa;YAC5B,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;;AAhIH,4CAiIC;;;AAmCD;;;GAGG;AACH,MAAa,QAAS,SAAQ,gBAAgB;IAM5C,YAAY,KAAgB,EAAE,EAAU,EAAE,QAAuB,EAAE;QACjE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,2FAA2F;QAC3F,mGAAmG;QACnG,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,QAAQ,CAAC;QACjC,IAAI,CAAC,0BAA0B,GAAG,KAAK,CAAC,0BAA0B,CAAC;QAEnE,IAAI,CAAC,yBAAyB,GAAG,KAAK,CAAC,yBAAyB,IAAI,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC;QAE7F,IAAI,KAAK,CAAC,0BAA0B,KAAK,SAAS,EAAE,CAAC;YAEnD,MAAM,aAAa,GAAG,GAAG,CAAC,eAAe,CAAC,uBAAuB,CAC/D,IAAI,EACJ,eAAe,EACf,mBAAmB,CACpB,CAAC;YACF,IAAI,CAAC,8BAA8B,GAAG,KAAK,CAAC,8BAA8B,IAAI,aAAa,CAAC,WAAW,CAAC;QAC1G,CAAC;QACD,yBAAW,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,cAAc,CAAC,uCAAuC,EAAE,oDAAoD,CAAC,CAAC;IACtI,CAAC;IAED;;;OAGG;IACI,WAAW,CAAC,KAAgB,EAAE,EAAe,EAAE,KAAuB;QAC3E,MAAM,EAAE,uBAAuB,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC,uBAAuB,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,CAAC;QACjI,IAAI,IAAI,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YAClC,MAAM,OAAO,GAAG,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,EAAE,EAAE,EAAE;gBAClD,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;gBACxC,SAAS,EAAE,uBAAuB,CAAC,CAAC,CAAC,SAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS;gBAC1E,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;gBAC5C,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;aAC3C,CAAC,CAAC;YACH,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,OAAO,CAAC,cAAc,CAAC,IAAI,UAAU,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;gBAClE,OAAO,CAAC,WAAW,CAAC,IAAI,UAAU,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;YACjE,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,0BAA0B,KAAK,SAAS,EAAE,CAAC;YAClD,IAAI,qBAAkB,CAAC,KAAK,EAAE,EAAE,EAAE;gBAChC,YAAY,EAAE,IAAI,CAAC,0BAA0B,IAAI,EAAE;gBACnD,WAAW,EAAE,IAAI,CAAC,yBAAyB;gBAC3C,8BAA8B,EAAE,IAAI,CAAC,8BAA8B,IAAI,EAAE;gBACzE,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;gBACxC,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;gBAC5C,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;gBAC1C,MAAM,EAAE,KAAK,CAAC,MAAM;gBACpB,SAAS,EAAE,uBAAuB,CAAC,CAAC,CAAC,SAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS;gBAC1E,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;gBACxC,QAAQ,EAAE,KAAK,CAAC,QAAQ;aACzB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;;OAGG;IACI,cAAc,CAAC,KAAgB,EAAE,EAAe,EAAE,KAAoB;QAC3E,MAAM,EAAE,uBAAuB,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC,uBAAuB,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,CAAC;QACjI,IAAI,IAAI,CAAC,0BAA0B,KAAK,SAAS,EAAE,CAAC;YAClD,IAAI,kBAAe,CAAC,KAAK,EAAE,EAAE,EAAE;gBAC7B,YAAY,EAAE,IAAI,CAAC,0BAA0B,IAAI,EAAE;gBACnD,WAAW,EAAE,KAAK,CAAC,yBAAyB,IAAI,IAAI,CAAC,yBAAyB;gBAC9E,uBAAuB,EAAE,KAAK,CAAC,uBAAuB;gBACtD,MAAM,EAAE,KAAK,CAAC,MAAM;gBACpB,YAAY,EAAE,KAAK,CAAC,YAAY,EAAE,yDAAyD;gBAC3F,cAAc,EAAE,KAAK,CAAC,cAAc;gBACpC,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,YAAY,EAAE,KAAK,CAAC,YAAY,EAAE,yDAAyD;gBAC3F,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;gBACxC,SAAS,EAAE,uBAAuB,CAAC,CAAC,CAAC,SAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS;gBAC1E,QAAQ,EAAE,KAAK,CAAC,QAAQ;gBACxB,KAAK,EAAE,KAAK,CAAC,KAAK;gBAClB,yBAAyB,EAAE,KAAK,CAAC,yBAAyB;aAC3D,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACI,UAAU,CAAC,KAAgB;QAChC,MAAM,MAAM,GAAG,IAAI,iBAAc,CAAC,IAAI,CAAC,CAAC;QACxC,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAChC,CAAC;;AA9FH,4BAgGC","sourcesContent":["import { Annotations, Aspects, Stack } from 'aws-cdk-lib';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as cw_actions from 'aws-cdk-lib/aws-cloudwatch-actions';\nimport * as docdb from 'aws-cdk-lib/aws-docdb';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport * as elbv2 from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as rds from 'aws-cdk-lib/aws-rds';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport { Construct, IConstruct } from 'constructs';\nimport {\n  WatchfulAspect,\n  WatchLambdaFunction,\n  WatchLambdaFunctionOptions,\n  WatchCloudfrontDistribution,\n  WatchCloudfrontDistributionOptions,\n  WatchApplicationLoadBalancer,\n  WatchApplicationLoadBalancerOptions,\n  WatchEcsService,\n  WatchEcsServiceOptions,\n  WatchApplicationTargetGroup,\n  WatchApplicationTargetGroupOptions,\n  DataDogMetricAlarm,\n  DataDogLogAlarm,\n  MetricAlarmProps,\n  WatchDocDbCluster,\n  WatchDocDbClusterOptions,\n  WatchDatabaseCluster,\n  WatchDatabaseClusterOptions,\n  WatchDynamoDbTable,\n  WatchDynamoDbTableOptions,\n  LogAlarmProps,\n  MonitorType,\n} from './';\n\n/**\n * Interface for Watchful implementation.\n */\nexport interface IWatchful extends IConstruct {\n  /**\n   * Adds and alert for error logs\n   * @param id - The name of the service\n   */\n  createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps): void;\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   * @param alarm - The watchful to add those resources to\n   *\n   */\n  createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void;\n}\n\nexport interface OverrideAlarmThresholdProps {\n  /**\n   * The scope of the resource for which the threshold needs to be overriden\n   */\n  readonly monitoredResourceScope: Construct;\n\n  /**\n   * Id of the monitor that needs to be overriden.\n   * Eg - Errors, Thresholds, LogErrorMonitor\n   */\n  readonly monitorType: MonitorType;\n\n  /**\n   * The threshold that needs to be set\n  */\n  readonly threshold: number;\n}\n\nexport interface ShouldOverrideThresholdProps {\n  readonly watchfulMonitorScope: Construct;\n  readonly watchfulNodeId: string;\n}\n\nexport interface ShouldOverrideThresholdReturnProps {\n  readonly shouldOverrideThreshold: boolean;\n  readonly threshold?: number;\n}\n\nexport abstract class AbstractWatchful extends Construct implements IWatchful {\n  protected readonly metricsAlarmOverrides: Array<OverrideAlarmThresholdProps> = [];\n  abstract createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps): void\n  abstract createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void;\n\n  /**\n   * Override the default thresholds for the watchful monitors\n   */\n  public overrideAlarmThreshold(props: OverrideAlarmThresholdProps) {\n    this.metricsAlarmOverrides.push({\n      monitoredResourceScope: props.monitoredResourceScope,\n      monitorType: props.monitorType,\n      threshold: props.threshold,\n    });\n  }\n\n  protected shouldOverrideThreshold(props: ShouldOverrideThresholdProps): ShouldOverrideThresholdReturnProps {\n    for (const metricsAlarmOverride of this.metricsAlarmOverrides) {\n      if (\n        props.watchfulMonitorScope.node.id === metricsAlarmOverride.monitoredResourceScope.node.id &&\n        props.watchfulNodeId === metricsAlarmOverride.monitorType) {\n        return { shouldOverrideThreshold: true, threshold: metricsAlarmOverride.threshold };\n      }\n    }\n    return { shouldOverrideThreshold: false };\n  };\n\n  /**\n   * Adds alarms for the given lambda function.\n   * @param fn - A lambda function that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchLambdaFunction(fn: lambda.Function, options: WatchLambdaFunctionOptions = {}): void {\n\n    new WatchLambdaFunction(this, fn.node.id, {\n      watchful: this,\n      fn,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given Cloudfront Distribution.\n   * @param cf - A Cloudfront Distribution that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchCloudfrontDistribution(\n    cf: cloudfront.Distribution,\n    options: WatchCloudfrontDistributionOptions = {},\n  ): void {\n    new WatchCloudfrontDistribution(this, cf.node.id, {\n      watchful: this,\n      cf,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given ECS service.\n   * @param service - An ECS service that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchEcsService(service: ecs.BaseService, options: WatchEcsServiceOptions = {}): void {\n    new WatchEcsService(this, service.node.id, {\n      watchful: this,\n      service,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given application load balancer.\n   * @param alb - An application load balancer that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchAlb(alb: elbv2.ApplicationLoadBalancer, options: WatchApplicationLoadBalancerOptions = {}): void {\n    new WatchApplicationLoadBalancer(this, alb.node.id, {\n      watchful: this,\n      alb,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given application target gorup.\n   * @param atg - An application target group that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchApplicationTargetGroup(\n    atg: elbv2.ApplicationTargetGroup,\n    options: WatchApplicationTargetGroupOptions = {},\n  ): void {\n    new WatchApplicationTargetGroup(this, atg.node.id, {\n      watchful: this,\n      atg,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given DocDb cluster.\n   * @param docDb - An DocDb cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDocDb(docDb: docdb.DatabaseCluster, options: WatchDocDbClusterOptions = {}): void {\n    new WatchDocDbCluster(this, docDb.node.id, {\n      watchful: this,\n      docDb,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given database cluster.\n   * @param databaseCluster - An database cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDatabaseCluster(databaseCluster: rds.DatabaseCluster, options: WatchDatabaseClusterOptions = {}): void {\n    new WatchDatabaseCluster(this, databaseCluster.node.id, {\n      watchful: this,\n      databaseCluster,\n      ...options,\n    });\n  }\n\n  /**\n   * Adds alarms for the given DynamoDb table.\n   * @param dynamoDbTable - An database cluster that should be watched [disable-awslint:ref-via-interface]\n   */\n  public watchDynamoDbTable(dynamoDbTable: dynamodb.Table, options: WatchDynamoDbTableOptions = {}): void {\n    new WatchDynamoDbTable(this, dynamoDbTable.node.id, {\n      watchful: this,\n      dynamoDbTable: dynamoDbTable,\n      ...options,\n    });\n  }\n}\n\n/**\n * Properties for defining Watchful\n */\nexport interface WatchfulProps {\n  /**\n   * An SNS topic as target for alarm actions.\n   *\n   * @defaultValue  The SNS topic provided by OpsGenie Integration account module.\n   */\n  readonly alarmSns?: sns.ITopic;\n\n  /**\n   * The service token for the DataDog Monitor custom resource.\n   *\n   * @defaultValue Empty.\n   */\n  readonly dataDogMonitorServiceToken?: string;\n\n  /**\n   * The service name.\n   *\n   * @defaultValue The stack name.\n   */\n  readonly dataDogMonitorServiceName?: string;\n\n  /**\n   * The OpsGenie handle used in DataDog to forward the DD monitor to OpsGenie.\n   *\n   * @defaultValue Empty if dataDogMonitorServiceToken is not set. Otherwise, team SSM parameter from LandingZone is used.\n   */\n  readonly dataDogOpsGenieIntegrationName?: string;\n}\n\n/**\n * A construct to watch given scope or resources. Opinionated DataDog alarms are automatically created for watched resources.\n * @deprecated Please use watchfulv2 instead\n */\nexport class Watchful extends AbstractWatchful {\n  private readonly alarmTopic?: sns.ITopic;\n  private readonly dataDogMonitorServiceToken?: string;\n  private readonly dataDogOpsGenieIntegrationName?: string;\n  private readonly dataDogMonitorServiceName: string;\n\n  constructor(scope: Construct, id: string, props: WatchfulProps = {}) {\n    super(scope, id);\n\n    // const opsGenieAlarmTopicArn = Fn.importValue('opsgenie-cloudwatch-alarm-sns-topic-arn');\n    // const opsGenieAlarmTopic = sns.Topic.fromTopicArn(this, 'OpsGenieTopic', opsGenieAlarmTopicArn);\n    this.alarmTopic = props.alarmSns;\n    this.dataDogMonitorServiceToken = props.dataDogMonitorServiceToken;\n\n    this.dataDogMonitorServiceName = props.dataDogMonitorServiceName ?? Stack.of(this).stackName;\n\n    if (props.dataDogMonitorServiceToken !== undefined) {\n\n      const teamNameParam = ssm.StringParameter.fromStringParameterName(\n        this,\n        'TeamNameParam',\n        '/config/team/name',\n      );\n      this.dataDogOpsGenieIntegrationName = props.dataDogOpsGenieIntegrationName ?? teamNameParam.stringValue;\n    }\n    Annotations.of(scope).addDeprecation('@rio-cloud/cdk-v2-constructs/watchful', 'watchful deprecated! Please use watchfulv2 instead');\n  }\n\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   * @param alarm The watchful to add those resources to\n   */\n  public createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    if (this.alarmTopic !== undefined) {\n      const cwAlarm = alarm.metric.createAlarm(scope, id, {\n        alarmDescription: alarm.alarmDescription,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        comparisonOperator: alarm.comparisonOperator,\n        evaluationPeriods: alarm.evaluationPeriods,\n      });\n      if (this.alarmTopic) {\n        cwAlarm.addAlarmAction(new cw_actions.SnsAction(this.alarmTopic));\n        cwAlarm.addOkAction(new cw_actions.SnsAction(this.alarmTopic));\n      }\n    }\n\n    if (this.dataDogMonitorServiceToken !== undefined) {\n      new DataDogMetricAlarm(scope, id, {\n        serviceToken: this.dataDogMonitorServiceToken ?? '',\n        serviceName: this.dataDogMonitorServiceName,\n        dataDogOpsGenieIntegrationName: this.dataDogOpsGenieIntegrationName ?? '',\n        alarmDescription: alarm.alarmDescription,\n        comparisonOperator: alarm.comparisonOperator,\n        evaluationPeriods: alarm.evaluationPeriods,\n        metric: alarm.metric,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        treatMissingData: alarm.treatMissingData,\n        priority: alarm.priority,\n      });\n    }\n  }\n\n  /**\n   * Adds an alert for error logs\n   * @param scope\n   */\n  public createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps):void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    if (this.dataDogMonitorServiceToken !== undefined) {\n      new DataDogLogAlarm(scope, id, {\n        serviceToken: this.dataDogMonitorServiceToken ?? '',\n        serviceName: alarm.lambdaCloudwatchGroupName ?? this.dataDogMonitorServiceName,\n        autoCloseOpsGenieAlerts: alarm.autoCloseOpsGenieAlerts,\n        period: alarm.period,\n        unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?\n        periodOperator: alarm.periodOperator,\n        statistic: alarm.statistic,\n        queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?\n        alarmDescription: alarm.alarmDescription,\n        threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n        priority: alarm.priority,\n        index: alarm.index,\n        lambdaCloudwatchGroupName: alarm.lambdaCloudwatchGroupName,\n      });\n    }\n  }\n\n  /**\n   * Watches the given scope and adds alarms for known resources.\n   */\n  public watchScope(scope: Construct): void {\n    const aspect = new WatchfulAspect(this);\n    Aspects.of(scope).add(aspect);\n  }\n\n}\n"]}