@liflig/cdk 2.21.2 → 2.21.4
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.
- package/README.md +18 -2
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +1 -1
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +46 -35
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +6 -6
- package/lib/index.d.ts +1 -4
- package/lib/index.js +2 -6
- package/lib/ses/configurationsetsnsdestination/handler.d.ts +1 -1
- package/lib/ses/configurationsetsnsdestination/handler.js +15 -25
- package/lib/ses/configurationsetsnsdestination/index.js +7 -7
- package/lib/ses/sesdomain/handler.d.ts +1 -1
- package/lib/ses/sesdomain/handler.js +31 -31
- package/lib/ses/sesdomain/index.js +6 -6
- package/lib/ses/sesverifyemail/handler.d.ts +1 -1
- package/lib/ses/sesverifyemail/handler.js +12 -10
- package/lib/ses/sesverifyemail/index.js +6 -6
- package/package.json +14 -3
- package/lib/cdk-deploy/cdk-deploy.d.ts +0 -63
- package/lib/cdk-deploy/cdk-deploy.js +0 -175
- package/lib/cdk-deploy/index.d.ts +0 -1
- package/lib/cdk-deploy/index.js +0 -6
- package/lib/cdk-deploy/start-deploy-handler.d.ts +0 -8
- package/lib/cdk-deploy/start-deploy-handler.js +0 -72
- package/lib/cdk-deploy/status-handler.d.ts +0 -6
- package/lib/cdk-deploy/status-handler.js +0 -83
- package/lib/ecs-update-image/artifact-status.d.ts +0 -39
- package/lib/ecs-update-image/artifact-status.js +0 -41
- package/lib/ecs-update-image/ecs-update-image.d.ts +0 -41
- package/lib/ecs-update-image/ecs-update-image.js +0 -98
- package/lib/ecs-update-image/index.d.ts +0 -3
- package/lib/ecs-update-image/index.js +0 -10
- package/lib/ecs-update-image/start-deploy-handler.d.ts +0 -6
- package/lib/ecs-update-image/start-deploy-handler.js +0 -104
- package/lib/ecs-update-image/status-handler.d.ts +0 -11
- package/lib/ecs-update-image/status-handler.js +0 -74
- package/lib/ecs-update-image/tag.d.ts +0 -47
- package/lib/ecs-update-image/tag.js +0 -67
- package/lib/pipelines/conventions.d.ts +0 -14
- package/lib/pipelines/conventions.js +0 -24
- package/lib/pipelines/deploy-env.d.ts +0 -18
- package/lib/pipelines/deploy-env.js +0 -96
- package/lib/pipelines/index.d.ts +0 -2
- package/lib/pipelines/index.js +0 -8
- package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +0 -13
- package/lib/pipelines/liflig-cdk-deployer-deps.js +0 -35
- package/lib/pipelines/pipeline.d.ts +0 -78
- package/lib/pipelines/pipeline.js +0 -224
package/README.md
CHANGED
|
@@ -32,18 +32,34 @@ are not yet resolved. Some relevant information:
|
|
|
32
32
|
|
|
33
33
|
## Testing library changes before releasing
|
|
34
34
|
|
|
35
|
-
1
|
|
35
|
+
### Alternative 1: Install liflig-cdk from tarball
|
|
36
|
+
|
|
37
|
+
1. Assemble artifact, which emits a tarball
|
|
36
38
|
|
|
37
39
|
```bash
|
|
38
40
|
npm pack
|
|
39
41
|
```
|
|
40
42
|
|
|
41
|
-
|
|
43
|
+
2. Install the library in an application from a tarball
|
|
42
44
|
|
|
43
45
|
```bash
|
|
44
46
|
npm install /path/to/liflig-cdk/liflig-cdk-0.0.0-development.tgz
|
|
45
47
|
```
|
|
46
48
|
|
|
49
|
+
### Alternative 2: Install liflig-cdk from a Git branch
|
|
50
|
+
|
|
51
|
+
1. Navigate to the project where you want to test the changes
|
|
52
|
+
|
|
53
|
+
```bash
|
|
54
|
+
cd /path/to/other/project
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
2. Install the library in an application from a Git branch
|
|
58
|
+
|
|
59
|
+
```bash
|
|
60
|
+
npm install "https://github.com/capralifecycle/liflig-cdk.git#my-feature-branch"
|
|
61
|
+
```
|
|
62
|
+
|
|
47
63
|
Note: `npm link` cannot be used, since it will lead to multiple
|
|
48
64
|
declarations of the same classes from CDK, breaking the `instanceof`
|
|
49
65
|
operator.
|
|
@@ -1,63 +1,74 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
4
|
-
|
|
5
|
-
const
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
const
|
|
3
|
+
exports.handler = void 0;
|
|
4
|
+
const client_s3_1 = require("@aws-sdk/client-s3");
|
|
5
|
+
const client_codepipeline_1 = require("@aws-sdk/client-codepipeline");
|
|
6
|
+
const lib_storage_1 = require("@aws-sdk/lib-storage");
|
|
7
|
+
const handler = async (event, context) => {
|
|
8
|
+
const s3Client = new client_s3_1.S3Client();
|
|
9
|
+
const codepipelineClient = new client_codepipeline_1.CodePipelineClient();
|
|
9
10
|
const jobId = event["CodePipeline.job"].id;
|
|
10
11
|
try {
|
|
11
12
|
const userParametersRaw = event["CodePipeline.job"].data.actionConfiguration.configuration
|
|
12
13
|
.UserParameters;
|
|
13
14
|
console.log("User parameters", userParametersRaw);
|
|
15
|
+
console.log("Parsing user parameters");
|
|
14
16
|
const userParameters = JSON.parse(userParametersRaw);
|
|
15
|
-
|
|
16
|
-
|
|
17
|
+
console.log("Fetching reference data from S3");
|
|
18
|
+
const getReferenceDataResp = await s3Client.send(new client_s3_1.GetObjectCommand({
|
|
17
19
|
Bucket: userParameters.bucketName,
|
|
18
20
|
Key: userParameters.objectKey,
|
|
19
|
-
})
|
|
20
|
-
|
|
21
|
-
|
|
21
|
+
}));
|
|
22
|
+
const referenceData = await getReferenceDataResp.Body.transformToString("utf-8");
|
|
23
|
+
console.log("Parsing reference data from S3");
|
|
22
24
|
const cloudAssemblyReference = JSON.parse(referenceData);
|
|
23
25
|
const outputArtifact = event["CodePipeline.job"].data.outputArtifacts[0];
|
|
24
26
|
const s3Loc = outputArtifact.location.s3Location;
|
|
25
|
-
|
|
26
|
-
|
|
27
|
+
console.log("Fetching Cloud Assembly ZIP data from S3");
|
|
28
|
+
const cloudAssemblyZipData = await s3Client.send(new client_s3_1.GetObjectCommand({
|
|
27
29
|
Bucket: cloudAssemblyReference.cloudAssemblyBucketName,
|
|
28
30
|
Key: cloudAssemblyReference.cloudAssemblyBucketKey,
|
|
29
|
-
})
|
|
30
|
-
|
|
31
|
-
console.log("
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
})
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
31
|
+
}));
|
|
32
|
+
console.log("Content-Length of Cloud Assembly (bytes)", cloudAssemblyZipData.ContentLength);
|
|
33
|
+
console.log("Creating authenticated S3 client for artifact upload");
|
|
34
|
+
const artifactCreds = event["CodePipeline.job"].data.artifactCredentials;
|
|
35
|
+
if (artifactCreds === undefined) {
|
|
36
|
+
console.error("No artifact credentials found in job event");
|
|
37
|
+
}
|
|
38
|
+
const authedS3Client = new client_s3_1.S3Client({
|
|
39
|
+
credentials: artifactCreds,
|
|
40
|
+
});
|
|
41
|
+
console.log(`Uploading Cloud Assembly ZIP to s3://${s3Loc.bucketName}/${s3Loc.objectKey}`);
|
|
42
|
+
const upload = new lib_storage_1.Upload({
|
|
43
|
+
client: authedS3Client,
|
|
44
|
+
params: {
|
|
45
|
+
Bucket: s3Loc.bucketName,
|
|
46
|
+
Key: s3Loc.objectKey,
|
|
47
|
+
Body: cloudAssemblyZipData.Body,
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
upload.on("httpUploadProgress", (progress) => {
|
|
51
|
+
console.log(progress);
|
|
52
|
+
});
|
|
53
|
+
await upload.done();
|
|
54
|
+
console.log("Sending success result to CodePipeline");
|
|
55
|
+
await codepipelineClient.send(new client_codepipeline_1.PutJobSuccessResultCommand({
|
|
43
56
|
jobId,
|
|
44
|
-
})
|
|
45
|
-
.promise();
|
|
57
|
+
}));
|
|
46
58
|
console.log("Success");
|
|
47
59
|
}
|
|
48
60
|
catch (e) {
|
|
49
|
-
|
|
50
|
-
|
|
61
|
+
console.log("Sending failure result to CodePipeline");
|
|
62
|
+
await codepipelineClient.send(new client_codepipeline_1.PutJobFailureResultCommand({
|
|
51
63
|
failureDetails: {
|
|
52
64
|
message: JSON.stringify(e),
|
|
53
65
|
type: "JobFailed",
|
|
54
66
|
externalExecutionId: context.awsRequestId,
|
|
55
67
|
},
|
|
56
68
|
jobId,
|
|
57
|
-
})
|
|
58
|
-
.promise();
|
|
69
|
+
}));
|
|
59
70
|
console.error("Failed", e);
|
|
60
71
|
}
|
|
61
72
|
};
|
|
62
|
-
exports.
|
|
63
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
73
|
+
exports.handler = handler;
|
|
74
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly-lookup-handler.js","sourceRoot":"","sources":["../../src/cdk-pipelines/cloud-assembly-lookup-handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-var-requires */\nimport type { Handler } from \"aws-lambda\"\nimport { GetObjectCommand, S3Client } from \"@aws-sdk/client-s3\"\nimport {\n  CodePipelineClient,\n  PutJobFailureResultCommand,\n  PutJobSuccessResultCommand,\n} from \"@aws-sdk/client-codepipeline\"\nimport { Upload } from \"@aws-sdk/lib-storage\"\n\n// Relevant fields from\n// https://docs.amazonaws.cn/en_us/lambda/latest/dg/services-codepipeline.html\ninterface CodePipelineEvent {\n  \"CodePipeline.job\": {\n    id: string\n    data: {\n      actionConfiguration: {\n        configuration: {\n          // JSON\n          UserParameters: string\n        }\n      }\n      outputArtifacts: {\n        name: string\n        location: {\n          s3Location: {\n            bucketName: string\n            objectKey: string\n          }\n        }\n      }[]\n      artifactCredentials: {\n        accessKeyId: string\n        secretAccessKey: string\n        sessionToken: string\n      }\n    }\n  }\n}\n\ninterface CloudAssemblyReference {\n  cloudAssemblyBucketName: string\n  cloudAssemblyBucketKey: string\n}\n\nexport interface CloudAssemblyLookupUserParameters {\n  bucketName: string\n  objectKey: string\n}\n\nexport const handler: Handler = async (event: CodePipelineEvent, context) => {\n  const s3Client = new S3Client()\n  const codepipelineClient = new CodePipelineClient()\n\n  const jobId = event[\"CodePipeline.job\"].id\n\n  try {\n    const userParametersRaw =\n      event[\"CodePipeline.job\"].data.actionConfiguration.configuration\n        .UserParameters\n\n    console.log(\"User parameters\", userParametersRaw)\n\n    console.log(\"Parsing user parameters\")\n    const userParameters = JSON.parse(\n      userParametersRaw,\n    ) as CloudAssemblyLookupUserParameters\n\n    console.log(\"Fetching reference data from S3\")\n    const getReferenceDataResp = await s3Client.send(\n      new GetObjectCommand({\n        Bucket: userParameters.bucketName,\n        Key: userParameters.objectKey,\n      }),\n    )\n    const referenceData =\n      await getReferenceDataResp.Body!.transformToString(\"utf-8\")\n\n    console.log(\"Parsing reference data from S3\")\n    const cloudAssemblyReference = JSON.parse(\n      referenceData,\n    ) as CloudAssemblyReference\n\n    const outputArtifact = event[\"CodePipeline.job\"].data.outputArtifacts[0]\n    const s3Loc = outputArtifact.location.s3Location\n\n    console.log(\"Fetching Cloud Assembly ZIP data from S3\")\n    const cloudAssemblyZipData = await s3Client.send(\n      new GetObjectCommand({\n        Bucket: cloudAssemblyReference.cloudAssemblyBucketName,\n        Key: cloudAssemblyReference.cloudAssemblyBucketKey,\n      }),\n    )\n    console.log(\n      \"Content-Length of Cloud Assembly (bytes)\",\n      cloudAssemblyZipData.ContentLength,\n    )\n\n    console.log(\"Creating authenticated S3 client for artifact upload\")\n    const artifactCreds = event[\"CodePipeline.job\"].data.artifactCredentials\n    if (artifactCreds === undefined) {\n      console.error(\"No artifact credentials found in job event\")\n    }\n    const authedS3Client = new S3Client({\n      credentials: artifactCreds,\n    })\n\n    console.log(\n      `Uploading Cloud Assembly ZIP to s3://${s3Loc.bucketName}/${s3Loc.objectKey}`,\n    )\n\n    const upload = new Upload({\n      client: authedS3Client,\n      params: {\n        Bucket: s3Loc.bucketName,\n        Key: s3Loc.objectKey,\n        Body: cloudAssemblyZipData.Body,\n      },\n    })\n\n    upload.on(\"httpUploadProgress\", (progress) => {\n      console.log(progress)\n    })\n\n    await upload.done()\n\n    console.log(\"Sending success result to CodePipeline\")\n    await codepipelineClient.send(\n      new PutJobSuccessResultCommand({\n        jobId,\n      }),\n    )\n    console.log(\"Success\")\n  } catch (e) {\n    console.log(\"Sending failure result to CodePipeline\")\n    await codepipelineClient.send(\n      new PutJobFailureResultCommand({\n        failureDetails: {\n          message: JSON.stringify(e),\n          type: \"JobFailed\",\n          externalExecutionId: context.awsRequestId,\n        },\n        jobId,\n      }),\n    )\n    console.error(\"Failed\", e)\n  }\n}\n"]}
|
|
@@ -13,8 +13,8 @@ const pipelines = require("aws-cdk-lib/pipelines");
|
|
|
13
13
|
const fs = require("fs");
|
|
14
14
|
const path = require("path");
|
|
15
15
|
const artefact_bucket_1 = require("../griid/artefact-bucket");
|
|
16
|
-
const cloud_assembly_lookup_handler_1 = require("./cloud-assembly-lookup-handler");
|
|
17
16
|
const slack_notification_1 = require("./slack-notification");
|
|
17
|
+
const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
|
|
18
18
|
/**
|
|
19
19
|
* CDK Pipeline for Liflig.
|
|
20
20
|
*
|
|
@@ -147,12 +147,12 @@ class LifligCdkPipeline extends constructs.Construct {
|
|
|
147
147
|
return undefined;
|
|
148
148
|
}
|
|
149
149
|
cloudAssemblyStage(cloudAssemblyArtifact, cdkBucket, pipelineName) {
|
|
150
|
-
const cloudAssemblyLookupFn = new
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
runtime: lambda.Runtime.NODEJS_16_X,
|
|
150
|
+
const cloudAssemblyLookupFn = new aws_lambda_nodejs_1.NodejsFunction(this, "CloudAssemblyLookupFn", {
|
|
151
|
+
entry: require.resolve("./cloud-assembly-lookup-handler"),
|
|
152
|
+
runtime: lambda.Runtime.NODEJS_18_X,
|
|
154
153
|
timeout: cdk.Duration.minutes(1),
|
|
155
154
|
memorySize: 512,
|
|
155
|
+
awsSdkConnectionReuse: false,
|
|
156
156
|
});
|
|
157
157
|
cdkBucket.grantReadWrite(cloudAssemblyLookupFn);
|
|
158
158
|
const userParameters = {
|
|
@@ -229,4 +229,4 @@ class LifligCdkPipeline extends constructs.Construct {
|
|
|
229
229
|
}
|
|
230
230
|
}
|
|
231
231
|
exports.LifligCdkPipeline = LifligCdkPipeline;
|
|
232
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"liflig-cdk-pipeline.js","sourceRoot":"","sources":["../../src/cdk-pipelines/liflig-cdk-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as codepipeline from \"aws-cdk-lib/aws-codepipeline\"\nimport * as codepipelineActions from \"aws-cdk-lib/aws-codepipeline-actions\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as targets from \"aws-cdk-lib/aws-events-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as pipelines from \"aws-cdk-lib/pipelines\"\nimport * as fs from \"fs\"\nimport * as path from \"path\"\nimport { getGriidArtefactBucket } from \"../griid/artefact-bucket\"\nimport {\n  cloudAssemblyLookupHandler,\n  CloudAssemblyLookupUserParameters,\n} from \"./cloud-assembly-lookup-handler\"\nimport { SlackNotification, SlackNotificationProps } from \"./slack-notification\"\n\nexport interface LifligCdkPipelineProps {\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   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Type of uploaded artifact. This changes the behaviour of the\n   * pipeline and what kind of process it performs.\n   *\n   * Two types are supported:\n   *\n   *   - cdk-source: The uploaded artifact represents a CDK application\n   *     as source code. A build step will compile this into a\n   *     CDK Cloud Assembly.\n   *\n   *     As part of synthesizing this into a CDK Cloud Assembly,\n   *     a file \"variables.json\" will be written for the\n   *     CDK application to parameterize the build if any\n   *     variables are found in the pipeline source.\n   *\n   *   - cloud-assembly: The uploaded artifact represents a\n   *     CDK Cloud Assembly which is ready for deployment.\n   *\n   *     This does not support reading variables at the current time\n   *     since CDK Pipelines don't support parameterized deploys.\n   *     See https://github.com/aws/aws-cdk/issues/9560\n   */\n  sourceType: \"cdk-source\" | \"cloud-assembly\"\n  /**\n   * The namespace used for parameters in Parameter Store.\n   *\n   * Only relevant for sourceType of \"cdk-soruce\".\n   *\n   * @default default\n   */\n  parametersNamespace?: string\n}\n\n/**\n * CDK Pipeline for Liflig.\n *\n * Avoid putting multiple pipelines in a stack, since the pipeline\n * will also keep the hosting stack up-to-date.\n *\n * The pipeline is executed by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n * For upload type \"cdk-source\":\n *\n *   - cdk-source.json holding a pointer to the active CDK source\n *     that should be used. Schema:\n *\n *     {\n *       bucketName: string\n *       bucketKey: string\n *     }\n *\n *   - variables*.json which can be zero or more files\n *     with string-string map holding variables that will\n *     be written to variables.json and can be read by the\n *     the CDK application during synthesize.\n *\n * For upload type \"cloud-assembly\":\n *\n *   - cloud-assembly.json holding a pointer to the active\n *     CDK Cloud Assembly that should be used: Schema:\n *\n *     {\n *       cloudAssemblyBucketName: string\n *       cloudAssemblyBucketKey: string\n *     }\n *\n * Variables enables separation of IaC code and application code if\n * they are not colocated in the same repository.\n */\nexport class LifligCdkPipeline extends constructs.Construct {\n  /**\n   * Path on S3 for pipeline configuration.\n   */\n  static pipelineS3Prefix(pipelineName: string): string {\n    return `pipelines/${pipelineName}/`\n  }\n\n  /**\n   * Key in S3 bucket used to trigger pipeline.\n   *\n   * This is an empty file within the pipeline path.\n   */\n  static pipelineS3TriggerKey(pipelineName: string): string {\n    return `pipelines/${pipelineName}/trigger`\n  }\n\n  public readonly cdkPipeline: pipelines.CodePipeline\n  public readonly codePipeline: codepipeline.Pipeline\n  public readonly artifactsBucket: s3.IBucket\n  public readonly triggerObjectKey: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: LifligCdkPipelineProps,\n  ) {\n    super(scope, id)\n\n    this.artifactsBucket = props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const cloudAssemblyArtifact = new codepipeline.Artifact()\n\n    let synth: pipelines.IFileSetProducer\n    let stages: codepipeline.StageProps[]\n\n    switch (props.sourceType) {\n      case \"cloud-assembly\":\n        const cloudAssembly = this.cloudAssemblyStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n        )\n        synth = cloudAssembly.synth\n        stages = cloudAssembly.stages\n        break\n      case \"cdk-source\":\n        const cdkSource = this.cdkSourceStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n          props.parametersNamespace ?? \"default\",\n        )\n        synth = cdkSource.synth\n        stages = cdkSource.stages\n        break\n    }\n\n    const dummyArtifact = new codepipeline.Artifact()\n\n    this.triggerObjectKey = LifligCdkPipeline.pipelineS3TriggerKey(\n      props.pipelineName,\n    )\n\n    this.codePipeline = new codepipeline.Pipeline(this, \"CodePipeline\", {\n      pipelineName: props.pipelineName,\n      stages: [\n        {\n          stageName: \"Source\",\n          actions: [\n            new codepipelineActions.S3SourceAction({\n              actionName: \"source\",\n              bucket: this.artifactsBucket,\n              trigger: codepipelineActions.S3Trigger.NONE,\n              bucketKey: this.triggerObjectKey,\n              output: dummyArtifact,\n            }),\n          ],\n        },\n        ...stages,\n      ],\n      restartExecutionOnUpdate: true,\n    })\n\n    new events.Rule(this, \"PipelineTrigger\", {\n      eventPattern: {\n        source: [\"aws.s3\"],\n        detailType: [\"Object Created\"],\n        detail: {\n          bucket: {\n            name: [this.artifactsBucket.bucketName],\n          },\n          object: {\n            key: [this.triggerObjectKey],\n          },\n        },\n      },\n      targets: [new targets.CodePipeline(this.codePipeline)],\n    })\n\n    this.cdkPipeline = new pipelines.CodePipeline(this, \"CdkPipeline\", {\n      synth,\n      useChangeSets: false,\n      codePipeline: this.codePipeline,\n    })\n  }\n\n  private static getAwsCdkPackageJsonFile(): string | undefined {\n    // Also look up the tree a bit to handle yarn workspaces.\n    const candidates = [\n      path.join(process.cwd(), \"node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../../node_modules/aws-cdk/package.json\"),\n    ]\n\n    for (const candidate of candidates) {\n      if (fs.existsSync(candidate)) {\n        return candidate\n      }\n    }\n\n    return undefined\n  }\n\n  private cloudAssemblyStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const cloudAssemblyLookupFn = new lambda.Function(\n      this,\n      \"CloudAssemblyLookupFn\",\n      {\n        code: new lambda.InlineCode(\n          `exports.handler = ${cloudAssemblyLookupHandler.toString()};`,\n        ),\n        handler: \"index.handler\",\n        runtime: lambda.Runtime.NODEJS_16_X,\n        timeout: cdk.Duration.minutes(1),\n        memorySize: 512,\n      },\n    )\n\n    cdkBucket.grantReadWrite(cloudAssemblyLookupFn)\n\n    const userParameters: CloudAssemblyLookupUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      objectKey: `pipelines/${pipelineName}/cloud-assembly.json`,\n    }\n\n    const synth = pipelines.CodePipelineFileSet.fromArtifact(\n      cloudAssemblyArtifact,\n    )\n\n    const stages = [\n      {\n        stageName: \"PrepareCloudAssembly\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"cloud-assembly-lookup\",\n            lambda: cloudAssemblyLookupFn,\n            outputs: [cloudAssemblyArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  private cdkSourceStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n    parametersNamespace: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const prepareCdkSourceFn = new lambda.Function(this, \"PrepareCdkSourceFn\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dirname, \"../../assets/prepare-cdk-source-lambda\"),\n      ),\n      handler: \"index.handler\",\n      // Using python instead if NodeJS due to zip-support in stdlib.\n      runtime: lambda.Runtime.PYTHON_3_8,\n      timeout: cdk.Duration.minutes(1),\n      memorySize: 512,\n    })\n\n    const account = cdk.Stack.of(this).account\n    const region = cdk.Stack.of(this).region\n\n    prepareCdkSourceFn.grantPrincipal.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\"ssm:GetParametersByPath\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      }),\n    )\n\n    cdkBucket.grantReadWrite(prepareCdkSourceFn)\n\n    const cdkSourceArtifact = new codepipeline.Artifact()\n\n    const userParameters: PrepareCdkSourceUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      prefix: `pipelines/${pipelineName}/`,\n      parametersNamespace: parametersNamespace,\n    }\n\n    const synth = new pipelines.ShellStep(\"GenerateCloudAssembly\", {\n      input: pipelines.CodePipelineFileSet.fromArtifact(cdkSourceArtifact),\n      installCommands: [\"npm ci\"],\n      commands: [\"npx cdk synth\"],\n    })\n    const stages = [\n      {\n        stageName: \"PrepareCdkSource\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"prepare-cdk-source\",\n            lambda: prepareCdkSourceFn,\n            outputs: [cdkSourceArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  addSlackNotification(\n    props: Omit<\n      SlackNotificationProps,\n      \"pipeline\" | \"artifactsBucket\" | \"triggerObjectKey\"\n    >,\n  ): void {\n    new SlackNotification(this, \"Slack\", {\n      pipeline: this.codePipeline,\n      artifactsBucket: this.artifactsBucket,\n      triggerObjectKey: this.triggerObjectKey,\n      ...props,\n    })\n  }\n}\n\ninterface PrepareCdkSourceUserParameters {\n  bucketName: string\n  prefix: string\n  parametersNamespace: string\n}\n"]}
|
|
232
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"liflig-cdk-pipeline.js","sourceRoot":"","sources":["../../src/cdk-pipelines/liflig-cdk-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as codepipeline from \"aws-cdk-lib/aws-codepipeline\"\nimport * as codepipelineActions from \"aws-cdk-lib/aws-codepipeline-actions\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as targets from \"aws-cdk-lib/aws-events-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as pipelines from \"aws-cdk-lib/pipelines\"\nimport * as fs from \"fs\"\nimport * as path from \"path\"\nimport { getGriidArtefactBucket } from \"../griid/artefact-bucket\"\nimport { CloudAssemblyLookupUserParameters } from \"./cloud-assembly-lookup-handler\"\nimport { SlackNotification, SlackNotificationProps } from \"./slack-notification\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\n\nexport interface LifligCdkPipelineProps {\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   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Type of uploaded artifact. This changes the behaviour of the\n   * pipeline and what kind of process it performs.\n   *\n   * Two types are supported:\n   *\n   *   - cdk-source: The uploaded artifact represents a CDK application\n   *     as source code. A build step will compile this into a\n   *     CDK Cloud Assembly.\n   *\n   *     As part of synthesizing this into a CDK Cloud Assembly,\n   *     a file \"variables.json\" will be written for the\n   *     CDK application to parameterize the build if any\n   *     variables are found in the pipeline source.\n   *\n   *   - cloud-assembly: The uploaded artifact represents a\n   *     CDK Cloud Assembly which is ready for deployment.\n   *\n   *     This does not support reading variables at the current time\n   *     since CDK Pipelines don't support parameterized deploys.\n   *     See https://github.com/aws/aws-cdk/issues/9560\n   */\n  sourceType: \"cdk-source\" | \"cloud-assembly\"\n  /**\n   * The namespace used for parameters in Parameter Store.\n   *\n   * Only relevant for sourceType of \"cdk-soruce\".\n   *\n   * @default default\n   */\n  parametersNamespace?: string\n}\n\n/**\n * CDK Pipeline for Liflig.\n *\n * Avoid putting multiple pipelines in a stack, since the pipeline\n * will also keep the hosting stack up-to-date.\n *\n * The pipeline is executed by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n * For upload type \"cdk-source\":\n *\n *   - cdk-source.json holding a pointer to the active CDK source\n *     that should be used. Schema:\n *\n *     {\n *       bucketName: string\n *       bucketKey: string\n *     }\n *\n *   - variables*.json which can be zero or more files\n *     with string-string map holding variables that will\n *     be written to variables.json and can be read by the\n *     the CDK application during synthesize.\n *\n * For upload type \"cloud-assembly\":\n *\n *   - cloud-assembly.json holding a pointer to the active\n *     CDK Cloud Assembly that should be used: Schema:\n *\n *     {\n *       cloudAssemblyBucketName: string\n *       cloudAssemblyBucketKey: string\n *     }\n *\n * Variables enables separation of IaC code and application code if\n * they are not colocated in the same repository.\n */\nexport class LifligCdkPipeline extends constructs.Construct {\n  /**\n   * Path on S3 for pipeline configuration.\n   */\n  static pipelineS3Prefix(pipelineName: string): string {\n    return `pipelines/${pipelineName}/`\n  }\n\n  /**\n   * Key in S3 bucket used to trigger pipeline.\n   *\n   * This is an empty file within the pipeline path.\n   */\n  static pipelineS3TriggerKey(pipelineName: string): string {\n    return `pipelines/${pipelineName}/trigger`\n  }\n\n  public readonly cdkPipeline: pipelines.CodePipeline\n  public readonly codePipeline: codepipeline.Pipeline\n  public readonly artifactsBucket: s3.IBucket\n  public readonly triggerObjectKey: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: LifligCdkPipelineProps,\n  ) {\n    super(scope, id)\n\n    this.artifactsBucket = props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const cloudAssemblyArtifact = new codepipeline.Artifact()\n\n    let synth: pipelines.IFileSetProducer\n    let stages: codepipeline.StageProps[]\n\n    switch (props.sourceType) {\n      case \"cloud-assembly\":\n        const cloudAssembly = this.cloudAssemblyStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n        )\n        synth = cloudAssembly.synth\n        stages = cloudAssembly.stages\n        break\n      case \"cdk-source\":\n        const cdkSource = this.cdkSourceStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n          props.parametersNamespace ?? \"default\",\n        )\n        synth = cdkSource.synth\n        stages = cdkSource.stages\n        break\n    }\n\n    const dummyArtifact = new codepipeline.Artifact()\n\n    this.triggerObjectKey = LifligCdkPipeline.pipelineS3TriggerKey(\n      props.pipelineName,\n    )\n\n    this.codePipeline = new codepipeline.Pipeline(this, \"CodePipeline\", {\n      pipelineName: props.pipelineName,\n      stages: [\n        {\n          stageName: \"Source\",\n          actions: [\n            new codepipelineActions.S3SourceAction({\n              actionName: \"source\",\n              bucket: this.artifactsBucket,\n              trigger: codepipelineActions.S3Trigger.NONE,\n              bucketKey: this.triggerObjectKey,\n              output: dummyArtifact,\n            }),\n          ],\n        },\n        ...stages,\n      ],\n      restartExecutionOnUpdate: true,\n    })\n\n    new events.Rule(this, \"PipelineTrigger\", {\n      eventPattern: {\n        source: [\"aws.s3\"],\n        detailType: [\"Object Created\"],\n        detail: {\n          bucket: {\n            name: [this.artifactsBucket.bucketName],\n          },\n          object: {\n            key: [this.triggerObjectKey],\n          },\n        },\n      },\n      targets: [new targets.CodePipeline(this.codePipeline)],\n    })\n\n    this.cdkPipeline = new pipelines.CodePipeline(this, \"CdkPipeline\", {\n      synth,\n      useChangeSets: false,\n      codePipeline: this.codePipeline,\n    })\n  }\n\n  private static getAwsCdkPackageJsonFile(): string | undefined {\n    // Also look up the tree a bit to handle yarn workspaces.\n    const candidates = [\n      path.join(process.cwd(), \"node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../../node_modules/aws-cdk/package.json\"),\n    ]\n\n    for (const candidate of candidates) {\n      if (fs.existsSync(candidate)) {\n        return candidate\n      }\n    }\n\n    return undefined\n  }\n\n  private cloudAssemblyStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const cloudAssemblyLookupFn = new NodejsFunction(\n      this,\n      \"CloudAssemblyLookupFn\",\n      {\n        entry: require.resolve(\"./cloud-assembly-lookup-handler\"),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        timeout: cdk.Duration.minutes(1),\n        memorySize: 512,\n        awsSdkConnectionReuse: false,\n      },\n    )\n\n    cdkBucket.grantReadWrite(cloudAssemblyLookupFn)\n\n    const userParameters: CloudAssemblyLookupUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      objectKey: `pipelines/${pipelineName}/cloud-assembly.json`,\n    }\n\n    const synth = pipelines.CodePipelineFileSet.fromArtifact(\n      cloudAssemblyArtifact,\n    )\n\n    const stages = [\n      {\n        stageName: \"PrepareCloudAssembly\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"cloud-assembly-lookup\",\n            lambda: cloudAssemblyLookupFn,\n            outputs: [cloudAssemblyArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  private cdkSourceStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n    parametersNamespace: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const prepareCdkSourceFn = new lambda.Function(this, \"PrepareCdkSourceFn\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dirname, \"../../assets/prepare-cdk-source-lambda\"),\n      ),\n      handler: \"index.handler\",\n      // Using python instead if NodeJS due to zip-support in stdlib.\n      runtime: lambda.Runtime.PYTHON_3_8,\n      timeout: cdk.Duration.minutes(1),\n      memorySize: 512,\n    })\n\n    const account = cdk.Stack.of(this).account\n    const region = cdk.Stack.of(this).region\n\n    prepareCdkSourceFn.grantPrincipal.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\"ssm:GetParametersByPath\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      }),\n    )\n\n    cdkBucket.grantReadWrite(prepareCdkSourceFn)\n\n    const cdkSourceArtifact = new codepipeline.Artifact()\n\n    const userParameters: PrepareCdkSourceUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      prefix: `pipelines/${pipelineName}/`,\n      parametersNamespace: parametersNamespace,\n    }\n\n    const synth = new pipelines.ShellStep(\"GenerateCloudAssembly\", {\n      input: pipelines.CodePipelineFileSet.fromArtifact(cdkSourceArtifact),\n      installCommands: [\"npm ci\"],\n      commands: [\"npx cdk synth\"],\n    })\n    const stages = [\n      {\n        stageName: \"PrepareCdkSource\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"prepare-cdk-source\",\n            lambda: prepareCdkSourceFn,\n            outputs: [cdkSourceArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  addSlackNotification(\n    props: Omit<\n      SlackNotificationProps,\n      \"pipeline\" | \"artifactsBucket\" | \"triggerObjectKey\"\n    >,\n  ): void {\n    new SlackNotification(this, \"Slack\", {\n      pipeline: this.codePipeline,\n      artifactsBucket: this.artifactsBucket,\n      triggerObjectKey: this.triggerObjectKey,\n      ...props,\n    })\n  }\n}\n\ninterface PrepareCdkSourceUserParameters {\n  bucketName: string\n  prefix: string\n  parametersNamespace: string\n}\n"]}
|
package/lib/index.d.ts
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import * as alarms from "./alarms";
|
|
2
2
|
import * as cdkPipelines from "./cdk-pipelines";
|
|
3
3
|
import * as griid from "./griid";
|
|
4
|
-
import * as pipelines from "./pipelines";
|
|
5
4
|
import * as ses from "./ses";
|
|
6
5
|
import * as webapp from "./webapp";
|
|
7
6
|
import * as configureParameters from "./configure-parameters";
|
|
@@ -12,16 +11,14 @@ import * as rds from "./rds";
|
|
|
12
11
|
import * as platform from "./platform";
|
|
13
12
|
export { BastionHost } from "./bastion-host";
|
|
14
13
|
export * from "./build-artifacts";
|
|
15
|
-
export * from "./cdk-deploy";
|
|
16
14
|
export { CrossRegionSsmParameter } from "./cross-region-ssm-parameter";
|
|
17
|
-
export * from "./ecs-update-image";
|
|
18
15
|
export { HostedZoneWithParam } from "./hosted-zone-with-param";
|
|
19
16
|
export { createCloudAssemblySnapshot } from "./snapshots";
|
|
20
17
|
export { SsmParameterBackedResource } from "./ssm-parameter-backed-resource";
|
|
21
18
|
export { SsmParameterReader } from "./ssm-parameter-reader";
|
|
22
19
|
export { tagResources } from "./tags";
|
|
23
20
|
export { WebappDeployViaRole } from "./webapp-deploy-via-role";
|
|
24
|
-
export { alarms, cdkPipelines, griid,
|
|
21
|
+
export { alarms, cdkPipelines, griid, ses, webapp, configureParameters, ecs, loadBalancer, rds, platform, cloudTrailSlackIntegration, };
|
|
25
22
|
/**
|
|
26
23
|
* Check if we are synthesizing a snapshot by setting IS_SNAPSHOT
|
|
27
24
|
* environment variable to true.
|
package/lib/index.js
CHANGED
|
@@ -14,15 +14,13 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
exports.isSnapshot = exports.cloudTrailSlackIntegration = exports.platform = exports.rds = exports.loadBalancer = exports.ecs = exports.configureParameters = exports.webapp = exports.ses = exports.
|
|
17
|
+
exports.isSnapshot = exports.cloudTrailSlackIntegration = exports.platform = exports.rds = exports.loadBalancer = exports.ecs = exports.configureParameters = exports.webapp = exports.ses = exports.griid = exports.cdkPipelines = exports.alarms = exports.WebappDeployViaRole = exports.tagResources = exports.SsmParameterReader = exports.SsmParameterBackedResource = exports.createCloudAssemblySnapshot = exports.HostedZoneWithParam = exports.CrossRegionSsmParameter = exports.BastionHost = void 0;
|
|
18
18
|
const alarms = require("./alarms");
|
|
19
19
|
exports.alarms = alarms;
|
|
20
20
|
const cdkPipelines = require("./cdk-pipelines");
|
|
21
21
|
exports.cdkPipelines = cdkPipelines;
|
|
22
22
|
const griid = require("./griid");
|
|
23
23
|
exports.griid = griid;
|
|
24
|
-
const pipelines = require("./pipelines");
|
|
25
|
-
exports.pipelines = pipelines;
|
|
26
24
|
const ses = require("./ses");
|
|
27
25
|
exports.ses = ses;
|
|
28
26
|
const webapp = require("./webapp");
|
|
@@ -44,10 +42,8 @@ exports.platform = platform;
|
|
|
44
42
|
var bastion_host_1 = require("./bastion-host");
|
|
45
43
|
Object.defineProperty(exports, "BastionHost", { enumerable: true, get: function () { return bastion_host_1.BastionHost; } });
|
|
46
44
|
__exportStar(require("./build-artifacts"), exports);
|
|
47
|
-
__exportStar(require("./cdk-deploy"), exports);
|
|
48
45
|
var cross_region_ssm_parameter_1 = require("./cross-region-ssm-parameter");
|
|
49
46
|
Object.defineProperty(exports, "CrossRegionSsmParameter", { enumerable: true, get: function () { return cross_region_ssm_parameter_1.CrossRegionSsmParameter; } });
|
|
50
|
-
__exportStar(require("./ecs-update-image"), exports);
|
|
51
47
|
var hosted_zone_with_param_1 = require("./hosted-zone-with-param");
|
|
52
48
|
Object.defineProperty(exports, "HostedZoneWithParam", { enumerable: true, get: function () { return hosted_zone_with_param_1.HostedZoneWithParam; } });
|
|
53
49
|
var snapshots_1 = require("./snapshots");
|
|
@@ -68,4 +64,4 @@ Object.defineProperty(exports, "WebappDeployViaRole", { enumerable: true, get: f
|
|
|
68
64
|
* happen during snapshot creation.
|
|
69
65
|
*/
|
|
70
66
|
exports.isSnapshot = process.env.IS_SNAPSHOT === "true";
|
|
71
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
67
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxtQ0FBa0M7QUEwQmhDLHdCQUFNO0FBekJSLGdEQUErQztBQTBCN0Msb0NBQVk7QUF6QmQsaUNBQWdDO0FBMEI5QixzQkFBSztBQXpCUCw2QkFBNEI7QUEwQjFCLGtCQUFHO0FBekJMLG1DQUFrQztBQTBCaEMsd0JBQU07QUF6QlIsOERBQTZEO0FBMEIzRCxrREFBbUI7QUF6QnJCLDZCQUE0QjtBQTBCMUIsa0JBQUc7QUF6QkwsZ0RBQStDO0FBMEI3QyxvQ0FBWTtBQXpCZCw2RUFBNEU7QUE0QjFFLGdFQUEwQjtBQTNCNUIsNkJBQTRCO0FBeUIxQixrQkFBRztBQXhCTCx1Q0FBc0M7QUF5QnBDLDRCQUFRO0FBdkJWLGdFQUFnRTtBQUNoRSx1Q0FBdUM7QUFFdkMsK0NBQTRDO0FBQW5DLDJHQUFBLFdBQVcsT0FBQTtBQUNwQixvREFBaUM7QUFDakMsMkVBQXNFO0FBQTdELHFJQUFBLHVCQUF1QixPQUFBO0FBQ2hDLG1FQUE4RDtBQUFyRCw2SEFBQSxtQkFBbUIsT0FBQTtBQUM1Qix5Q0FBeUQ7QUFBaEQsd0hBQUEsMkJBQTJCLE9BQUE7QUFDcEMsaUZBQTRFO0FBQW5FLDJJQUFBLDBCQUEwQixPQUFBO0FBQ25DLCtEQUEyRDtBQUFsRCwwSEFBQSxrQkFBa0IsT0FBQTtBQUMzQiwrQkFBcUM7QUFBNUIsb0dBQUEsWUFBWSxPQUFBO0FBQ3JCLG1FQUE4RDtBQUFyRCw2SEFBQSxtQkFBbUIsT0FBQTtBQWdCNUI7Ozs7OztHQU1HO0FBQ1UsUUFBQSxVQUFVLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLEtBQUssTUFBTSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgYWxhcm1zIGZyb20gXCIuL2FsYXJtc1wiXG5pbXBvcnQgKiBhcyBjZGtQaXBlbGluZXMgZnJvbSBcIi4vY2RrLXBpcGVsaW5lc1wiXG5pbXBvcnQgKiBhcyBncmlpZCBmcm9tIFwiLi9ncmlpZFwiXG5pbXBvcnQgKiBhcyBzZXMgZnJvbSBcIi4vc2VzXCJcbmltcG9ydCAqIGFzIHdlYmFwcCBmcm9tIFwiLi93ZWJhcHBcIlxuaW1wb3J0ICogYXMgY29uZmlndXJlUGFyYW1ldGVycyBmcm9tIFwiLi9jb25maWd1cmUtcGFyYW1ldGVyc1wiXG5pbXBvcnQgKiBhcyBlY3MgZnJvbSBcIi4vZWNzXCJcbmltcG9ydCAqIGFzIGxvYWRCYWxhbmNlciBmcm9tIFwiLi9sb2FkLWJhbGFuY2VyXCJcbmltcG9ydCAqIGFzIGNsb3VkVHJhaWxTbGFja0ludGVncmF0aW9uIGZyb20gXCIuL2Nsb3VkdHJhaWwtc2xhY2staW50ZWdyYXRpb25cIlxuaW1wb3J0ICogYXMgcmRzIGZyb20gXCIuL3Jkc1wiXG5pbXBvcnQgKiBhcyBwbGF0Zm9ybSBmcm9tIFwiLi9wbGF0Zm9ybVwiXG5cbi8vIFRPRE86IFdlIHdhbnQgdG8gc3dpdGNoIGV4cG9ydHMgc28gdGhleSBldmVyeSBjb25zdHJ1Y3QgdW5kZXJcbi8vICBhIG5hbWVzcGFjZSBzdWNoIGFzIHRoZSBzbnMgZXhwb3J0LlxuXG5leHBvcnQgeyBCYXN0aW9uSG9zdCB9IGZyb20gXCIuL2Jhc3Rpb24taG9zdFwiXG5leHBvcnQgKiBmcm9tIFwiLi9idWlsZC1hcnRpZmFjdHNcIlxuZXhwb3J0IHsgQ3Jvc3NSZWdpb25Tc21QYXJhbWV0ZXIgfSBmcm9tIFwiLi9jcm9zcy1yZWdpb24tc3NtLXBhcmFtZXRlclwiXG5leHBvcnQgeyBIb3N0ZWRab25lV2l0aFBhcmFtIH0gZnJvbSBcIi4vaG9zdGVkLXpvbmUtd2l0aC1wYXJhbVwiXG5leHBvcnQgeyBjcmVhdGVDbG91ZEFzc2VtYmx5U25hcHNob3QgfSBmcm9tIFwiLi9zbmFwc2hvdHNcIlxuZXhwb3J0IHsgU3NtUGFyYW1ldGVyQmFja2VkUmVzb3VyY2UgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLWJhY2tlZC1yZXNvdXJjZVwiXG5leHBvcnQgeyBTc21QYXJhbWV0ZXJSZWFkZXIgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLXJlYWRlclwiXG5leHBvcnQgeyB0YWdSZXNvdXJjZXMgfSBmcm9tIFwiLi90YWdzXCJcbmV4cG9ydCB7IFdlYmFwcERlcGxveVZpYVJvbGUgfSBmcm9tIFwiLi93ZWJhcHAtZGVwbG95LXZpYS1yb2xlXCJcblxuZXhwb3J0IHtcbiAgYWxhcm1zLFxuICBjZGtQaXBlbGluZXMsXG4gIGdyaWlkLFxuICBzZXMsXG4gIHdlYmFwcCxcbiAgY29uZmlndXJlUGFyYW1ldGVycyxcbiAgZWNzLFxuICBsb2FkQmFsYW5jZXIsXG4gIHJkcyxcbiAgcGxhdGZvcm0sXG4gIGNsb3VkVHJhaWxTbGFja0ludGVncmF0aW9uLFxufVxuXG4vKipcbiAqIENoZWNrIGlmIHdlIGFyZSBzeW50aGVzaXppbmcgYSBzbmFwc2hvdCBieSBzZXR0aW5nIElTX1NOQVBTSE9UXG4gKiBlbnZpcm9ubWVudCB2YXJpYWJsZSB0byB0cnVlLlxuICpcbiAqIFRoaXMgYWxsb3dzIGZvciBzcGVjaWFsIGNvbmRpdGlvbmFsIGxvZ2ljIHRoYXQgc2hvdWxkIG9ubHlcbiAqIGhhcHBlbiBkdXJpbmcgc25hcHNob3QgY3JlYXRpb24uXG4gKi9cbmV4cG9ydCBjb25zdCBpc1NuYXBzaG90ID0gcHJvY2Vzcy5lbnYuSVNfU05BUFNIT1QgPT09IFwidHJ1ZVwiXG4iXX0=
|
|
@@ -1,15 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
const
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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.
|
|
75
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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.
|
|
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
|
|
55
|
-
|
|
56
|
-
|
|
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,
|
|
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"]}
|