token-injectable-docker-builder 0.1.3 → 0.1.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,6 +1,12 @@
1
1
  # TokenInjectableDockerBuilder
2
2
 
3
- The `TokenInjectableDockerBuilder` is a powerful AWS CDK construct that automates the building, pushing, and deployment of Docker images to Amazon Elastic Container Registry (ECR) using AWS CodeBuild and Lambda custom resources. This construct simplifies workflows by enabling token-based Docker image customization.
3
+ The `TokenInjectableDockerBuilder` is a flexible AWS CDK construct that enabled the usage of AWS CDK tokens in the building, pushing, and deployment of Docker images to Amazon Elastic Container Registry (ECR). It leverages AWS CodeBuild and Lambda custom resources.
4
+
5
+ ## Why?
6
+
7
+ AWS CDK already provides mechanisms for creating deployable assets using Docker, such as [DockerImageAsset](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecr_assets.DockerImageAsset.html) and [DockerImageCode](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_lambda.DockerImageCode.html), but these Constructs are limited because they cannot accept CDK tokens as build-args. With the TokenInjectableDockerBuilder, one can inject CDK tokens as build-time args into their Docker-based assets to satisfy a much larger range of dependency relationships.
8
+
9
+ For example, imagine a NextJS frontend Docker image that calls an API Gateway endpoint. Logically, one would first deploy the API Gateway, then deploy the NextJS frontend such that it has reference to the API Gateway endpoint through a [build-time environment variable](https://nextjs.org/docs/pages/building-your-application/configuring/environment-variables). In this case, building the Docker-based asset before deployment time doesn't work since it is dependent on the deployment of the API Gateway.
4
10
 
5
11
  ## Features
6
12
 
@@ -149,4 +155,12 @@ Ensure you have the following:
149
155
 
150
156
  1. **Build Errors**: Check the AWS CodeBuild logs in CloudWatch.
151
157
  2. **Lambda Function Errors**: Check the `onEvent` and `isComplete` Lambda logs in CloudWatch.
152
- 3. **Permissions**: Ensure the IAM role for CodeBuild has the required permissions to interact with ECR and CloudWatch.
158
+ 3. **Permissions**: Ensure the IAM role for CodeBuild has the required permissions to interact with ECR and CloudWatch.
159
+
160
+ ---
161
+
162
+ ## Support
163
+
164
+ Open an issue on [GitHub](https://github.com/AlexTech314/TokenInjectableDockerBuilder) :)
165
+
166
+
package/lib/index.d.ts CHANGED
@@ -1,16 +1,62 @@
1
1
  import { DockerImageCode } from 'aws-cdk-lib/aws-lambda';
2
2
  import { Construct } from 'constructs';
3
3
  import { ContainerImage } from 'aws-cdk-lib/aws-ecs';
4
+ /**
5
+ * Properties for the `TokenInjectableDockerBuilder` construct.
6
+ */
4
7
  export interface TokenInjectableDockerBuilderProps {
8
+ /**
9
+ * The path to the directory containing the Dockerfile or source code.
10
+ */
5
11
  path: string;
12
+ /**
13
+ * Build arguments to pass to the Docker build process.
14
+ * These are transformed into `--build-arg` flags.
15
+ * @example
16
+ * {
17
+ * TOKEN: 'my-secret-token',
18
+ * ENV: 'production'
19
+ * }
20
+ */
6
21
  buildArgs?: {
7
22
  [key: string]: string;
8
23
  };
9
24
  }
25
+ /**
26
+ * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
27
+ *
28
+ * @example
29
+ * const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {
30
+ * path: './docker',
31
+ * buildArgs: {
32
+ * TOKEN: 'my-secret-token',
33
+ * ENV: 'production'
34
+ * },
35
+ * });
36
+ *
37
+ * const containerImage = dockerBuilder.getContainerImage();
38
+ */
10
39
  export declare class TokenInjectableDockerBuilder extends Construct {
11
40
  private readonly ecrRepository;
12
41
  private readonly buildTriggerResource;
42
+ /**
43
+ * Creates a new `TokenInjectableDockerBuilder` instance.
44
+ *
45
+ * @param scope The parent construct/stack.
46
+ * @param id The unique ID of the construct.
47
+ * @param props Configuration properties for the construct.
48
+ */
13
49
  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
50
+ /**
51
+ * Retrieves the container image from the ECR repository.
52
+ *
53
+ * @returns A `ContainerImage` object representing the built image.
54
+ */
14
55
  getContainerImage(): ContainerImage;
56
+ /**
57
+ * Retrieves the Docker image code for use in AWS Lambda.
58
+ *
59
+ * @returns A `DockerImageCode` object for the Docker image.
60
+ */
15
61
  getDockerImageCode(): DockerImageCode;
16
62
  }
package/lib/index.js CHANGED
@@ -13,7 +13,28 @@ const constructs_1 = require("constructs");
13
13
  const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
14
14
  const crypto = require("crypto");
15
15
  const aws_lambda_2 = require("aws-cdk-lib/aws-lambda");
16
+ /**
17
+ * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
18
+ *
19
+ * @example
20
+ * const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {
21
+ * path: './docker',
22
+ * buildArgs: {
23
+ * TOKEN: 'my-secret-token',
24
+ * ENV: 'production'
25
+ * },
26
+ * });
27
+ *
28
+ * const containerImage = dockerBuilder.getContainerImage();
29
+ */
16
30
  class TokenInjectableDockerBuilder extends constructs_1.Construct {
31
+ /**
32
+ * Creates a new `TokenInjectableDockerBuilder` instance.
33
+ *
34
+ * @param scope The parent construct/stack.
35
+ * @param id The unique ID of the construct.
36
+ * @param props Configuration properties for the construct.
37
+ */
17
38
  constructor(scope, id, props) {
18
39
  super(scope, id);
19
40
  const { path: sourcePath, buildArgs } = props; // Default to linux/amd64
@@ -87,7 +108,7 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
87
108
  }));
88
109
  // Create Node.js Lambda function for onEvent
89
110
  const onEventHandlerFunction = new aws_lambda_2.Function(this, 'OnEventHandlerFunction', {
90
- runtime: aws_lambda_1.Runtime.NODEJS_18_X, // Use Node.js runtime
111
+ runtime: aws_lambda_1.Runtime.NODEJS_LATEST, // Use Node.js runtime
91
112
  code: aws_lambda_1.Code.fromAsset(onEventHandlerPath), // Path to handler code
92
113
  handler: 'index.handler', // Entry point (adjust as needed)
93
114
  timeout: aws_cdk_lib_1.Duration.minutes(15),
@@ -98,7 +119,7 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
98
119
  }));
99
120
  // Create Node.js Lambda function for isComplete
100
121
  const isCompleteHandlerFunction = new aws_lambda_2.Function(this, 'IsCompleteHandlerFunction', {
101
- runtime: aws_lambda_1.Runtime.NODEJS_18_X,
122
+ runtime: aws_lambda_1.Runtime.NODEJS_LATEST,
102
123
  code: aws_lambda_1.Code.fromAsset(isCompleteHandlerPath),
103
124
  handler: 'index.handler',
104
125
  timeout: aws_cdk_lib_1.Duration.minutes(15),
@@ -129,12 +150,22 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
129
150
  });
130
151
  this.buildTriggerResource.node.addDependency(codeBuildProject);
131
152
  }
153
+ /**
154
+ * Retrieves the container image from the ECR repository.
155
+ *
156
+ * @returns A `ContainerImage` object representing the built image.
157
+ */
132
158
  getContainerImage() {
133
159
  return aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');
134
160
  }
161
+ /**
162
+ * Retrieves the Docker image code for use in AWS Lambda.
163
+ *
164
+ * @returns A `DockerImageCode` object for the Docker image.
165
+ */
135
166
  getDockerImageCode() {
136
167
  return aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository);
137
168
  }
138
169
  }
139
170
  exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
140
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Duration, CustomResource, Stack } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from 'aws-cdk-lib/aws-ecr';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, DockerImageCode, Runtime } 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';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport * as crypto from 'crypto';\nimport { Function } from 'aws-cdk-lib/aws-lambda';\n\nexport interface TokenInjectableDockerBuilderProps {\n  path: string;\n  buildArgs?: { [key: string]: string };\n}\n\nexport class TokenInjectableDockerBuilder extends Construct {\n  private readonly ecrRepository: Repository;\n  private readonly buildTriggerResource: CustomResource;\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, '../src/onEventHandler');\n    const isCompleteHandlerPath = path.resolve(__dirname, '../src/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_18_X, // 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_18_X,\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.minutes(1),\n    });\n\n    // Define the custom resource\n    this.buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    this.buildTriggerResource.node.addDependency(codeBuildProject);\n  }\n\n  public getContainerImage(): ContainerImage {\n    return ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');\n  }\n\n  public getDockerImageCode(): DockerImageCode {\n    return DockerImageCode.fromEcr(this.ecrRepository);\n  }\n}\n"]}
171
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Duration, CustomResource, Stack } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from 'aws-cdk-lib/aws-ecr';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, DockerImageCode, Runtime } 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';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport * as crypto from 'crypto';\nimport { Function } from 'aws-cdk-lib/aws-lambda';\n\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  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  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  private readonly ecrRepository: Repository;\n  private readonly buildTriggerResource: CustomResource;\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, '../src/onEventHandler');\n    const isCompleteHandlerPath = path.resolve(__dirname, '../src/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.minutes(1),\n    });\n\n    // Define the custom resource\n    this.buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    this.buildTriggerResource.node.addDependency(codeBuildProject);\n  }\n\n  /**\n   * Retrieves the container image from the ECR repository.\n   *\n   * @returns A `ContainerImage` object representing the built image.\n   */\n  public getContainerImage(): ContainerImage {\n    return ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');\n  }\n\n  /**\n   * Retrieves the Docker image code for use in AWS Lambda.\n   *\n   * @returns A `DockerImageCode` object for the Docker image.\n   */\n  public getDockerImageCode(): DockerImageCode {\n    return DockerImageCode.fromEcr(this.ecrRepository);\n  }\n}\n"]}
package/lib/index.ts CHANGED
@@ -11,15 +11,54 @@ import { ContainerImage } from 'aws-cdk-lib/aws-ecs';
11
11
  import * as crypto from 'crypto';
12
12
  import { Function } from 'aws-cdk-lib/aws-lambda';
13
13
 
14
+
15
+ /**
16
+ * Properties for the `TokenInjectableDockerBuilder` construct.
17
+ */
14
18
  export interface TokenInjectableDockerBuilderProps {
19
+ /**
20
+ * The path to the directory containing the Dockerfile or source code.
21
+ */
15
22
  path: string;
23
+
24
+ /**
25
+ * Build arguments to pass to the Docker build process.
26
+ * These are transformed into `--build-arg` flags.
27
+ * @example
28
+ * {
29
+ * TOKEN: 'my-secret-token',
30
+ * ENV: 'production'
31
+ * }
32
+ */
16
33
  buildArgs?: { [key: string]: string };
17
34
  }
18
35
 
36
+
37
+ /**
38
+ * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
39
+ *
40
+ * @example
41
+ * const dockerBuilder = new TokenInjectableDockerBuilder(this, 'DockerBuilder', {
42
+ * path: './docker',
43
+ * buildArgs: {
44
+ * TOKEN: 'my-secret-token',
45
+ * ENV: 'production'
46
+ * },
47
+ * });
48
+ *
49
+ * const containerImage = dockerBuilder.getContainerImage();
50
+ */
19
51
  export class TokenInjectableDockerBuilder extends Construct {
20
52
  private readonly ecrRepository: Repository;
21
53
  private readonly buildTriggerResource: CustomResource;
22
54
 
55
+ /**
56
+ * Creates a new `TokenInjectableDockerBuilder` instance.
57
+ *
58
+ * @param scope The parent construct/stack.
59
+ * @param id The unique ID of the construct.
60
+ * @param props Configuration properties for the construct.
61
+ */
23
62
  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {
24
63
  super(scope, id);
25
64
 
@@ -40,8 +79,8 @@ export class TokenInjectableDockerBuilder extends Construct {
40
79
  // Transform buildArgs into a string of --build-arg KEY=VALUE
41
80
  const buildArgsString = buildArgs
42
81
  ? Object.entries(buildArgs)
43
- .map(([key, value]) => `--build-arg ${key}=${value}`)
44
- .join(' ')
82
+ .map(([key, value]) => `--build-arg ${key}=${value}`)
83
+ .join(' ')
45
84
  : '';
46
85
 
47
86
  // Pass the buildArgsString and platform as environment variables
@@ -108,7 +147,7 @@ export class TokenInjectableDockerBuilder extends Construct {
108
147
 
109
148
  // Create Node.js Lambda function for onEvent
110
149
  const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {
111
- runtime: Runtime.NODEJS_18_X, // Use Node.js runtime
150
+ runtime: Runtime.NODEJS_LATEST, // Use Node.js runtime
112
151
  code: Code.fromAsset(onEventHandlerPath), // Path to handler code
113
152
  handler: 'index.handler', // Entry point (adjust as needed)
114
153
  timeout: Duration.minutes(15),
@@ -123,7 +162,7 @@ export class TokenInjectableDockerBuilder extends Construct {
123
162
 
124
163
  // Create Node.js Lambda function for isComplete
125
164
  const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {
126
- runtime: Runtime.NODEJS_18_X,
165
+ runtime: Runtime.NODEJS_LATEST,
127
166
  code: Code.fromAsset(isCompleteHandlerPath),
128
167
  handler: 'index.handler',
129
168
  timeout: Duration.minutes(15),
@@ -161,10 +200,20 @@ export class TokenInjectableDockerBuilder extends Construct {
161
200
  this.buildTriggerResource.node.addDependency(codeBuildProject);
162
201
  }
163
202
 
203
+ /**
204
+ * Retrieves the container image from the ECR repository.
205
+ *
206
+ * @returns A `ContainerImage` object representing the built image.
207
+ */
164
208
  public getContainerImage(): ContainerImage {
165
209
  return ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');
166
210
  }
167
211
 
212
+ /**
213
+ * Retrieves the Docker image code for use in AWS Lambda.
214
+ *
215
+ * @returns A `DockerImageCode` object for the Docker image.
216
+ */
168
217
  public getDockerImageCode(): DockerImageCode {
169
218
  return DockerImageCode.fromEcr(this.ecrRepository);
170
219
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "token-injectable-docker-builder",
3
- "version": "0.1.3",
3
+ "version": "0.1.5",
4
4
  "main": "lib/index.js",
5
5
  "types": "lib/index.d.ts",
6
6
  "scripts": {