token-injectable-docker-builder 1.2.6 → 1.3.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/.jsii CHANGED
@@ -3860,7 +3860,7 @@
3860
3860
  "stability": "stable"
3861
3861
  },
3862
3862
  "homepage": "https://github.com/AlexTech314/TokenInjectableDockerBuilder.git",
3863
- "jsiiVersion": "5.5.12 (build 8b85294)",
3863
+ "jsiiVersion": "5.5.14 (build 992ecef)",
3864
3864
  "keywords": [
3865
3865
  "aws",
3866
3866
  "aws-cdk",
@@ -3920,7 +3920,7 @@
3920
3920
  "base": "constructs.Construct",
3921
3921
  "docs": {
3922
3922
  "stability": "stable",
3923
- "summary": "A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources."
3923
+ "summary": "A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources, retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references."
3924
3924
  },
3925
3925
  "fqn": "token-injectable-docker-builder.TokenInjectableDockerBuilder",
3926
3926
  "initializer": {
@@ -3929,7 +3929,7 @@
3929
3929
  },
3930
3930
  "locationInModule": {
3931
3931
  "filename": "src/index.ts",
3932
- "line": 123
3932
+ "line": 128
3933
3933
  },
3934
3934
  "parameters": [
3935
3935
  {
@@ -3955,18 +3955,19 @@
3955
3955
  "kind": "class",
3956
3956
  "locationInModule": {
3957
3957
  "filename": "src/index.ts",
3958
- "line": 118
3958
+ "line": 115
3959
3959
  },
3960
3960
  "name": "TokenInjectableDockerBuilder",
3961
3961
  "properties": [
3962
3962
  {
3963
3963
  "docs": {
3964
- "stability": "stable"
3964
+ "stability": "stable",
3965
+ "summary": "An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image."
3965
3966
  },
3966
3967
  "immutable": true,
3967
3968
  "locationInModule": {
3968
3969
  "filename": "src/index.ts",
3969
- "line": 120
3970
+ "line": 121
3970
3971
  },
3971
3972
  "name": "containerImage",
3972
3973
  "type": {
@@ -3975,12 +3976,13 @@
3975
3976
  },
3976
3977
  {
3977
3978
  "docs": {
3978
- "stability": "stable"
3979
+ "stability": "stable",
3980
+ "summary": "A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image."
3979
3981
  },
3980
3982
  "immutable": true,
3981
3983
  "locationInModule": {
3982
3984
  "filename": "src/index.ts",
3983
- "line": 121
3985
+ "line": 126
3984
3986
  },
3985
3987
  "name": "dockerImageCode",
3986
3988
  "type": {
@@ -4049,14 +4051,14 @@
4049
4051
  "abstract": true,
4050
4052
  "docs": {
4051
4053
  "example": "'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'",
4052
- "remarks": "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```\nIf not provided, the construct will skip Docker login during the build process.",
4054
+ "remarks": "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```\nIf not provided (or not needed), the construct will skip Docker Hub login.\nNOTE: The secret must be in the same region as the stack.",
4053
4055
  "stability": "stable",
4054
4056
  "summary": "The ARN of the AWS Secrets Manager secret containing Docker login credentials."
4055
4057
  },
4056
4058
  "immutable": true,
4057
4059
  "locationInModule": {
4058
4060
  "filename": "src/index.ts",
4059
- "line": 48
4061
+ "line": 49
4060
4062
  },
4061
4063
  "name": "dockerLoginSecretArn",
4062
4064
  "optional": true,
@@ -4068,14 +4070,14 @@
4068
4070
  "abstract": true,
4069
4071
  "docs": {
4070
4072
  "default": "- No additional install commands.",
4071
- "remarks": "**Example Usage:**\n```typescript\nnew TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n path: path.resolve(__dirname, '../app'),\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 // ... other properties ...\n});\n```\n*This example demonstrates how to install the `curl` and `dnsutils` packages during the install phase using `apt-get`, the package manager for Ubuntu-based CodeBuild environments.*",
4073
+ "remarks": "**Example Usage:**\n```typescript\nnew TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n path: path.resolve(__dirname, '../app'),\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 // ... other properties ...\n});\n```",
4072
4074
  "stability": "stable",
4073
4075
  "summary": "Custom commands to run during the install phase."
4074
4076
  },
4075
4077
  "immutable": true,
4076
4078
  "locationInModule": {
4077
4079
  "filename": "src/index.ts",
4078
- "line": 91
4080
+ "line": 90
4079
4081
  },
4080
4082
  "name": "installCommands",
4081
4083
  "optional": true,
@@ -4092,14 +4094,14 @@
4092
4094
  "abstract": true,
4093
4095
  "docs": {
4094
4096
  "default": "- No additional pre-build commands.",
4095
- "remarks": "**Example Usage:**\n```typescript\nnew TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n path: path.resolve(__dirname, '../app'),\n preBuildCommands: [\n 'echo \"Fetching configuration from private API...\"',\n 'curl -o config.json https://api.example.com/config',\n ],\n // ... other properties ...\n});\n```\n*In this example, the builder fetches a configuration file from a private API before starting the Docker build. This config file will be available in the same directory as your Dockerfile during CDK deployment.*",
4097
+ "remarks": "**Example Usage:**\n```typescript\nnew TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n path: path.resolve(__dirname, '../app'),\n preBuildCommands: [\n 'echo \"Fetching configuration from private API...\"',\n 'curl -o config.json https://api.example.com/config',\n ],\n // ... other properties ...\n});\n```",
4096
4098
  "stability": "stable",
4097
4099
  "summary": "Custom commands to run during the pre_build phase."
4098
4100
  },
4099
4101
  "immutable": true,
4100
4102
  "locationInModule": {
4101
4103
  "filename": "src/index.ts",
4102
- "line": 112
4104
+ "line": 108
4103
4105
  },
4104
4106
  "name": "preBuildCommands",
4105
4107
  "optional": true,
@@ -4123,7 +4125,7 @@
4123
4125
  "immutable": true,
4124
4126
  "locationInModule": {
4125
4127
  "filename": "src/index.ts",
4126
- "line": 62
4128
+ "line": 63
4127
4129
  },
4128
4130
  "name": "securityGroups",
4129
4131
  "optional": true,
@@ -4147,7 +4149,7 @@
4147
4149
  "immutable": true,
4148
4150
  "locationInModule": {
4149
4151
  "filename": "src/index.ts",
4150
- "line": 69
4152
+ "line": 70
4151
4153
  },
4152
4154
  "name": "subnetSelection",
4153
4155
  "optional": true,
@@ -4166,7 +4168,7 @@
4166
4168
  "immutable": true,
4167
4169
  "locationInModule": {
4168
4170
  "filename": "src/index.ts",
4169
- "line": 55
4171
+ "line": 56
4170
4172
  },
4171
4173
  "name": "vpc",
4172
4174
  "optional": true,
@@ -4178,6 +4180,6 @@
4178
4180
  "symbolId": "src/index:TokenInjectableDockerBuilderProps"
4179
4181
  }
4180
4182
  },
4181
- "version": "1.2.6",
4182
- "fingerprint": "WNcXGWyC7kcurolCEd97ibbPLxLoUQgd+lwmoF2Mb7U="
4183
+ "version": "1.3.0",
4184
+ "fingerprint": "ZL25JVTzJLNHeDV7j15vMt4HHx0Ae4Mvgy29nJsio6w="
4183
4185
  }
package/API.md CHANGED
@@ -4,7 +4,7 @@
4
4
 
5
5
  ### TokenInjectableDockerBuilder <a name="TokenInjectableDockerBuilder" id="token-injectable-docker-builder.TokenInjectableDockerBuilder"></a>
6
6
 
7
- A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
7
+ A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources, retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.
8
8
 
9
9
  #### Initializers <a name="Initializers" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer"></a>
10
10
 
@@ -87,8 +87,8 @@ Any object.
87
87
  | **Name** | **Type** | **Description** |
88
88
  | --- | --- | --- |
89
89
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.node">node</a></code> | <code>constructs.Node</code> | The tree node. |
90
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.containerImage">containerImage</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerImage</code> | *No description.* |
91
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode">dockerImageCode</a></code> | <code>aws-cdk-lib.aws_lambda.DockerImageCode</code> | *No description.* |
90
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.containerImage">containerImage</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerImage</code> | An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image. |
91
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode">dockerImageCode</a></code> | <code>aws-cdk-lib.aws_lambda.DockerImageCode</code> | A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image. |
92
92
 
93
93
  ---
94
94
 
@@ -112,6 +112,8 @@ public readonly containerImage: ContainerImage;
112
112
 
113
113
  - *Type:* aws-cdk-lib.aws_ecs.ContainerImage
114
114
 
115
+ An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.
116
+
115
117
  ---
116
118
 
117
119
  ##### `dockerImageCode`<sup>Required</sup> <a name="dockerImageCode" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode"></a>
@@ -122,6 +124,8 @@ public readonly dockerImageCode: DockerImageCode;
122
124
 
123
125
  - *Type:* aws-cdk-lib.aws_lambda.DockerImageCode
124
126
 
127
+ A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.
128
+
125
129
  ---
126
130
 
127
131
 
@@ -207,7 +211,8 @@ This secret should store a JSON object with the following structure:
207
211
  "password": "my-docker-password"
208
212
  }
209
213
  ```
210
- If not provided, the construct will skip Docker login during the build process.
214
+ If not provided (or not needed), the construct will skip Docker Hub login.
215
+ NOTE: The secret must be in the same region as the stack.
211
216
 
212
217
  ---
213
218
 
@@ -242,7 +247,6 @@ new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
242
247
  // ... other properties ...
243
248
  });
244
249
  ```
245
- *This example demonstrates how to install the `curl` and `dnsutils` packages during the install phase using `apt-get`, the package manager for Ubuntu-based CodeBuild environments.*
246
250
 
247
251
  ---
248
252
 
@@ -268,7 +272,6 @@ new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
268
272
  // ... other properties ...
269
273
  });
270
274
  ```
271
- *In this example, the builder fetches a configuration file from a private API before starting the Docker build. This config file will be available in the same directory as your Dockerfile during CDK deployment.*
272
275
 
273
276
  ---
274
277
 
package/lib/index.d.ts CHANGED
@@ -31,7 +31,8 @@ export interface TokenInjectableDockerBuilderProps {
31
31
  * "password": "my-docker-password"
32
32
  * }
33
33
  * ```
34
- * If not provided, the construct will skip Docker login during the build process.
34
+ * If not provided (or not needed), the construct will skip Docker Hub login.
35
+ * NOTE: The secret must be in the same region as the stack.
35
36
  *
36
37
  * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'
37
38
  */
@@ -70,8 +71,6 @@ export interface TokenInjectableDockerBuilderProps {
70
71
  * // ... other properties ...
71
72
  * });
72
73
  * ```
73
- * *This example demonstrates how to install the `curl` and `dnsutils` packages during the install phase using `apt-get`, the package manager for Ubuntu-based CodeBuild environments.*
74
- *
75
74
  * @default - No additional install commands.
76
75
  */
77
76
  readonly installCommands?: string[];
@@ -89,18 +88,23 @@ export interface TokenInjectableDockerBuilderProps {
89
88
  * // ... other properties ...
90
89
  * });
91
90
  * ```
92
- * *In this example, the builder fetches a configuration file from a private API before starting the Docker build. This config file will be available in the same directory as your Dockerfile during CDK deployment.*
93
- *
94
91
  * @default - No additional pre-build commands.
95
92
  */
96
93
  readonly preBuildCommands?: string[];
97
94
  }
98
95
  /**
99
- * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
96
+ * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources,
97
+ * retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.
100
98
  */
101
99
  export declare class TokenInjectableDockerBuilder extends Construct {
102
100
  private readonly ecrRepository;
101
+ /**
102
+ * An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.
103
+ */
103
104
  readonly containerImage: ContainerImage;
105
+ /**
106
+ * A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.
107
+ */
104
108
  readonly dockerImageCode: DockerImageCode;
105
109
  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
106
110
  }
package/lib/index.js CHANGED
@@ -16,17 +16,20 @@ const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
16
16
  const custom_resources_1 = require("aws-cdk-lib/custom-resources");
17
17
  const constructs_1 = require("constructs");
18
18
  /**
19
- * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
19
+ * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources,
20
+ * retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.
20
21
  */
21
22
  class TokenInjectableDockerBuilder extends constructs_1.Construct {
22
23
  constructor(scope, id, props) {
23
24
  super(scope, id);
24
25
  const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, } = props;
25
- // Define a KMS key for ECR encryption
26
+ // Generate a unique tag for this build.
27
+ const imageTag = crypto.randomUUID();
28
+ // KMS key for ECR encryption
26
29
  const encryptionKey = new aws_kms_1.Key(this, 'EcrEncryptionKey', {
27
30
  enableKeyRotation: true,
28
31
  });
29
- // Create an ECR repository with lifecycle rules, encryption, and image scanning enabled
32
+ // ECR repository
30
33
  this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository', {
31
34
  lifecycleRules: [
32
35
  {
@@ -37,31 +40,74 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
37
40
  },
38
41
  ],
39
42
  encryption: aws_ecr_1.RepositoryEncryption.KMS,
40
- encryptionKey: encryptionKey,
43
+ encryptionKey,
41
44
  imageScanOnPush: true,
42
45
  });
43
- // Package the source code as an asset
46
+ // Package source code
44
47
  const sourceAsset = new aws_s3_assets_1.Asset(this, 'SourceAsset', {
45
48
  path: sourcePath,
46
49
  });
47
- // Transform buildArgs into a string of --build-arg KEY=VALUE
50
+ // Build args
48
51
  const buildArgsString = buildArgs
49
52
  ? Object.entries(buildArgs)
50
- .map(([key, value]) => `--build-arg ${key}=${value}`)
53
+ .map(([k, v]) => `--build-arg ${k}=${v}`)
51
54
  .join(' ')
52
55
  : '';
53
- // Conditional Dockerhub login commands
56
+ // Docker Hub login commands
54
57
  const dockerLoginCommands = dockerLoginSecretArn
55
58
  ? [
56
59
  'echo "Retrieving Docker credentials from Secrets Manager..."',
60
+ 'apt-get update -y && apt-get install -y jq',
57
61
  `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,
58
62
  `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,
59
- 'echo "Logging in to Docker..."',
63
+ 'echo "Logging in to Docker Hub..."',
64
+ // Use non-stdin login to avoid TTY error:
60
65
  'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',
61
66
  ]
62
- : ['echo "No Docker credentials provided. Skipping login step."'];
63
- // Create a CodeBuild project
64
- const codeBuildProject = new aws_codebuild_1.Project(this, 'UICodeBuildProject', {
67
+ : ['echo "No Docker credentials provided. Skipping Docker Hub login."'];
68
+ // BuildSpec
69
+ const buildSpecObj = {
70
+ version: '0.2',
71
+ phases: {
72
+ install: {
73
+ commands: [
74
+ 'echo "Beginning install phase..."',
75
+ ...(installCommands || []),
76
+ ],
77
+ },
78
+ pre_build: {
79
+ commands: [
80
+ ...(preBuildCommands || []),
81
+ ...dockerLoginCommands,
82
+ 'echo "Retrieving AWS Account ID..."',
83
+ 'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
84
+ 'echo "Logging in to Amazon ECR..."',
85
+ 'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',
86
+ ],
87
+ },
88
+ build: {
89
+ commands: [
90
+ 'echo "Build phase: Building the Docker image..."',
91
+ `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,
92
+ ],
93
+ },
94
+ post_build: {
95
+ commands: [
96
+ `echo "Post-build phase: Pushing the Docker image with tag ${imageTag}..."`,
97
+ `docker push $ECR_REPO_URI:${imageTag}`,
98
+ `export IMAGE_DIGEST=$(docker inspect --format='{{index .RepoDigests 0}}' $ECR_REPO_URI:${imageTag})`,
99
+ 'echo "Image digest: $IMAGE_DIGEST"',
100
+ 'echo "{ \\"ImageDigest\\": \\"$IMAGE_DIGEST\\" }" > imageDetail.json',
101
+ ],
102
+ },
103
+ },
104
+ artifacts: {
105
+ files: ['imageDetail.json'],
106
+ name: 'imageDetail',
107
+ },
108
+ };
109
+ // CodeBuild project
110
+ const codeBuildProject = new aws_codebuild_1.Project(this, 'CodeBuildProject', {
65
111
  source: aws_codebuild_1.Source.s3({
66
112
  bucket: sourceAsset.bucket,
67
113
  path: sourceAsset.s3ObjectKey,
@@ -72,51 +118,15 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
72
118
  },
73
119
  environmentVariables: {
74
120
  ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },
75
- BUILD_ARGS: { value: buildArgsString },
76
- // Include build arguments as environment variables if needed
77
- ...(buildArgs &&
78
- Object.fromEntries(Object.entries(buildArgs).map(([key, value]) => [key, { value }]))),
79
121
  },
80
122
  vpc,
81
123
  securityGroups,
82
124
  subnetSelection,
83
- buildSpec: aws_codebuild_1.BuildSpec.fromObject({
84
- version: '0.2',
85
- phases: {
86
- install: {
87
- commands: [
88
- 'echo "Beginning install phase..."',
89
- ...(installCommands || []),
90
- ],
91
- },
92
- pre_build: {
93
- commands: [
94
- ...(preBuildCommands || []),
95
- ...dockerLoginCommands,
96
- 'echo "Retrieving AWS Account ID..."',
97
- 'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
98
- 'echo "Logging in to Amazon ECR..."',
99
- 'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',
100
- ],
101
- },
102
- build: {
103
- commands: [
104
- 'echo Build phase: Building the Docker image...',
105
- 'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',
106
- ],
107
- },
108
- post_build: {
109
- commands: [
110
- 'echo Post-build phase: Pushing the Docker image...',
111
- 'docker push $ECR_REPO_URI:latest',
112
- ],
113
- },
114
- },
115
- }),
125
+ buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
116
126
  });
117
- // Grant permissions to CodeBuild
127
+ // Permissions
118
128
  this.ecrRepository.grantPullPush(codeBuildProject);
119
- codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
129
+ codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
120
130
  actions: [
121
131
  'ecr:GetAuthorizationToken',
122
132
  'ecr:GetDownloadUrlForLayer',
@@ -125,14 +135,13 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
125
135
  resources: [this.ecrRepository.repositoryArn],
126
136
  }));
127
137
  if (dockerLoginSecretArn) {
128
- codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
138
+ codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
129
139
  actions: ['secretsmanager:GetSecretValue'],
130
140
  resources: [dockerLoginSecretArn],
131
141
  }));
132
142
  }
133
- // Grant CodeBuild access to the KMS key
134
143
  encryptionKey.grantEncryptDecrypt(codeBuildProject.role);
135
- // Create Lambda functions for onEvent and isComplete handlers
144
+ // onEvent handler
136
145
  const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
137
146
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
138
147
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
@@ -141,8 +150,9 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
141
150
  });
142
151
  onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
143
152
  actions: ['codebuild:StartBuild'],
144
- resources: [codeBuildProject.projectArn], // Restrict to specific project
153
+ resources: [codeBuildProject.projectArn],
145
154
  }));
155
+ // isComplete handler
146
156
  const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
147
157
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
148
158
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
@@ -156,34 +166,42 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
156
166
  'logs:GetLogEvents',
157
167
  'logs:DescribeLogStreams',
158
168
  'logs:DescribeLogGroups',
169
+ 's3:GetObject',
170
+ 's3:GetBucketLocation',
159
171
  ],
160
172
  resources: ['*'],
161
173
  }));
162
- // Grant Lambda functions access to KMS key and ECR
163
174
  encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
164
175
  encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
165
176
  this.ecrRepository.grantPullPush(onEventHandlerFunction);
166
177
  this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
167
- // Create a custom resource provider
178
+ // Provider
168
179
  const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
169
180
  onEventHandler: onEventHandlerFunction,
170
181
  isCompleteHandler: isCompleteHandlerFunction,
171
182
  queryInterval: aws_cdk_lib_1.Duration.seconds(30),
172
183
  });
173
- // Define the custom resource
184
+ // Custom resource
174
185
  const buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
175
186
  serviceToken: provider.serviceToken,
176
187
  properties: {
177
188
  ProjectName: codeBuildProject.projectName,
189
+ ImageTag: imageTag,
178
190
  Trigger: crypto.randomUUID(),
179
191
  },
180
192
  });
181
193
  buildTriggerResource.node.addDependency(codeBuildProject);
182
- this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository);
183
- this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository);
194
+ // Grab the SHA from the custom resource's response (set in isComplete handler)
195
+ const imageDigest = buildTriggerResource.getAttString('ImageDigest');
196
+ // ECS-compatible from ECR by digest
197
+ this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository, imageDigest);
198
+ // Lambda-compatible from ECR by digest
199
+ this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository, {
200
+ tagOrDigest: imageDigest,
201
+ });
184
202
  }
185
203
  }
186
204
  exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
187
205
  _a = JSII_RTTI_SYMBOL_1;
188
- TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.2.6" };
189
- //# 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 path from 'path';\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 { 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   * 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, the construct will skip Docker login during the build process.\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   * @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 should define the network access rules for the CodeBuild project.\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   * @default All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\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   *   // ... other properties ...\n   * });\n   * ```\n   * *This example demonstrates how to install the `curl` and `dnsutils` packages during the install phase using `apt-get`, the package manager for Ubuntu-based CodeBuild environments.*\n   *\n   * @default - No additional install commands.\n   */\n  readonly installCommands?: string[];\n\n\n  /**\n   * Custom commands to run during the pre_build phase.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\n   *   preBuildCommands: [\n   *     'echo \"Fetching configuration from private API...\"',\n   *     'curl -o config.json https://api.example.com/config',\n   *   ],\n   *   // ... other properties ...\n   * });\n   * ```\n   * *In this example, the builder fetches a configuration file from a private API before starting the Docker build. This config file will be available in the same directory as your Dockerfile during CDK deployment.*\n   *\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  private readonly ecrRepository: Repository;\n  public readonly containerImage: ContainerImage;\n  public readonly dockerImageCode: DockerImageCode;\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    } = props;\n\n    // Define a KMS key for ECR encryption\n    const encryptionKey = new Key(this, 'EcrEncryptionKey', {\n      enableKeyRotation: true,\n    });\n\n    // Create an ECR repository with lifecycle rules, encryption, and image scanning enabled\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: RepositoryEncryption.KMS,\n      encryptionKey: encryptionKey,\n      imageScanOnPush: true,\n    });\n\n    // Package the source code as an asset\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\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    // Conditional Dockerhub login commands\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials from Secrets Manager...\"',\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...\"',\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials provided. Skipping login step.\"'];\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,\n      },\n      environmentVariables: {\n        ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n        BUILD_ARGS: { value: buildArgsString },\n        // Include build arguments as environment variables if needed\n        ...(buildArgs &&\n          Object.fromEntries(\n            Object.entries(buildArgs).map(([key, value]) => [key, { value }]),\n          )),\n      },\n      vpc,\n      securityGroups,\n      subnetSelection,\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          install: {\n            commands: [\n              'echo \"Beginning install phase...\"',\n              ...(installCommands || []),\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 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 CodeBuild\n    this.ecrRepository.grantPullPush(codeBuildProject);\n\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:BatchCheckLayerAvailability',\n        ],\n        resources: [this.ecrRepository.repositoryArn],\n      }),\n    );\n\n    if (dockerLoginSecretArn) {\n      codeBuildProject.role!.addToPrincipalPolicy(\n        new PolicyStatement({\n          actions: ['secretsmanager:GetSecretValue'],\n          resources: [dockerLoginSecretArn],\n        }),\n      );\n    }\n\n    // Grant CodeBuild access to the KMS key\n    encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n\n    // Create Lambda functions for onEvent and isComplete handlers\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\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    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      handler: 'isComplete.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    // Grant Lambda functions access to KMS key and ECR\n    encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n    encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n    this.ecrRepository.grantPullPush(onEventHandlerFunction);\n    this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\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"]}
206
+ TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.3.0" };
207
+ //# 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 path from 'path';\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 { 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   * 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   * 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   * @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 should define the network access rules for the CodeBuild project.\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   * @default All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\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   *   // ... other properties ...\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.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\n   *   preBuildCommands: [\n   *     'echo \"Fetching configuration from private API...\"',\n   *     'curl -o config.json https://api.example.com/config',\n   *   ],\n   *   // ... other properties ...\n   * });\n   * ```\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources,\n * retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  private readonly ecrRepository: Repository;\n\n  /**\n   * An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.\n   */\n  public readonly containerImage: ContainerImage;\n\n  /**\n   * A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.\n   */\n  public readonly dockerImageCode: DockerImageCode;\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    } = props;\n\n    // Generate a unique tag for this build.\n    const imageTag = crypto.randomUUID();\n\n    // KMS key for ECR encryption\n    const encryptionKey = new Key(this, 'EcrEncryptionKey', {\n      enableKeyRotation: true,\n    });\n\n    // ECR repository\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: RepositoryEncryption.KMS,\n      encryptionKey,\n      imageScanOnPush: true,\n    });\n\n    // Package source code\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n    });\n\n    // Build args\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([k, v]) => `--build-arg ${k}=${v}`)\n        .join(' ')\n      : '';\n\n    // Docker Hub login commands\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials from Secrets Manager...\"',\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        // Use non-stdin login to avoid TTY error:\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials provided. Skipping Docker Hub login.\"'];\n\n    // BuildSpec\n    const buildSpecObj = {\n      version: '0.2',\n      phases: {\n        install: {\n          commands: [\n            'echo \"Beginning install phase...\"',\n            ...(installCommands || []),\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 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 ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,\n          ],\n        },\n        post_build: {\n          commands: [\n            `echo \"Post-build phase: Pushing the Docker image with tag ${imageTag}...\"`,\n            `docker push $ECR_REPO_URI:${imageTag}`,\n            `export IMAGE_DIGEST=$(docker inspect --format='{{index .RepoDigests 0}}' $ECR_REPO_URI:${imageTag})`,\n            'echo \"Image digest: $IMAGE_DIGEST\"',\n            'echo \"{ \\\\\"ImageDigest\\\\\": \\\\\"$IMAGE_DIGEST\\\\\" }\" > imageDetail.json',\n          ],\n        },\n      },\n      artifacts: {\n        files: ['imageDetail.json'],\n        name: 'imageDetail',\n      },\n    };\n\n    // 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      vpc,\n      securityGroups,\n      subnetSelection,\n      buildSpec: BuildSpec.fromObject(buildSpecObj),\n    });\n\n    // Permissions\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.role?.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:BatchCheckLayerAvailability',\n        ],\n        resources: [this.ecrRepository.repositoryArn],\n      }),\n    );\n\n    if (dockerLoginSecretArn) {\n      codeBuildProject.role?.addToPrincipalPolicy(\n        new PolicyStatement({\n          actions: ['secretsmanager:GetSecretValue'],\n          resources: [dockerLoginSecretArn],\n        }),\n      );\n    }\n\n    encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n\n    // onEvent handler\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn],\n      }),\n    );\n\n    // isComplete handler\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      handler: 'isComplete.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          's3:GetObject',\n          's3:GetBucketLocation',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n    encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n    this.ecrRepository.grantPullPush(onEventHandlerFunction);\n    this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\n\n    // Provider\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.seconds(30),\n    });\n\n    // Custom resource\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        ImageTag: imageTag,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Grab the SHA from the custom resource's response (set in isComplete handler)\n    const imageDigest = buildTriggerResource.getAttString('ImageDigest');\n\n    // ECS-compatible from ECR by digest\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository, imageDigest);\n\n    // Lambda-compatible from ECR by digest\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository, {\n      tagOrDigest: imageDigest,\n    });\n  }\n}\n"]}
package/package.json CHANGED
@@ -40,7 +40,7 @@
40
40
  "devDependencies": {
41
41
  "@stylistic/eslint-plugin": "^2",
42
42
  "@types/jest": "^29.5.14",
43
- "@types/node": "^22.10.1",
43
+ "@types/node": "^22.10.2",
44
44
  "@typescript-eslint/eslint-plugin": "^8",
45
45
  "@typescript-eslint/parser": "^8",
46
46
  "aws-cdk-lib": "2.166.0",
@@ -52,11 +52,11 @@
52
52
  "jest": "^29.7.0",
53
53
  "jest-junit": "^15",
54
54
  "jsii": "~5.5.0",
55
- "jsii-diff": "^1.105.0",
55
+ "jsii-diff": "^1.106.0",
56
56
  "jsii-docgen": "^10.5.0",
57
- "jsii-pacmak": "^1.105.0",
57
+ "jsii-pacmak": "^1.106.0",
58
58
  "jsii-rosetta": "~5.5.0",
59
- "projen": "^0.90.6",
59
+ "projen": "^0.91.1",
60
60
  "ts-jest": "^29.2.5",
61
61
  "ts-node": "^10.9.2",
62
62
  "typescript": "^5.7.2"
@@ -96,7 +96,7 @@
96
96
  "publishConfig": {
97
97
  "access": "public"
98
98
  },
99
- "version": "1.2.6",
99
+ "version": "1.3.0",
100
100
  "jest": {
101
101
  "coverageProvider": "v8",
102
102
  "testMatch": [