token-injectable-docker-builder 1.8.1 → 1.9.0

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.js CHANGED
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
- var _a;
2
+ var _a, _b;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
- exports.TokenInjectableDockerBuilder = void 0;
4
+ exports.TokenInjectableDockerBuilder = exports.TokenInjectableDockerBuilderProvider = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
6
  const crypto = require("crypto");
7
7
  const fs = require("fs");
@@ -16,6 +16,78 @@ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
16
16
  const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
17
17
  const custom_resources_1 = require("aws-cdk-lib/custom-resources");
18
18
  const constructs_1 = require("constructs");
19
+ const PROVIDER_SINGLETON_ID = 'TokenInjectableDockerBuilderProvider';
20
+ /**
21
+ * Shared provider for `TokenInjectableDockerBuilder` instances.
22
+ *
23
+ * Creates the onEvent and isComplete Lambda functions once per stack.
24
+ * Each builder instance registers its CodeBuild project ARN so the
25
+ * shared Lambdas have permission to start builds and read logs.
26
+ */
27
+ class TokenInjectableDockerBuilderProvider extends constructs_1.Construct {
28
+ /**
29
+ * Get or create the singleton provider for this stack.
30
+ * All `TokenInjectableDockerBuilder` instances in the same stack
31
+ * share a single pair of Lambda functions.
32
+ */
33
+ static getOrCreate(scope, props) {
34
+ const stack = aws_cdk_lib_1.Stack.of(scope);
35
+ const existing = stack.node.tryFindChild(PROVIDER_SINGLETON_ID);
36
+ if (existing)
37
+ return existing;
38
+ return new TokenInjectableDockerBuilderProvider(stack, PROVIDER_SINGLETON_ID, props);
39
+ }
40
+ constructor(scope, id, props) {
41
+ super(scope, id);
42
+ this.onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandler', {
43
+ runtime: aws_lambda_1.Runtime.NODEJS_22_X,
44
+ code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
45
+ handler: 'onEvent.handler',
46
+ timeout: aws_cdk_lib_1.Duration.minutes(15),
47
+ });
48
+ this.isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandler', {
49
+ runtime: aws_lambda_1.Runtime.NODEJS_22_X,
50
+ code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
51
+ handler: 'isComplete.handler',
52
+ timeout: aws_cdk_lib_1.Duration.minutes(15),
53
+ });
54
+ this.isCompleteHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
55
+ actions: [
56
+ 'codebuild:BatchGetBuilds',
57
+ 'codebuild:ListBuildsForProject',
58
+ 'logs:GetLogEvents',
59
+ 'logs:DescribeLogStreams',
60
+ 'logs:DescribeLogGroups',
61
+ ],
62
+ resources: ['*'],
63
+ }));
64
+ const provider = new custom_resources_1.Provider(this, 'Provider', {
65
+ onEventHandler: this.onEventHandlerFunction,
66
+ isCompleteHandler: this.isCompleteHandlerFunction,
67
+ queryInterval: props?.queryInterval ?? aws_cdk_lib_1.Duration.seconds(30),
68
+ });
69
+ this.serviceToken = provider.serviceToken;
70
+ }
71
+ /**
72
+ * Grant the shared Lambdas permission to start builds for a specific
73
+ * CodeBuild project and pull/push to its ECR repository.
74
+ */
75
+ registerProject(project, ecrRepo, encryptionKey) {
76
+ this.onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
77
+ actions: ['codebuild:StartBuild'],
78
+ resources: [project.projectArn],
79
+ }));
80
+ ecrRepo.grantPullPush(this.onEventHandlerFunction);
81
+ ecrRepo.grantPullPush(this.isCompleteHandlerFunction);
82
+ if (encryptionKey) {
83
+ encryptionKey.grantEncryptDecrypt(this.onEventHandlerFunction);
84
+ encryptionKey.grantEncryptDecrypt(this.isCompleteHandlerFunction);
85
+ }
86
+ }
87
+ }
88
+ exports.TokenInjectableDockerBuilderProvider = TokenInjectableDockerBuilderProvider;
89
+ _a = JSII_RTTI_SYMBOL_1;
90
+ TokenInjectableDockerBuilderProvider[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilderProvider", version: "1.9.0" };
19
91
  /**
20
92
  * A CDK construct to build and push Docker images to an ECR repository using
21
93
  * CodeBuild and Lambda custom resources, **then** retrieve the final image tag
@@ -31,7 +103,7 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
31
103
  */
32
104
  constructor(scope, id, props) {
33
105
  super(scope, id);
34
- const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, kmsEncryption = false, completenessQueryInterval, exclude, file: dockerFile, cacheDisabled = false, buildLogGroup: buildLogGroupProp, } = props;
106
+ const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, kmsEncryption = false, completenessQueryInterval, exclude, file: dockerFile, cacheDisabled = false, buildLogGroup: buildLogGroupProp, provider: sharedProvider, } = props;
35
107
  // Generate an ephemeral tag for CodeBuild
36
108
  const imageTag = crypto.randomUUID();
37
109
  // Optionally define a KMS key for ECR encryption if requested
@@ -188,52 +260,58 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
188
260
  if (encryptionKey) {
189
261
  encryptionKey.grantEncryptDecrypt(codeBuildProject.role);
190
262
  }
191
- // Define Lambda functions for custom resource event and completion handling
192
- const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
193
- runtime: aws_lambda_1.Runtime.NODEJS_22_X,
194
- code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
195
- handler: 'onEvent.handler',
196
- timeout: aws_cdk_lib_1.Duration.minutes(15),
197
- });
198
- onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
199
- actions: ['codebuild:StartBuild'],
200
- resources: [codeBuildProject.projectArn],
201
- }));
202
- const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
203
- runtime: aws_lambda_1.Runtime.NODEJS_22_X,
204
- code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
205
- environment: {
206
- IMAGE_TAG: imageTag,
207
- },
208
- handler: 'isComplete.handler',
209
- timeout: aws_cdk_lib_1.Duration.minutes(15),
210
- });
211
- isCompleteHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
212
- actions: [
213
- 'codebuild:BatchGetBuilds',
214
- 'codebuild:ListBuildsForProject',
215
- 'logs:GetLogEvents',
216
- 'logs:DescribeLogStreams',
217
- 'logs:DescribeLogGroups',
218
- ],
219
- resources: ['*'],
220
- }));
221
- // Conditionally allow encryption if a key is used
222
- if (encryptionKey) {
223
- encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
224
- encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
263
+ // Resolve the service token: shared provider or per-instance Lambdas
264
+ let serviceToken;
265
+ if (sharedProvider) {
266
+ sharedProvider.registerProject(codeBuildProject, this.ecrRepository, encryptionKey);
267
+ serviceToken = sharedProvider.serviceToken;
268
+ }
269
+ else {
270
+ const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
271
+ runtime: aws_lambda_1.Runtime.NODEJS_22_X,
272
+ code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
273
+ handler: 'onEvent.handler',
274
+ timeout: aws_cdk_lib_1.Duration.minutes(15),
275
+ });
276
+ onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
277
+ actions: ['codebuild:StartBuild'],
278
+ resources: [codeBuildProject.projectArn],
279
+ }));
280
+ const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
281
+ runtime: aws_lambda_1.Runtime.NODEJS_22_X,
282
+ code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
283
+ environment: {
284
+ IMAGE_TAG: imageTag,
285
+ },
286
+ handler: 'isComplete.handler',
287
+ timeout: aws_cdk_lib_1.Duration.minutes(15),
288
+ });
289
+ isCompleteHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
290
+ actions: [
291
+ 'codebuild:BatchGetBuilds',
292
+ 'codebuild:ListBuildsForProject',
293
+ 'logs:GetLogEvents',
294
+ 'logs:DescribeLogStreams',
295
+ 'logs:DescribeLogGroups',
296
+ ],
297
+ resources: ['*'],
298
+ }));
299
+ if (encryptionKey) {
300
+ encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
301
+ encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
302
+ }
303
+ this.ecrRepository.grantPullPush(onEventHandlerFunction);
304
+ this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
305
+ const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
306
+ onEventHandler: onEventHandlerFunction,
307
+ isCompleteHandler: isCompleteHandlerFunction,
308
+ queryInterval: completenessQueryInterval ?? aws_cdk_lib_1.Duration.seconds(30),
309
+ });
310
+ serviceToken = provider.serviceToken;
225
311
  }
226
- this.ecrRepository.grantPullPush(onEventHandlerFunction);
227
- this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
228
- // Create a custom resource provider that uses the above Lambdas
229
- const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
230
- onEventHandler: onEventHandlerFunction,
231
- isCompleteHandler: isCompleteHandlerFunction,
232
- queryInterval: completenessQueryInterval ?? aws_cdk_lib_1.Duration.seconds(30),
233
- });
234
312
  // Custom Resource that triggers the CodeBuild and waits for completion
235
313
  const buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
236
- serviceToken: provider.serviceToken,
314
+ serviceToken,
237
315
  properties: {
238
316
  ProjectName: codeBuildProject.projectName,
239
317
  ImageTag: imageTag,
@@ -250,6 +328,6 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
250
328
  }
251
329
  }
252
330
  exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
253
- _a = JSII_RTTI_SYMBOL_1;
254
- TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.8.1" };
255
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport * as path from 'path';\n\nimport { CustomResource, Duration } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { IVpc, ISecurityGroup, SubnetSelection } from 'aws-cdk-lib/aws-ec2';\nimport { Repository, RepositoryEncryption, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { ILogGroup } from 'aws-cdk-lib/aws-logs';\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 KEY=VALUE` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n\n  /**\n   * The ARN of the AWS Secrets Manager secret containing Docker login credentials.\n   * This secret should store a JSON object with the following structure:\n   * ```json\n   * {\n   *   \"username\": \"my-docker-username\",\n   *   \"password\": \"my-docker-password\"\n   * }\n   * ```\n   * If not provided (or not needed), the construct will skip Docker Hub login.\n   *\n   * **Note**: The secret must be in the same region as the stack.\n   *\n   * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'\n   */\n  readonly dockerLoginSecretArn?: string;\n\n  /**\n   * The VPC in which the CodeBuild project will be deployed.\n   * If provided, the CodeBuild project will be launched within the specified VPC.\n   *\n   * @default - No VPC is attached, and the CodeBuild project will use public internet.\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * The security groups to attach to the CodeBuild project.\n   * These define the network access rules for the CodeBuild project.\n   *\n   * @default - No security groups are attached.\n   */\n  readonly securityGroups?: ISecurityGroup[];\n\n  /**\n   * The subnet selection to specify which subnets to use within the VPC.\n   * Allows the user to select private, public, or isolated subnets.\n   *\n   * @default - All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * installCommands: [\n   *   'echo \"Updating package lists...\"',\n   *   'apt-get update -y',\n   *   'echo \"Installing required packages...\"',\n   *   'apt-get install -y curl dnsutils',\n   * ],\n   * ```\n   * @default - No additional install commands.\n   */\n  readonly installCommands?: string[];\n\n  /**\n   * Custom commands to run during the pre_build phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * preBuildCommands: [\n   *   'echo \"Fetching configuration from private API...\"',\n   *   'curl -o config.json https://api.example.com/config',\n   * ],\n   * ```\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n\n  /**\n   * Whether to enable KMS encryption for the ECR repository.\n   * If `true`, a KMS key will be created for encrypting ECR images.\n   * If `false`, the repository will use AES-256 encryption.\n   *\n   * @default - false\n   */\n  readonly kmsEncryption?: boolean;\n\n  /**\n   * The query interval for checking if the CodeBuild project has completed.\n   * This determines how frequently the custom resource polls for build completion.\n   *\n   * @default - Duration.seconds(30)\n   */\n  readonly completenessQueryInterval?: Duration;\n\n  /**\n   * A list of file paths in the Docker directory to exclude from build.\n   * Will use paths in .dockerignore file if present.\n   *\n   * @default - No file path exclusions\n   */\n  readonly exclude?: string[];\n\n  /**\n   * The name of the Dockerfile to use for the build.\n   * Passed as `--file` to `docker build`.\n   *\n   * @example 'Dockerfile.production'\n   * @default 'Dockerfile'\n   */\n  readonly file?: string;\n\n  /**\n   * When `true`, disables Docker layer caching. Every build runs from scratch.\n   * Use for debugging, corrupted cache, or major dependency changes.\n   *\n   * @default false\n   */\n  readonly cacheDisabled?: boolean;\n\n  /**\n   * CloudWatch log group for CodeBuild build logs.\n   * When provided with a RETAIN removal policy, build logs survive rollbacks\n   * and stack deletion for debugging.\n   *\n   * @default - CodeBuild default logging (logs are deleted on rollback)\n   */\n  readonly buildLogGroup?: ILogGroup;\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using\n * CodeBuild and Lambda custom resources, **then** retrieve the final image tag\n * so that ECS/Lambda references use the exact digest.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  /**\n   * The ECR repository that stores the resulting Docker image.\n   */\n  private readonly ecrRepository: Repository;\n\n  /**\n   * An ECS-compatible container image referencing the tag\n   * of the built Docker image.\n   */\n  public readonly containerImage: ContainerImage;\n\n  /**\n   * A Lambda-compatible DockerImageCode referencing the tag\n   * of the built Docker image.\n   */\n  public readonly dockerImageCode: DockerImageCode;\n\n  /**\n   * Creates a new `TokenInjectableDockerBuilder`.\n   *\n   * @param scope The scope in which to define this construct.\n   * @param id The scoped construct ID.\n   * @param props Configuration for building and pushing the Docker image.\n   */\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const {\n      path: sourcePath,\n      buildArgs,\n      dockerLoginSecretArn,\n      vpc,\n      securityGroups,\n      subnetSelection,\n      installCommands,\n      preBuildCommands,\n      kmsEncryption = false,\n      completenessQueryInterval,\n      exclude,\n      file: dockerFile,\n      cacheDisabled = false,\n      buildLogGroup: buildLogGroupProp,\n    } = props;\n\n    // Generate an ephemeral tag for CodeBuild\n    const imageTag = crypto.randomUUID();\n\n    // Optionally define a KMS key for ECR encryption if requested\n    let encryptionKey: Key | undefined;\n    if (kmsEncryption) {\n      encryptionKey = new Key(this, 'EcrEncryptionKey', {\n        enableKeyRotation: true,\n      });\n    }\n\n    // Create an ECR repository (optionally with KMS encryption)\n    this.ecrRepository = new Repository(this, 'ECRRepository', {\n      lifecycleRules: [\n        {\n          rulePriority: 1,\n          description: 'Remove untagged images after 1 day',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n      encryption: kmsEncryption ? RepositoryEncryption.KMS : RepositoryEncryption.AES_256,\n      encryptionKey: kmsEncryption ? encryptionKey : undefined,\n      imageScanOnPush: true,\n    });\n\n    let effectiveExclude = exclude;\n    if (!effectiveExclude) {\n      const dockerignorePath = path.join(sourcePath, '.dockerignore');\n      if (fs.existsSync(dockerignorePath)) {\n        const fileContent = fs.readFileSync(dockerignorePath, 'utf8');\n        effectiveExclude = fileContent\n          .split('\\n')\n          .map((line: string) => line.trim())\n          .filter((line: string) => line.length > 0 && !line.startsWith('#'));\n      }\n    }\n\n    // Ensure the target Dockerfile is never excluded (handles globs like \"Dockerfile*\")\n    const dockerFileName = dockerFile ?? 'Dockerfile';\n    if (effectiveExclude) {\n      effectiveExclude = effectiveExclude.filter((pattern: string) => {\n        const escaped = pattern.replace(/[.+^${}()|[\\]\\\\]/g, '\\\\$&');\n        const regex = new RegExp(`^${escaped.replace(/\\*/g, '.*').replace(/\\?/g, '.')}$`, 'i');\n        return !regex.test(dockerFileName);\n      });\n    }\n\n    // Wrap the source folder as an S3 asset for CodeBuild to use\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n      exclude: effectiveExclude,\n\n    });\n\n    // Convert buildArgs to a CLI-friendly string\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([k, v]) => `--build-arg ${k}=${v}`)\n        .join(' ')\n      : '';\n\n    const dockerFileFlag = dockerFile ? `-f $CODEBUILD_SRC_DIR/${dockerFile}` : '';\n\n    // Optional DockerHub login, if a secret ARN is provided\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials...\"',\n        'apt-get update -y && apt-get install -y jq',\n        `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,\n        `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,\n        'echo \"Logging in to Docker Hub...\"',\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials. Skipping Docker Hub login.\"'];\n\n    const buildxInstallCommands = cacheDisabled\n      ? []\n      : [\n        'echo \"Setting up Docker buildx for ECR layer cache...\"',\n        'docker buildx create --driver docker-container --name ecr-cache-builder --use 2>/dev/null || docker buildx use ecr-cache-builder',\n      ];\n\n    const buildCommand = cacheDisabled\n      ? `docker build ${dockerFileFlag} ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`\n      : `docker buildx build --push --cache-from type=registry,ref=$ECR_REPO_URI:cache --cache-to type=registry,ref=$ECR_REPO_URI:cache,mode=max,image-manifest=true ${dockerFileFlag} ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`;\n\n    const buildSpecObj = {\n      version: '0.2',\n      phases: {\n        install: {\n          commands: [\n            'echo \"Beginning install phase...\"',\n            ...(installCommands ?? []),\n            ...buildxInstallCommands,\n          ],\n        },\n        pre_build: {\n          commands: [\n            ...(preBuildCommands ?? []),\n            ...dockerLoginCommands,\n            'echo \"Retrieving AWS Account ID...\"',\n            'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n            'echo \"Logging into 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 \"Building Docker image with tag ${imageTag}...\"`,\n            buildCommand,\n          ],\n        },\n        ...(cacheDisabled && {\n          post_build: {\n            commands: [\n              `echo \"Pushing Docker image with tag ${imageTag}...\"`,\n              `docker push $ECR_REPO_URI:${imageTag}`,\n            ],\n          },\n        }),\n      },\n    };\n\n    // Create the CodeBuild project\n    const codeBuildProject = new Project(this, 'CodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true,\n      },\n      environmentVariables: {\n        ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      },\n      buildSpec: BuildSpec.fromObject(buildSpecObj),\n      ...(buildLogGroupProp && {\n        logging: {\n          cloudWatch: {\n            logGroup: buildLogGroupProp,\n          },\n        },\n      }),\n      vpc,\n      securityGroups,\n      subnetSelection,\n    });\n\n    // Grant CodeBuild the ability to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:BatchCheckLayerAvailability',\n        ],\n        resources: ['*'],\n      }),\n    );\n    if (dockerLoginSecretArn) {\n      codeBuildProject.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['secretsmanager:GetSecretValue'],\n          resources: [dockerLoginSecretArn],\n        }),\n      );\n    }\n\n    // Conditionally grant KMS encrypt/decrypt if a key is used\n    if (encryptionKey) {\n      encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n    }\n\n    // Define Lambda functions for custom resource event and completion handling\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_22_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\n      timeout: Duration.minutes(15),\n    });\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn],\n      }),\n    );\n\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_22_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      environment: {\n        IMAGE_TAG: imageTag,\n      },\n      handler: 'isComplete.handler',\n      timeout: Duration.minutes(15),\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    // Conditionally allow encryption if a key is used\n    if (encryptionKey) {\n      encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n      encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n    }\n    this.ecrRepository.grantPullPush(onEventHandlerFunction);\n    this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\n\n    // Create a custom resource provider that uses the above Lambdas\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: completenessQueryInterval ?? Duration.seconds(30),\n    });\n\n    // Custom Resource that triggers the CodeBuild and waits for completion\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        ImageTag: imageTag,\n        Trigger: sourceAsset.assetHash,\n      },\n    });\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Retrieve the final Docker image tag from Data.ImageTag\n    const imageTagRef = buildTriggerResource.getAttString('ImageTag');\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository, imageTagRef);\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository, {\n      tagOrDigest: imageTagRef,\n    });\n  }\n}\n"]}
331
+ _b = JSII_RTTI_SYMBOL_1;
332
+ TokenInjectableDockerBuilder[_b] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.9.0" };
333
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport * as path from 'path';\n\nimport { CustomResource, Duration, Stack } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { IVpc, ISecurityGroup, SubnetSelection } from 'aws-cdk-lib/aws-ec2';\nimport { Repository, RepositoryEncryption, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { ILogGroup } from 'aws-cdk-lib/aws-logs';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\nconst PROVIDER_SINGLETON_ID = 'TokenInjectableDockerBuilderProvider';\n\n/**\n * Options for creating a `TokenInjectableDockerBuilderProvider`.\n */\nexport interface TokenInjectableDockerBuilderProviderProps {\n  /**\n   * How often the provider polls for build completion.\n   *\n   * @default Duration.seconds(30)\n   */\n  readonly queryInterval?: Duration;\n}\n\n/**\n * Shared provider for `TokenInjectableDockerBuilder` instances.\n *\n * Creates the onEvent and isComplete Lambda functions once per stack.\n * Each builder instance registers its CodeBuild project ARN so the\n * shared Lambdas have permission to start builds and read logs.\n */\nexport class TokenInjectableDockerBuilderProvider extends Construct {\n  /**\n   * Get or create the singleton provider for this stack.\n   * All `TokenInjectableDockerBuilder` instances in the same stack\n   * share a single pair of Lambda functions.\n   */\n  public static getOrCreate(scope: Construct, props?: TokenInjectableDockerBuilderProviderProps): TokenInjectableDockerBuilderProvider {\n    const stack = Stack.of(scope);\n    const existing = stack.node.tryFindChild(PROVIDER_SINGLETON_ID) as TokenInjectableDockerBuilderProvider | undefined;\n    if (existing) return existing;\n    return new TokenInjectableDockerBuilderProvider(stack, PROVIDER_SINGLETON_ID, props);\n  }\n\n  /** The service token used by CustomResource instances. */\n  public readonly serviceToken: string;\n\n  private readonly onEventHandlerFunction: Function;\n  private readonly isCompleteHandlerFunction: Function;\n\n  private constructor(scope: Construct, id: string, props?: TokenInjectableDockerBuilderProviderProps) {\n    super(scope, id);\n\n    this.onEventHandlerFunction = new Function(this, 'OnEventHandler', {\n      runtime: Runtime.NODEJS_22_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    this.isCompleteHandlerFunction = new Function(this, 'IsCompleteHandler', {\n      runtime: Runtime.NODEJS_22_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      handler: 'isComplete.handler',\n      timeout: Duration.minutes(15),\n    });\n    this.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    const provider = new Provider(this, 'Provider', {\n      onEventHandler: this.onEventHandlerFunction,\n      isCompleteHandler: this.isCompleteHandlerFunction,\n      queryInterval: props?.queryInterval ?? Duration.seconds(30),\n    });\n\n    this.serviceToken = provider.serviceToken;\n  }\n\n  /**\n   * Grant the shared Lambdas permission to start builds for a specific\n   * CodeBuild project and pull/push to its ECR repository.\n   */\n  public registerProject(project: Project, ecrRepo: Repository, encryptionKey?: Key): void {\n    this.onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [project.projectArn],\n      }),\n    );\n    ecrRepo.grantPullPush(this.onEventHandlerFunction);\n    ecrRepo.grantPullPush(this.isCompleteHandlerFunction);\n\n    if (encryptionKey) {\n      encryptionKey.grantEncryptDecrypt(this.onEventHandlerFunction);\n      encryptionKey.grantEncryptDecrypt(this.isCompleteHandlerFunction);\n    }\n  }\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  readonly path: string;\n\n  /**\n   * Build arguments to pass to the Docker build process.\n   * These are transformed into `--build-arg KEY=VALUE` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n\n  /**\n   * The ARN of the AWS Secrets Manager secret containing Docker login credentials.\n   * This secret should store a JSON object with the following structure:\n   * ```json\n   * {\n   *   \"username\": \"my-docker-username\",\n   *   \"password\": \"my-docker-password\"\n   * }\n   * ```\n   * If not provided (or not needed), the construct will skip Docker Hub login.\n   *\n   * **Note**: The secret must be in the same region as the stack.\n   *\n   * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'\n   */\n  readonly dockerLoginSecretArn?: string;\n\n  /**\n   * The VPC in which the CodeBuild project will be deployed.\n   * If provided, the CodeBuild project will be launched within the specified VPC.\n   *\n   * @default - No VPC is attached, and the CodeBuild project will use public internet.\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * The security groups to attach to the CodeBuild project.\n   * These define the network access rules for the CodeBuild project.\n   *\n   * @default - No security groups are attached.\n   */\n  readonly securityGroups?: ISecurityGroup[];\n\n  /**\n   * The subnet selection to specify which subnets to use within the VPC.\n   * Allows the user to select private, public, or isolated subnets.\n   *\n   * @default - All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * installCommands: [\n   *   'echo \"Updating package lists...\"',\n   *   'apt-get update -y',\n   *   'echo \"Installing required packages...\"',\n   *   'apt-get install -y curl dnsutils',\n   * ],\n   * ```\n   * @default - No additional install commands.\n   */\n  readonly installCommands?: string[];\n\n  /**\n   * Custom commands to run during the pre_build phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * preBuildCommands: [\n   *   'echo \"Fetching configuration from private API...\"',\n   *   'curl -o config.json https://api.example.com/config',\n   * ],\n   * ```\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n\n  /**\n   * Whether to enable KMS encryption for the ECR repository.\n   * If `true`, a KMS key will be created for encrypting ECR images.\n   * If `false`, the repository will use AES-256 encryption.\n   *\n   * @default - false\n   */\n  readonly kmsEncryption?: boolean;\n\n  /**\n   * The query interval for checking if the CodeBuild project has completed.\n   * This determines how frequently the custom resource polls for build completion.\n   *\n   * @default - Duration.seconds(30)\n   */\n  readonly completenessQueryInterval?: Duration;\n\n  /**\n   * A list of file paths in the Docker directory to exclude from build.\n   * Will use paths in .dockerignore file if present.\n   *\n   * @default - No file path exclusions\n   */\n  readonly exclude?: string[];\n\n  /**\n   * The name of the Dockerfile to use for the build.\n   * Passed as `--file` to `docker build`.\n   *\n   * @example 'Dockerfile.production'\n   * @default 'Dockerfile'\n   */\n  readonly file?: string;\n\n  /**\n   * When `true`, disables Docker layer caching. Every build runs from scratch.\n   * Use for debugging, corrupted cache, or major dependency changes.\n   *\n   * @default false\n   */\n  readonly cacheDisabled?: boolean;\n\n  /**\n   * CloudWatch log group for CodeBuild build logs.\n   * When provided with a RETAIN removal policy, build logs survive rollbacks\n   * and stack deletion for debugging.\n   *\n   * @default - CodeBuild default logging (logs are deleted on rollback)\n   */\n  readonly buildLogGroup?: ILogGroup;\n\n  /**\n   * Shared provider for the custom resource Lambdas.\n   * Use `TokenInjectableDockerBuilderProvider.getOrCreate(this)` to create\n   * a singleton that is shared across all builders in the same stack.\n   *\n   * When omitted, each builder creates its own Lambdas (original behavior).\n   *\n   * @default - A new provider is created per builder instance\n   */\n  readonly provider?: TokenInjectableDockerBuilderProvider;\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using\n * CodeBuild and Lambda custom resources, **then** retrieve the final image tag\n * so that ECS/Lambda references use the exact digest.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  /**\n   * The ECR repository that stores the resulting Docker image.\n   */\n  private readonly ecrRepository: Repository;\n\n  /**\n   * An ECS-compatible container image referencing the tag\n   * of the built Docker image.\n   */\n  public readonly containerImage: ContainerImage;\n\n  /**\n   * A Lambda-compatible DockerImageCode referencing the tag\n   * of the built Docker image.\n   */\n  public readonly dockerImageCode: DockerImageCode;\n\n  /**\n   * Creates a new `TokenInjectableDockerBuilder`.\n   *\n   * @param scope The scope in which to define this construct.\n   * @param id The scoped construct ID.\n   * @param props Configuration for building and pushing the Docker image.\n   */\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const {\n      path: sourcePath,\n      buildArgs,\n      dockerLoginSecretArn,\n      vpc,\n      securityGroups,\n      subnetSelection,\n      installCommands,\n      preBuildCommands,\n      kmsEncryption = false,\n      completenessQueryInterval,\n      exclude,\n      file: dockerFile,\n      cacheDisabled = false,\n      buildLogGroup: buildLogGroupProp,\n      provider: sharedProvider,\n    } = props;\n\n    // Generate an ephemeral tag for CodeBuild\n    const imageTag = crypto.randomUUID();\n\n    // Optionally define a KMS key for ECR encryption if requested\n    let encryptionKey: Key | undefined;\n    if (kmsEncryption) {\n      encryptionKey = new Key(this, 'EcrEncryptionKey', {\n        enableKeyRotation: true,\n      });\n    }\n\n    // Create an ECR repository (optionally with KMS encryption)\n    this.ecrRepository = new Repository(this, 'ECRRepository', {\n      lifecycleRules: [\n        {\n          rulePriority: 1,\n          description: 'Remove untagged images after 1 day',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n      encryption: kmsEncryption ? RepositoryEncryption.KMS : RepositoryEncryption.AES_256,\n      encryptionKey: kmsEncryption ? encryptionKey : undefined,\n      imageScanOnPush: true,\n    });\n\n    let effectiveExclude = exclude;\n    if (!effectiveExclude) {\n      const dockerignorePath = path.join(sourcePath, '.dockerignore');\n      if (fs.existsSync(dockerignorePath)) {\n        const fileContent = fs.readFileSync(dockerignorePath, 'utf8');\n        effectiveExclude = fileContent\n          .split('\\n')\n          .map((line: string) => line.trim())\n          .filter((line: string) => line.length > 0 && !line.startsWith('#'));\n      }\n    }\n\n    // Ensure the target Dockerfile is never excluded (handles globs like \"Dockerfile*\")\n    const dockerFileName = dockerFile ?? 'Dockerfile';\n    if (effectiveExclude) {\n      effectiveExclude = effectiveExclude.filter((pattern: string) => {\n        const escaped = pattern.replace(/[.+^${}()|[\\]\\\\]/g, '\\\\$&');\n        const regex = new RegExp(`^${escaped.replace(/\\*/g, '.*').replace(/\\?/g, '.')}$`, 'i');\n        return !regex.test(dockerFileName);\n      });\n    }\n\n    // Wrap the source folder as an S3 asset for CodeBuild to use\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n      exclude: effectiveExclude,\n    });\n\n    // Convert buildArgs to a CLI-friendly string\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([k, v]) => `--build-arg ${k}=${v}`)\n        .join(' ')\n      : '';\n\n    const dockerFileFlag = dockerFile ? `-f $CODEBUILD_SRC_DIR/${dockerFile}` : '';\n\n    // Optional DockerHub login, if a secret ARN is provided\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials...\"',\n        'apt-get update -y && apt-get install -y jq',\n        `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,\n        `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,\n        'echo \"Logging in to Docker Hub...\"',\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials. Skipping Docker Hub login.\"'];\n\n    const buildxInstallCommands = cacheDisabled\n      ? []\n      : [\n        'echo \"Setting up Docker buildx for ECR layer cache...\"',\n        'docker buildx create --driver docker-container --name ecr-cache-builder --use 2>/dev/null || docker buildx use ecr-cache-builder',\n      ];\n\n    const buildCommand = cacheDisabled\n      ? `docker build ${dockerFileFlag} ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`\n      : `docker buildx build --push --cache-from type=registry,ref=$ECR_REPO_URI:cache --cache-to type=registry,ref=$ECR_REPO_URI:cache,mode=max,image-manifest=true ${dockerFileFlag} ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`;\n\n    const buildSpecObj = {\n      version: '0.2',\n      phases: {\n        install: {\n          commands: [\n            'echo \"Beginning install phase...\"',\n            ...(installCommands ?? []),\n            ...buildxInstallCommands,\n          ],\n        },\n        pre_build: {\n          commands: [\n            ...(preBuildCommands ?? []),\n            ...dockerLoginCommands,\n            'echo \"Retrieving AWS Account ID...\"',\n            'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n            'echo \"Logging into 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 \"Building Docker image with tag ${imageTag}...\"`,\n            buildCommand,\n          ],\n        },\n        ...(cacheDisabled && {\n          post_build: {\n            commands: [\n              `echo \"Pushing Docker image with tag ${imageTag}...\"`,\n              `docker push $ECR_REPO_URI:${imageTag}`,\n            ],\n          },\n        }),\n      },\n    };\n\n    // Create the CodeBuild project\n    const codeBuildProject = new Project(this, 'CodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true,\n      },\n      environmentVariables: {\n        ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      },\n      buildSpec: BuildSpec.fromObject(buildSpecObj),\n      ...(buildLogGroupProp && {\n        logging: {\n          cloudWatch: {\n            logGroup: buildLogGroupProp,\n          },\n        },\n      }),\n      vpc,\n      securityGroups,\n      subnetSelection,\n    });\n\n    // Grant CodeBuild the ability to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:BatchCheckLayerAvailability',\n        ],\n        resources: ['*'],\n      }),\n    );\n    if (dockerLoginSecretArn) {\n      codeBuildProject.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['secretsmanager:GetSecretValue'],\n          resources: [dockerLoginSecretArn],\n        }),\n      );\n    }\n\n    // Conditionally grant KMS encrypt/decrypt if a key is used\n    if (encryptionKey) {\n      encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n    }\n\n    // Resolve the service token: shared provider or per-instance Lambdas\n    let serviceToken: string;\n    if (sharedProvider) {\n      sharedProvider.registerProject(codeBuildProject, this.ecrRepository, encryptionKey);\n      serviceToken = sharedProvider.serviceToken;\n    } else {\n      const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n        runtime: Runtime.NODEJS_22_X,\n        code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n        handler: 'onEvent.handler',\n        timeout: Duration.minutes(15),\n      });\n      onEventHandlerFunction.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['codebuild:StartBuild'],\n          resources: [codeBuildProject.projectArn],\n        }),\n      );\n\n      const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n        runtime: Runtime.NODEJS_22_X,\n        code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n        environment: {\n          IMAGE_TAG: imageTag,\n        },\n        handler: 'isComplete.handler',\n        timeout: Duration.minutes(15),\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      if (encryptionKey) {\n        encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n        encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n      }\n      this.ecrRepository.grantPullPush(onEventHandlerFunction);\n      this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\n\n      const provider = new Provider(this, 'CustomResourceProvider', {\n        onEventHandler: onEventHandlerFunction,\n        isCompleteHandler: isCompleteHandlerFunction,\n        queryInterval: completenessQueryInterval ?? Duration.seconds(30),\n      });\n      serviceToken = provider.serviceToken;\n    }\n\n    // Custom Resource that triggers the CodeBuild and waits for completion\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        ImageTag: imageTag,\n        Trigger: sourceAsset.assetHash,\n      },\n    });\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Retrieve the final Docker image tag from Data.ImageTag\n    const imageTagRef = buildTriggerResource.getAttString('ImageTag');\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository, imageTagRef);\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository, {\n      tagOrDigest: imageTagRef,\n    });\n  }\n}\n"]}
package/package.json CHANGED
@@ -98,7 +98,7 @@
98
98
  "publishConfig": {
99
99
  "access": "public"
100
100
  },
101
- "version": "1.8.1",
101
+ "version": "1.9.0",
102
102
  "jest": {
103
103
  "coverageProvider": "v8",
104
104
  "testMatch": [