@liflig/cdk 2.21.1 → 2.21.3

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.
@@ -52,7 +52,7 @@ class Pipeline extends constructs.Construct {
52
52
  const checkCanRunFn = new lambda.SingletonFunction(this, "CheckCanRunFn", {
53
53
  uuid: "30ad3abb-f774-4804-a6ef-2c2f4a247362",
54
54
  code: new lambda.InlineCode(`exports.handler = ${checkCanRunHandler.toString()};`),
55
- runtime: lambda.Runtime.NODEJS_16_X,
55
+ runtime: lambda.Runtime.NODEJS_18_X,
56
56
  handler: "index.handler",
57
57
  timeout: cdk.Duration.seconds(10),
58
58
  });
@@ -71,7 +71,7 @@ class Pipeline extends constructs.Construct {
71
71
  const collectFilesFn = new lambda.SingletonFunction(this, "CollectFilesFn", {
72
72
  uuid: "c49cbfe1-50e0-4721-8964-fb20f4e5a7ad",
73
73
  code: new lambda.InlineCode(`exports.handler = ${collectFilesHandler.toString()};`),
74
- runtime: lambda.Runtime.NODEJS_16_X,
74
+ runtime: lambda.Runtime.NODEJS_18_X,
75
75
  handler: "index.handler",
76
76
  timeout: cdk.Duration.seconds(30),
77
77
  });
@@ -132,11 +132,10 @@ exports.Pipeline = Pipeline;
132
132
  // This is a self-contained function that will be serialized as a lambda.
133
133
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
134
134
  const collectFilesHandler = async (event) => {
135
- var _a;
136
135
  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
137
- const AWS = require("aws-sdk");
136
+ const { S3Client, ListObjectsV2Command, GetObjectCommand } = await Promise.resolve().then(() => require("@aws-sdk/client-s3"));
138
137
  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
139
- const s3 = new AWS.S3();
138
+ const s3Client = new S3Client();
140
139
  console.log("Event received: ", event);
141
140
  const bucketName = event.bucketName;
142
141
  const bucketPrefix = event.bucketPrefix;
@@ -144,37 +143,37 @@ const collectFilesHandler = async (event) => {
144
143
  if (bucketPrefix.slice(-1) !== "/") {
145
144
  throw new Error(`Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`);
146
145
  }
147
- const files = await s3
148
- .listObjectsV2({
149
- Bucket: bucketName,
150
- Prefix: bucketPrefix,
151
- })
152
- .promise();
153
- async function getData(key) {
154
- const result = await s3
155
- .getObject({
146
+ async function listObjects() {
147
+ const { Contents } = await s3Client.send(new ListObjectsV2Command({
148
+ Bucket: bucketName,
149
+ Prefix: bucketPrefix,
150
+ }));
151
+ return Contents;
152
+ }
153
+ async function getObject(key) {
154
+ const { Body } = await s3Client.send(new GetObjectCommand({
156
155
  Bucket: bucketName,
157
156
  Key: key,
158
- })
159
- .promise();
160
- return result.Body.toString();
157
+ }));
158
+ return Body.toString();
161
159
  }
162
160
  let cloudAssembly = null;
163
161
  let variables = {};
164
- for (const file of (_a = files.Contents) !== null && _a !== void 0 ? _a : []) {
162
+ const files = await listObjects();
163
+ for (const file of files !== null && files !== void 0 ? files : []) {
165
164
  const key = file.Key;
166
165
  const filename = key.slice(bucketPrefix.length);
167
166
  console.log(`File: ${filename}`);
168
167
  if (filename === "cloud-assembly.json") {
169
168
  console.log("Found Cloud Assembly");
170
169
  // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
171
- cloudAssembly = JSON.parse(await getData(key));
170
+ cloudAssembly = JSON.parse(await getObject(key));
172
171
  }
173
172
  else if (/^variables.*\.json$/.test(filename)) {
174
173
  console.log("Found variables file");
175
174
  variables = {
176
175
  ...variables,
177
- ...JSON.parse(await getData(key)),
176
+ ...JSON.parse(await getObject(key)),
178
177
  };
179
178
  }
180
179
  else {
@@ -199,26 +198,32 @@ const collectFilesHandler = async (event) => {
199
198
  // This is a self-contained function that will be serialized as a lambda.
200
199
  const checkCanRunHandler = async (event) => {
201
200
  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
202
- const AWS = require("aws-sdk");
203
201
  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
204
- const sf = new AWS.StepFunctions();
202
+ const { SFNClient, ListExecutionsCommand, ExecutionStatus } = await Promise.resolve().then(() => require("@aws-sdk/client-sfn"));
203
+ const sfnClient = new SFNClient();
205
204
  console.log("Event received: ", event);
206
205
  const stateMachineArn = event["stateMachineId"];
207
206
  const currentExecutionArn = event["executionId"];
208
- const executions = (await sf
209
- .listExecutions({
207
+ const { executions } = await sfnClient.send(new ListExecutionsCommand({
210
208
  stateMachineArn,
211
- statusFilter: "RUNNING",
212
- })
213
- .promise()).executions;
209
+ statusFilter: ExecutionStatus.RUNNING,
210
+ }));
211
+ if (!executions) {
212
+ throw new Error("Could not list executions");
213
+ }
214
214
  console.log("Executions: ", executions);
215
215
  const currentExecution = executions.find((it) => it.executionArn == currentExecutionArn);
216
216
  if (!currentExecution) {
217
217
  throw new Error("Could not find current execution");
218
218
  }
219
- const newer = executions.filter((it) => it.startDate > currentExecution.startDate).length;
219
+ const newer = executions.filter((it) => {
220
+ if (!it.startDate || !currentExecution.startDate) {
221
+ return false;
222
+ }
223
+ return it.startDate > currentExecution.startDate;
224
+ }).length;
220
225
  return {
221
226
  CanRunState: newer > 0 ? "SKIP" : executions.length == 1 ? "CONTINUE" : "WAIT",
222
227
  };
223
228
  };
224
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../../src/pipelines/pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as eventsTargets from \"aws-cdk-lib/aws-events-targets\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as sfn from \"aws-cdk-lib/aws-stepfunctions\"\nimport { Condition } from \"aws-cdk-lib/aws-stepfunctions\"\nimport * as tasks from \"aws-cdk-lib/aws-stepfunctions-tasks\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type { Handler } from \"aws-lambda\"\nimport type * as _AWS from \"aws-sdk\"\nimport { getGriidArtefactBucket } from \"../griid\"\nimport { pipelineS3Prefix, pipelineS3TriggerKey } from \"./conventions\"\nimport { DeployEnv } from \"./deploy-env\"\n\nexport interface PipelineProps {\n  /**\n   * Bucket holding pipeline configuration and trigger file.\n   *\n   * @default - use existing bucket based on Griid conventions\n   */\n  artifactsBucket?: s3.IBucket\n  /**\n   * Environments for this pipeline. Each environment is deployed sequentially\n   * in the order given.\n   */\n  environments: PipelineEnvironment[]\n  /**\n   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Trigger the pipeline when the trigger file is written.\n   *\n   * @default - true\n   */\n  triggerEnabled?: boolean\n  /**\n   * VPC used for Fargate resources.\n   */\n  vpc: ec2.IVpc\n}\n\nexport interface PipelineEnvironment {\n  /**\n   * Account number hosting the environment.\n   */\n  accountId: string\n  /**\n   * Additional tasks to run after the environment has been deployed.\n   */\n  afterSuccessfulDeploy?: sfn.Chain\n  /**\n   * Name of environment.\n   */\n  name: string\n}\n\n/**\n * Pipeline for doing a multi-account CDK deployment based\n * on a built CDK Cloud Assembly and parameters stored in S3.\n *\n * The accounts being deployed to must be provisioned with\n * the LifligCdkDeployerDeps construct so expected IAM\n * roles is present.\n *\n * The pipeline starts by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * The CDK deploy process is handled by liflig-cdk-deployer.\n * See https://github.com/capralifecycle/liflig-cdk-deployer\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n *  - cloud-assembly.json which has the format described as\n *    CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer\n *\n *  - variables*.json which can be zero or more files\n *    with string-string map that will be concatenated to\n *    form the format described as CDK_VARIABLES in\n *    liflig-cdk-deployer\n *\n * The separation of Cloud Assembly details and variables enables\n * separation of IaC code and application code if they are not\n * colocated in the same repository.\n */\nexport class Pipeline extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: PipelineProps) {\n    super(scope, id)\n\n    const s3Prefix = pipelineS3Prefix(props.pipelineName)\n    const s3TriggerKey = pipelineS3TriggerKey(props.pipelineName)\n\n    const artifactsBucket =\n      props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const checkCanRunFn = new lambda.SingletonFunction(this, \"CheckCanRunFn\", {\n      uuid: \"30ad3abb-f774-4804-a6ef-2c2f4a247362\",\n      code: new lambda.InlineCode(\n        `exports.handler = ${checkCanRunHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_16_X,\n      handler: \"index.handler\",\n      timeout: cdk.Duration.seconds(10),\n    })\n\n    const checkCanRunTask = new tasks.LambdaInvoke(\n      this,\n      \"Check if the process can run\",\n      {\n        lambdaFunction: checkCanRunFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          \"stateMachineId.$\": \"$$.StateMachine.Id\",\n          \"executionId.$\": \"$$.Execution.Id\",\n        }),\n      },\n    )\n\n    const wait = new sfn.Wait(this, \"Wait before rechecking status\", {\n      time: sfn.WaitTime.duration(cdk.Duration.seconds(15)),\n    })\n\n    const skip = new sfn.Succeed(this, \"Skip\")\n\n    const collectFilesFn = new lambda.SingletonFunction(\n      this,\n      \"CollectFilesFn\",\n      {\n        uuid: \"c49cbfe1-50e0-4721-8964-fb20f4e5a7ad\",\n        code: new lambda.InlineCode(\n          `exports.handler = ${collectFilesHandler.toString()};`,\n        ),\n        runtime: lambda.Runtime.NODEJS_16_X,\n        handler: \"index.handler\",\n        timeout: cdk.Duration.seconds(30),\n      },\n    )\n\n    artifactsBucket.grantRead(collectFilesFn)\n\n    const collectFilesTask = new tasks.LambdaInvoke(\n      this,\n      \"Collect files from S3\",\n      {\n        lambdaFunction: collectFilesFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          bucketName: artifactsBucket.bucketName,\n          bucketPrefix: s3Prefix,\n          envNames: props.environments.map((it) => it.name),\n        }),\n      },\n    )\n\n    let run: sfn.Chain = sfn.Chain.start(collectFilesTask)\n\n    const ifHavingStacks = (name: string, work: sfn.Chain) =>\n      new sfn.Choice(this, `Check if ${name} has stacks`)\n        .when(\n          Condition.or(\n            Condition.isNull(`$.StackCountPerEnv.${name}`),\n            Condition.numberEquals(`$.StackCountPerEnv.${name}`, 0),\n          ),\n          new sfn.Pass(this, `Skip ${name}`),\n        )\n        .otherwise(work)\n        .afterwards()\n\n    for (const env of props.environments) {\n      const it = new DeployEnv(this, env.name, {\n        accountId: env.accountId,\n        afterSuccessfulDeploy: env.afterSuccessfulDeploy,\n        artefactBucket: artifactsBucket,\n        envName: env.name,\n        vpc: props.vpc,\n      })\n\n      run = run.next(ifHavingStacks(env.name, it.chain))\n    }\n\n    const definition: sfn.Chain = sfn.Chain.start(checkCanRunTask).next(\n      new sfn.Choice(this, \"Can run?\")\n        .when(Condition.stringEquals(\"$.CanRunState\", \"CONTINUE\"), run)\n        .when(Condition.stringEquals(\"$.CanRunState\", \"SKIP\"), skip)\n        .otherwise(wait.next(checkCanRunTask)),\n    )\n\n    const machine = new sfn.StateMachine(this, \"StateMachine\", {\n      definition,\n      // https://docs.aws.amazon.com/step-functions/latest/dg/sfn-stuck-execution.html\n      timeout: cdk.Duration.hours(3),\n    })\n\n    new iam.Policy(this, \"CheckCanRunPolicy\", {\n      roles: [checkCanRunFn.role!],\n      statements: [\n        new iam.PolicyStatement({\n          actions: [\"states:ListExecutions\"],\n          resources: [machine.stateMachineArn],\n        }),\n      ],\n    })\n\n    if (props.triggerEnabled ?? true) {\n      artifactsBucket.onCloudTrailWriteObject(\"Trigger\", {\n        paths: [s3TriggerKey],\n        target: new eventsTargets.SfnStateMachine(machine, {\n          input: events.RuleTargetInput.fromObject({}),\n        }),\n      })\n    }\n  }\n}\n\ninterface CloudAssembly {\n  cloudAssemblyBucketName: string\n  cloudAssemblyBucketKey: string\n  environments: {\n    name: string\n    stackNames: string[]\n  }[]\n  parameters: {\n    name: string\n    value: unknown | { type: \"variable\"; variable: string }\n  }[]\n}\n\n// This is a self-contained function that will be serialized as a lambda.\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst collectFilesHandler: Handler = async (event: Record<string, any>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  const AWS = require(\"aws-sdk\")\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n  const s3 = new AWS.S3() as _AWS.S3\n\n  console.log(\"Event received: \", event)\n\n  const bucketName = event.bucketName as string\n  const bucketPrefix = event.bucketPrefix as string\n  const envNames = event.envNames as string[]\n\n  if (bucketPrefix.slice(-1) !== \"/\") {\n    throw new Error(\n      `Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`,\n    )\n  }\n\n  const files = await s3\n    .listObjectsV2({\n      Bucket: bucketName,\n      Prefix: bucketPrefix,\n    })\n    .promise()\n\n  async function getData(key: string): Promise<string> {\n    const result = await s3\n      .getObject({\n        Bucket: bucketName,\n        Key: key,\n      })\n      .promise()\n    return result.Body!.toString()\n  }\n\n  let cloudAssembly: CloudAssembly | null = null\n  let variables: Record<string, string> = {}\n\n  for (const file of files.Contents ?? []) {\n    const key = file.Key!\n    const filename = key.slice(bucketPrefix.length)\n\n    console.log(`File: ${filename}`)\n\n    if (filename === \"cloud-assembly.json\") {\n      console.log(\"Found Cloud Assembly\")\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n      cloudAssembly = JSON.parse(await getData(key))\n    } else if (/^variables.*\\.json$/.test(filename)) {\n      console.log(\"Found variables file\")\n      variables = {\n        ...variables,\n        ...(JSON.parse(await getData(key)) as Record<string, string>),\n      }\n    } else {\n      console.log(\"Ignoring unknown file\")\n    }\n  }\n\n  if (cloudAssembly === null) {\n    throw new Error(\"cloud-assembly.json not found\")\n  }\n\n  return {\n    CloudAssembly: JSON.stringify(cloudAssembly),\n    Variables: JSON.stringify(variables),\n    StackCountPerEnv: Object.fromEntries(\n      envNames.map((name) => [\n        name,\n        cloudAssembly.environments.find((it) => it.name === name)?.stackNames\n          .length ?? 0,\n      ]),\n    ),\n  }\n}\n\n// This is a self-contained function that will be serialized as a lambda.\nconst checkCanRunHandler: Handler = async (event: Record<string, string>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  const AWS = require(\"aws-sdk\")\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n  const sf = new AWS.StepFunctions() as _AWS.StepFunctions\n\n  console.log(\"Event received: \", event)\n\n  const stateMachineArn = event[\"stateMachineId\"]\n  const currentExecutionArn = event[\"executionId\"]\n\n  const executions = (\n    await sf\n      .listExecutions({\n        stateMachineArn,\n        statusFilter: \"RUNNING\",\n      })\n      .promise()\n  ).executions\n\n  console.log(\"Executions: \", executions)\n\n  const currentExecution = executions.find(\n    (it) => it.executionArn == currentExecutionArn,\n  )\n\n  if (!currentExecution) {\n    throw new Error(\"Could not find current execution\")\n  }\n\n  const newer = executions.filter(\n    (it) => it.startDate > currentExecution.startDate,\n  ).length\n\n  return {\n    CanRunState:\n      newer > 0 ? \"SKIP\" : executions.length == 1 ? \"CONTINUE\" : \"WAIT\",\n  }\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../../src/pipelines/pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as eventsTargets from \"aws-cdk-lib/aws-events-targets\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as sfn from \"aws-cdk-lib/aws-stepfunctions\"\nimport { Condition } from \"aws-cdk-lib/aws-stepfunctions\"\nimport * as tasks from \"aws-cdk-lib/aws-stepfunctions-tasks\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type { Handler } from \"aws-lambda\"\nimport { getGriidArtefactBucket } from \"../griid\"\nimport { pipelineS3Prefix, pipelineS3TriggerKey } from \"./conventions\"\nimport { DeployEnv } from \"./deploy-env\"\n\nexport interface PipelineProps {\n  /**\n   * Bucket holding pipeline configuration and trigger file.\n   *\n   * @default - use existing bucket based on Griid conventions\n   */\n  artifactsBucket?: s3.IBucket\n  /**\n   * Environments for this pipeline. Each environment is deployed sequentially\n   * in the order given.\n   */\n  environments: PipelineEnvironment[]\n  /**\n   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Trigger the pipeline when the trigger file is written.\n   *\n   * @default - true\n   */\n  triggerEnabled?: boolean\n  /**\n   * VPC used for Fargate resources.\n   */\n  vpc: ec2.IVpc\n}\n\nexport interface PipelineEnvironment {\n  /**\n   * Account number hosting the environment.\n   */\n  accountId: string\n  /**\n   * Additional tasks to run after the environment has been deployed.\n   */\n  afterSuccessfulDeploy?: sfn.Chain\n  /**\n   * Name of environment.\n   */\n  name: string\n}\n\n/**\n * Pipeline for doing a multi-account CDK deployment based\n * on a built CDK Cloud Assembly and parameters stored in S3.\n *\n * The accounts being deployed to must be provisioned with\n * the LifligCdkDeployerDeps construct so expected IAM\n * roles is present.\n *\n * The pipeline starts by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * The CDK deploy process is handled by liflig-cdk-deployer.\n * See https://github.com/capralifecycle/liflig-cdk-deployer\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n *  - cloud-assembly.json which has the format described as\n *    CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer\n *\n *  - variables*.json which can be zero or more files\n *    with string-string map that will be concatenated to\n *    form the format described as CDK_VARIABLES in\n *    liflig-cdk-deployer\n *\n * The separation of Cloud Assembly details and variables enables\n * separation of IaC code and application code if they are not\n * colocated in the same repository.\n */\nexport class Pipeline extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: PipelineProps) {\n    super(scope, id)\n\n    const s3Prefix = pipelineS3Prefix(props.pipelineName)\n    const s3TriggerKey = pipelineS3TriggerKey(props.pipelineName)\n\n    const artifactsBucket =\n      props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const checkCanRunFn = new lambda.SingletonFunction(this, \"CheckCanRunFn\", {\n      uuid: \"30ad3abb-f774-4804-a6ef-2c2f4a247362\",\n      code: new lambda.InlineCode(\n        `exports.handler = ${checkCanRunHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: \"index.handler\",\n      timeout: cdk.Duration.seconds(10),\n    })\n\n    const checkCanRunTask = new tasks.LambdaInvoke(\n      this,\n      \"Check if the process can run\",\n      {\n        lambdaFunction: checkCanRunFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          \"stateMachineId.$\": \"$$.StateMachine.Id\",\n          \"executionId.$\": \"$$.Execution.Id\",\n        }),\n      },\n    )\n\n    const wait = new sfn.Wait(this, \"Wait before rechecking status\", {\n      time: sfn.WaitTime.duration(cdk.Duration.seconds(15)),\n    })\n\n    const skip = new sfn.Succeed(this, \"Skip\")\n\n    const collectFilesFn = new lambda.SingletonFunction(\n      this,\n      \"CollectFilesFn\",\n      {\n        uuid: \"c49cbfe1-50e0-4721-8964-fb20f4e5a7ad\",\n        code: new lambda.InlineCode(\n          `exports.handler = ${collectFilesHandler.toString()};`,\n        ),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        handler: \"index.handler\",\n        timeout: cdk.Duration.seconds(30),\n      },\n    )\n\n    artifactsBucket.grantRead(collectFilesFn)\n\n    const collectFilesTask = new tasks.LambdaInvoke(\n      this,\n      \"Collect files from S3\",\n      {\n        lambdaFunction: collectFilesFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          bucketName: artifactsBucket.bucketName,\n          bucketPrefix: s3Prefix,\n          envNames: props.environments.map((it) => it.name),\n        }),\n      },\n    )\n\n    let run: sfn.Chain = sfn.Chain.start(collectFilesTask)\n\n    const ifHavingStacks = (name: string, work: sfn.Chain) =>\n      new sfn.Choice(this, `Check if ${name} has stacks`)\n        .when(\n          Condition.or(\n            Condition.isNull(`$.StackCountPerEnv.${name}`),\n            Condition.numberEquals(`$.StackCountPerEnv.${name}`, 0),\n          ),\n          new sfn.Pass(this, `Skip ${name}`),\n        )\n        .otherwise(work)\n        .afterwards()\n\n    for (const env of props.environments) {\n      const it = new DeployEnv(this, env.name, {\n        accountId: env.accountId,\n        afterSuccessfulDeploy: env.afterSuccessfulDeploy,\n        artefactBucket: artifactsBucket,\n        envName: env.name,\n        vpc: props.vpc,\n      })\n\n      run = run.next(ifHavingStacks(env.name, it.chain))\n    }\n\n    const definition: sfn.Chain = sfn.Chain.start(checkCanRunTask).next(\n      new sfn.Choice(this, \"Can run?\")\n        .when(Condition.stringEquals(\"$.CanRunState\", \"CONTINUE\"), run)\n        .when(Condition.stringEquals(\"$.CanRunState\", \"SKIP\"), skip)\n        .otherwise(wait.next(checkCanRunTask)),\n    )\n\n    const machine = new sfn.StateMachine(this, \"StateMachine\", {\n      definition,\n      // https://docs.aws.amazon.com/step-functions/latest/dg/sfn-stuck-execution.html\n      timeout: cdk.Duration.hours(3),\n    })\n\n    new iam.Policy(this, \"CheckCanRunPolicy\", {\n      roles: [checkCanRunFn.role!],\n      statements: [\n        new iam.PolicyStatement({\n          actions: [\"states:ListExecutions\"],\n          resources: [machine.stateMachineArn],\n        }),\n      ],\n    })\n\n    if (props.triggerEnabled ?? true) {\n      artifactsBucket.onCloudTrailWriteObject(\"Trigger\", {\n        paths: [s3TriggerKey],\n        target: new eventsTargets.SfnStateMachine(machine, {\n          input: events.RuleTargetInput.fromObject({}),\n        }),\n      })\n    }\n  }\n}\n\ninterface CloudAssembly {\n  cloudAssemblyBucketName: string\n  cloudAssemblyBucketKey: string\n  environments: {\n    name: string\n    stackNames: string[]\n  }[]\n  parameters: {\n    name: string\n    value: unknown | { type: \"variable\"; variable: string }\n  }[]\n}\n\n// This is a self-contained function that will be serialized as a lambda.\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst collectFilesHandler: Handler = async (event: Record<string, any>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  const { S3Client, ListObjectsV2Command, GetObjectCommand } = await import(\n    \"@aws-sdk/client-s3\"\n  )\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n\n  const s3Client = new S3Client()\n\n  console.log(\"Event received: \", event)\n\n  const bucketName = event.bucketName as string\n  const bucketPrefix = event.bucketPrefix as string\n  const envNames = event.envNames as string[]\n\n  if (bucketPrefix.slice(-1) !== \"/\") {\n    throw new Error(\n      `Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`,\n    )\n  }\n\n  async function listObjects() {\n    const { Contents } = await s3Client.send(\n      new ListObjectsV2Command({\n        Bucket: bucketName,\n        Prefix: bucketPrefix,\n      }),\n    )\n    return Contents\n  }\n\n  async function getObject(key: string): Promise<string> {\n    const { Body } = await s3Client.send(\n      new GetObjectCommand({\n        Bucket: bucketName,\n        Key: key,\n      }),\n    )\n    return Body!.toString()\n  }\n\n  let cloudAssembly: CloudAssembly | null = null\n  let variables: Record<string, string> = {}\n\n  const files = await listObjects()\n  for (const file of files ?? []) {\n    const key = file.Key!\n    const filename = key.slice(bucketPrefix.length)\n\n    console.log(`File: ${filename}`)\n\n    if (filename === \"cloud-assembly.json\") {\n      console.log(\"Found Cloud Assembly\")\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n      cloudAssembly = JSON.parse(await getObject(key))\n    } else if (/^variables.*\\.json$/.test(filename)) {\n      console.log(\"Found variables file\")\n      variables = {\n        ...variables,\n        ...(JSON.parse(await getObject(key)) as Record<string, string>),\n      }\n    } else {\n      console.log(\"Ignoring unknown file\")\n    }\n  }\n\n  if (cloudAssembly === null) {\n    throw new Error(\"cloud-assembly.json not found\")\n  }\n\n  return {\n    CloudAssembly: JSON.stringify(cloudAssembly),\n    Variables: JSON.stringify(variables),\n    StackCountPerEnv: Object.fromEntries(\n      envNames.map((name) => [\n        name,\n        cloudAssembly.environments.find((it) => it.name === name)?.stackNames\n          .length ?? 0,\n      ]),\n    ),\n  }\n}\n\n// This is a self-contained function that will be serialized as a lambda.\nconst checkCanRunHandler: Handler = async (event: Record<string, string>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n\n  const { SFNClient, ListExecutionsCommand, ExecutionStatus } = await import(\n    \"@aws-sdk/client-sfn\"\n  )\n  const sfnClient = new SFNClient()\n\n  console.log(\"Event received: \", event)\n\n  const stateMachineArn = event[\"stateMachineId\"]\n  const currentExecutionArn = event[\"executionId\"]\n\n  const { executions } = await sfnClient.send(\n    new ListExecutionsCommand({\n      stateMachineArn,\n      statusFilter: ExecutionStatus.RUNNING,\n    }),\n  )\n\n  if (!executions) {\n    throw new Error(\"Could not list executions\")\n  }\n\n  console.log(\"Executions: \", executions)\n\n  const currentExecution = executions.find(\n    (it) => it.executionArn == currentExecutionArn,\n  )\n\n  if (!currentExecution) {\n    throw new Error(\"Could not find current execution\")\n  }\n\n  const newer = executions.filter((it) => {\n    if (!it.startDate || !currentExecution.startDate) {\n      return false\n    }\n    return it.startDate > currentExecution.startDate\n  }).length\n\n  return {\n    CanRunState:\n      newer > 0 ? \"SKIP\" : executions.length == 1 ? \"CONTINUE\" : \"WAIT\",\n  }\n}\n"]}
@@ -13,5 +13,5 @@ type OnEventHandler = (event: {
13
13
  PhysicalResourceId?: string;
14
14
  Data?: Record<string, any>;
15
15
  }>;
16
- export declare const configurationSetSnsDestinationHandler: OnEventHandler;
16
+ export declare const handler: OnEventHandler;
17
17
  export {};
@@ -1,15 +1,13 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.configurationSetSnsDestinationHandler = void 0;
4
- // This function is inline-compiled for the lambda.
5
- // It must be self-contained.
6
- const configurationSetSnsDestinationHandler = async (event) => {
7
- const AWS = require("aws-sdk");
8
- const ses = new AWS.SESV2();
3
+ exports.handler = void 0;
4
+ const client_sesv2_1 = require("@aws-sdk/client-sesv2");
5
+ const handler = async (event) => {
6
+ const sesv2Client = new client_sesv2_1.SESv2Client();
9
7
  const configurationSetName = event.ResourceProperties.ConfigurationSetName;
10
8
  const eventDestinationName = event.ResourceProperties.EventDestinationName;
11
9
  const snsTopicArn = event.ResourceProperties.SnsTopicArn;
12
- const matchingEventTypes = event.ResourceProperties.MatchingEventTypes;
10
+ const matchingEventTypes = event.ResourceProperties.MatchingEventTypes.map((eventType) => eventType);
13
11
  const eventDestination = {
14
12
  MatchingEventTypes: matchingEventTypes,
15
13
  Enabled: true,
@@ -20,24 +18,20 @@ const configurationSetSnsDestinationHandler = async (event) => {
20
18
  console.log(`EventDestination ${JSON.stringify(eventDestination)}`);
21
19
  switch (event.RequestType) {
22
20
  case "Delete":
23
- const deleteResponse = await ses
24
- .deleteConfigurationSetEventDestination({
21
+ const deleteResponse = await sesv2Client.send(new client_sesv2_1.DeleteConfigurationSetEventDestinationCommand({
25
22
  ConfigurationSetName: configurationSetName,
26
23
  EventDestinationName: eventDestinationName,
27
- })
28
- .promise();
24
+ }));
29
25
  console.log(`ses.deleteConfigurationSetEventDestination: ${JSON.stringify(deleteResponse)}`);
30
26
  return {
31
27
  PhysicalResourceId: event.PhysicalResourceId,
32
28
  };
33
29
  case "Create":
34
- const createResponse = await ses
35
- .createConfigurationSetEventDestination({
30
+ const createResponse = await sesv2Client.send(new client_sesv2_1.CreateConfigurationSetEventDestinationCommand({
36
31
  ConfigurationSetName: configurationSetName,
37
32
  EventDestinationName: eventDestinationName,
38
33
  EventDestination: eventDestination,
39
- })
40
- .promise();
34
+ }));
41
35
  console.log(`ses.createConfigurationSetEventDestination: ${JSON.stringify(createResponse)}`);
42
36
  return {
43
37
  PhysicalResourceId: `ConfigurationSetSnsDestination-${configurationSetName}-${eventDestinationName}`,
@@ -47,23 +41,19 @@ const configurationSetSnsDestinationHandler = async (event) => {
47
41
  const previousConfigurationSetName = event.OldResourceProperties.ConfigurationSetName;
48
42
  if (configurationSetName !== previousConfigurationSetName ||
49
43
  eventDestinationName !== previousEventDestinationName) {
50
- const createResponse = await ses
51
- .createConfigurationSetEventDestination({
44
+ const createResponse = await sesv2Client.send(new client_sesv2_1.CreateConfigurationSetEventDestinationCommand({
52
45
  ConfigurationSetName: configurationSetName,
53
46
  EventDestinationName: eventDestinationName,
54
47
  EventDestination: eventDestination,
55
- })
56
- .promise();
48
+ }));
57
49
  console.log(`ses.createConfigurationSetEventDestination: ${JSON.stringify(createResponse)}`);
58
50
  }
59
51
  else {
60
- const updateResponse = await ses
61
- .updateConfigurationSetEventDestination({
52
+ const updateResponse = await sesv2Client.send(new client_sesv2_1.UpdateConfigurationSetEventDestinationCommand({
62
53
  ConfigurationSetName: configurationSetName,
63
54
  EventDestinationName: eventDestinationName,
64
55
  EventDestination: eventDestination,
65
- })
66
- .promise();
56
+ }));
67
57
  console.log(`ses.UpdateConfigurationSetEventDestination: ${JSON.stringify(updateResponse)}`);
68
58
  }
69
59
  return {
@@ -71,5 +61,5 @@ const configurationSetSnsDestinationHandler = async (event) => {
71
61
  };
72
62
  }
73
63
  };
74
- exports.configurationSetSnsDestinationHandler = configurationSetSnsDestinationHandler;
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/configurationsetsnsdestination/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type * as _AWS from \"aws-sdk\"\n\ninterface ResourceProps {\n  ConfigurationSetName: string\n  EventDestinationName: string\n  SnsTopicArn: string\n  MatchingEventTypes: string[]\n}\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: ResourceProps\n  OldResourceProperties?: ResourceProps\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const configurationSetSnsDestinationHandler: OnEventHandler = async (\n  event,\n) => {\n  const AWS = require(\"aws-sdk\")\n\n  const ses: AWS.SESV2 = new AWS.SESV2() as _AWS.SESV2\n\n  const configurationSetName = event.ResourceProperties.ConfigurationSetName\n  const eventDestinationName = event.ResourceProperties.EventDestinationName\n  const snsTopicArn = event.ResourceProperties.SnsTopicArn\n  const matchingEventTypes = event.ResourceProperties.MatchingEventTypes\n\n  const eventDestination: _AWS.SESV2.Types.EventDestinationDefinition = {\n    MatchingEventTypes: matchingEventTypes,\n    Enabled: true,\n    SnsDestination: {\n      TopicArn: snsTopicArn,\n    },\n  }\n\n  console.log(`EventDestination ${JSON.stringify(eventDestination)}`)\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const deleteResponse = await ses\n        .deleteConfigurationSetEventDestination({\n          ConfigurationSetName: configurationSetName,\n          EventDestinationName: eventDestinationName,\n        })\n        .promise()\n      console.log(\n        `ses.deleteConfigurationSetEventDestination: ${JSON.stringify(\n          deleteResponse,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n      const createResponse = await ses\n        .createConfigurationSetEventDestination({\n          ConfigurationSetName: configurationSetName,\n          EventDestinationName: eventDestinationName,\n          EventDestination: eventDestination,\n        })\n        .promise()\n      console.log(\n        `ses.createConfigurationSetEventDestination: ${JSON.stringify(\n          createResponse,\n        )}`,\n      )\n      return {\n        PhysicalResourceId: `ConfigurationSetSnsDestination-${configurationSetName}-${eventDestinationName}`,\n      }\n\n    case \"Update\":\n      const previousEventDestinationName =\n        event.OldResourceProperties!.EventDestinationName\n      const previousConfigurationSetName =\n        event.OldResourceProperties!.ConfigurationSetName\n\n      if (\n        configurationSetName !== previousConfigurationSetName ||\n        eventDestinationName !== previousEventDestinationName\n      ) {\n        const createResponse = await ses\n          .createConfigurationSetEventDestination({\n            ConfigurationSetName: configurationSetName,\n            EventDestinationName: eventDestinationName,\n            EventDestination: eventDestination,\n          })\n          .promise()\n        console.log(\n          `ses.createConfigurationSetEventDestination: ${JSON.stringify(\n            createResponse,\n          )}`,\n        )\n      } else {\n        const updateResponse = await ses\n          .updateConfigurationSetEventDestination({\n            ConfigurationSetName: configurationSetName,\n            EventDestinationName: eventDestinationName,\n            EventDestination: eventDestination,\n          })\n          .promise()\n        console.log(\n          `ses.UpdateConfigurationSetEventDestination: ${JSON.stringify(\n            updateResponse,\n          )}`,\n        )\n      }\n\n      return {\n        PhysicalResourceId: `ConfigurationSetSnsDestination-${configurationSetName}-${eventDestinationName}`,\n      }\n  }\n}\n"]}
64
+ exports.handler = handler;
65
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/configurationsetsnsdestination/handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SESv2Client,\n  CreateConfigurationSetEventDestinationCommand,\n  UpdateConfigurationSetEventDestinationCommand,\n  DeleteConfigurationSetEventDestinationCommand,\n  EventType,\n  EventDestinationDefinition,\n} from \"@aws-sdk/client-sesv2\"\n\ninterface ResourceProps {\n  ConfigurationSetName: string\n  EventDestinationName: string\n  SnsTopicArn: string\n  MatchingEventTypes: string[]\n}\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: ResourceProps\n  OldResourceProperties?: ResourceProps\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\nexport const handler: OnEventHandler = async (event) => {\n  const sesv2Client = new SESv2Client()\n  const configurationSetName = event.ResourceProperties.ConfigurationSetName\n  const eventDestinationName = event.ResourceProperties.EventDestinationName\n  const snsTopicArn = event.ResourceProperties.SnsTopicArn\n  const matchingEventTypes = event.ResourceProperties.MatchingEventTypes.map(\n    (eventType) => eventType as EventType,\n  )\n\n  const eventDestination: EventDestinationDefinition = {\n    MatchingEventTypes: matchingEventTypes,\n    Enabled: true,\n    SnsDestination: {\n      TopicArn: snsTopicArn,\n    },\n  }\n\n  console.log(`EventDestination ${JSON.stringify(eventDestination)}`)\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const deleteResponse = await sesv2Client.send(\n        new DeleteConfigurationSetEventDestinationCommand({\n          ConfigurationSetName: configurationSetName,\n          EventDestinationName: eventDestinationName,\n        }),\n      )\n      console.log(\n        `ses.deleteConfigurationSetEventDestination: ${JSON.stringify(\n          deleteResponse,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n      const createResponse = await sesv2Client.send(\n        new CreateConfigurationSetEventDestinationCommand({\n          ConfigurationSetName: configurationSetName,\n          EventDestinationName: eventDestinationName,\n          EventDestination: eventDestination,\n        }),\n      )\n      console.log(\n        `ses.createConfigurationSetEventDestination: ${JSON.stringify(\n          createResponse,\n        )}`,\n      )\n      return {\n        PhysicalResourceId: `ConfigurationSetSnsDestination-${configurationSetName}-${eventDestinationName}`,\n      }\n\n    case \"Update\":\n      const previousEventDestinationName =\n        event.OldResourceProperties!.EventDestinationName\n      const previousConfigurationSetName =\n        event.OldResourceProperties!.ConfigurationSetName\n\n      if (\n        configurationSetName !== previousConfigurationSetName ||\n        eventDestinationName !== previousEventDestinationName\n      ) {\n        const createResponse = await sesv2Client.send(\n          new CreateConfigurationSetEventDestinationCommand({\n            ConfigurationSetName: configurationSetName,\n            EventDestinationName: eventDestinationName,\n            EventDestination: eventDestination,\n          }),\n        )\n        console.log(\n          `ses.createConfigurationSetEventDestination: ${JSON.stringify(\n            createResponse,\n          )}`,\n        )\n      } else {\n        const updateResponse = await sesv2Client.send(\n          new UpdateConfigurationSetEventDestinationCommand({\n            ConfigurationSetName: configurationSetName,\n            EventDestinationName: eventDestinationName,\n            EventDestination: eventDestination,\n          }),\n        )\n        console.log(\n          `ses.UpdateConfigurationSetEventDestination: ${JSON.stringify(\n            updateResponse,\n          )}`,\n        )\n      }\n\n      return {\n        PhysicalResourceId: `ConfigurationSetSnsDestination-${configurationSetName}-${eventDestinationName}`,\n      }\n  }\n}\n"]}
@@ -6,9 +6,9 @@ const iam = require("aws-cdk-lib/aws-iam");
6
6
  const lambda = require("aws-cdk-lib/aws-lambda");
7
7
  const cdk = require("aws-cdk-lib");
8
8
  const cr = require("aws-cdk-lib/custom-resources");
9
- const handler_1 = require("./handler");
10
9
  const aws_logs_1 = require("aws-cdk-lib/aws-logs");
11
10
  const snsSubscriptions = require("aws-cdk-lib/aws-sns-subscriptions");
11
+ const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
12
12
  class ConfigurationSetSnsDestination extends constructs.Construct {
13
13
  constructor(scope, id, props) {
14
14
  var _a;
@@ -26,7 +26,7 @@ class ConfigurationSetSnsDestination extends constructs.Construct {
26
26
  const sesEventLoggerFunction = new lambda.Function(this, "EventsHandler", {
27
27
  code: new lambda.InlineCode(`exports.handler = ${sesEventLoggerHandler.toString()};`),
28
28
  handler: "index.handler",
29
- runtime: lambda.Runtime.NODEJS_16_X,
29
+ runtime: lambda.Runtime.NODEJS_18_X,
30
30
  logRetention: aws_logs_1.RetentionDays.THREE_MONTHS,
31
31
  });
32
32
  if ((_a = props.logEvents) !== null && _a !== void 0 ? _a : true) {
@@ -51,11 +51,11 @@ class ConfigurationSetSnsDestinationProvider extends constructs.Construct {
51
51
  constructor(scope, id) {
52
52
  super(scope, id);
53
53
  this.provider = new cr.Provider(this, "Provider", {
54
- onEventHandler: new lambda.Function(this, "Function", {
55
- code: new lambda.InlineCode(`exports.handler = ${handler_1.configurationSetSnsDestinationHandler.toString()};`),
56
- handler: "index.handler",
57
- runtime: lambda.Runtime.NODEJS_16_X,
54
+ onEventHandler: new aws_lambda_nodejs_1.NodejsFunction(this, "Function", {
55
+ entry: require.resolve("./handler"),
56
+ runtime: lambda.Runtime.NODEJS_18_X,
58
57
  timeout: cdk.Duration.minutes(5),
58
+ awsSdkConnectionReuse: false,
59
59
  initialPolicy: [
60
60
  new iam.PolicyStatement({
61
61
  actions: [
@@ -72,4 +72,4 @@ class ConfigurationSetSnsDestinationProvider extends constructs.Construct {
72
72
  this.serviceToken = this.provider.serviceToken;
73
73
  }
74
74
  }
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/configurationsetsnsdestination/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as sns from \"aws-cdk-lib/aws-sns\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { configurationSetSnsDestinationHandler } from \"./handler\"\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\"\nimport * as snsSubscriptions from \"aws-cdk-lib/aws-sns-subscriptions\"\nimport { SNSHandler } from \"aws-lambda\"\n\nexport type ConfigurationSetSnsDestinationEventType =\n  | \"SEND\"\n  | \"REJECT\"\n  | \"BOUNCE\"\n  | \"COMPLAINT\"\n  | \"DELIVERY\"\n  | \"OPEN\"\n  | \"CLICK\"\n  | \"RENDERING_FAILURE\"\n  | \"DELIVERY_DELAY\"\n  | \"SUBSCRIPTION\"\n\nexport interface ConfigurationSetSnsDestinationProps {\n  /**\n   * Whether SES events will be logged to CloudWatch\n   * @default true\n   */\n  logEvents?: boolean\n  /**\n   * The SES configuration set name\n   */\n  configurationSetName: string\n  /**\n   * The SES configuration set event destination name\n   */\n  eventDestinationName: string\n  /**\n   * SNS topic to send bounces to\n   */\n  snsTopic: sns.ITopic\n  /**\n   * Event types to match\n   */\n  matchingEventTypes: ConfigurationSetSnsDestinationEventType[]\n}\n\nexport class ConfigurationSetSnsDestination extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ConfigurationSetSnsDestinationProps,\n  ) {\n    super(scope, id)\n\n    new cdk.CustomResource(this, \"Resource\", {\n      serviceToken:\n        ConfigurationSetSnsDestinationProvider.getOrCreate(this).serviceToken,\n      properties: {\n        ConfigurationSetName: props.configurationSetName,\n        EventDestinationName: props.eventDestinationName,\n        SnsTopicArn: props.snsTopic.topicArn,\n        MatchingEventTypes: props.matchingEventTypes,\n        Serial: 1,\n      },\n    })\n\n    const sesEventLoggerFunction = new lambda.Function(this, \"EventsHandler\", {\n      code: new lambda.InlineCode(\n        `exports.handler = ${sesEventLoggerHandler.toString()};`,\n      ),\n      handler: \"index.handler\",\n      runtime: lambda.Runtime.NODEJS_16_X,\n      logRetention: RetentionDays.THREE_MONTHS,\n    })\n\n    if (props.logEvents ?? true) {\n      props.snsTopic.addSubscription(\n        new snsSubscriptions.LambdaSubscription(sesEventLoggerFunction),\n      )\n    }\n  }\n}\n\nconst sesEventLoggerHandler: SNSHandler = (event) => {\n  event.Records.forEach((record) =>\n    console.log(`SES event message from SNS: ${record.Sns.Message}`),\n  )\n}\n\nclass ConfigurationSetSnsDestinationProvider extends constructs.Construct {\n  /**\n   * Returns the singleton provider.\n   */\n  public static getOrCreate(scope: constructs.Construct) {\n    const stack = cdk.Stack.of(scope)\n    const id = \"liflig-cdk.configuration-set-sns-destination\"\n    return (\n      (stack.node.tryFindChild(id) as ConfigurationSetSnsDestinationProvider) ||\n      new ConfigurationSetSnsDestinationProvider(stack, id)\n    )\n  }\n\n  private readonly provider: cr.Provider\n  public readonly serviceToken: string\n\n  constructor(scope: constructs.Construct, id: string) {\n    super(scope, id)\n\n    this.provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler: new lambda.Function(this, \"Function\", {\n        code: new lambda.InlineCode(\n          `exports.handler = ${configurationSetSnsDestinationHandler.toString()};`,\n        ),\n        handler: \"index.handler\",\n        runtime: lambda.Runtime.NODEJS_16_X,\n        timeout: cdk.Duration.minutes(5),\n        initialPolicy: [\n          new iam.PolicyStatement({\n            actions: [\n              \"ses:CreateConfigurationSetEventDestination\",\n              \"ses:UpdateConfigurationSetEventDestination\",\n              \"ses:DeleteConfigurationSetEventDestination\",\n              \"ses:GetConfigurationSetEventDestinations\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/configurationsetsnsdestination/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as sns from \"aws-cdk-lib/aws-sns\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\"\nimport * as snsSubscriptions from \"aws-cdk-lib/aws-sns-subscriptions\"\nimport { SNSHandler } from \"aws-lambda\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\n\nexport type ConfigurationSetSnsDestinationEventType =\n  | \"SEND\"\n  | \"REJECT\"\n  | \"BOUNCE\"\n  | \"COMPLAINT\"\n  | \"DELIVERY\"\n  | \"OPEN\"\n  | \"CLICK\"\n  | \"RENDERING_FAILURE\"\n  | \"DELIVERY_DELAY\"\n  | \"SUBSCRIPTION\"\n\nexport interface ConfigurationSetSnsDestinationProps {\n  /**\n   * Whether SES events will be logged to CloudWatch\n   * @default true\n   */\n  logEvents?: boolean\n  /**\n   * The SES configuration set name\n   */\n  configurationSetName: string\n  /**\n   * The SES configuration set event destination name\n   */\n  eventDestinationName: string\n  /**\n   * SNS topic to send bounces to\n   */\n  snsTopic: sns.ITopic\n  /**\n   * Event types to match\n   */\n  matchingEventTypes: ConfigurationSetSnsDestinationEventType[]\n}\n\nexport class ConfigurationSetSnsDestination extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ConfigurationSetSnsDestinationProps,\n  ) {\n    super(scope, id)\n\n    new cdk.CustomResource(this, \"Resource\", {\n      serviceToken:\n        ConfigurationSetSnsDestinationProvider.getOrCreate(this).serviceToken,\n      properties: {\n        ConfigurationSetName: props.configurationSetName,\n        EventDestinationName: props.eventDestinationName,\n        SnsTopicArn: props.snsTopic.topicArn,\n        MatchingEventTypes: props.matchingEventTypes,\n        Serial: 1,\n      },\n    })\n\n    const sesEventLoggerFunction = new lambda.Function(this, \"EventsHandler\", {\n      code: new lambda.InlineCode(\n        `exports.handler = ${sesEventLoggerHandler.toString()};`,\n      ),\n      handler: \"index.handler\",\n      runtime: lambda.Runtime.NODEJS_18_X,\n      logRetention: RetentionDays.THREE_MONTHS,\n    })\n\n    if (props.logEvents ?? true) {\n      props.snsTopic.addSubscription(\n        new snsSubscriptions.LambdaSubscription(sesEventLoggerFunction),\n      )\n    }\n  }\n}\n\nconst sesEventLoggerHandler: SNSHandler = (event) => {\n  event.Records.forEach((record) =>\n    console.log(`SES event message from SNS: ${record.Sns.Message}`),\n  )\n}\n\nclass ConfigurationSetSnsDestinationProvider extends constructs.Construct {\n  /**\n   * Returns the singleton provider.\n   */\n  public static getOrCreate(scope: constructs.Construct) {\n    const stack = cdk.Stack.of(scope)\n    const id = \"liflig-cdk.configuration-set-sns-destination\"\n    return (\n      (stack.node.tryFindChild(id) as ConfigurationSetSnsDestinationProvider) ||\n      new ConfigurationSetSnsDestinationProvider(stack, id)\n    )\n  }\n\n  private readonly provider: cr.Provider\n  public readonly serviceToken: string\n\n  constructor(scope: constructs.Construct, id: string) {\n    super(scope, id)\n\n    this.provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler: new NodejsFunction(this, \"Function\", {\n        entry: require.resolve(\"./handler\"),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        timeout: cdk.Duration.minutes(5),\n        awsSdkConnectionReuse: false,\n        initialPolicy: [\n          new iam.PolicyStatement({\n            actions: [\n              \"ses:CreateConfigurationSetEventDestination\",\n              \"ses:UpdateConfigurationSetEventDestination\",\n              \"ses:DeleteConfigurationSetEventDestination\",\n              \"ses:GetConfigurationSetEventDestinations\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
@@ -6,5 +6,5 @@ type OnEventHandler = (event: {
6
6
  PhysicalResourceId?: string;
7
7
  Data?: Record<string, any>;
8
8
  }>;
9
- export declare const sesDomainHandler: OnEventHandler;
9
+ export declare const handler: OnEventHandler;
10
10
  export {};
@@ -1,13 +1,13 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.sesDomainHandler = void 0;
4
- // This function is inline-compiled for the lambda.
5
- // It must be self-contained.
6
- const sesDomainHandler = async (event) => {
7
- const AWS = require("aws-sdk");
3
+ exports.handler = void 0;
4
+ const client_ses_1 = require("@aws-sdk/client-ses");
5
+ const client_sesv2_1 = require("@aws-sdk/client-sesv2");
6
+ const handler = async (event) => {
7
+ var _a;
8
+ const sesClient = new client_ses_1.SESClient();
9
+ const sesv2Client = new client_sesv2_1.SESv2Client();
8
10
  const ttl = "1800";
9
- const ses = new AWS.SES();
10
- const sesv2 = new AWS.SESV2();
11
11
  const domainName = event.ResourceProperties["DomainName"];
12
12
  const includeVerificationRecord = event.ResourceProperties["IncludeVerificationRecord"] == "true";
13
13
  const defaultConfigurationSetName = event.ResourceProperties["DefaultConfigurationSetName"];
@@ -36,39 +36,39 @@ const sesDomainHandler = async (event) => {
36
36
  }
37
37
  switch (event.RequestType) {
38
38
  case "Delete":
39
- const response = await ses
40
- .deleteIdentity({ Identity: domainName })
41
- .promise();
42
- console.log(`ses.deleteIdentity: ${JSON.stringify(response)}`);
39
+ const deleteIdentityResp = await sesClient.send(new client_ses_1.DeleteIdentityCommand({
40
+ Identity: domainName,
41
+ }));
42
+ console.log(`ses.deleteIdentity: ${JSON.stringify(deleteIdentityResp)}`);
43
43
  return {
44
44
  PhysicalResourceId: event.PhysicalResourceId,
45
45
  };
46
46
  case "Create":
47
47
  case "Update":
48
48
  // Idempotent.
49
- const response1 = await ses
50
- .verifyDomainIdentity({
49
+ const verifyDomainIdentityResp = await sesClient.send(new client_ses_1.VerifyDomainIdentityCommand({
51
50
  Domain: domainName,
52
- })
53
- .promise();
54
- console.log(`ses.verifyDomainIdentity: ${JSON.stringify(response1)}`);
55
- const verificationToken = response1["VerificationToken"];
51
+ }));
52
+ console.log(`ses.verifyDomainIdentity: ${JSON.stringify(verifyDomainIdentityResp)}`);
53
+ const verificationToken = verifyDomainIdentityResp.VerificationToken;
54
+ if (!verificationToken) {
55
+ throw new Error("Verification token not returned");
56
+ }
56
57
  // Idempotent.
57
- const response2 = await ses
58
- .verifyDomainDkim({ Domain: domainName })
59
- .promise();
60
- console.log(`ses.verifyDomainDkim: ${JSON.stringify(response2)}`);
61
- const dkimTokens = response2["DkimTokens"];
58
+ const verifyDomainDkimResp = await sesClient.send(new client_ses_1.VerifyDomainDkimCommand({
59
+ Domain: domainName,
60
+ }));
61
+ console.log(`ses.verifyDomainDkim: ${JSON.stringify(verifyDomainDkimResp)}`);
62
+ const dkimTokens = (_a = verifyDomainDkimResp.DkimTokens) !== null && _a !== void 0 ? _a : [];
63
+ if (!dkimTokens) {
64
+ throw new Error("DKIM tokens not returned");
65
+ }
62
66
  // Idempotent.
63
- const response3 = await sesv2
64
- .putEmailIdentityConfigurationSetAttributes({
67
+ const putEmailIdentityConfigResp = await sesv2Client.send(new client_sesv2_1.PutEmailIdentityConfigurationSetAttributesCommand({
65
68
  EmailIdentity: domainName,
66
- // ConfigurationSetName can be set to undefined to remove
67
- // the default configuration set from the identity.
68
69
  ConfigurationSetName: defaultConfigurationSetName,
69
- })
70
- .promise();
71
- console.log(`sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(response3)}`);
70
+ }));
71
+ console.log(`sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(putEmailIdentityConfigResp)}`);
72
72
  return {
73
73
  PhysicalResourceId: `SesDomain${domainName}`,
74
74
  Data: {
@@ -78,5 +78,5 @@ const sesDomainHandler = async (event) => {
78
78
  };
79
79
  }
80
80
  };
81
- exports.sesDomainHandler = sesDomainHandler;
82
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type * as _AWS from \"aws-sdk\"\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: Record<string, string>\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\n// Type to mach CloudFormation AWS::Route53::RecordSetGroup RecordSet\ninterface RecordSetProperty {\n  Name: string\n  Type: string\n  ResourceRecords: string[]\n  TTL: string\n}\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const sesDomainHandler: OnEventHandler = async (event) => {\n  const AWS = require(\"aws-sdk\")\n\n  const ttl = \"1800\"\n\n  const ses: AWS.SES = new AWS.SES() as _AWS.SES\n  const sesv2: AWS.SESV2 = new AWS.SESV2() as _AWS.SESV2\n\n  const domainName = event.ResourceProperties[\"DomainName\"]\n  const includeVerificationRecord =\n    event.ResourceProperties[\"IncludeVerificationRecord\"] == \"true\"\n\n  const defaultConfigurationSetName =\n    event.ResourceProperties[\"DefaultConfigurationSetName\"]\n\n  if (!includeVerificationRecord) {\n    console.log(\"Excluding verification TXT record\")\n  }\n\n  function createRoute53RecordSets(\n    verificationToken: string,\n    dkimTokens: string[],\n  ) {\n    const records: RecordSetProperty[] = []\n\n    if (includeVerificationRecord) {\n      records.push({\n        Name: `_amazonses.${domainName}.`,\n        Type: \"TXT\",\n        ResourceRecords: [`\"${verificationToken}\"`],\n        TTL: ttl,\n      })\n    }\n\n    for (const token of dkimTokens) {\n      records.push({\n        Name: `${token}._domainkey.${domainName}.`,\n        Type: \"CNAME\",\n        ResourceRecords: [`${token}.dkim.amazonses.com.`],\n        TTL: ttl,\n      })\n    }\n    return records\n  }\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const response = await ses\n        .deleteIdentity({ Identity: domainName })\n        .promise()\n      console.log(`ses.deleteIdentity: ${JSON.stringify(response)}`)\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n    case \"Update\":\n      // Idempotent.\n      const response1 = await ses\n        .verifyDomainIdentity({\n          Domain: domainName,\n        })\n        .promise()\n\n      console.log(`ses.verifyDomainIdentity: ${JSON.stringify(response1)}`)\n      const verificationToken = response1[\"VerificationToken\"]\n\n      // Idempotent.\n      const response2 = await ses\n        .verifyDomainDkim({ Domain: domainName })\n        .promise()\n      console.log(`ses.verifyDomainDkim: ${JSON.stringify(response2)}`)\n      const dkimTokens = response2[\"DkimTokens\"]\n\n      // Idempotent.\n      const response3 = await sesv2\n        .putEmailIdentityConfigurationSetAttributes({\n          EmailIdentity: domainName,\n          // ConfigurationSetName can be set to undefined to remove\n          // the default configuration set from the identity.\n          ConfigurationSetName: defaultConfigurationSetName,\n        })\n        .promise()\n      console.log(\n        `sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(\n          response3,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: `SesDomain${domainName}`,\n        Data: {\n          Route53RecordSets: createRoute53RecordSets(\n            verificationToken,\n            dkimTokens,\n          ),\n          VerificationToken: verificationToken,\n        },\n      }\n  }\n}\n"]}
81
+ exports.handler = handler;
82
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SESClient,\n  VerifyDomainIdentityCommand,\n  VerifyDomainDkimCommand,\n  DeleteIdentityCommand,\n} from \"@aws-sdk/client-ses\"\n\nimport {\n  SESv2Client,\n  PutEmailIdentityConfigurationSetAttributesCommand,\n} from \"@aws-sdk/client-sesv2\"\n\ntype OnEventHandler = (event: {\n  PhysicalResourceId?: string\n  RequestType: \"Create\" | \"Update\" | \"Delete\"\n  ResourceProperties: Record<string, string>\n}) => Promise<{\n  PhysicalResourceId?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  Data?: Record<string, any>\n}>\n\n// Type to mach CloudFormation AWS::Route53::RecordSetGroup RecordSet\ninterface RecordSetProperty {\n  Name: string\n  Type: string\n  ResourceRecords: string[]\n  TTL: string\n}\n\nexport const handler: OnEventHandler = async (event) => {\n  const sesClient = new SESClient()\n  const sesv2Client = new SESv2Client()\n\n  const ttl = \"1800\"\n\n  const domainName = event.ResourceProperties[\"DomainName\"]\n  const includeVerificationRecord =\n    event.ResourceProperties[\"IncludeVerificationRecord\"] == \"true\"\n\n  const defaultConfigurationSetName =\n    event.ResourceProperties[\"DefaultConfigurationSetName\"]\n\n  if (!includeVerificationRecord) {\n    console.log(\"Excluding verification TXT record\")\n  }\n\n  function createRoute53RecordSets(\n    verificationToken: string,\n    dkimTokens: string[],\n  ) {\n    const records: RecordSetProperty[] = []\n\n    if (includeVerificationRecord) {\n      records.push({\n        Name: `_amazonses.${domainName}.`,\n        Type: \"TXT\",\n        ResourceRecords: [`\"${verificationToken}\"`],\n        TTL: ttl,\n      })\n    }\n\n    for (const token of dkimTokens) {\n      records.push({\n        Name: `${token}._domainkey.${domainName}.`,\n        Type: \"CNAME\",\n        ResourceRecords: [`${token}.dkim.amazonses.com.`],\n        TTL: ttl,\n      })\n    }\n    return records\n  }\n\n  switch (event.RequestType) {\n    case \"Delete\":\n      const deleteIdentityResp = await sesClient.send(\n        new DeleteIdentityCommand({\n          Identity: domainName,\n        }),\n      )\n      console.log(`ses.deleteIdentity: ${JSON.stringify(deleteIdentityResp)}`)\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n      }\n\n    case \"Create\":\n    case \"Update\":\n      // Idempotent.\n      const verifyDomainIdentityResp = await sesClient.send(\n        new VerifyDomainIdentityCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainIdentity: ${JSON.stringify(verifyDomainIdentityResp)}`,\n      )\n      const verificationToken = verifyDomainIdentityResp.VerificationToken\n      if (!verificationToken) {\n        throw new Error(\"Verification token not returned\")\n      }\n\n      // Idempotent.\n      const verifyDomainDkimResp = await sesClient.send(\n        new VerifyDomainDkimCommand({\n          Domain: domainName,\n        }),\n      )\n      console.log(\n        `ses.verifyDomainDkim: ${JSON.stringify(verifyDomainDkimResp)}`,\n      )\n      const dkimTokens = verifyDomainDkimResp.DkimTokens ?? []\n      if (!dkimTokens) {\n        throw new Error(\"DKIM tokens not returned\")\n      }\n\n      // Idempotent.\n      const putEmailIdentityConfigResp = await sesv2Client.send(\n        new PutEmailIdentityConfigurationSetAttributesCommand({\n          EmailIdentity: domainName,\n          ConfigurationSetName: defaultConfigurationSetName,\n        }),\n      )\n      console.log(\n        `sesv2.putEmailIdentityConfigurationSetAttributes ${JSON.stringify(\n          putEmailIdentityConfigResp,\n        )}`,\n      )\n\n      return {\n        PhysicalResourceId: `SesDomain${domainName}`,\n        Data: {\n          Route53RecordSets: createRoute53RecordSets(\n            verificationToken,\n            dkimTokens,\n          ),\n          VerificationToken: verificationToken,\n        },\n      }\n  }\n}\n"]}