token-injectable-docker-builder 1.0.0 → 1.0.1
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/.jsii +9 -9
- package/lib/index.js +6 -10
- package/package.json +1 -1
- package/src/index.ts +197 -0
- package/src/isComplete.js +97 -0
- package/src/onEvent.js +39 -0
package/.jsii
CHANGED
|
@@ -3927,7 +3927,7 @@
|
|
|
3927
3927
|
},
|
|
3928
3928
|
"locationInModule": {
|
|
3929
3929
|
"filename": "src/index.ts",
|
|
3930
|
-
"line":
|
|
3930
|
+
"line": 59
|
|
3931
3931
|
},
|
|
3932
3932
|
"parameters": [
|
|
3933
3933
|
{
|
|
@@ -3962,7 +3962,7 @@
|
|
|
3962
3962
|
"kind": "class",
|
|
3963
3963
|
"locationInModule": {
|
|
3964
3964
|
"filename": "src/index.ts",
|
|
3965
|
-
"line":
|
|
3965
|
+
"line": 47
|
|
3966
3966
|
},
|
|
3967
3967
|
"name": "TokenInjectableDockerBuilder",
|
|
3968
3968
|
"properties": [
|
|
@@ -3973,7 +3973,7 @@
|
|
|
3973
3973
|
"immutable": true,
|
|
3974
3974
|
"locationInModule": {
|
|
3975
3975
|
"filename": "src/index.ts",
|
|
3976
|
-
"line":
|
|
3976
|
+
"line": 48
|
|
3977
3977
|
},
|
|
3978
3978
|
"name": "containerImage",
|
|
3979
3979
|
"type": {
|
|
@@ -3987,7 +3987,7 @@
|
|
|
3987
3987
|
"immutable": true,
|
|
3988
3988
|
"locationInModule": {
|
|
3989
3989
|
"filename": "src/index.ts",
|
|
3990
|
-
"line":
|
|
3990
|
+
"line": 49
|
|
3991
3991
|
},
|
|
3992
3992
|
"name": "dockerImageCode",
|
|
3993
3993
|
"type": {
|
|
@@ -4008,7 +4008,7 @@
|
|
|
4008
4008
|
"kind": "interface",
|
|
4009
4009
|
"locationInModule": {
|
|
4010
4010
|
"filename": "src/index.ts",
|
|
4011
|
-
"line":
|
|
4011
|
+
"line": 14
|
|
4012
4012
|
},
|
|
4013
4013
|
"name": "TokenInjectableDockerBuilderProps",
|
|
4014
4014
|
"properties": [
|
|
@@ -4021,7 +4021,7 @@
|
|
|
4021
4021
|
"immutable": true,
|
|
4022
4022
|
"locationInModule": {
|
|
4023
4023
|
"filename": "src/index.ts",
|
|
4024
|
-
"line":
|
|
4024
|
+
"line": 18
|
|
4025
4025
|
},
|
|
4026
4026
|
"name": "path",
|
|
4027
4027
|
"type": {
|
|
@@ -4039,7 +4039,7 @@
|
|
|
4039
4039
|
"immutable": true,
|
|
4040
4040
|
"locationInModule": {
|
|
4041
4041
|
"filename": "src/index.ts",
|
|
4042
|
-
"line":
|
|
4042
|
+
"line": 29
|
|
4043
4043
|
},
|
|
4044
4044
|
"name": "buildArgs",
|
|
4045
4045
|
"optional": true,
|
|
@@ -4056,6 +4056,6 @@
|
|
|
4056
4056
|
"symbolId": "src/index:TokenInjectableDockerBuilderProps"
|
|
4057
4057
|
}
|
|
4058
4058
|
},
|
|
4059
|
-
"version": "1.0.
|
|
4060
|
-
"fingerprint": "
|
|
4059
|
+
"version": "1.0.1",
|
|
4060
|
+
"fingerprint": "bAw/kr8wy+8DBOatIsJQcEwY7gqDQMe9vQ1zuXwtsxU="
|
|
4061
4061
|
}
|
package/lib/index.js
CHANGED
|
@@ -3,7 +3,6 @@ var _a;
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
4
|
exports.TokenInjectableDockerBuilder = void 0;
|
|
5
5
|
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
-
const path = require("path");
|
|
7
6
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
7
|
const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild");
|
|
9
8
|
const aws_ecr_1 = require("aws-cdk-lib/aws-ecr");
|
|
@@ -38,9 +37,6 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
38
37
|
constructor(scope, id, props) {
|
|
39
38
|
super(scope, id);
|
|
40
39
|
const { path: sourcePath, buildArgs } = props; // Default to linux/amd64
|
|
41
|
-
// Define absolute paths for Lambda handlers
|
|
42
|
-
const onEventHandlerPath = path.resolve(__dirname, './onEventHandler');
|
|
43
|
-
const isCompleteHandlerPath = path.resolve(__dirname, './isCompleteHandler');
|
|
44
40
|
// Create an ECR repository
|
|
45
41
|
this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository');
|
|
46
42
|
// Package the source code as an asset
|
|
@@ -109,8 +105,8 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
109
105
|
// Create Node.js Lambda function for onEvent
|
|
110
106
|
const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
|
|
111
107
|
runtime: aws_lambda_1.Runtime.NODEJS_LATEST, // Use Node.js runtime
|
|
112
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
113
|
-
handler: '
|
|
108
|
+
code: aws_lambda_1.Code.fromAsset('.'), // Path to handler code
|
|
109
|
+
handler: 'onEvent.handler', // Entry point (adjust as needed)
|
|
114
110
|
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
115
111
|
});
|
|
116
112
|
onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
|
|
@@ -120,8 +116,8 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
120
116
|
// Create Node.js Lambda function for isComplete
|
|
121
117
|
const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
|
|
122
118
|
runtime: aws_lambda_1.Runtime.NODEJS_LATEST,
|
|
123
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
124
|
-
handler: '
|
|
119
|
+
code: aws_lambda_1.Code.fromAsset('.'), // Path to handler code
|
|
120
|
+
handler: 'isComplete.handler', // Entry point (adjust as needed)
|
|
125
121
|
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
126
122
|
});
|
|
127
123
|
isCompleteHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
|
|
@@ -155,5 +151,5 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
155
151
|
}
|
|
156
152
|
exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
|
|
157
153
|
_a = JSII_RTTI_SYMBOL_1;
|
|
158
|
-
TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.0.
|
|
159
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { CustomResource, Stack, Duration } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\n/**\n * Properties for the `TokenInjectableDockerBuilder` construct.\n */\nexport interface TokenInjectableDockerBuilderProps {\n  /**\n   * The path to the directory containing the Dockerfile or source code.\n   */\n  readonly path: string;\n\n  /**\n   * Build arguments to pass to the Docker build process.\n   * These are transformed into `--build-arg` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n}\n\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.\n *\n * @example\n * const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {\n *   path: './docker',\n *   buildArgs: {\n *     TOKEN: 'my-secret-token',\n *     ENV: 'production'\n *   },\n * });\n *\n * const containerImage = dockerBuilder.getContainerImage();\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  public readonly containerImage: ContainerImage;\n  public readonly dockerImageCode: DockerImageCode;\n  private readonly ecrRepository: Repository;\n\n  /**\n   * Creates a new `TokenInjectableDockerBuilder` instance.\n   *\n   * @param scope The parent construct/stack.\n   * @param id The unique ID of the construct.\n   * @param props Configuration properties for the construct.\n   */\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const { path: sourcePath, buildArgs } = props; // Default to linux/amd64\n\n    // Define absolute paths for Lambda handlers\n    const onEventHandlerPath = path.resolve(__dirname, './onEventHandler');\n    const isCompleteHandlerPath = path.resolve(__dirname, './isCompleteHandler');\n\n    // Create an ECR repository\n    this.ecrRepository = new Repository(this, 'ECRRepository');\n\n    // Package the source code as an asset\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath, // Path to the Dockerfile or source code\n    });\n\n    // Transform buildArgs into a string of --build-arg KEY=VALUE\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([key, value]) => `--build-arg ${key}=${value}`)\n        .join(' ')\n      : '';\n\n    // Pass the buildArgsString and platform as environment variables\n    const environmentVariables: { [name: string]: { value: string } } = {\n      ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      BUILD_ARGS: { value: buildArgsString },\n    };\n\n    // Create a CodeBuild project\n    const codeBuildProject = new Project(this, 'UICodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true, // Required for Docker builds\n      },\n      environmentVariables: environmentVariables,\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          pre_build: {\n            commands: [\n              'echo \"Retrieving AWS Account ID...\"',\n              'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n              'echo \"Logging in to Amazon ECR...\"',\n              'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',\n            ],\n          },\n          build: {\n            commands: [\n              'echo Build phase: Building the Docker image...',\n              'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',\n            ],\n          },\n          post_build: {\n            commands: [\n              'echo Post-build phase: Pushing the Docker image...',\n              'docker push $ECR_REPO_URI:latest',\n            ],\n          },\n        },\n      }),\n    });\n\n    // Grant permissions to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['ecr:GetAuthorizationToken'],\n        resources: ['*'],\n      }),\n    );\n\n    // Grant permissions to CodeBuild for CloudWatch Logs\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['logs:PutLogEvents', 'logs:CreateLogGroup', 'logs:CreateLogStream'],\n        resources: [`arn:aws:logs:${Stack.of(this).region}:${Stack.of(this).account}:*`],\n      }),\n    );\n\n    // Create Node.js Lambda function for onEvent\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_LATEST, // Use Node.js runtime\n      code: Code.fromAsset(onEventHandlerPath), // Path to handler code\n      handler: 'index.handler', // Entry point (adjust as needed)\n      timeout: Duration.minutes(15),\n    });\n\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn], // Restrict to specific project\n      }),\n    );\n\n    // Create Node.js Lambda function for isComplete\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_LATEST,\n      code: Code.fromAsset(isCompleteHandlerPath),\n      handler: 'index.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    isCompleteHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'codebuild:BatchGetBuilds',\n          'codebuild:ListBuildsForProject',\n          'logs:GetLogEvents',\n          'logs:DescribeLogStreams',\n          'logs:DescribeLogGroups',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    // Create a custom resource provider\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.seconds(30),\n    });\n\n    // Define the custom resource\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    buildTriggerResource.node.addDependency(codeBuildProject);\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository);\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository);\n  }\n}\n"]}
|
|
154
|
+
TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.0.1" };
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { CustomResource, Stack, Duration } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\n/**\n * Properties for the `TokenInjectableDockerBuilder` construct.\n */\nexport interface TokenInjectableDockerBuilderProps {\n  /**\n   * The path to the directory containing the Dockerfile or source code.\n   */\n  readonly path: string;\n\n  /**\n   * Build arguments to pass to the Docker build process.\n   * These are transformed into `--build-arg` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n}\n\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.\n *\n * @example\n * const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {\n *   path: './docker',\n *   buildArgs: {\n *     TOKEN: 'my-secret-token',\n *     ENV: 'production'\n *   },\n * });\n *\n * const containerImage = dockerBuilder.getContainerImage();\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  public readonly containerImage: ContainerImage;\n  public readonly dockerImageCode: DockerImageCode;\n  private readonly ecrRepository: Repository;\n\n  /**\n   * Creates a new `TokenInjectableDockerBuilder` instance.\n   *\n   * @param scope The parent construct/stack.\n   * @param id The unique ID of the construct.\n   * @param props Configuration properties for the construct.\n   */\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const { path: sourcePath, buildArgs } = props; // Default to linux/amd64\n\n    // Create an ECR repository\n    this.ecrRepository = new Repository(this, 'ECRRepository');\n\n    // Package the source code as an asset\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath, // Path to the Dockerfile or source code\n    });\n\n    // Transform buildArgs into a string of --build-arg KEY=VALUE\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([key, value]) => `--build-arg ${key}=${value}`)\n        .join(' ')\n      : '';\n\n    // Pass the buildArgsString and platform as environment variables\n    const environmentVariables: { [name: string]: { value: string } } = {\n      ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      BUILD_ARGS: { value: buildArgsString },\n    };\n\n    // Create a CodeBuild project\n    const codeBuildProject = new Project(this, 'UICodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true, // Required for Docker builds\n      },\n      environmentVariables: environmentVariables,\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          pre_build: {\n            commands: [\n              'echo \"Retrieving AWS Account ID...\"',\n              'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n              'echo \"Logging in to Amazon ECR...\"',\n              'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',\n            ],\n          },\n          build: {\n            commands: [\n              'echo Build phase: Building the Docker image...',\n              'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',\n            ],\n          },\n          post_build: {\n            commands: [\n              'echo Post-build phase: Pushing the Docker image...',\n              'docker push $ECR_REPO_URI:latest',\n            ],\n          },\n        },\n      }),\n    });\n\n    // Grant permissions to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['ecr:GetAuthorizationToken'],\n        resources: ['*'],\n      }),\n    );\n\n    // Grant permissions to CodeBuild for CloudWatch Logs\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['logs:PutLogEvents', 'logs:CreateLogGroup', 'logs:CreateLogStream'],\n        resources: [`arn:aws:logs:${Stack.of(this).region}:${Stack.of(this).account}:*`],\n      }),\n    );\n\n    // Create Node.js Lambda function for onEvent\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_LATEST, // Use Node.js runtime\n      code: Code.fromAsset('.'), // Path to handler code\n      handler: 'onEvent.handler', // Entry point (adjust as needed)\n      timeout: Duration.minutes(15),\n    });\n\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn], // Restrict to specific project\n      }),\n    );\n\n    // Create Node.js Lambda function for isComplete\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_LATEST,\n      code: Code.fromAsset('.'), // Path to handler code\n      handler: 'isComplete.handler', // Entry point (adjust as needed)\n      timeout: Duration.minutes(15),\n    });\n\n    isCompleteHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'codebuild:BatchGetBuilds',\n          'codebuild:ListBuildsForProject',\n          'logs:GetLogEvents',\n          'logs:DescribeLogStreams',\n          'logs:DescribeLogGroups',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    // Create a custom resource provider\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.seconds(30),\n    });\n\n    // Define the custom resource\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    buildTriggerResource.node.addDependency(codeBuildProject);\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository);\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository);\n  }\n}\n"]}
|
package/package.json
CHANGED
package/src/index.ts
ADDED
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
import { CustomResource, Stack, Duration } from 'aws-cdk-lib';
|
|
2
|
+
import { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';
|
|
3
|
+
import { Repository } from 'aws-cdk-lib/aws-ecr';
|
|
4
|
+
import { ContainerImage } from 'aws-cdk-lib/aws-ecs';
|
|
5
|
+
import { PolicyStatement } from 'aws-cdk-lib/aws-iam';
|
|
6
|
+
import { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';
|
|
7
|
+
import { Asset } from 'aws-cdk-lib/aws-s3-assets';
|
|
8
|
+
import { Provider } from 'aws-cdk-lib/custom-resources';
|
|
9
|
+
import { Construct } from 'constructs';
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Properties for the `TokenInjectableDockerBuilder` construct.
|
|
13
|
+
*/
|
|
14
|
+
export interface TokenInjectableDockerBuilderProps {
|
|
15
|
+
/**
|
|
16
|
+
* The path to the directory containing the Dockerfile or source code.
|
|
17
|
+
*/
|
|
18
|
+
readonly path: string;
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Build arguments to pass to the Docker build process.
|
|
22
|
+
* These are transformed into `--build-arg` flags.
|
|
23
|
+
* @example
|
|
24
|
+
* {
|
|
25
|
+
* TOKEN: 'my-secret-token',
|
|
26
|
+
* ENV: 'production'
|
|
27
|
+
* }
|
|
28
|
+
*/
|
|
29
|
+
readonly buildArgs?: { [key: string]: string };
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
|
|
35
|
+
*
|
|
36
|
+
* @example
|
|
37
|
+
* const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {
|
|
38
|
+
* path: './docker',
|
|
39
|
+
* buildArgs: {
|
|
40
|
+
* TOKEN: 'my-secret-token',
|
|
41
|
+
* ENV: 'production'
|
|
42
|
+
* },
|
|
43
|
+
* });
|
|
44
|
+
*
|
|
45
|
+
* const containerImage = dockerBuilder.getContainerImage();
|
|
46
|
+
*/
|
|
47
|
+
export class TokenInjectableDockerBuilder extends Construct {
|
|
48
|
+
public readonly containerImage: ContainerImage;
|
|
49
|
+
public readonly dockerImageCode: DockerImageCode;
|
|
50
|
+
private readonly ecrRepository: Repository;
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Creates a new `TokenInjectableDockerBuilder` instance.
|
|
54
|
+
*
|
|
55
|
+
* @param scope The parent construct/stack.
|
|
56
|
+
* @param id The unique ID of the construct.
|
|
57
|
+
* @param props Configuration properties for the construct.
|
|
58
|
+
*/
|
|
59
|
+
constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {
|
|
60
|
+
super(scope, id);
|
|
61
|
+
|
|
62
|
+
const { path: sourcePath, buildArgs } = props; // Default to linux/amd64
|
|
63
|
+
|
|
64
|
+
// Create an ECR repository
|
|
65
|
+
this.ecrRepository = new Repository(this, 'ECRRepository');
|
|
66
|
+
|
|
67
|
+
// Package the source code as an asset
|
|
68
|
+
const sourceAsset = new Asset(this, 'SourceAsset', {
|
|
69
|
+
path: sourcePath, // Path to the Dockerfile or source code
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
// Transform buildArgs into a string of --build-arg KEY=VALUE
|
|
73
|
+
const buildArgsString = buildArgs
|
|
74
|
+
? Object.entries(buildArgs)
|
|
75
|
+
.map(([key, value]) => `--build-arg ${key}=${value}`)
|
|
76
|
+
.join(' ')
|
|
77
|
+
: '';
|
|
78
|
+
|
|
79
|
+
// Pass the buildArgsString and platform as environment variables
|
|
80
|
+
const environmentVariables: { [name: string]: { value: string } } = {
|
|
81
|
+
ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },
|
|
82
|
+
BUILD_ARGS: { value: buildArgsString },
|
|
83
|
+
};
|
|
84
|
+
|
|
85
|
+
// Create a CodeBuild project
|
|
86
|
+
const codeBuildProject = new Project(this, 'UICodeBuildProject', {
|
|
87
|
+
source: Source.s3({
|
|
88
|
+
bucket: sourceAsset.bucket,
|
|
89
|
+
path: sourceAsset.s3ObjectKey,
|
|
90
|
+
}),
|
|
91
|
+
environment: {
|
|
92
|
+
buildImage: LinuxBuildImage.STANDARD_7_0,
|
|
93
|
+
privileged: true, // Required for Docker builds
|
|
94
|
+
},
|
|
95
|
+
environmentVariables: environmentVariables,
|
|
96
|
+
buildSpec: BuildSpec.fromObject({
|
|
97
|
+
version: '0.2',
|
|
98
|
+
phases: {
|
|
99
|
+
pre_build: {
|
|
100
|
+
commands: [
|
|
101
|
+
'echo "Retrieving AWS Account ID..."',
|
|
102
|
+
'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
|
|
103
|
+
'echo "Logging in to Amazon ECR..."',
|
|
104
|
+
'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',
|
|
105
|
+
],
|
|
106
|
+
},
|
|
107
|
+
build: {
|
|
108
|
+
commands: [
|
|
109
|
+
'echo Build phase: Building the Docker image...',
|
|
110
|
+
'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',
|
|
111
|
+
],
|
|
112
|
+
},
|
|
113
|
+
post_build: {
|
|
114
|
+
commands: [
|
|
115
|
+
'echo Post-build phase: Pushing the Docker image...',
|
|
116
|
+
'docker push $ECR_REPO_URI:latest',
|
|
117
|
+
],
|
|
118
|
+
},
|
|
119
|
+
},
|
|
120
|
+
}),
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
// Grant permissions to interact with ECR
|
|
124
|
+
this.ecrRepository.grantPullPush(codeBuildProject);
|
|
125
|
+
|
|
126
|
+
codeBuildProject.role!.addToPrincipalPolicy(
|
|
127
|
+
new PolicyStatement({
|
|
128
|
+
actions: ['ecr:GetAuthorizationToken'],
|
|
129
|
+
resources: ['*'],
|
|
130
|
+
}),
|
|
131
|
+
);
|
|
132
|
+
|
|
133
|
+
// Grant permissions to CodeBuild for CloudWatch Logs
|
|
134
|
+
codeBuildProject.role!.addToPrincipalPolicy(
|
|
135
|
+
new PolicyStatement({
|
|
136
|
+
actions: ['logs:PutLogEvents', 'logs:CreateLogGroup', 'logs:CreateLogStream'],
|
|
137
|
+
resources: [`arn:aws:logs:${Stack.of(this).region}:${Stack.of(this).account}:*`],
|
|
138
|
+
}),
|
|
139
|
+
);
|
|
140
|
+
|
|
141
|
+
// Create Node.js Lambda function for onEvent
|
|
142
|
+
const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {
|
|
143
|
+
runtime: Runtime.NODEJS_LATEST, // Use Node.js runtime
|
|
144
|
+
code: Code.fromAsset('.'), // Path to handler code
|
|
145
|
+
handler: 'onEvent.handler', // Entry point (adjust as needed)
|
|
146
|
+
timeout: Duration.minutes(15),
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
onEventHandlerFunction.addToRolePolicy(
|
|
150
|
+
new PolicyStatement({
|
|
151
|
+
actions: ['codebuild:StartBuild'],
|
|
152
|
+
resources: [codeBuildProject.projectArn], // Restrict to specific project
|
|
153
|
+
}),
|
|
154
|
+
);
|
|
155
|
+
|
|
156
|
+
// Create Node.js Lambda function for isComplete
|
|
157
|
+
const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {
|
|
158
|
+
runtime: Runtime.NODEJS_LATEST,
|
|
159
|
+
code: Code.fromAsset('.'), // Path to handler code
|
|
160
|
+
handler: 'isComplete.handler', // Entry point (adjust as needed)
|
|
161
|
+
timeout: Duration.minutes(15),
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
isCompleteHandlerFunction.addToRolePolicy(
|
|
165
|
+
new PolicyStatement({
|
|
166
|
+
actions: [
|
|
167
|
+
'codebuild:BatchGetBuilds',
|
|
168
|
+
'codebuild:ListBuildsForProject',
|
|
169
|
+
'logs:GetLogEvents',
|
|
170
|
+
'logs:DescribeLogStreams',
|
|
171
|
+
'logs:DescribeLogGroups',
|
|
172
|
+
],
|
|
173
|
+
resources: ['*'],
|
|
174
|
+
}),
|
|
175
|
+
);
|
|
176
|
+
|
|
177
|
+
// Create a custom resource provider
|
|
178
|
+
const provider = new Provider(this, 'CustomResourceProvider', {
|
|
179
|
+
onEventHandler: onEventHandlerFunction,
|
|
180
|
+
isCompleteHandler: isCompleteHandlerFunction,
|
|
181
|
+
queryInterval: Duration.seconds(30),
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
// Define the custom resource
|
|
185
|
+
const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {
|
|
186
|
+
serviceToken: provider.serviceToken,
|
|
187
|
+
properties: {
|
|
188
|
+
ProjectName: codeBuildProject.projectName,
|
|
189
|
+
Trigger: crypto.randomUUID(),
|
|
190
|
+
},
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
buildTriggerResource.node.addDependency(codeBuildProject);
|
|
194
|
+
this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository);
|
|
195
|
+
this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository);
|
|
196
|
+
}
|
|
197
|
+
}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
const { CodeBuildClient, ListBuildsForProjectCommand, BatchGetBuildsCommand } = require('@aws-sdk/client-codebuild');
|
|
2
|
+
const { CloudWatchLogsClient, GetLogEventsCommand } = require('@aws-sdk/client-cloudwatch-logs');
|
|
3
|
+
|
|
4
|
+
exports.handler = async (event, context) => {
|
|
5
|
+
console.log('isCompleteHandler Event:', JSON.stringify(event, null, 2));
|
|
6
|
+
|
|
7
|
+
// Initialize AWS SDK v3 clients
|
|
8
|
+
const codebuildClient = new CodeBuildClient({ region: process.env.AWS_REGION });
|
|
9
|
+
const cloudwatchlogsClient = new CloudWatchLogsClient({ region: process.env.AWS_REGION });
|
|
10
|
+
|
|
11
|
+
try {
|
|
12
|
+
const projectName = event.ResourceProperties.ProjectName;
|
|
13
|
+
|
|
14
|
+
if (!projectName) {
|
|
15
|
+
throw new Error('ProjectName is required in ResourceProperties');
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
console.log(`Checking status for CodeBuild project: ${projectName}`);
|
|
19
|
+
|
|
20
|
+
// Retrieve the latest build for the given project
|
|
21
|
+
const listBuildsCommand = new ListBuildsForProjectCommand({
|
|
22
|
+
projectName: projectName,
|
|
23
|
+
sortOrder: 'DESCENDING',
|
|
24
|
+
maxResults: 1,
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
const listBuildsResp = await codebuildClient.send(listBuildsCommand);
|
|
28
|
+
const buildIds = listBuildsResp.ids;
|
|
29
|
+
|
|
30
|
+
if (!buildIds || buildIds.length === 0) {
|
|
31
|
+
throw new Error(`No builds found for project: ${projectName}`);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const buildId = buildIds[0];
|
|
35
|
+
console.log(`Latest Build ID: ${buildId}`);
|
|
36
|
+
|
|
37
|
+
// Get build details
|
|
38
|
+
const batchGetBuildsCommand = new BatchGetBuildsCommand({
|
|
39
|
+
ids: [buildId],
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
const buildDetailsResp = await codebuildClient.send(batchGetBuildsCommand);
|
|
43
|
+
const build = buildDetailsResp.builds[0];
|
|
44
|
+
|
|
45
|
+
if (!build) {
|
|
46
|
+
throw new Error(`Build details not found for Build ID: ${buildId}`);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
const buildStatus = build.buildStatus;
|
|
50
|
+
console.log(`Build Status: ${buildStatus}`);
|
|
51
|
+
|
|
52
|
+
if (buildStatus === 'IN_PROGRESS') {
|
|
53
|
+
// Build is still in progress
|
|
54
|
+
console.log('Build is still in progress.');
|
|
55
|
+
return { IsComplete: false };
|
|
56
|
+
} else if (buildStatus === 'SUCCEEDED') {
|
|
57
|
+
// Build succeeded
|
|
58
|
+
console.log('Build succeeded.');
|
|
59
|
+
return { IsComplete: true };
|
|
60
|
+
} else if (['FAILED', 'FAULT', 'STOPPED', 'TIMED_OUT'].includes(buildStatus)) {
|
|
61
|
+
// Build failed; retrieve last 5 log lines
|
|
62
|
+
const logsInfo = build.logs;
|
|
63
|
+
if (logsInfo && logsInfo.groupName && logsInfo.streamName) {
|
|
64
|
+
console.log(`Retrieving logs from CloudWatch Logs Group: ${logsInfo.groupName}, Stream: ${logsInfo.streamName}`);
|
|
65
|
+
|
|
66
|
+
const getLogEventsCommand = new GetLogEventsCommand({
|
|
67
|
+
logGroupName: logsInfo.groupName,
|
|
68
|
+
logStreamName: logsInfo.streamName,
|
|
69
|
+
startFromHead: false, // Start from the end to get latest logs
|
|
70
|
+
limit: 5,
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
const logEventsResp = await cloudwatchlogsClient.send(getLogEventsCommand);
|
|
74
|
+
const logEvents = logEventsResp.events;
|
|
75
|
+
const lastFiveMessages = logEvents.map((event) => event.message).reverse().join('\n');
|
|
76
|
+
|
|
77
|
+
const errorMessage = `Build failed with status: ${buildStatus}\nLast 5 build logs:\n${lastFiveMessages}`;
|
|
78
|
+
console.error(errorMessage);
|
|
79
|
+
|
|
80
|
+
// Throw an error to indicate failure to the CDK provider
|
|
81
|
+
throw new Error(errorMessage);
|
|
82
|
+
} else {
|
|
83
|
+
const errorMessage = `Build failed with status: ${buildStatus}, but logs are not available.`;
|
|
84
|
+
console.error(errorMessage);
|
|
85
|
+
throw new Error(errorMessage);
|
|
86
|
+
}
|
|
87
|
+
} else {
|
|
88
|
+
const errorMessage = `Unknown build status: ${buildStatus}`;
|
|
89
|
+
console.error(errorMessage);
|
|
90
|
+
throw new Error(errorMessage);
|
|
91
|
+
}
|
|
92
|
+
} catch (error) {
|
|
93
|
+
console.error('Error in isCompleteHandler:', error);
|
|
94
|
+
// Rethrow the error to inform the CDK provider of the failure
|
|
95
|
+
throw error;
|
|
96
|
+
}
|
|
97
|
+
};
|
package/src/onEvent.js
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
const { CodeBuildClient, StartBuildCommand } = require('@aws-sdk/client-codebuild');
|
|
2
|
+
|
|
3
|
+
exports.handler = async (event, context) => {
|
|
4
|
+
console.log('Event:', JSON.stringify(event, null, 2));
|
|
5
|
+
|
|
6
|
+
// Initialize the AWS SDK v3 CodeBuild client
|
|
7
|
+
const codebuildClient = new CodeBuildClient({ region: process.env.AWS_REGION });
|
|
8
|
+
|
|
9
|
+
// Set the PhysicalResourceId
|
|
10
|
+
let physicalResourceId = event.PhysicalResourceId || event.LogicalResourceId;
|
|
11
|
+
|
|
12
|
+
if (event.RequestType === 'Create' || event.RequestType === 'Update') {
|
|
13
|
+
const params = {
|
|
14
|
+
projectName: event.ResourceProperties.ProjectName,
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
try {
|
|
18
|
+
const command = new StartBuildCommand(params); // Create the command
|
|
19
|
+
const build = await codebuildClient.send(command); // Send the command
|
|
20
|
+
console.log('Started build:', JSON.stringify(build, null, 2));
|
|
21
|
+
} catch (error) {
|
|
22
|
+
console.error('Error starting build:', error);
|
|
23
|
+
|
|
24
|
+
return {
|
|
25
|
+
PhysicalResourceId: physicalResourceId,
|
|
26
|
+
Data: {},
|
|
27
|
+
Reason: error.message,
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
} else if (event.RequestType === 'Delete') {
|
|
31
|
+
// No action needed for delete, but ensure PhysicalResourceId remains the same
|
|
32
|
+
console.log('Delete request received. No action required.');
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
return {
|
|
36
|
+
PhysicalResourceId: physicalResourceId,
|
|
37
|
+
Data: {},
|
|
38
|
+
};
|
|
39
|
+
};
|