@liflig/cdk 2.18.5 → 2.18.6

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 (140) hide show
  1. package/assets/cloudtrail-slack-integration-lambda/main.py +267 -0
  2. package/assets/pipeline-slack-notification-lambda/index.py +300 -0
  3. package/assets/prepare-cdk-source-lambda/index.py +159 -0
  4. package/assets/slack-alarm-lambda/index.py +103 -0
  5. package/lib/alarms/database-alarms.d.ts +125 -0
  6. package/lib/alarms/database-alarms.js +171 -0
  7. package/lib/alarms/index.d.ts +3 -0
  8. package/lib/alarms/index.js +10 -0
  9. package/lib/alarms/service-alarms.d.ts +145 -0
  10. package/lib/alarms/service-alarms.js +148 -0
  11. package/lib/alarms/ses-alarms.d.ts +67 -0
  12. package/lib/alarms/ses-alarms.js +49 -0
  13. package/lib/alarms/slack-alarm.d.ts +25 -0
  14. package/lib/alarms/slack-alarm.js +47 -0
  15. package/lib/bastion-host.d.ts +41 -0
  16. package/lib/bastion-host.js +86 -0
  17. package/lib/bin/cdk-create-snapshots.d.ts +2 -0
  18. package/lib/bin/fetch-pipeline-variables.d.ts +2 -0
  19. package/lib/build-artifacts/index.d.ts +68 -0
  20. package/lib/build-artifacts/index.js +118 -0
  21. package/lib/cdk-deploy/cdk-deploy.d.ts +63 -0
  22. package/lib/cdk-deploy/cdk-deploy.js +175 -0
  23. package/lib/cdk-deploy/index.d.ts +1 -0
  24. package/lib/cdk-deploy/index.js +6 -0
  25. package/lib/cdk-deploy/start-deploy-handler.d.ts +8 -0
  26. package/lib/cdk-deploy/start-deploy-handler.js +72 -0
  27. package/lib/cdk-deploy/status-handler.d.ts +6 -0
  28. package/lib/cdk-deploy/status-handler.js +83 -0
  29. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +6 -0
  30. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +63 -0
  31. package/lib/cdk-pipelines/index.d.ts +3 -0
  32. package/lib/cdk-pipelines/index.js +10 -0
  33. package/lib/cdk-pipelines/liflig-cdk-pipeline.d.ts +110 -0
  34. package/lib/cdk-pipelines/liflig-cdk-pipeline.js +232 -0
  35. package/lib/cdk-pipelines/slack-notification.d.ts +51 -0
  36. package/lib/cdk-pipelines/slack-notification.js +54 -0
  37. package/lib/cdk-pipelines/variables.d.ts +15 -0
  38. package/lib/cdk-pipelines/variables.js +80 -0
  39. package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.d.ts +47 -0
  40. package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.js +211 -0
  41. package/lib/cloudtrail-slack-integration/index.d.ts +1 -0
  42. package/lib/cloudtrail-slack-integration/index.js +6 -0
  43. package/lib/configure-parameters/configure-parameters.d.ts +61 -0
  44. package/lib/configure-parameters/configure-parameters.js +94 -0
  45. package/lib/configure-parameters/index.d.ts +1 -0
  46. package/lib/configure-parameters/index.js +6 -0
  47. package/lib/cross-region-ssm-parameter.d.ts +13 -0
  48. package/lib/cross-region-ssm-parameter.js +46 -0
  49. package/lib/ecs/cluster.d.ts +25 -0
  50. package/lib/ecs/cluster.js +70 -0
  51. package/lib/ecs/fargate-service.d.ts +63 -0
  52. package/lib/ecs/fargate-service.js +98 -0
  53. package/lib/ecs/index.d.ts +3 -0
  54. package/lib/ecs/index.js +10 -0
  55. package/lib/ecs/listener-rule.d.ts +25 -0
  56. package/lib/ecs/listener-rule.js +27 -0
  57. package/lib/ecs-update-image/artifact-status.d.ts +39 -0
  58. package/lib/ecs-update-image/artifact-status.js +41 -0
  59. package/lib/ecs-update-image/ecs-update-image.d.ts +41 -0
  60. package/lib/ecs-update-image/ecs-update-image.js +98 -0
  61. package/lib/ecs-update-image/index.d.ts +3 -0
  62. package/lib/ecs-update-image/index.js +10 -0
  63. package/lib/ecs-update-image/start-deploy-handler.d.ts +6 -0
  64. package/lib/ecs-update-image/start-deploy-handler.js +104 -0
  65. package/lib/ecs-update-image/status-handler.d.ts +11 -0
  66. package/lib/ecs-update-image/status-handler.js +74 -0
  67. package/lib/ecs-update-image/tag.d.ts +47 -0
  68. package/lib/ecs-update-image/tag.js +67 -0
  69. package/lib/feature-flags.d.ts +18 -0
  70. package/lib/feature-flags.js +48 -0
  71. package/lib/griid/artefact-bucket.d.ts +7 -0
  72. package/lib/griid/artefact-bucket.js +30 -0
  73. package/lib/griid/index.d.ts +4 -0
  74. package/lib/griid/index.js +18 -0
  75. package/lib/hosted-zone-with-param.d.ts +29 -0
  76. package/lib/hosted-zone-with-param.js +65 -0
  77. package/lib/index.d.ts +32 -0
  78. package/lib/kinesis/index.d.ts +1 -0
  79. package/lib/kinesis/index.js +6 -0
  80. package/lib/kinesis/kinesis-to-datadog-stream.d.ts +28 -0
  81. package/lib/kinesis/kinesis-to-datadog-stream.js +126 -0
  82. package/lib/load-balancer/index.d.ts +1 -0
  83. package/lib/load-balancer/index.js +6 -0
  84. package/lib/load-balancer/load-balancer.d.ts +16 -0
  85. package/lib/load-balancer/load-balancer.js +60 -0
  86. package/lib/pipelines/conventions.d.ts +14 -0
  87. package/lib/pipelines/conventions.js +24 -0
  88. package/lib/pipelines/deploy-env.d.ts +18 -0
  89. package/lib/pipelines/deploy-env.js +96 -0
  90. package/lib/pipelines/index.d.ts +2 -0
  91. package/lib/pipelines/index.js +8 -0
  92. package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +13 -0
  93. package/lib/pipelines/liflig-cdk-deployer-deps.js +35 -0
  94. package/lib/pipelines/pipeline.d.ts +78 -0
  95. package/lib/pipelines/pipeline.js +224 -0
  96. package/lib/platform/index.d.ts +1 -0
  97. package/lib/platform/index.js +7 -0
  98. package/lib/platform/platform.d.ts +37 -0
  99. package/lib/platform/platform.js +57 -0
  100. package/lib/rds/database.d.ts +49 -0
  101. package/lib/rds/database.js +60 -0
  102. package/lib/rds/index.d.ts +1 -0
  103. package/lib/rds/index.js +6 -0
  104. package/lib/ses/configurationsetdeliveryoptions/index.d.ts +26 -0
  105. package/lib/ses/configurationsetdeliveryoptions/index.js +48 -0
  106. package/lib/ses/configurationsetsnsdestination/handler.d.ts +17 -0
  107. package/lib/ses/configurationsetsnsdestination/handler.js +75 -0
  108. package/lib/ses/configurationsetsnsdestination/index.d.ts +29 -0
  109. package/lib/ses/configurationsetsnsdestination/index.js +75 -0
  110. package/lib/ses/index.d.ts +4 -0
  111. package/lib/ses/index.js +12 -0
  112. package/lib/ses/sesdomain/handler.d.ts +10 -0
  113. package/lib/ses/sesdomain/handler.js +82 -0
  114. package/lib/ses/sesdomain/index.d.ts +57 -0
  115. package/lib/ses/sesdomain/index.js +94 -0
  116. package/lib/ses/sesverifyemail/handler.d.ts +9 -0
  117. package/lib/ses/sesverifyemail/handler.js +25 -0
  118. package/lib/ses/sesverifyemail/index.d.ts +13 -0
  119. package/lib/ses/sesverifyemail/index.js +51 -0
  120. package/lib/snapshots.d.ts +4 -0
  121. package/lib/snapshots.js +214 -0
  122. package/lib/ssm-parameter-backed-resource.d.ts +45 -0
  123. package/lib/ssm-parameter-backed-resource.js +67 -0
  124. package/lib/ssm-parameter-reader.d.ts +21 -0
  125. package/lib/ssm-parameter-reader.js +48 -0
  126. package/lib/tags.d.ts +8 -0
  127. package/lib/tags.js +36 -0
  128. package/lib/utils.d.ts +2 -0
  129. package/lib/utils.js +17 -0
  130. package/lib/webapp/index.d.ts +3 -0
  131. package/lib/webapp/index.js +10 -0
  132. package/lib/webapp/monitor.d.ts +187 -0
  133. package/lib/webapp/monitor.js +156 -0
  134. package/lib/webapp/security-headers.d.ts +38 -0
  135. package/lib/webapp/security-headers.js +129 -0
  136. package/lib/webapp/webapp.d.ts +116 -0
  137. package/lib/webapp/webapp.js +118 -0
  138. package/lib/webapp-deploy-via-role.d.ts +25 -0
  139. package/lib/webapp-deploy-via-role.js +32 -0
  140. package/package.json +3 -2
@@ -0,0 +1,63 @@
1
+ import * as cdk from "aws-cdk-lib";
2
+ import * as ec2 from "aws-cdk-lib/aws-ec2";
3
+ import * as ecs from "aws-cdk-lib/aws-ecs";
4
+ import * as elb from "aws-cdk-lib/aws-elasticloadbalancingv2";
5
+ import * as logs from "aws-cdk-lib/aws-logs";
6
+ import * as constructs from "constructs";
7
+ import { Parameter } from "../configure-parameters/configure-parameters";
8
+ export interface FargateServiceProps {
9
+ serviceName: string;
10
+ vpc: ec2.IVpc;
11
+ cluster: ecs.ICluster;
12
+ desiredCount: number;
13
+ ecsImage: ecs.ContainerImage;
14
+ /**
15
+ * @default 256
16
+ */
17
+ cpu?: number;
18
+ /**
19
+ * @default 512
20
+ */
21
+ memoryLimitMiB?: number;
22
+ /**
23
+ * @default 2 weeks
24
+ */
25
+ logsRetention?: logs.RetentionDays;
26
+ /**
27
+ * @default 15 seconds
28
+ */
29
+ deregistrationDelay?: cdk.Duration;
30
+ /**
31
+ * @default 8080
32
+ */
33
+ containerPort?: number;
34
+ /**
35
+ * @default 60 seconds
36
+ */
37
+ healthCheckGracePeriod?: cdk.Duration;
38
+ /**
39
+ * Use this as workaround when adding the service to a load balancer after
40
+ * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'
41
+ * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842
42
+ */
43
+ skipHealthCheckGracePeriod?: boolean;
44
+ parameters?: Parameter[];
45
+ overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>;
46
+ overrideHealthCheck?: Partial<elb.HealthCheck>;
47
+ overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>;
48
+ overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>;
49
+ secrets?: Record<string, ecs.Secret>;
50
+ environment?: Record<string, string>;
51
+ /**
52
+ * @default false
53
+ */
54
+ skipTargetGroup?: boolean;
55
+ }
56
+ export declare class FargateService extends constructs.Construct {
57
+ readonly fargateService: ecs.FargateService;
58
+ readonly securityGroup: ec2.SecurityGroup;
59
+ readonly taskDefinition: ecs.TaskDefinition;
60
+ readonly targetGroup: elb.ApplicationTargetGroup | undefined;
61
+ readonly logGroup: logs.LogGroup;
62
+ constructor(scope: constructs.Construct, id: string, props: FargateServiceProps);
63
+ }
@@ -0,0 +1,98 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.FargateService = void 0;
4
+ const cdk = require("aws-cdk-lib");
5
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
6
+ const ec2 = require("aws-cdk-lib/aws-ec2");
7
+ const ecs = require("aws-cdk-lib/aws-ecs");
8
+ const elb = require("aws-cdk-lib/aws-elasticloadbalancingv2");
9
+ const logs = require("aws-cdk-lib/aws-logs");
10
+ const constructs = require("constructs");
11
+ const configure_parameters_1 = require("../configure-parameters");
12
+ class FargateService extends constructs.Construct {
13
+ constructor(scope, id, props) {
14
+ var _a, _b, _c, _d, _e, _f, _g;
15
+ super(scope, id);
16
+ const parameters = new configure_parameters_1.ConfigureParameters(this, {
17
+ ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${this.node.addr}/parameters`,
18
+ parameters: (_a = props.parameters) !== null && _a !== void 0 ? _a : [],
19
+ });
20
+ this.logGroup = new logs.LogGroup(this, "LogGroup", {
21
+ retention: (_b = props.logsRetention) !== null && _b !== void 0 ? _b : logs.RetentionDays.TWO_WEEKS,
22
+ });
23
+ this.securityGroup = new ec2.SecurityGroup(this, "SecurityGroup", {
24
+ vpc: props.vpc,
25
+ });
26
+ this.taskDefinition = new ecs.FargateTaskDefinition(this, "TaskDefinition", {
27
+ cpu: (_c = props.cpu) !== null && _c !== void 0 ? _c : 256,
28
+ memoryLimitMiB: (_d = props.memoryLimitMiB) !== null && _d !== void 0 ? _d : 512,
29
+ });
30
+ parameters.grantRead(this.taskDefinition.taskRole);
31
+ const container = this.taskDefinition.addContainer("Container", {
32
+ logging: ecs.LogDriver.awsLogs({
33
+ logGroup: this.logGroup,
34
+ streamPrefix: "ecs",
35
+ datetimeFormat: "%Y-%m-%dT%H:%M:%S",
36
+ }),
37
+ image: props.ecsImage,
38
+ secrets: props.secrets,
39
+ environment: {
40
+ SSM_PREFIX: parameters.ssmPrefix,
41
+ // Not read by the application, only used to help with redeployments.
42
+ PARAMS_HASH: parameters.hashValue,
43
+ ...((_e = props.environment) !== null && _e !== void 0 ? _e : {}),
44
+ },
45
+ linuxParameters: new ecs.LinuxParameters(this, "LinuxParameters", {
46
+ initProcessEnabled: true,
47
+ }),
48
+ ...props.overrideContainerProps,
49
+ });
50
+ const port = (_f = props.containerPort) !== null && _f !== void 0 ? _f : 8080;
51
+ container.addPortMappings({
52
+ containerPort: port,
53
+ hostPort: port,
54
+ });
55
+ this.fargateService = new ecs.FargateService(this, "Service", {
56
+ serviceName: props.serviceName,
57
+ vpcSubnets: {
58
+ subnetType: ec2.SubnetType.PUBLIC,
59
+ },
60
+ taskDefinition: this.taskDefinition,
61
+ cluster: props.cluster,
62
+ minHealthyPercent: 100,
63
+ healthCheckGracePeriod: props.healthCheckGracePeriod,
64
+ desiredCount: props.desiredCount,
65
+ assignPublicIp: true,
66
+ securityGroups: [this.securityGroup],
67
+ platformVersion: ecs.FargatePlatformVersion.VERSION1_4,
68
+ enableExecuteCommand: true,
69
+ ...props.overrideFargateServiceProps,
70
+ });
71
+ if (props.skipHealthCheckGracePeriod) {
72
+ ;
73
+ this.fargateService.node.defaultChild.healthCheckGracePeriodSeconds = undefined;
74
+ }
75
+ for (const param of parameters.parameters) {
76
+ this.fargateService.node.addDependency(param);
77
+ }
78
+ if (!props.skipTargetGroup) {
79
+ this.targetGroup = new elb.ApplicationTargetGroup(this, "TargetGroup", {
80
+ protocol: elb.ApplicationProtocol.HTTP,
81
+ port: port,
82
+ vpc: props.vpc,
83
+ targetType: elb.TargetType.IP,
84
+ targets: [this.fargateService],
85
+ deregistrationDelay: (_g = props.deregistrationDelay) !== null && _g !== void 0 ? _g : cdk.Duration.seconds(15),
86
+ ...props.overrideTargetGroupProps,
87
+ });
88
+ this.targetGroup.configureHealthCheck({
89
+ interval: aws_cdk_lib_1.Duration.seconds(10),
90
+ path: "/health",
91
+ healthyThresholdCount: 2,
92
+ ...props.overrideHealthCheck,
93
+ });
94
+ }
95
+ }
96
+ }
97
+ exports.FargateService = FargateService;
98
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-service.js","sourceRoot":"","sources":["../../src/ecs/fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport { Duration } from \"aws-cdk-lib\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport { CfnService } from \"aws-cdk-lib/aws-ecs\"\nimport * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\nimport { ConfigureParameters } from \"../configure-parameters\"\nimport { Parameter } from \"../configure-parameters/configure-parameters\"\n\nexport interface FargateServiceProps {\n  serviceName: string\n  vpc: ec2.IVpc\n  cluster: ecs.ICluster\n  desiredCount: number\n  ecsImage: ecs.ContainerImage\n  /**\n   * @default 256\n   */\n  cpu?: number\n  /**\n   * @default 512\n   */\n  memoryLimitMiB?: number\n  /**\n   * @default 2 weeks\n   */\n  logsRetention?: logs.RetentionDays\n  /**\n   * @default 15 seconds\n   */\n  deregistrationDelay?: cdk.Duration\n  /**\n   * @default 8080\n   */\n  containerPort?: number\n  /**\n   * @default 60 seconds\n   */\n  healthCheckGracePeriod?: cdk.Duration\n  /**\n   * Use this as workaround when adding the service to a load balancer after\n   * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'\n   * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842\n   */\n  skipHealthCheckGracePeriod?: boolean\n  parameters?: Parameter[]\n  overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>\n  overrideHealthCheck?: Partial<elb.HealthCheck>\n  overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>\n  overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>\n  secrets?: Record<string, ecs.Secret>\n  environment?: Record<string, string>\n  /**\n   * @default false\n   */\n  skipTargetGroup?: boolean\n}\n\nexport class FargateService extends constructs.Construct {\n  public readonly fargateService: ecs.FargateService\n  public readonly securityGroup: ec2.SecurityGroup\n  public readonly taskDefinition: ecs.TaskDefinition\n  public readonly targetGroup: elb.ApplicationTargetGroup | undefined\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: FargateServiceProps,\n  ) {\n    super(scope, id)\n\n    const parameters = new ConfigureParameters(this, {\n      ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${\n        this.node.addr\n      }/parameters`,\n      parameters: props.parameters ?? [],\n    })\n\n    this.logGroup = new logs.LogGroup(this, \"LogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.TWO_WEEKS,\n    })\n\n    this.securityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: props.vpc,\n    })\n\n    this.taskDefinition = new ecs.FargateTaskDefinition(\n      this,\n      \"TaskDefinition\",\n      {\n        cpu: props.cpu ?? 256,\n        memoryLimitMiB: props.memoryLimitMiB ?? 512,\n      },\n    )\n\n    parameters.grantRead(this.taskDefinition.taskRole)\n\n    const container = this.taskDefinition.addContainer(\"Container\", {\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: this.logGroup,\n        streamPrefix: \"ecs\",\n        datetimeFormat: \"%Y-%m-%dT%H:%M:%S\",\n      }),\n      image: props.ecsImage,\n      secrets: props.secrets,\n      environment: {\n        SSM_PREFIX: parameters.ssmPrefix,\n        // Not read by the application, only used to help with redeployments.\n        PARAMS_HASH: parameters.hashValue,\n        ...(props.environment ?? {}),\n      },\n      linuxParameters: new ecs.LinuxParameters(this, \"LinuxParameters\", {\n        initProcessEnabled: true,\n      }),\n      ...props.overrideContainerProps,\n    })\n\n    const port = props.containerPort ?? 8080\n\n    container.addPortMappings({\n      containerPort: port,\n      hostPort: port,\n    })\n\n    this.fargateService = new ecs.FargateService(this, \"Service\", {\n      serviceName: props.serviceName,\n      vpcSubnets: {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      taskDefinition: this.taskDefinition,\n      cluster: props.cluster,\n      minHealthyPercent: 100,\n      healthCheckGracePeriod: props.healthCheckGracePeriod,\n      desiredCount: props.desiredCount,\n      assignPublicIp: true,\n      securityGroups: [this.securityGroup],\n      platformVersion: ecs.FargatePlatformVersion.VERSION1_4,\n      enableExecuteCommand: true,\n      ...props.overrideFargateServiceProps,\n    })\n\n    if (props.skipHealthCheckGracePeriod) {\n      ;(\n        this.fargateService.node.defaultChild as CfnService\n      ).healthCheckGracePeriodSeconds = undefined\n    }\n\n    for (const param of parameters.parameters) {\n      this.fargateService.node.addDependency(param)\n    }\n\n    if (!props.skipTargetGroup) {\n      this.targetGroup = new elb.ApplicationTargetGroup(this, \"TargetGroup\", {\n        protocol: elb.ApplicationProtocol.HTTP,\n        port: port,\n        vpc: props.vpc,\n        targetType: elb.TargetType.IP,\n        targets: [this.fargateService],\n        deregistrationDelay:\n          props.deregistrationDelay ?? cdk.Duration.seconds(15),\n        ...props.overrideTargetGroupProps,\n      })\n\n      this.targetGroup.configureHealthCheck({\n        interval: Duration.seconds(10),\n        path: \"/health\",\n        healthyThresholdCount: 2,\n        ...props.overrideHealthCheck,\n      })\n    }\n  }\n}\n"]}
@@ -0,0 +1,3 @@
1
+ export { Cluster, ClusterProps } from "./cluster";
2
+ export { FargateService, FargateServiceProps } from "./fargate-service";
3
+ export { ListenerRule, ListenerRuleProps } from "./listener-rule";
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ListenerRule = exports.FargateService = exports.Cluster = void 0;
4
+ var cluster_1 = require("./cluster");
5
+ Object.defineProperty(exports, "Cluster", { enumerable: true, get: function () { return cluster_1.Cluster; } });
6
+ var fargate_service_1 = require("./fargate-service");
7
+ Object.defineProperty(exports, "FargateService", { enumerable: true, get: function () { return fargate_service_1.FargateService; } });
8
+ var listener_rule_1 = require("./listener-rule");
9
+ Object.defineProperty(exports, "ListenerRule", { enumerable: true, get: function () { return listener_rule_1.ListenerRule; } });
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZWNzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHFDQUFpRDtBQUF4QyxrR0FBQSxPQUFPLE9BQUE7QUFDaEIscURBQXVFO0FBQTlELGlIQUFBLGNBQWMsT0FBQTtBQUN2QixpREFBaUU7QUFBeEQsNkdBQUEsWUFBWSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgQ2x1c3RlciwgQ2x1c3RlclByb3BzIH0gZnJvbSBcIi4vY2x1c3RlclwiXG5leHBvcnQgeyBGYXJnYXRlU2VydmljZSwgRmFyZ2F0ZVNlcnZpY2VQcm9wcyB9IGZyb20gXCIuL2ZhcmdhdGUtc2VydmljZVwiXG5leHBvcnQgeyBMaXN0ZW5lclJ1bGUsIExpc3RlbmVyUnVsZVByb3BzIH0gZnJvbSBcIi4vbGlzdGVuZXItcnVsZVwiXG4iXX0=
@@ -0,0 +1,25 @@
1
+ import * as constructs from "constructs";
2
+ import * as elb from "aws-cdk-lib/aws-elasticloadbalancingv2";
3
+ import * as route53 from "aws-cdk-lib/aws-route53";
4
+ export interface ListenerRuleProps {
5
+ httpsListener: elb.IApplicationListener;
6
+ loadBalancer: elb.IApplicationLoadBalancer;
7
+ domainName: string;
8
+ listenerPriority: number;
9
+ targetGroup: elb.IApplicationTargetGroup;
10
+ /**
11
+ * If 'hostedZone' is an A record for 'domainName' is created with
12
+ * the 'loadBalancer' as target
13
+ */
14
+ hostedZone?: route53.IHostedZone;
15
+ }
16
+ export declare class ListenerRule extends constructs.Construct {
17
+ /**
18
+ * The rule created in the ALB Listener.
19
+ *
20
+ * Use {@link elb.ApplicationListenerRule.addCondition} to add [other conditions](https://docs.aws.amazon.com/elasticloadbalancing/latest/application/load-balancer-listeners.html#rule-condition-types)
21
+ * than Host-header.
22
+ */
23
+ readonly applicationListenerRule: elb.ApplicationListenerRule;
24
+ constructor(scope: constructs.Construct, id: string, props: ListenerRuleProps);
25
+ }
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ListenerRule = void 0;
4
+ const constructs = require("constructs");
5
+ const elb = require("aws-cdk-lib/aws-elasticloadbalancingv2");
6
+ const route53 = require("aws-cdk-lib/aws-route53");
7
+ const route53Targets = require("aws-cdk-lib/aws-route53-targets");
8
+ class ListenerRule extends constructs.Construct {
9
+ constructor(scope, id, props) {
10
+ super(scope, id);
11
+ this.applicationListenerRule = new elb.ApplicationListenerRule(this, "ListenerRule", {
12
+ listener: props.httpsListener,
13
+ priority: props.listenerPriority,
14
+ conditions: [elb.ListenerCondition.hostHeaders([props.domainName])],
15
+ targetGroups: [props.targetGroup],
16
+ });
17
+ if (props.hostedZone != null) {
18
+ new route53.ARecord(this, "ARecord", {
19
+ zone: props.hostedZone,
20
+ recordName: `${props.domainName}.`,
21
+ target: route53.RecordTarget.fromAlias(new route53Targets.LoadBalancerTarget(props.loadBalancer)),
22
+ });
23
+ }
24
+ }
25
+ }
26
+ exports.ListenerRule = ListenerRule;
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdGVuZXItcnVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9lY3MvbGlzdGVuZXItcnVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBd0M7QUFDeEMsOERBQTZEO0FBQzdELG1EQUFrRDtBQUNsRCxrRUFBaUU7QUFlakUsTUFBYSxZQUFhLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFTcEQsWUFDRSxLQUEyQixFQUMzQixFQUFVLEVBQ1YsS0FBd0I7UUFFeEIsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUVoQixJQUFJLENBQUMsdUJBQXVCLEdBQUcsSUFBSSxHQUFHLENBQUMsdUJBQXVCLENBQzVELElBQUksRUFDSixjQUFjLEVBQ2Q7WUFDRSxRQUFRLEVBQUUsS0FBSyxDQUFDLGFBQWE7WUFDN0IsUUFBUSxFQUFFLEtBQUssQ0FBQyxnQkFBZ0I7WUFDaEMsVUFBVSxFQUFFLENBQUMsR0FBRyxDQUFDLGlCQUFpQixDQUFDLFdBQVcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDO1lBQ25FLFlBQVksRUFBRSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUM7U0FDbEMsQ0FDRixDQUFBO1FBRUQsSUFBSSxLQUFLLENBQUMsVUFBVSxJQUFJLElBQUksRUFBRSxDQUFDO1lBQzdCLElBQUksT0FBTyxDQUFDLE9BQU8sQ0FBQyxJQUFJLEVBQUUsU0FBUyxFQUFFO2dCQUNuQyxJQUFJLEVBQUUsS0FBSyxDQUFDLFVBQVU7Z0JBQ3RCLFVBQVUsRUFBRSxHQUFHLEtBQUssQ0FBQyxVQUFVLEdBQUc7Z0JBQ2xDLE1BQU0sRUFBRSxPQUFPLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FDcEMsSUFBSSxjQUFjLENBQUMsa0JBQWtCLENBQUMsS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUMxRDthQUNGLENBQUMsQ0FBQTtRQUNKLENBQUM7SUFDSCxDQUFDO0NBQ0Y7QUFyQ0Qsb0NBcUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBlbGIgZnJvbSBcImF3cy1jZGstbGliL2F3cy1lbGFzdGljbG9hZGJhbGFuY2luZ3YyXCJcbmltcG9ydCAqIGFzIHJvdXRlNTMgZnJvbSBcImF3cy1jZGstbGliL2F3cy1yb3V0ZTUzXCJcbmltcG9ydCAqIGFzIHJvdXRlNTNUYXJnZXRzIGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtcm91dGU1My10YXJnZXRzXCJcblxuZXhwb3J0IGludGVyZmFjZSBMaXN0ZW5lclJ1bGVQcm9wcyB7XG4gIGh0dHBzTGlzdGVuZXI6IGVsYi5JQXBwbGljYXRpb25MaXN0ZW5lclxuICBsb2FkQmFsYW5jZXI6IGVsYi5JQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJcbiAgZG9tYWluTmFtZTogc3RyaW5nXG4gIGxpc3RlbmVyUHJpb3JpdHk6IG51bWJlclxuICB0YXJnZXRHcm91cDogZWxiLklBcHBsaWNhdGlvblRhcmdldEdyb3VwXG4gIC8qKlxuICAgKiBJZiAnaG9zdGVkWm9uZScgaXMgYW4gQSByZWNvcmQgZm9yICdkb21haW5OYW1lJyBpcyBjcmVhdGVkIHdpdGhcbiAgICogdGhlICdsb2FkQmFsYW5jZXInIGFzIHRhcmdldFxuICAgKi9cbiAgaG9zdGVkWm9uZT86IHJvdXRlNTMuSUhvc3RlZFpvbmVcbn1cblxuZXhwb3J0IGNsYXNzIExpc3RlbmVyUnVsZSBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgLyoqXG4gICAqIFRoZSBydWxlIGNyZWF0ZWQgaW4gdGhlIEFMQiBMaXN0ZW5lci5cbiAgICpcbiAgICogVXNlIHtAbGluayBlbGIuQXBwbGljYXRpb25MaXN0ZW5lclJ1bGUuYWRkQ29uZGl0aW9ufSB0byBhZGQgW290aGVyIGNvbmRpdGlvbnNdKGh0dHBzOi8vZG9jcy5hd3MuYW1hem9uLmNvbS9lbGFzdGljbG9hZGJhbGFuY2luZy9sYXRlc3QvYXBwbGljYXRpb24vbG9hZC1iYWxhbmNlci1saXN0ZW5lcnMuaHRtbCNydWxlLWNvbmRpdGlvbi10eXBlcylcbiAgICogdGhhbiBIb3N0LWhlYWRlci5cbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBhcHBsaWNhdGlvbkxpc3RlbmVyUnVsZTogZWxiLkFwcGxpY2F0aW9uTGlzdGVuZXJSdWxlXG5cbiAgY29uc3RydWN0b3IoXG4gICAgc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LFxuICAgIGlkOiBzdHJpbmcsXG4gICAgcHJvcHM6IExpc3RlbmVyUnVsZVByb3BzLFxuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmFwcGxpY2F0aW9uTGlzdGVuZXJSdWxlID0gbmV3IGVsYi5BcHBsaWNhdGlvbkxpc3RlbmVyUnVsZShcbiAgICAgIHRoaXMsXG4gICAgICBcIkxpc3RlbmVyUnVsZVwiLFxuICAgICAge1xuICAgICAgICBsaXN0ZW5lcjogcHJvcHMuaHR0cHNMaXN0ZW5lcixcbiAgICAgICAgcHJpb3JpdHk6IHByb3BzLmxpc3RlbmVyUHJpb3JpdHksXG4gICAgICAgIGNvbmRpdGlvbnM6IFtlbGIuTGlzdGVuZXJDb25kaXRpb24uaG9zdEhlYWRlcnMoW3Byb3BzLmRvbWFpbk5hbWVdKV0sXG4gICAgICAgIHRhcmdldEdyb3VwczogW3Byb3BzLnRhcmdldEdyb3VwXSxcbiAgICAgIH0sXG4gICAgKVxuXG4gICAgaWYgKHByb3BzLmhvc3RlZFpvbmUgIT0gbnVsbCkge1xuICAgICAgbmV3IHJvdXRlNTMuQVJlY29yZCh0aGlzLCBcIkFSZWNvcmRcIiwge1xuICAgICAgICB6b25lOiBwcm9wcy5ob3N0ZWRab25lLFxuICAgICAgICByZWNvcmROYW1lOiBgJHtwcm9wcy5kb21haW5OYW1lfS5gLFxuICAgICAgICB0YXJnZXQ6IHJvdXRlNTMuUmVjb3JkVGFyZ2V0LmZyb21BbGlhcyhcbiAgICAgICAgICBuZXcgcm91dGU1M1RhcmdldHMuTG9hZEJhbGFuY2VyVGFyZ2V0KHByb3BzLmxvYWRCYWxhbmNlciksXG4gICAgICAgICksXG4gICAgICB9KVxuICAgIH1cbiAgfVxufVxuIl19
@@ -0,0 +1,39 @@
1
+ interface Props {
2
+ artifactPushedAndTagUpdated: boolean;
3
+ }
4
+ /**
5
+ * The responsibility of this class is to be a Value Object to cause
6
+ * more consistent usage and the importance of an artifact being published.
7
+ *
8
+ * Due to the way we build and deploy ECS services, the ECR image will
9
+ * not be available during the first initial account/service deployment.
10
+ *
11
+ * This causes the following sequence:
12
+ *
13
+ * 1. The ECR repository must be provisioned.
14
+ *
15
+ * 2. The tag container must be provisioned in every target account.
16
+ * Before we can provision the ECS service this must be updated
17
+ * to reference an existing artifact.
18
+ *
19
+ * 3. The build system must produce and upload an artifact.
20
+ *
21
+ * 4. The tag container must be updated in every target account
22
+ * to contain the newly built image. This is easiest done by also
23
+ * provisioning the ECS update image construct as part of step 2,
24
+ * without including service and task definition details (which
25
+ * does not exist yet before we can resolve the ECR image).
26
+ * When the ECS update image construct is provisioned the deployment
27
+ * pipeline can start a deployment, which under this state will
28
+ * only store the updated tag but cause no ECS service update.
29
+ *
30
+ * 5. The ECS service can be provisioned, which will use the image
31
+ * reference in the tag container as the first deployment.
32
+ *
33
+ * 6. The next time a deploy is started, the ECS service will be updated.
34
+ */
35
+ export declare class EcsUpdateImageArtifactStatus {
36
+ readonly artifactPushedAndTagUpdated: boolean;
37
+ constructor(props: Props);
38
+ }
39
+ export {};
@@ -0,0 +1,41 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.EcsUpdateImageArtifactStatus = void 0;
4
+ /**
5
+ * The responsibility of this class is to be a Value Object to cause
6
+ * more consistent usage and the importance of an artifact being published.
7
+ *
8
+ * Due to the way we build and deploy ECS services, the ECR image will
9
+ * not be available during the first initial account/service deployment.
10
+ *
11
+ * This causes the following sequence:
12
+ *
13
+ * 1. The ECR repository must be provisioned.
14
+ *
15
+ * 2. The tag container must be provisioned in every target account.
16
+ * Before we can provision the ECS service this must be updated
17
+ * to reference an existing artifact.
18
+ *
19
+ * 3. The build system must produce and upload an artifact.
20
+ *
21
+ * 4. The tag container must be updated in every target account
22
+ * to contain the newly built image. This is easiest done by also
23
+ * provisioning the ECS update image construct as part of step 2,
24
+ * without including service and task definition details (which
25
+ * does not exist yet before we can resolve the ECR image).
26
+ * When the ECS update image construct is provisioned the deployment
27
+ * pipeline can start a deployment, which under this state will
28
+ * only store the updated tag but cause no ECS service update.
29
+ *
30
+ * 5. The ECS service can be provisioned, which will use the image
31
+ * reference in the tag container as the first deployment.
32
+ *
33
+ * 6. The next time a deploy is started, the ECS service will be updated.
34
+ */
35
+ class EcsUpdateImageArtifactStatus {
36
+ constructor(props) {
37
+ this.artifactPushedAndTagUpdated = props.artifactPushedAndTagUpdated;
38
+ }
39
+ }
40
+ exports.EcsUpdateImageArtifactStatus = EcsUpdateImageArtifactStatus;
41
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXJ0aWZhY3Qtc3RhdHVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Vjcy11cGRhdGUtaW1hZ2UvYXJ0aWZhY3Qtc3RhdHVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUlBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4Qkc7QUFDSCxNQUFhLDRCQUE0QjtJQUd2QyxZQUFZLEtBQVk7UUFDdEIsSUFBSSxDQUFDLDJCQUEyQixHQUFHLEtBQUssQ0FBQywyQkFBMkIsQ0FBQTtJQUN0RSxDQUFDO0NBQ0Y7QUFORCxvRUFNQyIsInNvdXJjZXNDb250ZW50IjpbImludGVyZmFjZSBQcm9wcyB7XG4gIGFydGlmYWN0UHVzaGVkQW5kVGFnVXBkYXRlZDogYm9vbGVhblxufVxuXG4vKipcbiAqIFRoZSByZXNwb25zaWJpbGl0eSBvZiB0aGlzIGNsYXNzIGlzIHRvIGJlIGEgVmFsdWUgT2JqZWN0IHRvIGNhdXNlXG4gKiBtb3JlIGNvbnNpc3RlbnQgdXNhZ2UgYW5kIHRoZSBpbXBvcnRhbmNlIG9mIGFuIGFydGlmYWN0IGJlaW5nIHB1Ymxpc2hlZC5cbiAqXG4gKiBEdWUgdG8gdGhlIHdheSB3ZSBidWlsZCBhbmQgZGVwbG95IEVDUyBzZXJ2aWNlcywgdGhlIEVDUiBpbWFnZSB3aWxsXG4gKiBub3QgYmUgYXZhaWxhYmxlIGR1cmluZyB0aGUgZmlyc3QgaW5pdGlhbCBhY2NvdW50L3NlcnZpY2UgZGVwbG95bWVudC5cbiAqXG4gKiBUaGlzIGNhdXNlcyB0aGUgZm9sbG93aW5nIHNlcXVlbmNlOlxuICpcbiAqIDEuIFRoZSBFQ1IgcmVwb3NpdG9yeSBtdXN0IGJlIHByb3Zpc2lvbmVkLlxuICpcbiAqIDIuIFRoZSB0YWcgY29udGFpbmVyIG11c3QgYmUgcHJvdmlzaW9uZWQgaW4gZXZlcnkgdGFyZ2V0IGFjY291bnQuXG4gKiAgICBCZWZvcmUgd2UgY2FuIHByb3Zpc2lvbiB0aGUgRUNTIHNlcnZpY2UgdGhpcyBtdXN0IGJlIHVwZGF0ZWRcbiAqICAgIHRvIHJlZmVyZW5jZSBhbiBleGlzdGluZyBhcnRpZmFjdC5cbiAqXG4gKiAzLiBUaGUgYnVpbGQgc3lzdGVtIG11c3QgcHJvZHVjZSBhbmQgdXBsb2FkIGFuIGFydGlmYWN0LlxuICpcbiAqIDQuIFRoZSB0YWcgY29udGFpbmVyIG11c3QgYmUgdXBkYXRlZCBpbiBldmVyeSB0YXJnZXQgYWNjb3VudFxuICogICAgdG8gY29udGFpbiB0aGUgbmV3bHkgYnVpbHQgaW1hZ2UuIFRoaXMgaXMgZWFzaWVzdCBkb25lIGJ5IGFsc29cbiAqICAgIHByb3Zpc2lvbmluZyB0aGUgRUNTIHVwZGF0ZSBpbWFnZSBjb25zdHJ1Y3QgYXMgcGFydCBvZiBzdGVwIDIsXG4gKiAgICB3aXRob3V0IGluY2x1ZGluZyBzZXJ2aWNlIGFuZCB0YXNrIGRlZmluaXRpb24gZGV0YWlscyAod2hpY2hcbiAqICAgIGRvZXMgbm90IGV4aXN0IHlldCBiZWZvcmUgd2UgY2FuIHJlc29sdmUgdGhlIEVDUiBpbWFnZSkuXG4gKiAgICBXaGVuIHRoZSBFQ1MgdXBkYXRlIGltYWdlIGNvbnN0cnVjdCBpcyBwcm92aXNpb25lZCB0aGUgZGVwbG95bWVudFxuICogICAgcGlwZWxpbmUgY2FuIHN0YXJ0IGEgZGVwbG95bWVudCwgd2hpY2ggdW5kZXIgdGhpcyBzdGF0ZSB3aWxsXG4gKiAgICBvbmx5IHN0b3JlIHRoZSB1cGRhdGVkIHRhZyBidXQgY2F1c2Ugbm8gRUNTIHNlcnZpY2UgdXBkYXRlLlxuICpcbiAqIDUuIFRoZSBFQ1Mgc2VydmljZSBjYW4gYmUgcHJvdmlzaW9uZWQsIHdoaWNoIHdpbGwgdXNlIHRoZSBpbWFnZVxuICogICAgcmVmZXJlbmNlIGluIHRoZSB0YWcgY29udGFpbmVyIGFzIHRoZSBmaXJzdCBkZXBsb3ltZW50LlxuICpcbiAqIDYuIFRoZSBuZXh0IHRpbWUgYSBkZXBsb3kgaXMgc3RhcnRlZCwgdGhlIEVDUyBzZXJ2aWNlIHdpbGwgYmUgdXBkYXRlZC5cbiAqL1xuZXhwb3J0IGNsYXNzIEVjc1VwZGF0ZUltYWdlQXJ0aWZhY3RTdGF0dXMge1xuICByZWFkb25seSBhcnRpZmFjdFB1c2hlZEFuZFRhZ1VwZGF0ZWQ6IGJvb2xlYW5cblxuICBjb25zdHJ1Y3Rvcihwcm9wczogUHJvcHMpIHtcbiAgICB0aGlzLmFydGlmYWN0UHVzaGVkQW5kVGFnVXBkYXRlZCA9IHByb3BzLmFydGlmYWN0UHVzaGVkQW5kVGFnVXBkYXRlZFxuICB9XG59XG4iXX0=
@@ -0,0 +1,41 @@
1
+ import * as constructs from "constructs";
2
+ import * as ecr from "aws-cdk-lib/aws-ecr";
3
+ import * as ecs from "aws-cdk-lib/aws-ecs";
4
+ import * as iam from "aws-cdk-lib/aws-iam";
5
+ import { EcsUpdateImageTag } from "./tag";
6
+ interface Props {
7
+ cluster: ecs.ICluster;
8
+ service?: ecs.IService;
9
+ taskRole?: iam.IRole;
10
+ executionRole?: iam.IRole;
11
+ /**
12
+ * The role that will be granted permission to assume the deploy
13
+ * role. This role must have permission to assume the deploy role.
14
+ */
15
+ callerRoleArn: string;
16
+ /**
17
+ * The name that will be used for the deploy role. This is the role
18
+ * that the caller will assume in order to have permission to invoke
19
+ * the Lambda Functions.
20
+ */
21
+ roleName: string;
22
+ startDeployFunctionName: string;
23
+ statusFunctionName: string;
24
+ tagContainer: EcsUpdateImageTag;
25
+ ecrRepository: ecr.IRepository;
26
+ }
27
+ /**
28
+ * Dynamically update image of an ECS Service outside CDK.
29
+ *
30
+ * This construct sets up a Lambda Function that can be invoked
31
+ * from the CD pipeline to update the ECS Service to a new image.
32
+ *
33
+ * To keep resources in sync, a Secret is used to keep track of
34
+ * the current image, which is dynamically resolved in the CloudFormation
35
+ * templates to avoid infrastructure changes to cause an older/different
36
+ * image to be used.
37
+ */
38
+ export declare class EcsUpdateImage extends constructs.Construct {
39
+ constructor(scope: constructs.Construct, id: string, props: Props);
40
+ }
41
+ export {};
@@ -0,0 +1,98 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.EcsUpdateImage = void 0;
4
+ const constructs = require("constructs");
5
+ const iam = require("aws-cdk-lib/aws-iam");
6
+ const lambda = require("aws-cdk-lib/aws-lambda");
7
+ const cdk = require("aws-cdk-lib");
8
+ const start_deploy_handler_1 = require("./start-deploy-handler");
9
+ const status_handler_1 = require("./status-handler");
10
+ /**
11
+ * Dynamically update image of an ECS Service outside CDK.
12
+ *
13
+ * This construct sets up a Lambda Function that can be invoked
14
+ * from the CD pipeline to update the ECS Service to a new image.
15
+ *
16
+ * To keep resources in sync, a Secret is used to keep track of
17
+ * the current image, which is dynamically resolved in the CloudFormation
18
+ * templates to avoid infrastructure changes to cause an older/different
19
+ * image to be used.
20
+ */
21
+ class EcsUpdateImage extends constructs.Construct {
22
+ constructor(scope, id, props) {
23
+ super(scope, id);
24
+ const account = cdk.Stack.of(this).account;
25
+ const region = cdk.Stack.of(this).region;
26
+ const roleToBeAssumed = new iam.Role(this, "Role", {
27
+ roleName: props.roleName,
28
+ assumedBy: new iam.ArnPrincipal(props.callerRoleArn),
29
+ });
30
+ const startDeployFn = new lambda.Function(this, "StartDeployFunction", {
31
+ functionName: props.startDeployFunctionName,
32
+ code: new lambda.InlineCode(`exports.handler = ${start_deploy_handler_1.startDeployHandler.toString()};`),
33
+ runtime: lambda.Runtime.NODEJS_16_X,
34
+ handler: "index.handler",
35
+ timeout: cdk.Duration.seconds(60),
36
+ environment: {
37
+ CLUSTER_NAME: props.cluster.clusterName,
38
+ SERVICE_NAME: props.service == null ? "" : props.service.serviceName,
39
+ REPOSITORY_URL: props.ecrRepository.repositoryUri,
40
+ ECR_TAG_SECRET_ARN: props.tagContainer.secretArn,
41
+ },
42
+ reservedConcurrentExecutions: 1,
43
+ });
44
+ startDeployFn.grantInvoke(roleToBeAssumed);
45
+ props.tagContainer.grantUpdate(startDeployFn);
46
+ if (props.service != null) {
47
+ startDeployFn.addToRolePolicy(new iam.PolicyStatement({
48
+ actions: [
49
+ "ecs:DeregisterTaskDefinition",
50
+ "ecs:DescribeServices",
51
+ "ecs:DescribeTaskDefinition",
52
+ "ecs:DescribeTasks",
53
+ "ecs:ListTasks",
54
+ "ecs:ListTaskDefinitions",
55
+ "ecs:RegisterTaskDefinition",
56
+ ],
57
+ resources: ["*"],
58
+ }));
59
+ startDeployFn.addToRolePolicy(new iam.PolicyStatement({
60
+ actions: ["ecs:UpdateService"],
61
+ resources: [
62
+ `arn:aws:ecs:${region}:${account}:service/${props.cluster.clusterName}/${props.service.serviceName}`,
63
+ ],
64
+ }));
65
+ }
66
+ if (props.taskRole != null) {
67
+ props.taskRole.grantPassRole(startDeployFn.role);
68
+ }
69
+ if (props.executionRole != null) {
70
+ props.executionRole.grantPassRole(startDeployFn.role);
71
+ }
72
+ const statusFn = new lambda.Function(this, "StatusFunction", {
73
+ functionName: props.statusFunctionName,
74
+ code: new lambda.InlineCode(`exports.handler = ${status_handler_1.statusHandler.toString()};`),
75
+ runtime: lambda.Runtime.NODEJS_16_X,
76
+ handler: "index.handler",
77
+ timeout: cdk.Duration.seconds(60),
78
+ environment: {
79
+ CLUSTER_NAME: props.cluster.clusterName,
80
+ SERVICE_NAME: props.service == null ? "" : props.service.serviceName,
81
+ },
82
+ });
83
+ statusFn.grantInvoke(roleToBeAssumed);
84
+ statusFn.addToRolePolicy(new iam.PolicyStatement({
85
+ actions: [
86
+ "ecs:DeregisterTaskDefinition",
87
+ "ecs:DescribeServices",
88
+ "ecs:DescribeTaskDefinition",
89
+ "ecs:DescribeTasks",
90
+ "ecs:ListTasks",
91
+ "ecs:ListTaskDefinitions",
92
+ ],
93
+ resources: ["*"],
94
+ }));
95
+ }
96
+ }
97
+ exports.EcsUpdateImage = EcsUpdateImage;
98
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-update-image.js","sourceRoot":"","sources":["../../src/ecs-update-image/ecs-update-image.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ecr from \"aws-cdk-lib/aws-ecr\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as cdk from \"aws-cdk-lib\"\nimport { startDeployHandler } from \"./start-deploy-handler\"\nimport { statusHandler } from \"./status-handler\"\nimport { EcsUpdateImageTag } from \"./tag\"\n\ninterface Props {\n  cluster: ecs.ICluster\n  service?: ecs.IService\n  taskRole?: iam.IRole\n  executionRole?: iam.IRole\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  startDeployFunctionName: string\n  statusFunctionName: string\n  tagContainer: EcsUpdateImageTag\n  ecrRepository: ecr.IRepository\n}\n\n/**\n * Dynamically update image of an ECS Service outside CDK.\n *\n * This construct sets up a Lambda Function that can be invoked\n * from the CD pipeline to update the ECS Service to a new image.\n *\n * To keep resources in sync, a Secret is used to keep track of\n * the current image, which is dynamically resolved in the CloudFormation\n * templates to avoid infrastructure changes to cause an older/different\n * image to be used.\n */\nexport class EcsUpdateImage 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 roleToBeAssumed = new iam.Role(this, \"Role\", {\n      roleName: props.roleName,\n      assumedBy: new iam.ArnPrincipal(props.callerRoleArn),\n    })\n\n    const startDeployFn = new lambda.Function(this, \"StartDeployFunction\", {\n      functionName: props.startDeployFunctionName,\n      code: new lambda.InlineCode(\n        `exports.handler = ${startDeployHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_16_X,\n      handler: \"index.handler\",\n      timeout: cdk.Duration.seconds(60),\n      environment: {\n        CLUSTER_NAME: props.cluster.clusterName,\n        SERVICE_NAME: props.service == null ? \"\" : props.service.serviceName,\n        REPOSITORY_URL: props.ecrRepository.repositoryUri,\n        ECR_TAG_SECRET_ARN: props.tagContainer.secretArn,\n      },\n      reservedConcurrentExecutions: 1,\n    })\n\n    startDeployFn.grantInvoke(roleToBeAssumed)\n    props.tagContainer.grantUpdate(startDeployFn)\n\n    if (props.service != null) {\n      startDeployFn.addToRolePolicy(\n        new iam.PolicyStatement({\n          actions: [\n            \"ecs:DeregisterTaskDefinition\",\n            \"ecs:DescribeServices\",\n            \"ecs:DescribeTaskDefinition\",\n            \"ecs:DescribeTasks\",\n            \"ecs:ListTasks\",\n            \"ecs:ListTaskDefinitions\",\n            \"ecs:RegisterTaskDefinition\",\n          ],\n          resources: [\"*\"],\n        }),\n      )\n\n      startDeployFn.addToRolePolicy(\n        new iam.PolicyStatement({\n          actions: [\"ecs:UpdateService\"],\n          resources: [\n            `arn:aws:ecs:${region}:${account}:service/${props.cluster.clusterName}/${props.service.serviceName}`,\n          ],\n        }),\n      )\n    }\n\n    if (props.taskRole != null) {\n      props.taskRole.grantPassRole(startDeployFn.role!)\n    }\n\n    if (props.executionRole != null) {\n      props.executionRole.grantPassRole(startDeployFn.role!)\n    }\n\n    const statusFn = new lambda.Function(this, \"StatusFunction\", {\n      functionName: props.statusFunctionName,\n      code: new lambda.InlineCode(\n        `exports.handler = ${statusHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_16_X,\n      handler: \"index.handler\",\n      timeout: cdk.Duration.seconds(60),\n      environment: {\n        CLUSTER_NAME: props.cluster.clusterName,\n        SERVICE_NAME: props.service == null ? \"\" : props.service.serviceName,\n      },\n    })\n\n    statusFn.grantInvoke(roleToBeAssumed)\n\n    statusFn.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecs:DeregisterTaskDefinition\",\n          \"ecs:DescribeServices\",\n          \"ecs:DescribeTaskDefinition\",\n          \"ecs:DescribeTasks\",\n          \"ecs:ListTasks\",\n          \"ecs:ListTaskDefinitions\",\n        ],\n        resources: [\"*\"],\n      }),\n    )\n  }\n}\n"]}
@@ -0,0 +1,3 @@
1
+ export { EcsUpdateImageArtifactStatus } from "./artifact-status";
2
+ export { EcsUpdateImage } from "./ecs-update-image";
3
+ export { EcsUpdateImageTag } from "./tag";
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.EcsUpdateImageTag = exports.EcsUpdateImage = exports.EcsUpdateImageArtifactStatus = void 0;
4
+ var artifact_status_1 = require("./artifact-status");
5
+ Object.defineProperty(exports, "EcsUpdateImageArtifactStatus", { enumerable: true, get: function () { return artifact_status_1.EcsUpdateImageArtifactStatus; } });
6
+ var ecs_update_image_1 = require("./ecs-update-image");
7
+ Object.defineProperty(exports, "EcsUpdateImage", { enumerable: true, get: function () { return ecs_update_image_1.EcsUpdateImage; } });
8
+ var tag_1 = require("./tag");
9
+ Object.defineProperty(exports, "EcsUpdateImageTag", { enumerable: true, get: function () { return tag_1.EcsUpdateImageTag; } });
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZWNzLXVwZGF0ZS1pbWFnZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxxREFBZ0U7QUFBdkQsK0hBQUEsNEJBQTRCLE9BQUE7QUFDckMsdURBQW1EO0FBQTFDLGtIQUFBLGNBQWMsT0FBQTtBQUN2Qiw2QkFBeUM7QUFBaEMsd0dBQUEsaUJBQWlCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBFY3NVcGRhdGVJbWFnZUFydGlmYWN0U3RhdHVzIH0gZnJvbSBcIi4vYXJ0aWZhY3Qtc3RhdHVzXCJcbmV4cG9ydCB7IEVjc1VwZGF0ZUltYWdlIH0gZnJvbSBcIi4vZWNzLXVwZGF0ZS1pbWFnZVwiXG5leHBvcnQgeyBFY3NVcGRhdGVJbWFnZVRhZyB9IGZyb20gXCIuL3RhZ1wiXG4iXX0=
@@ -0,0 +1,6 @@
1
+ import type { Handler } from "aws-lambda";
2
+ interface ExpectedInput {
3
+ tag: string;
4
+ }
5
+ export declare const startDeployHandler: Handler<Partial<ExpectedInput>>;
6
+ export {};