@liflig/cdk 2.21.3 → 2.21.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/index.d.ts +1 -4
- package/lib/index.js +2 -6
- package/package.json +12 -12
- package/lib/cdk-deploy/cdk-deploy.d.ts +0 -63
- package/lib/cdk-deploy/cdk-deploy.js +0 -174
- package/lib/cdk-deploy/index.d.ts +0 -1
- package/lib/cdk-deploy/index.js +0 -6
- package/lib/cdk-deploy/start-deploy-handler.d.ts +0 -8
- package/lib/cdk-deploy/start-deploy-handler.js +0 -65
- package/lib/cdk-deploy/status-handler.d.ts +0 -6
- package/lib/cdk-deploy/status-handler.js +0 -80
- package/lib/ecs-update-image/artifact-status.d.ts +0 -39
- package/lib/ecs-update-image/artifact-status.js +0 -41
- package/lib/ecs-update-image/ecs-update-image.d.ts +0 -41
- package/lib/ecs-update-image/ecs-update-image.js +0 -97
- package/lib/ecs-update-image/index.d.ts +0 -3
- package/lib/ecs-update-image/index.js +0 -10
- package/lib/ecs-update-image/start-deploy-handler.d.ts +0 -6
- package/lib/ecs-update-image/start-deploy-handler.js +0 -95
- package/lib/ecs-update-image/status-handler.d.ts +0 -11
- package/lib/ecs-update-image/status-handler.js +0 -67
- package/lib/ecs-update-image/tag.d.ts +0 -47
- package/lib/ecs-update-image/tag.js +0 -67
- package/lib/pipelines/conventions.d.ts +0 -14
- package/lib/pipelines/conventions.js +0 -24
- package/lib/pipelines/deploy-env.d.ts +0 -18
- package/lib/pipelines/deploy-env.js +0 -96
- package/lib/pipelines/index.d.ts +0 -2
- package/lib/pipelines/index.js +0 -8
- package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +0 -13
- package/lib/pipelines/liflig-cdk-deployer-deps.js +0 -35
- package/lib/pipelines/pipeline.d.ts +0 -78
- package/lib/pipelines/pipeline.js +0 -229
|
@@ -1,96 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.DeployEnv = void 0;
|
|
4
|
-
const constructs = require("constructs");
|
|
5
|
-
const ec2 = require("aws-cdk-lib/aws-ec2");
|
|
6
|
-
const ecr = require("aws-cdk-lib/aws-ecr");
|
|
7
|
-
const ecs = require("aws-cdk-lib/aws-ecs");
|
|
8
|
-
const iam = require("aws-cdk-lib/aws-iam");
|
|
9
|
-
const logs = require("aws-cdk-lib/aws-logs");
|
|
10
|
-
const sfn = require("aws-cdk-lib/aws-stepfunctions");
|
|
11
|
-
const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
12
|
-
const cdk = require("aws-cdk-lib");
|
|
13
|
-
const conventions_1 = require("./conventions");
|
|
14
|
-
class DeployEnv extends constructs.Construct {
|
|
15
|
-
constructor(scope, id, props) {
|
|
16
|
-
super(scope, id);
|
|
17
|
-
const cluster = this.getOrCreateCluster(props.vpc);
|
|
18
|
-
// We don't reuse the task definition across multiple pipelines
|
|
19
|
-
// so that we can easier find the correct logs for each pipeline.
|
|
20
|
-
const taskDefinition = new ecs.TaskDefinition(this, "TaskDefinition", {
|
|
21
|
-
memoryMiB: "1024",
|
|
22
|
-
cpu: "256",
|
|
23
|
-
compatibility: ecs.Compatibility.FARGATE,
|
|
24
|
-
});
|
|
25
|
-
const containerDefinition = taskDefinition.addContainer("app", {
|
|
26
|
-
image: ecs.ContainerImage.fromEcrRepository(ecr.Repository.fromRepositoryArn(this, "Repository",
|
|
27
|
-
// See https://github.com/capralifecycle/liflig-cdk-deployer
|
|
28
|
-
"arn:aws:ecr:eu-west-1:001112238813:repository/incub-common-liflig-cdk-deployer"), "1-experimental.2"),
|
|
29
|
-
logging: ecs.LogDriver.awsLogs({
|
|
30
|
-
logGroup: new logs.LogGroup(this, "LogGroup", {
|
|
31
|
-
removalPolicy: cdk.RemovalPolicy.DESTROY,
|
|
32
|
-
retention: logs.RetentionDays.ONE_MONTH,
|
|
33
|
-
}),
|
|
34
|
-
streamPrefix: "app",
|
|
35
|
-
}),
|
|
36
|
-
});
|
|
37
|
-
const cdkRole = iam.Role.fromRoleArn(this, `CdkRole-${props.envName}`, `arn:aws:iam::${props.accountId}:role/${conventions_1.cdkDeployRoleName}`);
|
|
38
|
-
cdkRole.grant(taskDefinition.taskRole, "sts:AssumeRole");
|
|
39
|
-
props.artefactBucket.grantRead(taskDefinition.taskRole);
|
|
40
|
-
this.chain = sfn.Chain.start(new tasks.EcsRunTask(this, `Deploy ${props.envName}`, {
|
|
41
|
-
resultPath: sfn.JsonPath.DISCARD,
|
|
42
|
-
securityGroups: [this.getOrCreateTaskSecurityGroup(props.vpc)],
|
|
43
|
-
integrationPattern: sfn.IntegrationPattern.RUN_JOB,
|
|
44
|
-
cluster,
|
|
45
|
-
assignPublicIp: true,
|
|
46
|
-
launchTarget: new tasks.EcsFargateLaunchTarget(),
|
|
47
|
-
taskDefinition,
|
|
48
|
-
containerOverrides: [
|
|
49
|
-
{
|
|
50
|
-
containerDefinition,
|
|
51
|
-
environment: [
|
|
52
|
-
{
|
|
53
|
-
name: "CDK_TARGET_ROLE_ARN",
|
|
54
|
-
value: cdkRole.roleArn,
|
|
55
|
-
},
|
|
56
|
-
{
|
|
57
|
-
name: "CDK_ENV_NAME",
|
|
58
|
-
value: props.envName,
|
|
59
|
-
},
|
|
60
|
-
{
|
|
61
|
-
name: "CDK_CLOUD_ASSEMBLY",
|
|
62
|
-
value: sfn.JsonPath.stringAt("$.CloudAssembly"),
|
|
63
|
-
},
|
|
64
|
-
{
|
|
65
|
-
name: "CDK_VARIABLES",
|
|
66
|
-
value: sfn.JsonPath.stringAt("$.Variables"),
|
|
67
|
-
},
|
|
68
|
-
],
|
|
69
|
-
},
|
|
70
|
-
],
|
|
71
|
-
}));
|
|
72
|
-
if (props.afterSuccessfulDeploy != null) {
|
|
73
|
-
this.chain = this.chain.next(props.afterSuccessfulDeploy);
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
// Reuse ECS cluster for multiple pipelines in same stack.
|
|
77
|
-
getOrCreateCluster(vpc) {
|
|
78
|
-
var _a;
|
|
79
|
-
const stack = cdk.Stack.of(this);
|
|
80
|
-
const uniqueId = "pipeline.04ad36b1.cluster";
|
|
81
|
-
return ((_a = stack.node.tryFindChild(uniqueId)) !== null && _a !== void 0 ? _a : new ecs.Cluster(stack, uniqueId, {
|
|
82
|
-
vpc,
|
|
83
|
-
}));
|
|
84
|
-
}
|
|
85
|
-
// Reuse security group for multiple pipelines in same stack.
|
|
86
|
-
getOrCreateTaskSecurityGroup(vpc) {
|
|
87
|
-
var _a;
|
|
88
|
-
const stack = cdk.Stack.of(this);
|
|
89
|
-
const uniqueId = "pipeline.04ad36b1.security-group";
|
|
90
|
-
return ((_a = stack.node.tryFindChild(uniqueId)) !== null && _a !== void 0 ? _a : new ec2.SecurityGroup(stack, uniqueId, {
|
|
91
|
-
vpc,
|
|
92
|
-
}));
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
exports.DeployEnv = DeployEnv;
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-env.js","sourceRoot":"","sources":["../../src/pipelines/deploy-env.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\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 logs from \"aws-cdk-lib/aws-logs\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as sfn from \"aws-cdk-lib/aws-stepfunctions\"\nimport * as tasks from \"aws-cdk-lib/aws-stepfunctions-tasks\"\nimport * as cdk from \"aws-cdk-lib\"\nimport { cdkDeployRoleName } from \"./conventions\"\n\ninterface DeployEnvProps {\n  accountId: string\n  afterSuccessfulDeploy?: sfn.Chain\n  artefactBucket: s3.IBucket\n  envName: string\n  vpc: ec2.IVpc\n}\n\nexport class DeployEnv extends constructs.Construct {\n  public chain: sfn.Chain\n\n  constructor(scope: constructs.Construct, id: string, props: DeployEnvProps) {\n    super(scope, id)\n\n    const cluster = this.getOrCreateCluster(props.vpc)\n\n    // We don't reuse the task definition across multiple pipelines\n    // so that we can easier find the correct logs for each pipeline.\n\n    const taskDefinition = new ecs.TaskDefinition(this, \"TaskDefinition\", {\n      memoryMiB: \"1024\",\n      cpu: \"256\",\n      compatibility: ecs.Compatibility.FARGATE,\n    })\n\n    const containerDefinition = taskDefinition.addContainer(\"app\", {\n      image: ecs.ContainerImage.fromEcrRepository(\n        ecr.Repository.fromRepositoryArn(\n          this,\n          \"Repository\",\n          // See https://github.com/capralifecycle/liflig-cdk-deployer\n          \"arn:aws:ecr:eu-west-1:001112238813:repository/incub-common-liflig-cdk-deployer\",\n        ),\n        \"1-experimental.2\",\n      ),\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: new logs.LogGroup(this, \"LogGroup\", {\n          removalPolicy: cdk.RemovalPolicy.DESTROY,\n          retention: logs.RetentionDays.ONE_MONTH,\n        }),\n        streamPrefix: \"app\",\n      }),\n    })\n\n    const cdkRole = iam.Role.fromRoleArn(\n      this,\n      `CdkRole-${props.envName}`,\n      `arn:aws:iam::${props.accountId}:role/${cdkDeployRoleName}`,\n    )\n\n    cdkRole.grant(taskDefinition.taskRole, \"sts:AssumeRole\")\n\n    props.artefactBucket.grantRead(taskDefinition.taskRole)\n\n    this.chain = sfn.Chain.start(\n      new tasks.EcsRunTask(this, `Deploy ${props.envName}`, {\n        resultPath: sfn.JsonPath.DISCARD,\n        securityGroups: [this.getOrCreateTaskSecurityGroup(props.vpc)],\n        integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n        cluster,\n        assignPublicIp: true,\n        launchTarget: new tasks.EcsFargateLaunchTarget(),\n        taskDefinition,\n        containerOverrides: [\n          {\n            containerDefinition,\n            environment: [\n              {\n                name: \"CDK_TARGET_ROLE_ARN\",\n                value: cdkRole.roleArn,\n              },\n              {\n                name: \"CDK_ENV_NAME\",\n                value: props.envName,\n              },\n              {\n                name: \"CDK_CLOUD_ASSEMBLY\",\n                value: sfn.JsonPath.stringAt(\"$.CloudAssembly\"),\n              },\n              {\n                name: \"CDK_VARIABLES\",\n                value: sfn.JsonPath.stringAt(\"$.Variables\"),\n              },\n            ],\n          },\n        ],\n      }),\n    )\n\n    if (props.afterSuccessfulDeploy != null) {\n      this.chain = this.chain.next(props.afterSuccessfulDeploy)\n    }\n  }\n\n  // Reuse ECS cluster for multiple pipelines in same stack.\n  private getOrCreateCluster(vpc: ec2.IVpc): ecs.Cluster {\n    const stack = cdk.Stack.of(this)\n    const uniqueId = \"pipeline.04ad36b1.cluster\"\n    return (\n      (stack.node.tryFindChild(uniqueId) as ecs.Cluster) ??\n      new ecs.Cluster(stack, uniqueId, {\n        vpc,\n      })\n    )\n  }\n\n  // Reuse security group for multiple pipelines in same stack.\n  private getOrCreateTaskSecurityGroup(vpc: ec2.IVpc): ec2.SecurityGroup {\n    const stack = cdk.Stack.of(this)\n    const uniqueId = \"pipeline.04ad36b1.security-group\"\n    return (\n      (stack.node.tryFindChild(uniqueId) as ec2.SecurityGroup) ??\n      new ec2.SecurityGroup(stack, uniqueId, {\n        vpc,\n      })\n    )\n  }\n}\n"]}
|
package/lib/pipelines/index.d.ts
DELETED
package/lib/pipelines/index.js
DELETED
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.Pipeline = exports.LifligCdkDeployerDeps = void 0;
|
|
4
|
-
var liflig_cdk_deployer_deps_1 = require("./liflig-cdk-deployer-deps");
|
|
5
|
-
Object.defineProperty(exports, "LifligCdkDeployerDeps", { enumerable: true, get: function () { return liflig_cdk_deployer_deps_1.LifligCdkDeployerDeps; } });
|
|
6
|
-
var pipeline_1 = require("./pipeline");
|
|
7
|
-
Object.defineProperty(exports, "Pipeline", { enumerable: true, get: function () { return pipeline_1.Pipeline; } });
|
|
8
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGlwZWxpbmVzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHVFQUdtQztBQUZqQyxpSUFBQSxxQkFBcUIsT0FBQTtBQUd2Qix1Q0FBeUU7QUFBaEUsb0dBQUEsUUFBUSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHtcbiAgTGlmbGlnQ2RrRGVwbG95ZXJEZXBzLFxuICBMaWZsaWdDZGtEZXBsb3llckRlcHNQcm9wcyxcbn0gZnJvbSBcIi4vbGlmbGlnLWNkay1kZXBsb3llci1kZXBzXCJcbmV4cG9ydCB7IFBpcGVsaW5lLCBQaXBlbGluZUVudmlyb25tZW50LCBQaXBlbGluZVByb3BzIH0gZnJvbSBcIi4vcGlwZWxpbmVcIlxuIl19
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
import * as constructs from "constructs";
|
|
2
|
-
export interface LifligCdkDeployerDepsProps {
|
|
3
|
-
trustedAccountIds: string[];
|
|
4
|
-
}
|
|
5
|
-
/**
|
|
6
|
-
* Resources needed so liflig-cdk-deployer can deploy to the account.
|
|
7
|
-
*
|
|
8
|
-
* This must exist in each target account that the pipeline should
|
|
9
|
-
* be able to deploy into.
|
|
10
|
-
*/
|
|
11
|
-
export declare class LifligCdkDeployerDeps extends constructs.Construct {
|
|
12
|
-
constructor(scope: constructs.Construct, id: string, props: LifligCdkDeployerDepsProps);
|
|
13
|
-
}
|
|
@@ -1,35 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.LifligCdkDeployerDeps = void 0;
|
|
4
|
-
const constructs = require("constructs");
|
|
5
|
-
const iam = require("aws-cdk-lib/aws-iam");
|
|
6
|
-
const cdk = require("aws-cdk-lib");
|
|
7
|
-
const conventions_1 = require("./conventions");
|
|
8
|
-
/**
|
|
9
|
-
* Resources needed so liflig-cdk-deployer can deploy to the account.
|
|
10
|
-
*
|
|
11
|
-
* This must exist in each target account that the pipeline should
|
|
12
|
-
* be able to deploy into.
|
|
13
|
-
*/
|
|
14
|
-
class LifligCdkDeployerDeps extends constructs.Construct {
|
|
15
|
-
constructor(scope, id, props) {
|
|
16
|
-
super(scope, id);
|
|
17
|
-
const account = cdk.Stack.of(this).account;
|
|
18
|
-
// The role used when running "cdk deploy".
|
|
19
|
-
const cdkRole = new iam.Role(this, "CdkRole", {
|
|
20
|
-
roleName: conventions_1.cdkDeployRoleName,
|
|
21
|
-
assumedBy: new iam.CompositePrincipal(...props.trustedAccountIds.map((it) => new iam.AccountPrincipal(it))),
|
|
22
|
-
});
|
|
23
|
-
// Roles used by CDK CLI for the actual deployment.
|
|
24
|
-
// (For use under new-style synthesize.)
|
|
25
|
-
cdkRole.addToPolicy(new iam.PolicyStatement({
|
|
26
|
-
actions: ["sts:AssumeRole"],
|
|
27
|
-
resources: [
|
|
28
|
-
`arn:aws:iam::${account}:role/*-deploy-role-*`,
|
|
29
|
-
`arn:aws:iam::${account}:role/*-publishing-role-*`,
|
|
30
|
-
],
|
|
31
|
-
}));
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
exports.LifligCdkDeployerDeps = LifligCdkDeployerDeps;
|
|
35
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlmbGlnLWNkay1kZXBsb3llci1kZXBzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3BpcGVsaW5lcy9saWZsaWctY2RrLWRlcGxveWVyLWRlcHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQXdDO0FBQ3hDLDJDQUEwQztBQUMxQyxtQ0FBa0M7QUFDbEMsK0NBQWlEO0FBTWpEOzs7OztHQUtHO0FBQ0gsTUFBYSxxQkFBc0IsU0FBUSxVQUFVLENBQUMsU0FBUztJQUM3RCxZQUNFLEtBQTJCLEVBQzNCLEVBQVUsRUFDVixLQUFpQztRQUVqQyxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLE1BQU0sT0FBTyxHQUFHLEdBQUcsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLE9BQU8sQ0FBQTtRQUUxQywyQ0FBMkM7UUFDM0MsTUFBTSxPQUFPLEdBQUcsSUFBSSxHQUFHLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxTQUFTLEVBQUU7WUFDNUMsUUFBUSxFQUFFLCtCQUFpQjtZQUMzQixTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsa0JBQWtCLENBQ25DLEdBQUcsS0FBSyxDQUFDLGlCQUFpQixDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsSUFBSSxHQUFHLENBQUMsZ0JBQWdCLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FDckU7U0FDRixDQUFDLENBQUE7UUFFRixtREFBbUQ7UUFDbkQsd0NBQXdDO1FBQ3hDLE9BQU8sQ0FBQyxXQUFXLENBQ2pCLElBQUksR0FBRyxDQUFDLGVBQWUsQ0FBQztZQUN0QixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQztZQUMzQixTQUFTLEVBQUU7Z0JBQ1QsZ0JBQWdCLE9BQU8sdUJBQXVCO2dCQUM5QyxnQkFBZ0IsT0FBTywyQkFBMkI7YUFDbkQ7U0FDRixDQUFDLENBQ0gsQ0FBQTtJQUNILENBQUM7Q0FDRjtBQTlCRCxzREE4QkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBjZGsgZnJvbSBcImF3cy1jZGstbGliXCJcbmltcG9ydCB7IGNka0RlcGxveVJvbGVOYW1lIH0gZnJvbSBcIi4vY29udmVudGlvbnNcIlxuXG5leHBvcnQgaW50ZXJmYWNlIExpZmxpZ0Nka0RlcGxveWVyRGVwc1Byb3BzIHtcbiAgdHJ1c3RlZEFjY291bnRJZHM6IHN0cmluZ1tdXG59XG5cbi8qKlxuICogUmVzb3VyY2VzIG5lZWRlZCBzbyBsaWZsaWctY2RrLWRlcGxveWVyIGNhbiBkZXBsb3kgdG8gdGhlIGFjY291bnQuXG4gKlxuICogVGhpcyBtdXN0IGV4aXN0IGluIGVhY2ggdGFyZ2V0IGFjY291bnQgdGhhdCB0aGUgcGlwZWxpbmUgc2hvdWxkXG4gKiBiZSBhYmxlIHRvIGRlcGxveSBpbnRvLlxuICovXG5leHBvcnQgY2xhc3MgTGlmbGlnQ2RrRGVwbG95ZXJEZXBzIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBjb25zdHJ1Y3RvcihcbiAgICBzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogTGlmbGlnQ2RrRGVwbG95ZXJEZXBzUHJvcHMsXG4gICkge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIGNvbnN0IGFjY291bnQgPSBjZGsuU3RhY2sub2YodGhpcykuYWNjb3VudFxuXG4gICAgLy8gVGhlIHJvbGUgdXNlZCB3aGVuIHJ1bm5pbmcgXCJjZGsgZGVwbG95XCIuXG4gICAgY29uc3QgY2RrUm9sZSA9IG5ldyBpYW0uUm9sZSh0aGlzLCBcIkNka1JvbGVcIiwge1xuICAgICAgcm9sZU5hbWU6IGNka0RlcGxveVJvbGVOYW1lLFxuICAgICAgYXNzdW1lZEJ5OiBuZXcgaWFtLkNvbXBvc2l0ZVByaW5jaXBhbChcbiAgICAgICAgLi4ucHJvcHMudHJ1c3RlZEFjY291bnRJZHMubWFwKChpdCkgPT4gbmV3IGlhbS5BY2NvdW50UHJpbmNpcGFsKGl0KSksXG4gICAgICApLFxuICAgIH0pXG5cbiAgICAvLyBSb2xlcyB1c2VkIGJ5IENESyBDTEkgZm9yIHRoZSBhY3R1YWwgZGVwbG95bWVudC5cbiAgICAvLyAoRm9yIHVzZSB1bmRlciBuZXctc3R5bGUgc3ludGhlc2l6ZS4pXG4gICAgY2RrUm9sZS5hZGRUb1BvbGljeShcbiAgICAgIG5ldyBpYW0uUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgYWN0aW9uczogW1wic3RzOkFzc3VtZVJvbGVcIl0sXG4gICAgICAgIHJlc291cmNlczogW1xuICAgICAgICAgIGBhcm46YXdzOmlhbTo6JHthY2NvdW50fTpyb2xlLyotZGVwbG95LXJvbGUtKmAsXG4gICAgICAgICAgYGFybjphd3M6aWFtOjoke2FjY291bnR9OnJvbGUvKi1wdWJsaXNoaW5nLXJvbGUtKmAsXG4gICAgICAgIF0sXG4gICAgICB9KSxcbiAgICApXG4gIH1cbn1cbiJdfQ==
|
|
@@ -1,78 +0,0 @@
|
|
|
1
|
-
import * as constructs from "constructs";
|
|
2
|
-
import * as ec2 from "aws-cdk-lib/aws-ec2";
|
|
3
|
-
import * as s3 from "aws-cdk-lib/aws-s3";
|
|
4
|
-
import * as sfn from "aws-cdk-lib/aws-stepfunctions";
|
|
5
|
-
export interface PipelineProps {
|
|
6
|
-
/**
|
|
7
|
-
* Bucket holding pipeline configuration and trigger file.
|
|
8
|
-
*
|
|
9
|
-
* @default - use existing bucket based on Griid conventions
|
|
10
|
-
*/
|
|
11
|
-
artifactsBucket?: s3.IBucket;
|
|
12
|
-
/**
|
|
13
|
-
* Environments for this pipeline. Each environment is deployed sequentially
|
|
14
|
-
* in the order given.
|
|
15
|
-
*/
|
|
16
|
-
environments: PipelineEnvironment[];
|
|
17
|
-
/**
|
|
18
|
-
* Name of pipeline. This is used for the path where configuration
|
|
19
|
-
* is stored in S3.
|
|
20
|
-
*/
|
|
21
|
-
pipelineName: string;
|
|
22
|
-
/**
|
|
23
|
-
* Trigger the pipeline when the trigger file is written.
|
|
24
|
-
*
|
|
25
|
-
* @default - true
|
|
26
|
-
*/
|
|
27
|
-
triggerEnabled?: boolean;
|
|
28
|
-
/**
|
|
29
|
-
* VPC used for Fargate resources.
|
|
30
|
-
*/
|
|
31
|
-
vpc: ec2.IVpc;
|
|
32
|
-
}
|
|
33
|
-
export interface PipelineEnvironment {
|
|
34
|
-
/**
|
|
35
|
-
* Account number hosting the environment.
|
|
36
|
-
*/
|
|
37
|
-
accountId: string;
|
|
38
|
-
/**
|
|
39
|
-
* Additional tasks to run after the environment has been deployed.
|
|
40
|
-
*/
|
|
41
|
-
afterSuccessfulDeploy?: sfn.Chain;
|
|
42
|
-
/**
|
|
43
|
-
* Name of environment.
|
|
44
|
-
*/
|
|
45
|
-
name: string;
|
|
46
|
-
}
|
|
47
|
-
/**
|
|
48
|
-
* Pipeline for doing a multi-account CDK deployment based
|
|
49
|
-
* on a built CDK Cloud Assembly and parameters stored in S3.
|
|
50
|
-
*
|
|
51
|
-
* The accounts being deployed to must be provisioned with
|
|
52
|
-
* the LifligCdkDeployerDeps construct so expected IAM
|
|
53
|
-
* roles is present.
|
|
54
|
-
*
|
|
55
|
-
* The pipeline starts by writing an empty file to
|
|
56
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger
|
|
57
|
-
*
|
|
58
|
-
* The CDK deploy process is handled by liflig-cdk-deployer.
|
|
59
|
-
* See https://github.com/capralifecycle/liflig-cdk-deployer
|
|
60
|
-
*
|
|
61
|
-
* Configuration files are read from S3 at the path
|
|
62
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/
|
|
63
|
-
*
|
|
64
|
-
* - cloud-assembly.json which has the format described as
|
|
65
|
-
* CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer
|
|
66
|
-
*
|
|
67
|
-
* - variables*.json which can be zero or more files
|
|
68
|
-
* with string-string map that will be concatenated to
|
|
69
|
-
* form the format described as CDK_VARIABLES in
|
|
70
|
-
* liflig-cdk-deployer
|
|
71
|
-
*
|
|
72
|
-
* The separation of Cloud Assembly details and variables enables
|
|
73
|
-
* separation of IaC code and application code if they are not
|
|
74
|
-
* colocated in the same repository.
|
|
75
|
-
*/
|
|
76
|
-
export declare class Pipeline extends constructs.Construct {
|
|
77
|
-
constructor(scope: constructs.Construct, id: string, props: PipelineProps);
|
|
78
|
-
}
|
|
@@ -1,229 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.Pipeline = void 0;
|
|
4
|
-
const constructs = require("constructs");
|
|
5
|
-
const events = require("aws-cdk-lib/aws-events");
|
|
6
|
-
const eventsTargets = require("aws-cdk-lib/aws-events-targets");
|
|
7
|
-
const iam = require("aws-cdk-lib/aws-iam");
|
|
8
|
-
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
9
|
-
const sfn = require("aws-cdk-lib/aws-stepfunctions");
|
|
10
|
-
const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
|
|
11
|
-
const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
12
|
-
const cdk = require("aws-cdk-lib");
|
|
13
|
-
const griid_1 = require("../griid");
|
|
14
|
-
const conventions_1 = require("./conventions");
|
|
15
|
-
const deploy_env_1 = require("./deploy-env");
|
|
16
|
-
/**
|
|
17
|
-
* Pipeline for doing a multi-account CDK deployment based
|
|
18
|
-
* on a built CDK Cloud Assembly and parameters stored in S3.
|
|
19
|
-
*
|
|
20
|
-
* The accounts being deployed to must be provisioned with
|
|
21
|
-
* the LifligCdkDeployerDeps construct so expected IAM
|
|
22
|
-
* roles is present.
|
|
23
|
-
*
|
|
24
|
-
* The pipeline starts by writing an empty file to
|
|
25
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger
|
|
26
|
-
*
|
|
27
|
-
* The CDK deploy process is handled by liflig-cdk-deployer.
|
|
28
|
-
* See https://github.com/capralifecycle/liflig-cdk-deployer
|
|
29
|
-
*
|
|
30
|
-
* Configuration files are read from S3 at the path
|
|
31
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/
|
|
32
|
-
*
|
|
33
|
-
* - cloud-assembly.json which has the format described as
|
|
34
|
-
* CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer
|
|
35
|
-
*
|
|
36
|
-
* - variables*.json which can be zero or more files
|
|
37
|
-
* with string-string map that will be concatenated to
|
|
38
|
-
* form the format described as CDK_VARIABLES in
|
|
39
|
-
* liflig-cdk-deployer
|
|
40
|
-
*
|
|
41
|
-
* The separation of Cloud Assembly details and variables enables
|
|
42
|
-
* separation of IaC code and application code if they are not
|
|
43
|
-
* colocated in the same repository.
|
|
44
|
-
*/
|
|
45
|
-
class Pipeline extends constructs.Construct {
|
|
46
|
-
constructor(scope, id, props) {
|
|
47
|
-
var _a, _b;
|
|
48
|
-
super(scope, id);
|
|
49
|
-
const s3Prefix = (0, conventions_1.pipelineS3Prefix)(props.pipelineName);
|
|
50
|
-
const s3TriggerKey = (0, conventions_1.pipelineS3TriggerKey)(props.pipelineName);
|
|
51
|
-
const artifactsBucket = (_a = props.artifactsBucket) !== null && _a !== void 0 ? _a : (0, griid_1.getGriidArtefactBucket)(this);
|
|
52
|
-
const checkCanRunFn = new lambda.SingletonFunction(this, "CheckCanRunFn", {
|
|
53
|
-
uuid: "30ad3abb-f774-4804-a6ef-2c2f4a247362",
|
|
54
|
-
code: new lambda.InlineCode(`exports.handler = ${checkCanRunHandler.toString()};`),
|
|
55
|
-
runtime: lambda.Runtime.NODEJS_18_X,
|
|
56
|
-
handler: "index.handler",
|
|
57
|
-
timeout: cdk.Duration.seconds(10),
|
|
58
|
-
});
|
|
59
|
-
const checkCanRunTask = new tasks.LambdaInvoke(this, "Check if the process can run", {
|
|
60
|
-
lambdaFunction: checkCanRunFn,
|
|
61
|
-
outputPath: "$.Payload",
|
|
62
|
-
payload: sfn.TaskInput.fromObject({
|
|
63
|
-
"stateMachineId.$": "$$.StateMachine.Id",
|
|
64
|
-
"executionId.$": "$$.Execution.Id",
|
|
65
|
-
}),
|
|
66
|
-
});
|
|
67
|
-
const wait = new sfn.Wait(this, "Wait before rechecking status", {
|
|
68
|
-
time: sfn.WaitTime.duration(cdk.Duration.seconds(15)),
|
|
69
|
-
});
|
|
70
|
-
const skip = new sfn.Succeed(this, "Skip");
|
|
71
|
-
const collectFilesFn = new lambda.SingletonFunction(this, "CollectFilesFn", {
|
|
72
|
-
uuid: "c49cbfe1-50e0-4721-8964-fb20f4e5a7ad",
|
|
73
|
-
code: new lambda.InlineCode(`exports.handler = ${collectFilesHandler.toString()};`),
|
|
74
|
-
runtime: lambda.Runtime.NODEJS_18_X,
|
|
75
|
-
handler: "index.handler",
|
|
76
|
-
timeout: cdk.Duration.seconds(30),
|
|
77
|
-
});
|
|
78
|
-
artifactsBucket.grantRead(collectFilesFn);
|
|
79
|
-
const collectFilesTask = new tasks.LambdaInvoke(this, "Collect files from S3", {
|
|
80
|
-
lambdaFunction: collectFilesFn,
|
|
81
|
-
outputPath: "$.Payload",
|
|
82
|
-
payload: sfn.TaskInput.fromObject({
|
|
83
|
-
bucketName: artifactsBucket.bucketName,
|
|
84
|
-
bucketPrefix: s3Prefix,
|
|
85
|
-
envNames: props.environments.map((it) => it.name),
|
|
86
|
-
}),
|
|
87
|
-
});
|
|
88
|
-
let run = sfn.Chain.start(collectFilesTask);
|
|
89
|
-
const ifHavingStacks = (name, work) => new sfn.Choice(this, `Check if ${name} has stacks`)
|
|
90
|
-
.when(aws_stepfunctions_1.Condition.or(aws_stepfunctions_1.Condition.isNull(`$.StackCountPerEnv.${name}`), aws_stepfunctions_1.Condition.numberEquals(`$.StackCountPerEnv.${name}`, 0)), new sfn.Pass(this, `Skip ${name}`))
|
|
91
|
-
.otherwise(work)
|
|
92
|
-
.afterwards();
|
|
93
|
-
for (const env of props.environments) {
|
|
94
|
-
const it = new deploy_env_1.DeployEnv(this, env.name, {
|
|
95
|
-
accountId: env.accountId,
|
|
96
|
-
afterSuccessfulDeploy: env.afterSuccessfulDeploy,
|
|
97
|
-
artefactBucket: artifactsBucket,
|
|
98
|
-
envName: env.name,
|
|
99
|
-
vpc: props.vpc,
|
|
100
|
-
});
|
|
101
|
-
run = run.next(ifHavingStacks(env.name, it.chain));
|
|
102
|
-
}
|
|
103
|
-
const definition = sfn.Chain.start(checkCanRunTask).next(new sfn.Choice(this, "Can run?")
|
|
104
|
-
.when(aws_stepfunctions_1.Condition.stringEquals("$.CanRunState", "CONTINUE"), run)
|
|
105
|
-
.when(aws_stepfunctions_1.Condition.stringEquals("$.CanRunState", "SKIP"), skip)
|
|
106
|
-
.otherwise(wait.next(checkCanRunTask)));
|
|
107
|
-
const machine = new sfn.StateMachine(this, "StateMachine", {
|
|
108
|
-
definition,
|
|
109
|
-
// https://docs.aws.amazon.com/step-functions/latest/dg/sfn-stuck-execution.html
|
|
110
|
-
timeout: cdk.Duration.hours(3),
|
|
111
|
-
});
|
|
112
|
-
new iam.Policy(this, "CheckCanRunPolicy", {
|
|
113
|
-
roles: [checkCanRunFn.role],
|
|
114
|
-
statements: [
|
|
115
|
-
new iam.PolicyStatement({
|
|
116
|
-
actions: ["states:ListExecutions"],
|
|
117
|
-
resources: [machine.stateMachineArn],
|
|
118
|
-
}),
|
|
119
|
-
],
|
|
120
|
-
});
|
|
121
|
-
if ((_b = props.triggerEnabled) !== null && _b !== void 0 ? _b : true) {
|
|
122
|
-
artifactsBucket.onCloudTrailWriteObject("Trigger", {
|
|
123
|
-
paths: [s3TriggerKey],
|
|
124
|
-
target: new eventsTargets.SfnStateMachine(machine, {
|
|
125
|
-
input: events.RuleTargetInput.fromObject({}),
|
|
126
|
-
}),
|
|
127
|
-
});
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
exports.Pipeline = Pipeline;
|
|
132
|
-
// This is a self-contained function that will be serialized as a lambda.
|
|
133
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
134
|
-
const collectFilesHandler = async (event) => {
|
|
135
|
-
// eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
|
|
136
|
-
const { S3Client, ListObjectsV2Command, GetObjectCommand } = await Promise.resolve().then(() => require("@aws-sdk/client-s3"));
|
|
137
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
|
138
|
-
const s3Client = new S3Client();
|
|
139
|
-
console.log("Event received: ", event);
|
|
140
|
-
const bucketName = event.bucketName;
|
|
141
|
-
const bucketPrefix = event.bucketPrefix;
|
|
142
|
-
const envNames = event.envNames;
|
|
143
|
-
if (bucketPrefix.slice(-1) !== "/") {
|
|
144
|
-
throw new Error(`Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`);
|
|
145
|
-
}
|
|
146
|
-
async function listObjects() {
|
|
147
|
-
const { Contents } = await s3Client.send(new ListObjectsV2Command({
|
|
148
|
-
Bucket: bucketName,
|
|
149
|
-
Prefix: bucketPrefix,
|
|
150
|
-
}));
|
|
151
|
-
return Contents;
|
|
152
|
-
}
|
|
153
|
-
async function getObject(key) {
|
|
154
|
-
const { Body } = await s3Client.send(new GetObjectCommand({
|
|
155
|
-
Bucket: bucketName,
|
|
156
|
-
Key: key,
|
|
157
|
-
}));
|
|
158
|
-
return Body.toString();
|
|
159
|
-
}
|
|
160
|
-
let cloudAssembly = null;
|
|
161
|
-
let variables = {};
|
|
162
|
-
const files = await listObjects();
|
|
163
|
-
for (const file of files !== null && files !== void 0 ? files : []) {
|
|
164
|
-
const key = file.Key;
|
|
165
|
-
const filename = key.slice(bucketPrefix.length);
|
|
166
|
-
console.log(`File: ${filename}`);
|
|
167
|
-
if (filename === "cloud-assembly.json") {
|
|
168
|
-
console.log("Found Cloud Assembly");
|
|
169
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
|
|
170
|
-
cloudAssembly = JSON.parse(await getObject(key));
|
|
171
|
-
}
|
|
172
|
-
else if (/^variables.*\.json$/.test(filename)) {
|
|
173
|
-
console.log("Found variables file");
|
|
174
|
-
variables = {
|
|
175
|
-
...variables,
|
|
176
|
-
...JSON.parse(await getObject(key)),
|
|
177
|
-
};
|
|
178
|
-
}
|
|
179
|
-
else {
|
|
180
|
-
console.log("Ignoring unknown file");
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
if (cloudAssembly === null) {
|
|
184
|
-
throw new Error("cloud-assembly.json not found");
|
|
185
|
-
}
|
|
186
|
-
return {
|
|
187
|
-
CloudAssembly: JSON.stringify(cloudAssembly),
|
|
188
|
-
Variables: JSON.stringify(variables),
|
|
189
|
-
StackCountPerEnv: Object.fromEntries(envNames.map((name) => {
|
|
190
|
-
var _a, _b;
|
|
191
|
-
return [
|
|
192
|
-
name,
|
|
193
|
-
(_b = (_a = cloudAssembly.environments.find((it) => it.name === name)) === null || _a === void 0 ? void 0 : _a.stackNames.length) !== null && _b !== void 0 ? _b : 0,
|
|
194
|
-
];
|
|
195
|
-
})),
|
|
196
|
-
};
|
|
197
|
-
};
|
|
198
|
-
// This is a self-contained function that will be serialized as a lambda.
|
|
199
|
-
const checkCanRunHandler = async (event) => {
|
|
200
|
-
// eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
|
|
201
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
|
202
|
-
const { SFNClient, ListExecutionsCommand, ExecutionStatus } = await Promise.resolve().then(() => require("@aws-sdk/client-sfn"));
|
|
203
|
-
const sfnClient = new SFNClient();
|
|
204
|
-
console.log("Event received: ", event);
|
|
205
|
-
const stateMachineArn = event["stateMachineId"];
|
|
206
|
-
const currentExecutionArn = event["executionId"];
|
|
207
|
-
const { executions } = await sfnClient.send(new ListExecutionsCommand({
|
|
208
|
-
stateMachineArn,
|
|
209
|
-
statusFilter: ExecutionStatus.RUNNING,
|
|
210
|
-
}));
|
|
211
|
-
if (!executions) {
|
|
212
|
-
throw new Error("Could not list executions");
|
|
213
|
-
}
|
|
214
|
-
console.log("Executions: ", executions);
|
|
215
|
-
const currentExecution = executions.find((it) => it.executionArn == currentExecutionArn);
|
|
216
|
-
if (!currentExecution) {
|
|
217
|
-
throw new Error("Could not find current execution");
|
|
218
|
-
}
|
|
219
|
-
const newer = executions.filter((it) => {
|
|
220
|
-
if (!it.startDate || !currentExecution.startDate) {
|
|
221
|
-
return false;
|
|
222
|
-
}
|
|
223
|
-
return it.startDate > currentExecution.startDate;
|
|
224
|
-
}).length;
|
|
225
|
-
return {
|
|
226
|
-
CanRunState: newer > 0 ? "SKIP" : executions.length == 1 ? "CONTINUE" : "WAIT",
|
|
227
|
-
};
|
|
228
|
-
};
|
|
229
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../../src/pipelines/pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as eventsTargets from \"aws-cdk-lib/aws-events-targets\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as sfn from \"aws-cdk-lib/aws-stepfunctions\"\nimport { Condition } from \"aws-cdk-lib/aws-stepfunctions\"\nimport * as tasks from \"aws-cdk-lib/aws-stepfunctions-tasks\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type { Handler } from \"aws-lambda\"\nimport { getGriidArtefactBucket } from \"../griid\"\nimport { pipelineS3Prefix, pipelineS3TriggerKey } from \"./conventions\"\nimport { DeployEnv } from \"./deploy-env\"\n\nexport interface PipelineProps {\n  /**\n   * Bucket holding pipeline configuration and trigger file.\n   *\n   * @default - use existing bucket based on Griid conventions\n   */\n  artifactsBucket?: s3.IBucket\n  /**\n   * Environments for this pipeline. Each environment is deployed sequentially\n   * in the order given.\n   */\n  environments: PipelineEnvironment[]\n  /**\n   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Trigger the pipeline when the trigger file is written.\n   *\n   * @default - true\n   */\n  triggerEnabled?: boolean\n  /**\n   * VPC used for Fargate resources.\n   */\n  vpc: ec2.IVpc\n}\n\nexport interface PipelineEnvironment {\n  /**\n   * Account number hosting the environment.\n   */\n  accountId: string\n  /**\n   * Additional tasks to run after the environment has been deployed.\n   */\n  afterSuccessfulDeploy?: sfn.Chain\n  /**\n   * Name of environment.\n   */\n  name: string\n}\n\n/**\n * Pipeline for doing a multi-account CDK deployment based\n * on a built CDK Cloud Assembly and parameters stored in S3.\n *\n * The accounts being deployed to must be provisioned with\n * the LifligCdkDeployerDeps construct so expected IAM\n * roles is present.\n *\n * The pipeline starts by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * The CDK deploy process is handled by liflig-cdk-deployer.\n * See https://github.com/capralifecycle/liflig-cdk-deployer\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n *  - cloud-assembly.json which has the format described as\n *    CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer\n *\n *  - variables*.json which can be zero or more files\n *    with string-string map that will be concatenated to\n *    form the format described as CDK_VARIABLES in\n *    liflig-cdk-deployer\n *\n * The separation of Cloud Assembly details and variables enables\n * separation of IaC code and application code if they are not\n * colocated in the same repository.\n */\nexport class Pipeline extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: PipelineProps) {\n    super(scope, id)\n\n    const s3Prefix = pipelineS3Prefix(props.pipelineName)\n    const s3TriggerKey = pipelineS3TriggerKey(props.pipelineName)\n\n    const artifactsBucket =\n      props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const checkCanRunFn = new lambda.SingletonFunction(this, \"CheckCanRunFn\", {\n      uuid: \"30ad3abb-f774-4804-a6ef-2c2f4a247362\",\n      code: new lambda.InlineCode(\n        `exports.handler = ${checkCanRunHandler.toString()};`,\n      ),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      handler: \"index.handler\",\n      timeout: cdk.Duration.seconds(10),\n    })\n\n    const checkCanRunTask = new tasks.LambdaInvoke(\n      this,\n      \"Check if the process can run\",\n      {\n        lambdaFunction: checkCanRunFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          \"stateMachineId.$\": \"$$.StateMachine.Id\",\n          \"executionId.$\": \"$$.Execution.Id\",\n        }),\n      },\n    )\n\n    const wait = new sfn.Wait(this, \"Wait before rechecking status\", {\n      time: sfn.WaitTime.duration(cdk.Duration.seconds(15)),\n    })\n\n    const skip = new sfn.Succeed(this, \"Skip\")\n\n    const collectFilesFn = new lambda.SingletonFunction(\n      this,\n      \"CollectFilesFn\",\n      {\n        uuid: \"c49cbfe1-50e0-4721-8964-fb20f4e5a7ad\",\n        code: new lambda.InlineCode(\n          `exports.handler = ${collectFilesHandler.toString()};`,\n        ),\n        runtime: lambda.Runtime.NODEJS_18_X,\n        handler: \"index.handler\",\n        timeout: cdk.Duration.seconds(30),\n      },\n    )\n\n    artifactsBucket.grantRead(collectFilesFn)\n\n    const collectFilesTask = new tasks.LambdaInvoke(\n      this,\n      \"Collect files from S3\",\n      {\n        lambdaFunction: collectFilesFn,\n        outputPath: \"$.Payload\",\n        payload: sfn.TaskInput.fromObject({\n          bucketName: artifactsBucket.bucketName,\n          bucketPrefix: s3Prefix,\n          envNames: props.environments.map((it) => it.name),\n        }),\n      },\n    )\n\n    let run: sfn.Chain = sfn.Chain.start(collectFilesTask)\n\n    const ifHavingStacks = (name: string, work: sfn.Chain) =>\n      new sfn.Choice(this, `Check if ${name} has stacks`)\n        .when(\n          Condition.or(\n            Condition.isNull(`$.StackCountPerEnv.${name}`),\n            Condition.numberEquals(`$.StackCountPerEnv.${name}`, 0),\n          ),\n          new sfn.Pass(this, `Skip ${name}`),\n        )\n        .otherwise(work)\n        .afterwards()\n\n    for (const env of props.environments) {\n      const it = new DeployEnv(this, env.name, {\n        accountId: env.accountId,\n        afterSuccessfulDeploy: env.afterSuccessfulDeploy,\n        artefactBucket: artifactsBucket,\n        envName: env.name,\n        vpc: props.vpc,\n      })\n\n      run = run.next(ifHavingStacks(env.name, it.chain))\n    }\n\n    const definition: sfn.Chain = sfn.Chain.start(checkCanRunTask).next(\n      new sfn.Choice(this, \"Can run?\")\n        .when(Condition.stringEquals(\"$.CanRunState\", \"CONTINUE\"), run)\n        .when(Condition.stringEquals(\"$.CanRunState\", \"SKIP\"), skip)\n        .otherwise(wait.next(checkCanRunTask)),\n    )\n\n    const machine = new sfn.StateMachine(this, \"StateMachine\", {\n      definition,\n      // https://docs.aws.amazon.com/step-functions/latest/dg/sfn-stuck-execution.html\n      timeout: cdk.Duration.hours(3),\n    })\n\n    new iam.Policy(this, \"CheckCanRunPolicy\", {\n      roles: [checkCanRunFn.role!],\n      statements: [\n        new iam.PolicyStatement({\n          actions: [\"states:ListExecutions\"],\n          resources: [machine.stateMachineArn],\n        }),\n      ],\n    })\n\n    if (props.triggerEnabled ?? true) {\n      artifactsBucket.onCloudTrailWriteObject(\"Trigger\", {\n        paths: [s3TriggerKey],\n        target: new eventsTargets.SfnStateMachine(machine, {\n          input: events.RuleTargetInput.fromObject({}),\n        }),\n      })\n    }\n  }\n}\n\ninterface CloudAssembly {\n  cloudAssemblyBucketName: string\n  cloudAssemblyBucketKey: string\n  environments: {\n    name: string\n    stackNames: string[]\n  }[]\n  parameters: {\n    name: string\n    value: unknown | { type: \"variable\"; variable: string }\n  }[]\n}\n\n// This is a self-contained function that will be serialized as a lambda.\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst collectFilesHandler: Handler = async (event: Record<string, any>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  const { S3Client, ListObjectsV2Command, GetObjectCommand } = await import(\n    \"@aws-sdk/client-s3\"\n  )\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n\n  const s3Client = new S3Client()\n\n  console.log(\"Event received: \", event)\n\n  const bucketName = event.bucketName as string\n  const bucketPrefix = event.bucketPrefix as string\n  const envNames = event.envNames as string[]\n\n  if (bucketPrefix.slice(-1) !== \"/\") {\n    throw new Error(\n      `Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`,\n    )\n  }\n\n  async function listObjects() {\n    const { Contents } = await s3Client.send(\n      new ListObjectsV2Command({\n        Bucket: bucketName,\n        Prefix: bucketPrefix,\n      }),\n    )\n    return Contents\n  }\n\n  async function getObject(key: string): Promise<string> {\n    const { Body } = await s3Client.send(\n      new GetObjectCommand({\n        Bucket: bucketName,\n        Key: key,\n      }),\n    )\n    return Body!.toString()\n  }\n\n  let cloudAssembly: CloudAssembly | null = null\n  let variables: Record<string, string> = {}\n\n  const files = await listObjects()\n  for (const file of files ?? []) {\n    const key = file.Key!\n    const filename = key.slice(bucketPrefix.length)\n\n    console.log(`File: ${filename}`)\n\n    if (filename === \"cloud-assembly.json\") {\n      console.log(\"Found Cloud Assembly\")\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n      cloudAssembly = JSON.parse(await getObject(key))\n    } else if (/^variables.*\\.json$/.test(filename)) {\n      console.log(\"Found variables file\")\n      variables = {\n        ...variables,\n        ...(JSON.parse(await getObject(key)) as Record<string, string>),\n      }\n    } else {\n      console.log(\"Ignoring unknown file\")\n    }\n  }\n\n  if (cloudAssembly === null) {\n    throw new Error(\"cloud-assembly.json not found\")\n  }\n\n  return {\n    CloudAssembly: JSON.stringify(cloudAssembly),\n    Variables: JSON.stringify(variables),\n    StackCountPerEnv: Object.fromEntries(\n      envNames.map((name) => [\n        name,\n        cloudAssembly.environments.find((it) => it.name === name)?.stackNames\n          .length ?? 0,\n      ]),\n    ),\n  }\n}\n\n// This is a self-contained function that will be serialized as a lambda.\nconst checkCanRunHandler: Handler = async (event: Record<string, string>) => {\n  // eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access\n\n  const { SFNClient, ListExecutionsCommand, ExecutionStatus } = await import(\n    \"@aws-sdk/client-sfn\"\n  )\n  const sfnClient = new SFNClient()\n\n  console.log(\"Event received: \", event)\n\n  const stateMachineArn = event[\"stateMachineId\"]\n  const currentExecutionArn = event[\"executionId\"]\n\n  const { executions } = await sfnClient.send(\n    new ListExecutionsCommand({\n      stateMachineArn,\n      statusFilter: ExecutionStatus.RUNNING,\n    }),\n  )\n\n  if (!executions) {\n    throw new Error(\"Could not list executions\")\n  }\n\n  console.log(\"Executions: \", executions)\n\n  const currentExecution = executions.find(\n    (it) => it.executionArn == currentExecutionArn,\n  )\n\n  if (!currentExecution) {\n    throw new Error(\"Could not find current execution\")\n  }\n\n  const newer = executions.filter((it) => {\n    if (!it.startDate || !currentExecution.startDate) {\n      return false\n    }\n    return it.startDate > currentExecution.startDate\n  }).length\n\n  return {\n    CanRunState:\n      newer > 0 ? \"SKIP\" : executions.length == 1 ? \"CONTINUE\" : \"WAIT\",\n  }\n}\n"]}
|