@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.
Files changed (46) hide show
  1. package/README.md +18 -2
  2. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +1 -1
  3. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +46 -35
  4. package/lib/cdk-pipelines/liflig-cdk-pipeline.js +6 -6
  5. package/lib/index.d.ts +1 -4
  6. package/lib/index.js +2 -6
  7. package/lib/ses/configurationsetsnsdestination/handler.d.ts +1 -1
  8. package/lib/ses/configurationsetsnsdestination/handler.js +15 -25
  9. package/lib/ses/configurationsetsnsdestination/index.js +7 -7
  10. package/lib/ses/sesdomain/handler.d.ts +1 -1
  11. package/lib/ses/sesdomain/handler.js +31 -31
  12. package/lib/ses/sesdomain/index.js +6 -6
  13. package/lib/ses/sesverifyemail/handler.d.ts +1 -1
  14. package/lib/ses/sesverifyemail/handler.js +12 -10
  15. package/lib/ses/sesverifyemail/index.js +6 -6
  16. package/package.json +14 -3
  17. package/lib/cdk-deploy/cdk-deploy.d.ts +0 -63
  18. package/lib/cdk-deploy/cdk-deploy.js +0 -175
  19. package/lib/cdk-deploy/index.d.ts +0 -1
  20. package/lib/cdk-deploy/index.js +0 -6
  21. package/lib/cdk-deploy/start-deploy-handler.d.ts +0 -8
  22. package/lib/cdk-deploy/start-deploy-handler.js +0 -72
  23. package/lib/cdk-deploy/status-handler.d.ts +0 -6
  24. package/lib/cdk-deploy/status-handler.js +0 -83
  25. package/lib/ecs-update-image/artifact-status.d.ts +0 -39
  26. package/lib/ecs-update-image/artifact-status.js +0 -41
  27. package/lib/ecs-update-image/ecs-update-image.d.ts +0 -41
  28. package/lib/ecs-update-image/ecs-update-image.js +0 -98
  29. package/lib/ecs-update-image/index.d.ts +0 -3
  30. package/lib/ecs-update-image/index.js +0 -10
  31. package/lib/ecs-update-image/start-deploy-handler.d.ts +0 -6
  32. package/lib/ecs-update-image/start-deploy-handler.js +0 -104
  33. package/lib/ecs-update-image/status-handler.d.ts +0 -11
  34. package/lib/ecs-update-image/status-handler.js +0 -74
  35. package/lib/ecs-update-image/tag.d.ts +0 -47
  36. package/lib/ecs-update-image/tag.js +0 -67
  37. package/lib/pipelines/conventions.d.ts +0 -14
  38. package/lib/pipelines/conventions.js +0 -24
  39. package/lib/pipelines/deploy-env.d.ts +0 -18
  40. package/lib/pipelines/deploy-env.js +0 -96
  41. package/lib/pipelines/index.d.ts +0 -2
  42. package/lib/pipelines/index.js +0 -8
  43. package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +0 -13
  44. package/lib/pipelines/liflig-cdk-deployer-deps.js +0 -35
  45. package/lib/pipelines/pipeline.d.ts +0 -78
  46. package/lib/pipelines/pipeline.js +0 -224
@@ -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"]}
@@ -7,7 +7,7 @@ const lambda = require("aws-cdk-lib/aws-lambda");
7
7
  const r53 = require("aws-cdk-lib/aws-route53");
8
8
  const cdk = require("aws-cdk-lib");
9
9
  const cr = require("aws-cdk-lib/custom-resources");
10
- const handler_1 = require("./handler");
10
+ const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
11
11
  class SesDomain extends constructs.Construct {
12
12
  constructor(scope, id, props) {
13
13
  var _a, _b, _c, _d;
@@ -65,11 +65,11 @@ class SesDomainProvider extends constructs.Construct {
65
65
  constructor(scope, id) {
66
66
  super(scope, id);
67
67
  this.provider = new cr.Provider(this, "Provider", {
68
- onEventHandler: new lambda.Function(this, "Function", {
69
- code: new lambda.InlineCode(`exports.handler = ${handler_1.sesDomainHandler.toString()};`),
70
- handler: "index.handler",
71
- runtime: lambda.Runtime.NODEJS_16_X,
68
+ onEventHandler: new aws_lambda_nodejs_1.NodejsFunction(this, "Function", {
69
+ entry: require.resolve("./handler"),
70
+ runtime: lambda.Runtime.NODEJS_18_X,
72
71
  timeout: cdk.Duration.minutes(5),
72
+ awsSdkConnectionReuse: false,
73
73
  initialPolicy: [
74
74
  new iam.PolicyStatement({
75
75
  actions: [
@@ -91,4 +91,4 @@ class SesDomainProvider extends constructs.Construct {
91
91
  this.serviceToken = this.provider.serviceToken;
92
92
  }
93
93
  }
94
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/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 r53 from \"aws-cdk-lib/aws-route53\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { sesDomainHandler } from \"./handler\"\n\ninterface Props {\n  /**\n   * The domain name to register in SES.\n   */\n  domainName: string\n  /**\n   * Hosted Zone to attach DNS records. If not given it must\n   * be performed manually.\n   */\n  hostedZone?: r53.IHostedZone\n  /**\n   * Include or exclude verification TXT record.\n   *\n   * CNAME records for DKIM tokens will still be created.\n   *\n   * Route 53 will not allow multiple TXT records with the same name.\n   * This option allows to \"opt-out\" of the records and leaving\n   * the caller responsible of handling it.\n   *\n   * @default true\n   */\n  includeVerificationRecord?: boolean\n  /**\n   * Default configuration set for emails sent from this domain.\n   */\n  defaultConfigurationSetName?: string\n  /**\n   * Configuration for an SPF record.\n   *\n   * @default - an SPF record with a default value is created.\n   */\n  spfRecord?: {\n    /**\n     * Whether to create the record or not.\n     *\n     * @default true\n     */\n    include?: boolean\n    /**\n     * The value of the SPF record.\n     *\n     * NOTE: The value will be enclosed in double quotes for you.\n     *\n     * @default \"v=spf1 include:amazonses.com ~all\"\n     */\n    value?: string\n  }\n}\n\nexport class SesDomain extends constructs.Construct {\n  public route53RecordSets: cdk.IResolvable\n  public verificationToken: string\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: SesDomainProvider.getOrCreate(this).serviceToken,\n      properties: {\n        DomainName: props.domainName,\n        IncludeVerificationRecord: (\n          props.includeVerificationRecord ?? true\n        ).toString(),\n        DefaultConfigurationSetName: props.defaultConfigurationSetName,\n        // Bump this if changing logic in the lambda that should be\n        // re-evaluated.\n        Serial: 1,\n      },\n    })\n\n    const staticRecordSets: r53.CfnRecordSetGroup.RecordSetProperty[] =\n      props.spfRecord?.include ?? true\n        ? [\n            {\n              name: props.domainName,\n              type: r53.RecordType.TXT,\n              ttl: \"60\",\n              resourceRecords: [\n                JSON.stringify(\n                  props.spfRecord?.value || \"v=spf1 include:amazonses.com ~all\",\n                ),\n              ],\n            },\n          ]\n        : []\n\n    this.route53RecordSets = resource.getAtt(\"Route53RecordSets\")\n    this.verificationToken = resource.getAttString(\"VerificationToken\")\n\n    if (props.hostedZone) {\n      new r53.CfnRecordSetGroup(this, \"RecordSetGroup\", {\n        hostedZoneId: props.hostedZone.hostedZoneId,\n        recordSets: this.route53RecordSets,\n      })\n      if (staticRecordSets.length) {\n        new r53.CfnRecordSetGroup(this, \"StaticRecordSetGroup\", {\n          hostedZoneId: props.hostedZone.hostedZoneId,\n          recordSets: staticRecordSets,\n        })\n      }\n    }\n  }\n}\n\nclass SesDomainProvider 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.ses-domain.provider\"\n    return (\n      (stack.node.tryFindChild(id) as SesDomainProvider) ||\n      new SesDomainProvider(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 = ${sesDomainHandler.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:DeleteIdentity\",\n              \"ses:GetIdentityDkimAttributes\",\n              \"ses:GetIdentityMailFromDomainAttributes\",\n              \"ses:GetIdentityVerificationAttributes\",\n              \"ses:SetIdentityDkimEnabled\",\n              \"ses:SetIdentityMailFromDomain\",\n              \"ses:VerifyDomainDkim\",\n              \"ses:VerifyDomainIdentity\",\n              \"ses:PutEmailIdentityConfigurationSetAttributes\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
94
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/ses/sesdomain/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 r53 from \"aws-cdk-lib/aws-route53\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\n\ninterface Props {\n  /**\n   * The domain name to register in SES.\n   */\n  domainName: string\n  /**\n   * Hosted Zone to attach DNS records. If not given it must\n   * be performed manually.\n   */\n  hostedZone?: r53.IHostedZone\n  /**\n   * Include or exclude verification TXT record.\n   *\n   * CNAME records for DKIM tokens will still be created.\n   *\n   * Route 53 will not allow multiple TXT records with the same name.\n   * This option allows to \"opt-out\" of the records and leaving\n   * the caller responsible of handling it.\n   *\n   * @default true\n   */\n  includeVerificationRecord?: boolean\n  /**\n   * Default configuration set for emails sent from this domain.\n   */\n  defaultConfigurationSetName?: string\n  /**\n   * Configuration for an SPF record.\n   *\n   * @default - an SPF record with a default value is created.\n   */\n  spfRecord?: {\n    /**\n     * Whether to create the record or not.\n     *\n     * @default true\n     */\n    include?: boolean\n    /**\n     * The value of the SPF record.\n     *\n     * NOTE: The value will be enclosed in double quotes for you.\n     *\n     * @default \"v=spf1 include:amazonses.com ~all\"\n     */\n    value?: string\n  }\n}\n\nexport class SesDomain extends constructs.Construct {\n  public route53RecordSets: cdk.IResolvable\n  public verificationToken: string\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: SesDomainProvider.getOrCreate(this).serviceToken,\n      properties: {\n        DomainName: props.domainName,\n        IncludeVerificationRecord: (\n          props.includeVerificationRecord ?? true\n        ).toString(),\n        DefaultConfigurationSetName: props.defaultConfigurationSetName,\n        // Bump this if changing logic in the lambda that should be\n        // re-evaluated.\n        Serial: 1,\n      },\n    })\n\n    const staticRecordSets: r53.CfnRecordSetGroup.RecordSetProperty[] =\n      props.spfRecord?.include ?? true\n        ? [\n            {\n              name: props.domainName,\n              type: r53.RecordType.TXT,\n              ttl: \"60\",\n              resourceRecords: [\n                JSON.stringify(\n                  props.spfRecord?.value || \"v=spf1 include:amazonses.com ~all\",\n                ),\n              ],\n            },\n          ]\n        : []\n\n    this.route53RecordSets = resource.getAtt(\"Route53RecordSets\")\n    this.verificationToken = resource.getAttString(\"VerificationToken\")\n\n    if (props.hostedZone) {\n      new r53.CfnRecordSetGroup(this, \"RecordSetGroup\", {\n        hostedZoneId: props.hostedZone.hostedZoneId,\n        recordSets: this.route53RecordSets,\n      })\n      if (staticRecordSets.length) {\n        new r53.CfnRecordSetGroup(this, \"StaticRecordSetGroup\", {\n          hostedZoneId: props.hostedZone.hostedZoneId,\n          recordSets: staticRecordSets,\n        })\n      }\n    }\n  }\n}\n\nclass SesDomainProvider 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.ses-domain.provider\"\n    return (\n      (stack.node.tryFindChild(id) as SesDomainProvider) ||\n      new SesDomainProvider(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:DeleteIdentity\",\n              \"ses:GetIdentityDkimAttributes\",\n              \"ses:GetIdentityMailFromDomainAttributes\",\n              \"ses:GetIdentityVerificationAttributes\",\n              \"ses:SetIdentityDkimEnabled\",\n              \"ses:SetIdentityMailFromDomain\",\n              \"ses:VerifyDomainDkim\",\n              \"ses:VerifyDomainIdentity\",\n              \"ses:PutEmailIdentityConfigurationSetAttributes\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    })\n\n    this.serviceToken = this.provider.serviceToken\n  }\n}\n"]}
@@ -5,5 +5,5 @@ type OnEventHandler = (event: {
5
5
  }) => Promise<{
6
6
  PhysicalResourceId?: string;
7
7
  }>;
8
- export declare const sesVerifyEmailHandler: OnEventHandler;
8
+ export declare const handler: OnEventHandler;
9
9
  export {};
@@ -1,25 +1,27 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.sesVerifyEmailHandler = void 0;
4
- // This function is inline-compiled for the lambda.
5
- // It must be self-contained.
6
- const sesVerifyEmailHandler = async (event) => {
7
- const AWS = require("aws-sdk");
8
- const ses = new AWS.SES();
3
+ exports.handler = void 0;
4
+ const client_ses_1 = require("@aws-sdk/client-ses");
5
+ const handler = async (event) => {
6
+ const sesClient = new client_ses_1.SESClient();
9
7
  const emailAddress = event.ResourceProperties["EmailAddress"];
10
8
  switch (event.RequestType) {
11
9
  case "Delete":
12
- await ses.deleteIdentity({ Identity: emailAddress }).promise();
10
+ await sesClient.send(new client_ses_1.DeleteIdentityCommand({
11
+ Identity: emailAddress,
12
+ }));
13
13
  return {
14
14
  PhysicalResourceId: event.PhysicalResourceId,
15
15
  };
16
16
  case "Create":
17
17
  case "Update":
18
- await ses.verifyEmailIdentity({ EmailAddress: emailAddress }).promise();
18
+ await sesClient.send(new client_ses_1.VerifyEmailIdentityCommand({
19
+ EmailAddress: emailAddress,
20
+ }));
19
21
  return {
20
22
  PhysicalResourceId: `SesVerifyEmail:${emailAddress}`,
21
23
  };
22
24
  }
23
25
  };
24
- exports.sesVerifyEmailHandler = sesVerifyEmailHandler;
25
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXMvc2VzdmVyaWZ5ZW1haWwvaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFjQSxtREFBbUQ7QUFDbkQsNkJBQTZCO0FBQ3RCLE1BQU0scUJBQXFCLEdBQW1CLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRTtJQUNuRSxNQUFNLEdBQUcsR0FBRyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUE7SUFFOUIsTUFBTSxHQUFHLEdBQVksSUFBSSxHQUFHLENBQUMsR0FBRyxFQUFjLENBQUE7SUFFOUMsTUFBTSxZQUFZLEdBQUcsS0FBSyxDQUFDLGtCQUFrQixDQUFDLGNBQWMsQ0FBQyxDQUFBO0lBRTdELFFBQVEsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzFCLEtBQUssUUFBUTtZQUNYLE1BQU0sR0FBRyxDQUFDLGNBQWMsQ0FBQyxFQUFFLFFBQVEsRUFBRSxZQUFZLEVBQUUsQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFBO1lBRTlELE9BQU87Z0JBQ0wsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLGtCQUFrQjthQUM3QyxDQUFBO1FBRUgsS0FBSyxRQUFRLENBQUM7UUFDZCxLQUFLLFFBQVE7WUFDWCxNQUFNLEdBQUcsQ0FBQyxtQkFBbUIsQ0FBQyxFQUFFLFlBQVksRUFBRSxZQUFZLEVBQUUsQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFBO1lBRXZFLE9BQU87Z0JBQ0wsa0JBQWtCLEVBQUUsa0JBQWtCLFlBQVksRUFBRTthQUNyRCxDQUFBO0lBQ0wsQ0FBQztBQUNILENBQUMsQ0FBQTtBQXZCWSxRQUFBLHFCQUFxQix5QkF1QmpDIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1hc3NpZ25tZW50ICovXG4vKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWNhbGwgKi9cbi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtbWVtYmVyLWFjY2VzcyAqL1xuLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXZhci1yZXF1aXJlcyAqL1xuaW1wb3J0IHR5cGUgKiBhcyBfQVdTIGZyb20gXCJhd3Mtc2RrXCJcblxudHlwZSBPbkV2ZW50SGFuZGxlciA9IChldmVudDoge1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn0pID0+IFByb21pc2U8e1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbn0+XG5cbi8vIFRoaXMgZnVuY3Rpb24gaXMgaW5saW5lLWNvbXBpbGVkIGZvciB0aGUgbGFtYmRhLlxuLy8gSXQgbXVzdCBiZSBzZWxmLWNvbnRhaW5lZC5cbmV4cG9ydCBjb25zdCBzZXNWZXJpZnlFbWFpbEhhbmRsZXI6IE9uRXZlbnRIYW5kbGVyID0gYXN5bmMgKGV2ZW50KSA9PiB7XG4gIGNvbnN0IEFXUyA9IHJlcXVpcmUoXCJhd3Mtc2RrXCIpXG5cbiAgY29uc3Qgc2VzOiBBV1MuU0VTID0gbmV3IEFXUy5TRVMoKSBhcyBfQVdTLlNFU1xuXG4gIGNvbnN0IGVtYWlsQWRkcmVzcyA9IGV2ZW50LlJlc291cmNlUHJvcGVydGllc1tcIkVtYWlsQWRkcmVzc1wiXVxuXG4gIHN3aXRjaCAoZXZlbnQuUmVxdWVzdFR5cGUpIHtcbiAgICBjYXNlIFwiRGVsZXRlXCI6XG4gICAgICBhd2FpdCBzZXMuZGVsZXRlSWRlbnRpdHkoeyBJZGVudGl0eTogZW1haWxBZGRyZXNzIH0pLnByb21pc2UoKVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBQaHlzaWNhbFJlc291cmNlSWQ6IGV2ZW50LlBoeXNpY2FsUmVzb3VyY2VJZCxcbiAgICAgIH1cblxuICAgIGNhc2UgXCJDcmVhdGVcIjpcbiAgICBjYXNlIFwiVXBkYXRlXCI6XG4gICAgICBhd2FpdCBzZXMudmVyaWZ5RW1haWxJZGVudGl0eSh7IEVtYWlsQWRkcmVzczogZW1haWxBZGRyZXNzIH0pLnByb21pc2UoKVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBQaHlzaWNhbFJlc291cmNlSWQ6IGBTZXNWZXJpZnlFbWFpbDoke2VtYWlsQWRkcmVzc31gLFxuICAgICAgfVxuICB9XG59XG4iXX0=
26
+ exports.handler = handler;
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXMvc2VzdmVyaWZ5ZW1haWwvaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxvREFJNEI7QUFVckIsTUFBTSxPQUFPLEdBQW1CLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRTtJQUNyRCxNQUFNLFNBQVMsR0FBRyxJQUFJLHNCQUFTLEVBQUUsQ0FBQTtJQUNqQyxNQUFNLFlBQVksR0FBRyxLQUFLLENBQUMsa0JBQWtCLENBQUMsY0FBYyxDQUFDLENBQUE7SUFFN0QsUUFBUSxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDMUIsS0FBSyxRQUFRO1lBQ1gsTUFBTSxTQUFTLENBQUMsSUFBSSxDQUNsQixJQUFJLGtDQUFxQixDQUFDO2dCQUN4QixRQUFRLEVBQUUsWUFBWTthQUN2QixDQUFDLENBQ0gsQ0FBQTtZQUVELE9BQU87Z0JBQ0wsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLGtCQUFrQjthQUM3QyxDQUFBO1FBRUgsS0FBSyxRQUFRLENBQUM7UUFDZCxLQUFLLFFBQVE7WUFDWCxNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQ2xCLElBQUksdUNBQTBCLENBQUM7Z0JBQzdCLFlBQVksRUFBRSxZQUFZO2FBQzNCLENBQUMsQ0FDSCxDQUFBO1lBRUQsT0FBTztnQkFDTCxrQkFBa0IsRUFBRSxrQkFBa0IsWUFBWSxFQUFFO2FBQ3JELENBQUE7SUFDTCxDQUFDO0FBQ0gsQ0FBQyxDQUFBO0FBNUJZLFFBQUEsT0FBTyxXQTRCbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBEZWxldGVJZGVudGl0eUNvbW1hbmQsXG4gIFNFU0NsaWVudCxcbiAgVmVyaWZ5RW1haWxJZGVudGl0eUNvbW1hbmQsXG59IGZyb20gXCJAYXdzLXNkay9jbGllbnQtc2VzXCJcblxudHlwZSBPbkV2ZW50SGFuZGxlciA9IChldmVudDoge1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn0pID0+IFByb21pc2U8e1xuICBQaHlzaWNhbFJlc291cmNlSWQ/OiBzdHJpbmdcbn0+XG5cbmV4cG9ydCBjb25zdCBoYW5kbGVyOiBPbkV2ZW50SGFuZGxlciA9IGFzeW5jIChldmVudCkgPT4ge1xuICBjb25zdCBzZXNDbGllbnQgPSBuZXcgU0VTQ2xpZW50KClcbiAgY29uc3QgZW1haWxBZGRyZXNzID0gZXZlbnQuUmVzb3VyY2VQcm9wZXJ0aWVzW1wiRW1haWxBZGRyZXNzXCJdXG5cbiAgc3dpdGNoIChldmVudC5SZXF1ZXN0VHlwZSkge1xuICAgIGNhc2UgXCJEZWxldGVcIjpcbiAgICAgIGF3YWl0IHNlc0NsaWVudC5zZW5kKFxuICAgICAgICBuZXcgRGVsZXRlSWRlbnRpdHlDb21tYW5kKHtcbiAgICAgICAgICBJZGVudGl0eTogZW1haWxBZGRyZXNzLFxuICAgICAgICB9KSxcbiAgICAgIClcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgUGh5c2ljYWxSZXNvdXJjZUlkOiBldmVudC5QaHlzaWNhbFJlc291cmNlSWQsXG4gICAgICB9XG5cbiAgICBjYXNlIFwiQ3JlYXRlXCI6XG4gICAgY2FzZSBcIlVwZGF0ZVwiOlxuICAgICAgYXdhaXQgc2VzQ2xpZW50LnNlbmQoXG4gICAgICAgIG5ldyBWZXJpZnlFbWFpbElkZW50aXR5Q29tbWFuZCh7XG4gICAgICAgICAgRW1haWxBZGRyZXNzOiBlbWFpbEFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgKVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBQaHlzaWNhbFJlc291cmNlSWQ6IGBTZXNWZXJpZnlFbWFpbDoke2VtYWlsQWRkcmVzc31gLFxuICAgICAgfVxuICB9XG59XG4iXX0=
@@ -6,7 +6,7 @@ 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");
9
+ const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
10
10
  class SesVerifyEmail extends constructs.Construct {
11
11
  constructor(scope, id, props) {
12
12
  super(scope, id);
@@ -32,11 +32,11 @@ class SesVerifyEmailProvider extends constructs.Construct {
32
32
  constructor(scope, id) {
33
33
  super(scope, id);
34
34
  this.provider = new cr.Provider(this, "Provider", {
35
- onEventHandler: new lambda.Function(this, "Function", {
36
- code: new lambda.InlineCode(`exports.handler = ${handler_1.sesVerifyEmailHandler.toString()};`),
37
- handler: "index.handler",
38
- runtime: lambda.Runtime.NODEJS_16_X,
35
+ onEventHandler: new aws_lambda_nodejs_1.NodejsFunction(this, "Function", {
36
+ entry: require.resolve("./handler"),
37
+ runtime: lambda.Runtime.NODEJS_18_X,
39
38
  timeout: cdk.Duration.minutes(5),
39
+ awsSdkConnectionReuse: false,
40
40
  initialPolicy: [
41
41
  new iam.PolicyStatement({
42
42
  actions: ["ses:DeleteIdentity", "ses:VerifyEmailIdentity"],
@@ -48,4 +48,4 @@ class SesVerifyEmailProvider extends constructs.Construct {
48
48
  this.serviceToken = this.provider.serviceToken;
49
49
  }
50
50
  }
51
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VzL3Nlc3ZlcmlmeWVtYWlsL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUN4QywyQ0FBMEM7QUFDMUMsaURBQWdEO0FBQ2hELG1DQUFrQztBQUNsQyxtREFBa0Q7QUFDbEQsdUNBQWlEO0FBU2pELE1BQWEsY0FBZSxTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBR3RELFlBQVksS0FBMkIsRUFBRSxFQUFVLEVBQUUsS0FBWTtRQUMvRCxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksR0FBRyxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3ZDLFlBQVksRUFBRSxzQkFBc0IsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUMsWUFBWTtZQUNuRSxVQUFVLEVBQUU7Z0JBQ1YsWUFBWSxFQUFFLEtBQUssQ0FBQyxZQUFZO2FBQ2pDO1NBQ0YsQ0FBQyxDQUFBO0lBQ0osQ0FBQztDQUNGO0FBYkQsd0NBYUM7QUFFRCxNQUFNLHNCQUF1QixTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQ3ZEOztPQUVHO0lBQ0ksTUFBTSxDQUFDLFdBQVcsQ0FBQyxLQUEyQjtRQUNuRCxNQUFNLEtBQUssR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQTtRQUNqQyxNQUFNLEVBQUUsR0FBRyxzQ0FBc0MsQ0FBQTtRQUNqRCxPQUFPLENBQ0osS0FBSyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsRUFBRSxDQUE0QjtZQUN2RCxJQUFJLHNCQUFzQixDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FDdEMsQ0FBQTtJQUNILENBQUM7SUFLRCxZQUFZLEtBQTJCLEVBQUUsRUFBVTtRQUNqRCxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxFQUFFLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDaEQsY0FBYyxFQUFFLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO2dCQUNwRCxJQUFJLEVBQUUsSUFBSSxNQUFNLENBQUMsVUFBVSxDQUN6QixxQkFBcUIsK0JBQXFCLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FDekQ7Z0JBQ0QsT0FBTyxFQUFFLGVBQWU7Z0JBQ3hCLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7Z0JBQ25DLE9BQU8sRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7Z0JBQ2hDLGFBQWEsRUFBRTtvQkFDYixJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7d0JBQ3RCLE9BQU8sRUFBRSxDQUFDLG9CQUFvQixFQUFFLHlCQUF5QixDQUFDO3dCQUMxRCxTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUM7cUJBQ2pCLENBQUM7aUJBQ0g7YUFDRixDQUFDO1NBQ0gsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQTtJQUNoRCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBjciBmcm9tIFwiYXdzLWNkay1saWIvY3VzdG9tLXJlc291cmNlc1wiXG5pbXBvcnQgeyBzZXNWZXJpZnlFbWFpbEhhbmRsZXIgfSBmcm9tIFwiLi9oYW5kbGVyXCJcblxuaW50ZXJmYWNlIFByb3BzIHtcbiAgLyoqXG4gICAqIFRoZSBlbWFpbCBhZGRyZXNzIHRvIGFkZCBhcyBhIHZlcmlmaWVkIGVtYWlsIGluIFNFUy5cbiAgICovXG4gIGVtYWlsQWRkcmVzczogc3RyaW5nXG59XG5cbmV4cG9ydCBjbGFzcyBTZXNWZXJpZnlFbWFpbCBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgcHVibGljIHJvdXRlNTNSZWNvcmRTZXRzOiBjZGsuSVJlc29sdmFibGVcblxuICBjb25zdHJ1Y3RvcihzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIG5ldyBjZGsuQ3VzdG9tUmVzb3VyY2UodGhpcywgXCJSZXNvdXJjZVwiLCB7XG4gICAgICBzZXJ2aWNlVG9rZW46IFNlc1ZlcmlmeUVtYWlsUHJvdmlkZXIuZ2V0T3JDcmVhdGUodGhpcykuc2VydmljZVRva2VuLFxuICAgICAgcHJvcGVydGllczoge1xuICAgICAgICBFbWFpbEFkZHJlc3M6IHByb3BzLmVtYWlsQWRkcmVzcyxcbiAgICAgIH0sXG4gICAgfSlcbiAgfVxufVxuXG5jbGFzcyBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICAvKipcbiAgICogUmV0dXJucyB0aGUgc2luZ2xldG9uIHByb3ZpZGVyLlxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBnZXRPckNyZWF0ZShzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QpIHtcbiAgICBjb25zdCBzdGFjayA9IGNkay5TdGFjay5vZihzY29wZSlcbiAgICBjb25zdCBpZCA9IFwibGlmbGlnLWNkay5zZXMtdmVyaWZ5LWVtYWlsLnByb3ZpZGVyXCJcbiAgICByZXR1cm4gKFxuICAgICAgKHN0YWNrLm5vZGUudHJ5RmluZENoaWxkKGlkKSBhcyBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyKSB8fFxuICAgICAgbmV3IFNlc1ZlcmlmeUVtYWlsUHJvdmlkZXIoc3RhY2ssIGlkKVxuICAgIClcbiAgfVxuXG4gIHByaXZhdGUgcmVhZG9ubHkgcHJvdmlkZXI6IGNyLlByb3ZpZGVyXG4gIHB1YmxpYyByZWFkb25seSBzZXJ2aWNlVG9rZW46IHN0cmluZ1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMucHJvdmlkZXIgPSBuZXcgY3IuUHJvdmlkZXIodGhpcywgXCJQcm92aWRlclwiLCB7XG4gICAgICBvbkV2ZW50SGFuZGxlcjogbmV3IGxhbWJkYS5GdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgICAgY29kZTogbmV3IGxhbWJkYS5JbmxpbmVDb2RlKFxuICAgICAgICAgIGBleHBvcnRzLmhhbmRsZXIgPSAke3Nlc1ZlcmlmeUVtYWlsSGFuZGxlci50b1N0cmluZygpfTtgLFxuICAgICAgICApLFxuICAgICAgICBoYW5kbGVyOiBcImluZGV4LmhhbmRsZXJcIixcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzE2X1gsXG4gICAgICAgIHRpbWVvdXQ6IGNkay5EdXJhdGlvbi5taW51dGVzKDUpLFxuICAgICAgICBpbml0aWFsUG9saWN5OiBbXG4gICAgICAgICAgbmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICAgICAgICAgICAgYWN0aW9uczogW1wic2VzOkRlbGV0ZUlkZW50aXR5XCIsIFwic2VzOlZlcmlmeUVtYWlsSWRlbnRpdHlcIl0sXG4gICAgICAgICAgICByZXNvdXJjZXM6IFtcIipcIl0sXG4gICAgICAgICAgfSksXG4gICAgICAgIF0sXG4gICAgICB9KSxcbiAgICB9KVxuXG4gICAgdGhpcy5zZXJ2aWNlVG9rZW4gPSB0aGlzLnByb3ZpZGVyLnNlcnZpY2VUb2tlblxuICB9XG59XG4iXX0=
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VzL3Nlc3ZlcmlmeWVtYWlsL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUN4QywyQ0FBMEM7QUFDMUMsaURBQWdEO0FBQ2hELG1DQUFrQztBQUNsQyxtREFBa0Q7QUFDbEQscUVBQThEO0FBUzlELE1BQWEsY0FBZSxTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBR3RELFlBQVksS0FBMkIsRUFBRSxFQUFVLEVBQUUsS0FBWTtRQUMvRCxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksR0FBRyxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3ZDLFlBQVksRUFBRSxzQkFBc0IsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUMsWUFBWTtZQUNuRSxVQUFVLEVBQUU7Z0JBQ1YsWUFBWSxFQUFFLEtBQUssQ0FBQyxZQUFZO2FBQ2pDO1NBQ0YsQ0FBQyxDQUFBO0lBQ0osQ0FBQztDQUNGO0FBYkQsd0NBYUM7QUFFRCxNQUFNLHNCQUF1QixTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQ3ZEOztPQUVHO0lBQ0ksTUFBTSxDQUFDLFdBQVcsQ0FBQyxLQUEyQjtRQUNuRCxNQUFNLEtBQUssR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQTtRQUNqQyxNQUFNLEVBQUUsR0FBRyxzQ0FBc0MsQ0FBQTtRQUNqRCxPQUFPLENBQ0osS0FBSyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsRUFBRSxDQUE0QjtZQUN2RCxJQUFJLHNCQUFzQixDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FDdEMsQ0FBQTtJQUNILENBQUM7SUFLRCxZQUFZLEtBQTJCLEVBQUUsRUFBVTtRQUNqRCxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxFQUFFLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDaEQsY0FBYyxFQUFFLElBQUksa0NBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO2dCQUNuRCxLQUFLLEVBQUUsT0FBTyxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUM7Z0JBQ25DLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7Z0JBQ25DLE9BQU8sRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7Z0JBQ2hDLHFCQUFxQixFQUFFLEtBQUs7Z0JBQzVCLGFBQWEsRUFBRTtvQkFDYixJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7d0JBQ3RCLE9BQU8sRUFBRSxDQUFDLG9CQUFvQixFQUFFLHlCQUF5QixDQUFDO3dCQUMxRCxTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUM7cUJBQ2pCLENBQUM7aUJBQ0g7YUFDRixDQUFDO1NBQ0gsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQTtJQUNoRCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBjciBmcm9tIFwiYXdzLWNkay1saWIvY3VzdG9tLXJlc291cmNlc1wiXG5pbXBvcnQgeyBOb2RlanNGdW5jdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtbGFtYmRhLW5vZGVqc1wiXG5cbmludGVyZmFjZSBQcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgZW1haWwgYWRkcmVzcyB0byBhZGQgYXMgYSB2ZXJpZmllZCBlbWFpbCBpbiBTRVMuXG4gICAqL1xuICBlbWFpbEFkZHJlc3M6IHN0cmluZ1xufVxuXG5leHBvcnQgY2xhc3MgU2VzVmVyaWZ5RW1haWwgZXh0ZW5kcyBjb25zdHJ1Y3RzLkNvbnN0cnVjdCB7XG4gIHB1YmxpYyByb3V0ZTUzUmVjb3JkU2V0czogY2RrLklSZXNvbHZhYmxlXG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICBuZXcgY2RrLkN1c3RvbVJlc291cmNlKHRoaXMsIFwiUmVzb3VyY2VcIiwge1xuICAgICAgc2VydmljZVRva2VuOiBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyLmdldE9yQ3JlYXRlKHRoaXMpLnNlcnZpY2VUb2tlbixcbiAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgRW1haWxBZGRyZXNzOiBwcm9wcy5lbWFpbEFkZHJlc3MsXG4gICAgICB9LFxuICAgIH0pXG4gIH1cbn1cblxuY2xhc3MgU2VzVmVyaWZ5RW1haWxQcm92aWRlciBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgLyoqXG4gICAqIFJldHVybnMgdGhlIHNpbmdsZXRvbiBwcm92aWRlci5cbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgZ2V0T3JDcmVhdGUoc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0KSB7XG4gICAgY29uc3Qgc3RhY2sgPSBjZGsuU3RhY2sub2Yoc2NvcGUpXG4gICAgY29uc3QgaWQgPSBcImxpZmxpZy1jZGsuc2VzLXZlcmlmeS1lbWFpbC5wcm92aWRlclwiXG4gICAgcmV0dXJuIChcbiAgICAgIChzdGFjay5ub2RlLnRyeUZpbmRDaGlsZChpZCkgYXMgU2VzVmVyaWZ5RW1haWxQcm92aWRlcikgfHxcbiAgICAgIG5ldyBTZXNWZXJpZnlFbWFpbFByb3ZpZGVyKHN0YWNrLCBpZClcbiAgICApXG4gIH1cblxuICBwcml2YXRlIHJlYWRvbmx5IHByb3ZpZGVyOiBjci5Qcm92aWRlclxuICBwdWJsaWMgcmVhZG9ubHkgc2VydmljZVRva2VuOiBzdHJpbmdcblxuICBjb25zdHJ1Y3RvcihzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGlkOiBzdHJpbmcpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLnByb3ZpZGVyID0gbmV3IGNyLlByb3ZpZGVyKHRoaXMsIFwiUHJvdmlkZXJcIiwge1xuICAgICAgb25FdmVudEhhbmRsZXI6IG5ldyBOb2RlanNGdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgICAgZW50cnk6IHJlcXVpcmUucmVzb2x2ZShcIi4vaGFuZGxlclwiKSxcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzE4X1gsXG4gICAgICAgIHRpbWVvdXQ6IGNkay5EdXJhdGlvbi5taW51dGVzKDUpLFxuICAgICAgICBhd3NTZGtDb25uZWN0aW9uUmV1c2U6IGZhbHNlLFxuICAgICAgICBpbml0aWFsUG9saWN5OiBbXG4gICAgICAgICAgbmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICAgICAgICAgICAgYWN0aW9uczogW1wic2VzOkRlbGV0ZUlkZW50aXR5XCIsIFwic2VzOlZlcmlmeUVtYWlsSWRlbnRpdHlcIl0sXG4gICAgICAgICAgICByZXNvdXJjZXM6IFtcIipcIl0sXG4gICAgICAgICAgfSksXG4gICAgICAgIF0sXG4gICAgICB9KSxcbiAgICB9KVxuXG4gICAgdGhpcy5zZXJ2aWNlVG9rZW4gPSB0aGlzLnByb3ZpZGVyLnNlcnZpY2VUb2tlblxuICB9XG59XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@liflig/cdk",
3
- "version": "2.21.2",
3
+ "version": "2.21.4",
4
4
  "description": "CDK library for Liflig",
5
5
  "repository": {
6
6
  "type": "git",
@@ -40,23 +40,34 @@
40
40
  },
41
41
  "devDependencies": {
42
42
  "@aws-cdk/assert": "2.68.0",
43
+ "@aws-sdk/client-cloudwatch-logs": "3.588.0",
44
+ "@aws-sdk/client-codebuild": "3.588.0",
45
+ "@aws-sdk/client-codepipeline": "3.588.0",
46
+ "@aws-sdk/client-ecs": "3.588.0",
47
+ "@aws-sdk/client-s3": "3.588.0",
48
+ "@aws-sdk/client-secrets-manager": "3.588.0",
49
+ "@aws-sdk/client-ses": "3.588.0",
50
+ "@aws-sdk/client-sesv2": "3.588.0",
51
+ "@aws-sdk/client-sfn": "3.588.0",
52
+ "@aws-sdk/lib-storage": "3.588.0",
43
53
  "@commitlint/cli": "19.3.0",
44
54
  "@commitlint/config-conventional": "19.2.2",
45
55
  "@types/aws-lambda": "8.10.138",
46
56
  "@types/jest": "29.5.12",
47
- "@types/node": "20.12.12",
57
+ "@types/node": "20.14.0",
48
58
  "@typescript-eslint/eslint-plugin": "5.62.0",
49
59
  "@typescript-eslint/parser": "5.62.0",
50
60
  "aws-cdk": "2.141.0",
51
61
  "aws-cdk-lib": "2.141.0",
52
62
  "constructs": "10.3.0",
63
+ "esbuild": "0.21.4",
53
64
  "eslint": "8.57.0",
54
65
  "eslint-config-prettier": "9.1.0",
55
66
  "eslint-plugin-prettier": "5.1.3",
56
67
  "husky": "9.0.11",
57
68
  "jest": "29.7.0",
58
69
  "jest-cdk-snapshot": "2.2.1",
59
- "prettier": "3.2.5",
70
+ "prettier": "3.3.0",
60
71
  "semantic-release": "23.1.1",
61
72
  "ts-jest": "29.1.4",
62
73
  "ts-node": "10.9.2",
@@ -1,63 +0,0 @@
1
- import * as constructs from "constructs";
2
- import * as cdk from "aws-cdk-lib";
3
- interface Props extends cdk.StackProps {
4
- /**
5
- * The role that will be granted permission to assume the deploy
6
- * role. This role must have permission to assume the deploy role.
7
- */
8
- callerRoleArn: string;
9
- /**
10
- * The name that will be used for the deploy role. This is the role
11
- * that the caller will assume in order to have permission to invoke
12
- * the Lambda Functions.
13
- */
14
- roleName: string;
15
- /**
16
- * The bucket used for storing artifacts. This is used to grant
17
- * permission to the role to read artifact. If the bucket is in
18
- * another account, it must have a policy which allows the target
19
- * account to use IAM permissions from target account.
20
- */
21
- artifactsBucketName: string;
22
- startDeployFunctionName: string;
23
- statusFunctionName: string;
24
- /**
25
- * This is the stack name used with `cdk bootstrap` and can e
26
- * found in cdk.json as "toolkitStackName".
27
- */
28
- cdkToolkitStackName: string;
29
- /**
30
- * We pass the CDK context values as they contain feature flags
31
- * used by the CDK CLI.
32
- */
33
- cdkContext: Record<string, string | string[]>;
34
- /**
35
- * The secret containing username and password (or access token)
36
- * for a valid docker user. This is used to access private
37
- * repositories or to handle docker hub's pull rate limiting.
38
- */
39
- dockerCredentialsSecretName?: string;
40
- }
41
- /**
42
- * This construct is responsible for the privileges and logic of
43
- * automatically deploying stack resources in an account.
44
- * Its resources are used from a deployment pipeline.
45
- *
46
- * The deployment is performed by invoking the "start deploy"
47
- * lambda with details of what should be deployed. As this is
48
- * responsible for deploying infrastructure, the principal invoking
49
- * might be able to cause privilege escalation. The principal invoking
50
- * should be assumed to have full administrator access.
51
- *
52
- * The process deploying the infrastructure is locked down so this
53
- * is only possibly by deployment through CloudFormation, and as
54
- * such removes a lot of possible escalation paths (e.g. no role
55
- * can be created by direct API call).
56
- *
57
- * The "status" lambda can be used to poll for completion, and will
58
- * also return logs from the job upon completion.
59
- */
60
- export declare class CdkDeploy extends constructs.Construct {
61
- constructor(scope: constructs.Construct, id: string, props: Props);
62
- }
63
- export {};
@@ -1,175 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.CdkDeploy = void 0;
4
- const constructs = require("constructs");
5
- const codebuild = require("aws-cdk-lib/aws-codebuild");
6
- const iam = require("aws-cdk-lib/aws-iam");
7
- const lambda = require("aws-cdk-lib/aws-lambda");
8
- const s3 = require("aws-cdk-lib/aws-s3");
9
- const cdk = require("aws-cdk-lib");
10
- const secretsmanager = require("aws-cdk-lib/aws-secretsmanager");
11
- const start_deploy_handler_1 = require("./start-deploy-handler");
12
- const status_handler_1 = require("./status-handler");
13
- /**
14
- * This construct is responsible for the privileges and logic of
15
- * automatically deploying stack resources in an account.
16
- * Its resources are used from a deployment pipeline.
17
- *
18
- * The deployment is performed by invoking the "start deploy"
19
- * lambda with details of what should be deployed. As this is
20
- * responsible for deploying infrastructure, the principal invoking
21
- * might be able to cause privilege escalation. The principal invoking
22
- * should be assumed to have full administrator access.
23
- *
24
- * The process deploying the infrastructure is locked down so this
25
- * is only possibly by deployment through CloudFormation, and as
26
- * such removes a lot of possible escalation paths (e.g. no role
27
- * can be created by direct API call).
28
- *
29
- * The "status" lambda can be used to poll for completion, and will
30
- * also return logs from the job upon completion.
31
- */
32
- class CdkDeploy extends constructs.Construct {
33
- constructor(scope, id, props) {
34
- super(scope, id);
35
- const account = cdk.Stack.of(this).account;
36
- const region = cdk.Stack.of(this).region;
37
- const artifactsBucket = s3.Bucket.fromBucketName(this, "ArtifactsBucket", props.artifactsBucketName);
38
- const roleToBeAssumed = new iam.Role(this, "Role", {
39
- roleName: props.roleName,
40
- assumedBy: new iam.ArnPrincipal(props.callerRoleArn),
41
- });
42
- // Bucked used for input to CodeBuild.
43
- // We let CloudFormation manage the bucket name.
44
- const codebuildBucket = new s3.Bucket(this, "CodebuildSourceBucket", {
45
- encryption: s3.BucketEncryption.S3_MANAGED,
46
- blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,
47
- lifecycleRules: [
48
- {
49
- expiration: cdk.Duration.days(5),
50
- },
51
- ],
52
- });
53
- // The role used for CloudFormation deployment.
54
- const cloudFormationRole = new iam.Role(this, "CloudFormationRole", {
55
- assumedBy: new iam.ServicePrincipal("cloudformation.amazonaws.com"),
56
- managedPolicies: [
57
- // TODO: Can we restrict this a bit more? E.g. look into how Griid has
58
- // limited what the individual stack deployments have permissions to do.
59
- iam.ManagedPolicy.fromAwsManagedPolicyName("AdministratorAccess"),
60
- ],
61
- });
62
- // Replace CodeBuild with ECS task?
63
- // See https://aws.amazon.com/blogs/devops/using-aws-codebuild-to-execute-administrative-tasks/
64
- const codebuildProject = new codebuild.Project(this, "CodebuildProject", {
65
- environment: {
66
- buildImage: props.dockerCredentialsSecretName == null
67
- ? codebuild.LinuxBuildImage.fromDockerRegistry("node:16")
68
- : codebuild.LinuxBuildImage.fromDockerRegistry("node:16", {
69
- secretsManagerCredentials: secretsmanager.Secret.fromSecretNameV2(this, "dockerCredentialsSecretName", props.dockerCredentialsSecretName),
70
- }),
71
- },
72
- buildSpec: codebuild.BuildSpec.fromObject({
73
- version: "0.2",
74
- env: {
75
- variables: {
76
- CDK_DEPLOY_ROLE_ARN: cloudFormationRole.roleArn,
77
- CDK_TOOLKIT_STACK_NAME: props.cdkToolkitStackName,
78
- },
79
- },
80
- phases: {
81
- build: {
82
- commands: [
83
- "npm install -g aws-cdk",
84
- 'cdk --app "$CODEBUILD_SRC_DIR_CLOUDASSEMBLY" --role-arn "$CDK_DEPLOY_ROLE_ARN" --toolkit-stack-name "$CDK_TOOLKIT_STACK_NAME" --require-approval never deploy --exclusively $(cat stack-names.txt)',
85
- ],
86
- },
87
- },
88
- }),
89
- timeout: cdk.Duration.hours(4),
90
- });
91
- // Grant access to CloudFormation.
92
- codebuildProject.addToRolePolicy(new iam.PolicyStatement({
93
- actions: [
94
- // For diff.
95
- "cloudformation:DescribeStacks",
96
- "cloudformation:GetTemplate",
97
- // For deploy.
98
- "cloudformation:CreateChangeSet",
99
- "cloudformation:DeleteStack",
100
- "cloudformation:DescribeChangeSet",
101
- "cloudformation:ExecuteChangeSet",
102
- "cloudformation:DescribeStackEvents",
103
- "cloudformation:DeleteChangeSet",
104
- ],
105
- resources: ["*"],
106
- }));
107
- // Grant access to the CDK Toolkit bucket.
108
- codebuildProject.addToRolePolicy(new iam.PolicyStatement({
109
- actions: [
110
- "s3:GetObject*",
111
- "s3:GetBucket*",
112
- "s3:List*",
113
- "s3:PutObject*",
114
- "s3:Abort*",
115
- "s3:DeleteObject*",
116
- ],
117
- resources: [
118
- `arn:aws:s3:::${props.cdkToolkitStackName.toLowerCase()}-stagingbucket-*`,
119
- ],
120
- }));
121
- artifactsBucket.grantRead(codebuildProject);
122
- cloudFormationRole.grantPassRole(codebuildProject.role);
123
- codebuildBucket.grantReadWrite(codebuildProject);
124
- const startDeployFn = new lambda.Function(this, "StartDeployFunction", {
125
- code: new lambda.InlineCode(`exports.handler = ${start_deploy_handler_1.startDeployHandler.toString()};`),
126
- runtime: lambda.Runtime.NODEJS_16_X,
127
- handler: "index.handler",
128
- functionName: props.startDeployFunctionName,
129
- environment: {
130
- PROJECT_NAME: codebuildProject.projectName,
131
- BUCKET_NAME: codebuildBucket.bucketName,
132
- CDK_CONTEXT: JSON.stringify(props.cdkContext),
133
- },
134
- timeout: cdk.Duration.seconds(30),
135
- });
136
- startDeployFn.grantInvoke(roleToBeAssumed);
137
- codebuildBucket.grantReadWrite(startDeployFn);
138
- startDeployFn.addToRolePolicy(new iam.PolicyStatement({
139
- actions: ["codebuild:StartBuild", "codebuild:BatchGetBuilds"],
140
- resources: [codebuildProject.projectArn],
141
- }));
142
- const statusFn = new lambda.Function(this, "StatusFunction", {
143
- code: new lambda.InlineCode(`exports.handler = ${status_handler_1.statusHandler.toString()};`),
144
- runtime: lambda.Runtime.NODEJS_16_X,
145
- handler: "index.handler",
146
- functionName: props.statusFunctionName,
147
- environment: {
148
- PROJECT_NAME: codebuildProject.projectName,
149
- },
150
- timeout: cdk.Duration.seconds(30),
151
- });
152
- statusFn.grantInvoke(roleToBeAssumed);
153
- statusFn.addToRolePolicy(new iam.PolicyStatement({
154
- actions: ["codebuild:BatchGetBuilds"],
155
- resources: [codebuildProject.projectArn],
156
- }));
157
- statusFn.addToRolePolicy(new iam.PolicyStatement({
158
- actions: ["logs:GetLogEvents"],
159
- resources: [
160
- `arn:aws:logs:${region}:${account}:log-group:/aws/codebuild/${codebuildProject.projectName}:log-stream:*`,
161
- ],
162
- }));
163
- new cdk.CfnOutput(this, "RoleToBeAssumedArn", {
164
- value: roleToBeAssumed.roleArn,
165
- });
166
- new cdk.CfnOutput(this, "StatusFunctionArn", {
167
- value: statusFn.functionArn,
168
- });
169
- new cdk.CfnOutput(this, "StartDeployFunctionArn", {
170
- value: startDeployFn.functionArn,
171
- });
172
- }
173
- }
174
- exports.CdkDeploy = CdkDeploy;
175
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-deploy.js","sourceRoot":"","sources":["../../src/cdk-deploy/cdk-deploy.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as codebuild from \"aws-cdk-lib/aws-codebuild\"\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 cdk from \"aws-cdk-lib\"\nimport * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport { startDeployHandler } from \"./start-deploy-handler\"\nimport { statusHandler } from \"./status-handler\"\n\ninterface Props extends cdk.StackProps {\n  /**\n   * The role that will be granted permission to assume the deploy\n   * role. This role must have permission to assume the deploy role.\n   */\n  callerRoleArn: string\n  /**\n   * The name that will be used for the deploy role. This is the role\n   * that the caller will assume in order to have permission to invoke\n   * the Lambda Functions.\n   */\n  roleName: string\n  /**\n   * The bucket used for storing artifacts. This is used to grant\n   * permission to the role to read artifact. If the bucket is in\n   * another account, it must have a policy which allows the target\n   * account to use IAM permissions from target account.\n   */\n  artifactsBucketName: string\n  startDeployFunctionName: string\n  statusFunctionName: string\n  /**\n   * This is the stack name used with `cdk bootstrap` and can e\n   * found in cdk.json as \"toolkitStackName\".\n   */\n  cdkToolkitStackName: string\n  /**\n   * We pass the CDK context values as they contain feature flags\n   * used by the CDK CLI.\n   */\n  cdkContext: Record<string, string | string[]>\n  /**\n   * The secret containing username and password (or access token)\n   * for a valid docker user. This is used to access private\n   * repositories or to handle docker hub's pull rate limiting.\n   */\n  dockerCredentialsSecretName?: string\n}\n\n/**\n * This construct is responsible for the privileges and logic of\n * automatically deploying stack resources in an account.\n * Its resources are used from a deployment pipeline.\n *\n * The deployment is performed by invoking the \"start deploy\"\n * lambda with details of what should be deployed. As this is\n * responsible for deploying infrastructure, the principal invoking\n * might be able to cause privilege escalation. The principal invoking\n * should be assumed to have full administrator access.\n *\n * The process deploying the infrastructure is locked down so this\n * is only possibly by deployment through CloudFormation, and as\n * such removes a lot of possible escalation paths (e.g. no role\n * can be created by direct API call).\n *\n * The \"status\" lambda can be used to poll for completion, and will\n * also return logs from the job upon completion.\n */\nexport class CdkDeploy extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const account = cdk.Stack.of(this).account\n    const region = cdk.Stack.of(this).region\n\n    const artifactsBucket = s3.Bucket.fromBucketName(\n      this,\n      \"ArtifactsBucket\",\n      props.artifactsBucketName,\n    )\n\n    const roleToBeAssumed = new iam.Role(this, \"Role\", {\n      roleName: props.roleName,\n      assumedBy: new iam.ArnPrincipal(props.callerRoleArn),\n    })\n\n    // Bucked used for input to CodeBuild.\n    // We let CloudFormation manage the bucket name.\n    const codebuildBucket = new s3.Bucket(this, \"CodebuildSourceBucket\", {\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      lifecycleRules: [\n        {\n          expiration: cdk.Duration.days(5),\n        },\n      ],\n    })\n\n    // The role used for CloudFormation deployment.\n    const cloudFormationRole = new iam.Role(this, \"CloudFormationRole\", {\n      assumedBy: new iam.ServicePrincipal(\"cloudformation.amazonaws.com\"),\n      managedPolicies: [\n        // TODO: Can we restrict this a bit more? E.g. look into how Griid has\n        //  limited what the individual stack deployments have permissions to do.\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\"AdministratorAccess\"),\n      ],\n    })\n\n    // Replace CodeBuild with ECS task?\n    // See https://aws.amazon.com/blogs/devops/using-aws-codebuild-to-execute-administrative-tasks/\n    const codebuildProject = new codebuild.Project(this, \"CodebuildProject\", {\n      environment: {\n        buildImage:\n          props.dockerCredentialsSecretName == null\n            ? codebuild.LinuxBuildImage.fromDockerRegistry(\"node:16\")\n            : codebuild.LinuxBuildImage.fromDockerRegistry(\"node:16\", {\n                secretsManagerCredentials:\n                  secretsmanager.Secret.fromSecretNameV2(\n                    this,\n                    \"dockerCredentialsSecretName\",\n                    props.dockerCredentialsSecretName,\n                  ),\n              }),\n      },\n      buildSpec: codebuild.BuildSpec.fromObject({\n        version: \"0.2\",\n        env: {\n          variables: {\n            CDK_DEPLOY_ROLE_ARN: cloudFormationRole.roleArn,\n            CDK_TOOLKIT_STACK_NAME: props.cdkToolkitStackName,\n          },\n        },\n        phases: {\n          build: {\n            commands: [\n              \"npm install -g aws-cdk\",\n              'cdk --app \"$CODEBUILD_SRC_DIR_CLOUDASSEMBLY\" --role-arn \"$CDK_DEPLOY_ROLE_ARN\" --toolkit-stack-name \"$CDK_TOOLKIT_STACK_NAME\" --require-approval never deploy --exclusively $(cat stack-names.txt)',\n            ],\n          },\n        },\n      }),\n      timeout: cdk.Duration.hours(4),\n    })\n\n    // Grant access to CloudFormation.\n    codebuildProject.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\n          // For diff.\n          \"cloudformation:DescribeStacks\",\n          \"cloudformation:GetTemplate\",\n          // For deploy.\n          \"cloudformation:CreateChangeSet\",\n          \"cloudformation:DeleteStack\",\n          \"cloudformation:DescribeChangeSet\",\n          \"cloudformation:ExecuteChangeSet\",\n          \"cloudformation:DescribeStackEvents\",\n          \"cloudformation:DeleteChangeSet\",\n        ],\n        resources: [\"*\"],\n      }),\n    )\n\n    // Grant access to the CDK Toolkit bucket.\n    codebuildProject.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"s3:GetObject*\",\n          \"s3:GetBucket*\",\n          \"s3:List*\",\n          \"s3:PutObject*\",\n          \"s3:Abort*\",\n          \"s3:DeleteObject*\",\n        ],\n        resources: [\n          `arn:aws:s3:::${props.cdkToolkitStackName.toLowerCase()}-stagingbucket-*`,\n        ],\n      }),\n    )\n\n    artifactsBucket.grantRead(codebuildProject)\n\n    cloudFormationRole.grantPassRole(codebuildProject.role!)\n\n    codebuildBucket.grantReadWrite(codebuildProject)\n\n    const startDeployFn = new lambda.Function(this, \"StartDeployFunction\", {\n      code: new lambda.InlineCode(\n        `exports.handler = ${startDeployHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_16_X,\n      handler: \"index.handler\",\n      functionName: props.startDeployFunctionName,\n      environment: {\n        PROJECT_NAME: codebuildProject.projectName,\n        BUCKET_NAME: codebuildBucket.bucketName,\n        CDK_CONTEXT: JSON.stringify(props.cdkContext),\n      },\n      timeout: cdk.Duration.seconds(30),\n    })\n\n    startDeployFn.grantInvoke(roleToBeAssumed)\n    codebuildBucket.grantReadWrite(startDeployFn)\n\n    startDeployFn.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\"codebuild:StartBuild\", \"codebuild:BatchGetBuilds\"],\n        resources: [codebuildProject.projectArn],\n      }),\n    )\n\n    const statusFn = new lambda.Function(this, \"StatusFunction\", {\n      code: new lambda.InlineCode(\n        `exports.handler = ${statusHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_16_X,\n      handler: \"index.handler\",\n      functionName: props.statusFunctionName,\n      environment: {\n        PROJECT_NAME: codebuildProject.projectName,\n      },\n      timeout: cdk.Duration.seconds(30),\n    })\n\n    statusFn.grantInvoke(roleToBeAssumed)\n\n    statusFn.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\"codebuild:BatchGetBuilds\"],\n        resources: [codebuildProject.projectArn],\n      }),\n    )\n\n    statusFn.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\"logs:GetLogEvents\"],\n        resources: [\n          `arn:aws:logs:${region}:${account}:log-group:/aws/codebuild/${codebuildProject.projectName}:log-stream:*`,\n        ],\n      }),\n    )\n\n    new cdk.CfnOutput(this, \"RoleToBeAssumedArn\", {\n      value: roleToBeAssumed.roleArn,\n    })\n    new cdk.CfnOutput(this, \"StatusFunctionArn\", {\n      value: statusFn.functionArn,\n    })\n    new cdk.CfnOutput(this, \"StartDeployFunctionArn\", {\n      value: startDeployFn.functionArn,\n    })\n  }\n}\n"]}