token-injectable-docker-builder 1.3.0 → 1.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.jsii CHANGED
@@ -7,7 +7,7 @@
7
7
  ]
8
8
  },
9
9
  "dependencies": {
10
- "aws-cdk-lib": "^2.166.0",
10
+ "aws-cdk-lib": "^2.173.2",
11
11
  "constructs": "^10.0.5"
12
12
  },
13
13
  "dependencyClosure": {
@@ -1019,6 +1019,19 @@
1019
1019
  }
1020
1020
  }
1021
1021
  },
1022
+ "aws-cdk-lib.aws_connectcampaignsv2": {
1023
+ "targets": {
1024
+ "dotnet": {
1025
+ "package": "Amazon.CDK.AWS.ConnectCampaignsV2"
1026
+ },
1027
+ "java": {
1028
+ "package": "software.amazon.awscdk.services.connectcampaignsv2"
1029
+ },
1030
+ "python": {
1031
+ "module": "aws_cdk.aws_connectcampaignsv2"
1032
+ }
1033
+ }
1034
+ },
1022
1035
  "aws-cdk-lib.aws_controltower": {
1023
1036
  "targets": {
1024
1037
  "dotnet": {
@@ -1838,6 +1851,19 @@
1838
1851
  }
1839
1852
  }
1840
1853
  },
1854
+ "aws-cdk-lib.aws_invoicing": {
1855
+ "targets": {
1856
+ "dotnet": {
1857
+ "package": "Amazon.CDK.AWS.Invoicing"
1858
+ },
1859
+ "java": {
1860
+ "package": "software.amazon.awscdk.services.invoicing"
1861
+ },
1862
+ "python": {
1863
+ "module": "aws_cdk.aws_invoicing"
1864
+ }
1865
+ }
1866
+ },
1841
1867
  "aws-cdk-lib.aws_iot": {
1842
1868
  "targets": {
1843
1869
  "dotnet": {
@@ -2826,6 +2852,19 @@
2826
2852
  }
2827
2853
  }
2828
2854
  },
2855
+ "aws-cdk-lib.aws_rbin": {
2856
+ "targets": {
2857
+ "dotnet": {
2858
+ "package": "Amazon.CDK.AWS.Rbin"
2859
+ },
2860
+ "java": {
2861
+ "package": "software.amazon.awscdk.services.rbin"
2862
+ },
2863
+ "python": {
2864
+ "module": "aws_cdk.aws_rbin"
2865
+ }
2866
+ }
2867
+ },
2829
2868
  "aws-cdk-lib.aws_rds": {
2830
2869
  "targets": {
2831
2870
  "dotnet": {
@@ -3920,31 +3959,41 @@
3920
3959
  "base": "constructs.Construct",
3921
3960
  "docs": {
3922
3961
  "stability": "stable",
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."
3962
+ "summary": "A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources, **then** retrieve the final image tag so that ECS/Lambda references use the exact digest."
3924
3963
  },
3925
3964
  "fqn": "token-injectable-docker-builder.TokenInjectableDockerBuilder",
3926
3965
  "initializer": {
3927
3966
  "docs": {
3928
- "stability": "stable"
3967
+ "stability": "stable",
3968
+ "summary": "Creates a new `TokenInjectableDockerBuilder`."
3929
3969
  },
3930
3970
  "locationInModule": {
3931
3971
  "filename": "src/index.ts",
3932
- "line": 128
3972
+ "line": 138
3933
3973
  },
3934
3974
  "parameters": [
3935
3975
  {
3976
+ "docs": {
3977
+ "summary": "The scope in which to define this construct."
3978
+ },
3936
3979
  "name": "scope",
3937
3980
  "type": {
3938
3981
  "fqn": "constructs.Construct"
3939
3982
  }
3940
3983
  },
3941
3984
  {
3985
+ "docs": {
3986
+ "summary": "The scoped construct ID."
3987
+ },
3942
3988
  "name": "id",
3943
3989
  "type": {
3944
3990
  "primitive": "string"
3945
3991
  }
3946
3992
  },
3947
3993
  {
3994
+ "docs": {
3995
+ "summary": "Configuration for building and pushing the Docker image."
3996
+ },
3948
3997
  "name": "props",
3949
3998
  "type": {
3950
3999
  "fqn": "token-injectable-docker-builder.TokenInjectableDockerBuilderProps"
@@ -3955,19 +4004,19 @@
3955
4004
  "kind": "class",
3956
4005
  "locationInModule": {
3957
4006
  "filename": "src/index.ts",
3958
- "line": 115
4007
+ "line": 113
3959
4008
  },
3960
4009
  "name": "TokenInjectableDockerBuilder",
3961
4010
  "properties": [
3962
4011
  {
3963
4012
  "docs": {
3964
4013
  "stability": "stable",
3965
- "summary": "An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image."
4014
+ "summary": "An ECS-compatible container image referencing the tag of the built Docker image."
3966
4015
  },
3967
4016
  "immutable": true,
3968
4017
  "locationInModule": {
3969
4018
  "filename": "src/index.ts",
3970
- "line": 121
4019
+ "line": 123
3971
4020
  },
3972
4021
  "name": "containerImage",
3973
4022
  "type": {
@@ -3977,12 +4026,12 @@
3977
4026
  {
3978
4027
  "docs": {
3979
4028
  "stability": "stable",
3980
- "summary": "A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image."
4029
+ "summary": "A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image."
3981
4030
  },
3982
4031
  "immutable": true,
3983
4032
  "locationInModule": {
3984
4033
  "filename": "src/index.ts",
3985
- "line": 126
4034
+ "line": 129
3986
4035
  },
3987
4036
  "name": "dockerImageCode",
3988
4037
  "type": {
@@ -4003,7 +4052,7 @@
4003
4052
  "kind": "interface",
4004
4053
  "locationInModule": {
4005
4054
  "filename": "src/index.ts",
4006
- "line": 18
4055
+ "line": 19
4007
4056
  },
4008
4057
  "name": "TokenInjectableDockerBuilderProps",
4009
4058
  "properties": [
@@ -4016,7 +4065,7 @@
4016
4065
  "immutable": true,
4017
4066
  "locationInModule": {
4018
4067
  "filename": "src/index.ts",
4019
- "line": 22
4068
+ "line": 23
4020
4069
  },
4021
4070
  "name": "path",
4022
4071
  "type": {
@@ -4027,14 +4076,14 @@
4027
4076
  "abstract": true,
4028
4077
  "docs": {
4029
4078
  "example": "{\n TOKEN: 'my-secret-token',\n ENV: 'production'\n}",
4030
- "remarks": "These are transformed into `--build-arg` flags.",
4079
+ "remarks": "These are transformed into `--build-arg KEY=VALUE` flags.",
4031
4080
  "stability": "stable",
4032
4081
  "summary": "Build arguments to pass to the Docker build process."
4033
4082
  },
4034
4083
  "immutable": true,
4035
4084
  "locationInModule": {
4036
4085
  "filename": "src/index.ts",
4037
- "line": 33
4086
+ "line": 34
4038
4087
  },
4039
4088
  "name": "buildArgs",
4040
4089
  "optional": true,
@@ -4051,14 +4100,14 @@
4051
4100
  "abstract": true,
4052
4101
  "docs": {
4053
4102
  "example": "'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'",
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.",
4103
+ "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.\n\n**Note**: The secret must be in the same region as the stack.",
4055
4104
  "stability": "stable",
4056
4105
  "summary": "The ARN of the AWS Secrets Manager secret containing Docker login credentials."
4057
4106
  },
4058
4107
  "immutable": true,
4059
4108
  "locationInModule": {
4060
4109
  "filename": "src/index.ts",
4061
- "line": 49
4110
+ "line": 51
4062
4111
  },
4063
4112
  "name": "dockerLoginSecretArn",
4064
4113
  "optional": true,
@@ -4070,14 +4119,14 @@
4070
4119
  "abstract": true,
4071
4120
  "docs": {
4072
4121
  "default": "- No additional install commands.",
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```",
4122
+ "remarks": "**Example**:\n```ts\ninstallCommands: [\n 'echo \"Updating package lists...\"',\n 'apt-get update -y',\n 'echo \"Installing required packages...\"',\n 'apt-get install -y curl dnsutils',\n],\n```",
4074
4123
  "stability": "stable",
4075
- "summary": "Custom commands to run during the install phase."
4124
+ "summary": "Custom commands to run during the install phase of CodeBuild."
4076
4125
  },
4077
4126
  "immutable": true,
4078
4127
  "locationInModule": {
4079
4128
  "filename": "src/index.ts",
4080
- "line": 90
4129
+ "line": 91
4081
4130
  },
4082
4131
  "name": "installCommands",
4083
4132
  "optional": true,
@@ -4094,14 +4143,14 @@
4094
4143
  "abstract": true,
4095
4144
  "docs": {
4096
4145
  "default": "- No additional pre-build commands.",
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```",
4146
+ "remarks": "**Example**:\n```ts\npreBuildCommands: [\n 'echo \"Fetching configuration from private API...\"',\n 'curl -o config.json https://api.example.com/config',\n],\n```",
4098
4147
  "stability": "stable",
4099
- "summary": "Custom commands to run during the pre_build phase."
4148
+ "summary": "Custom commands to run during the pre_build phase of CodeBuild."
4100
4149
  },
4101
4150
  "immutable": true,
4102
4151
  "locationInModule": {
4103
4152
  "filename": "src/index.ts",
4104
- "line": 108
4153
+ "line": 105
4105
4154
  },
4106
4155
  "name": "preBuildCommands",
4107
4156
  "optional": true,
@@ -4117,15 +4166,15 @@
4117
4166
  {
4118
4167
  "abstract": true,
4119
4168
  "docs": {
4120
- "default": "No security groups are attached.",
4121
- "remarks": "These should define the network access rules for the CodeBuild project.",
4169
+ "default": "- No security groups are attached.",
4170
+ "remarks": "These define the network access rules for the CodeBuild project.",
4122
4171
  "stability": "stable",
4123
4172
  "summary": "The security groups to attach to the CodeBuild project."
4124
4173
  },
4125
4174
  "immutable": true,
4126
4175
  "locationInModule": {
4127
4176
  "filename": "src/index.ts",
4128
- "line": 63
4177
+ "line": 67
4129
4178
  },
4130
4179
  "name": "securityGroups",
4131
4180
  "optional": true,
@@ -4141,7 +4190,7 @@
4141
4190
  {
4142
4191
  "abstract": true,
4143
4192
  "docs": {
4144
- "default": "All subnets in the VPC are used.",
4193
+ "default": "- All subnets in the VPC are used.",
4145
4194
  "remarks": "Allows the user to select private, public, or isolated subnets.",
4146
4195
  "stability": "stable",
4147
4196
  "summary": "The subnet selection to specify which subnets to use within the VPC."
@@ -4149,7 +4198,7 @@
4149
4198
  "immutable": true,
4150
4199
  "locationInModule": {
4151
4200
  "filename": "src/index.ts",
4152
- "line": 70
4201
+ "line": 75
4153
4202
  },
4154
4203
  "name": "subnetSelection",
4155
4204
  "optional": true,
@@ -4160,7 +4209,7 @@
4160
4209
  {
4161
4210
  "abstract": true,
4162
4211
  "docs": {
4163
- "default": "No VPC is attached, and the CodeBuild project will use public internet.",
4212
+ "default": "- No VPC is attached, and the CodeBuild project will use public internet.",
4164
4213
  "remarks": "If provided, the CodeBuild project will be launched within the specified VPC.",
4165
4214
  "stability": "stable",
4166
4215
  "summary": "The VPC in which the CodeBuild project will be deployed."
@@ -4168,7 +4217,7 @@
4168
4217
  "immutable": true,
4169
4218
  "locationInModule": {
4170
4219
  "filename": "src/index.ts",
4171
- "line": 56
4220
+ "line": 59
4172
4221
  },
4173
4222
  "name": "vpc",
4174
4223
  "optional": true,
@@ -4180,6 +4229,6 @@
4180
4229
  "symbolId": "src/index:TokenInjectableDockerBuilderProps"
4181
4230
  }
4182
4231
  },
4183
- "version": "1.3.0",
4184
- "fingerprint": "ZL25JVTzJLNHeDV7j15vMt4HHx0Ae4Mvgy29nJsio6w="
4232
+ "version": "1.3.1",
4233
+ "fingerprint": "Q9DfzcrA3DzkFuZ3QghI1ZV3uoLI/0611iFI3ZPoifc="
4185
4234
  }
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, retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.
7
+ A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources, **then** retrieve the final image tag so that ECS/Lambda references use the exact digest.
8
8
 
9
9
  #### Initializers <a name="Initializers" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer"></a>
10
10
 
@@ -16,9 +16,9 @@ new TokenInjectableDockerBuilder(scope: Construct, id: string, props: TokenInjec
16
16
 
17
17
  | **Name** | **Type** | **Description** |
18
18
  | --- | --- | --- |
19
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.scope">scope</a></code> | <code>constructs.Construct</code> | *No description.* |
20
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.id">id</a></code> | <code>string</code> | *No description.* |
21
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.props">props</a></code> | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps">TokenInjectableDockerBuilderProps</a></code> | *No description.* |
19
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.scope">scope</a></code> | <code>constructs.Construct</code> | The scope in which to define this construct. |
20
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.id">id</a></code> | <code>string</code> | The scoped construct ID. |
21
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.props">props</a></code> | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps">TokenInjectableDockerBuilderProps</a></code> | Configuration for building and pushing the Docker image. |
22
22
 
23
23
  ---
24
24
 
@@ -26,18 +26,24 @@ new TokenInjectableDockerBuilder(scope: Construct, id: string, props: TokenInjec
26
26
 
27
27
  - *Type:* constructs.Construct
28
28
 
29
+ The scope in which to define this construct.
30
+
29
31
  ---
30
32
 
31
33
  ##### `id`<sup>Required</sup> <a name="id" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.id"></a>
32
34
 
33
35
  - *Type:* string
34
36
 
37
+ The scoped construct ID.
38
+
35
39
  ---
36
40
 
37
41
  ##### `props`<sup>Required</sup> <a name="props" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.props"></a>
38
42
 
39
43
  - *Type:* <a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps">TokenInjectableDockerBuilderProps</a>
40
44
 
45
+ Configuration for building and pushing the Docker image.
46
+
41
47
  ---
42
48
 
43
49
  #### Methods <a name="Methods" id="Methods"></a>
@@ -87,8 +93,8 @@ Any object.
87
93
  | **Name** | **Type** | **Description** |
88
94
  | --- | --- | --- |
89
95
  | <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> | 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. |
96
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.containerImage">containerImage</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerImage</code> | An ECS-compatible container image referencing the tag of the built Docker image. |
97
+ | <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 the tag of the built Docker image. |
92
98
 
93
99
  ---
94
100
 
@@ -112,7 +118,7 @@ public readonly containerImage: ContainerImage;
112
118
 
113
119
  - *Type:* aws-cdk-lib.aws_ecs.ContainerImage
114
120
 
115
- An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.
121
+ An ECS-compatible container image referencing the tag of the built Docker image.
116
122
 
117
123
  ---
118
124
 
@@ -124,7 +130,7 @@ public readonly dockerImageCode: DockerImageCode;
124
130
 
125
131
  - *Type:* aws-cdk-lib.aws_lambda.DockerImageCode
126
132
 
127
- A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.
133
+ A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image.
128
134
 
129
135
  ---
130
136
 
@@ -150,8 +156,8 @@ const tokenInjectableDockerBuilderProps: TokenInjectableDockerBuilderProps = { .
150
156
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.path">path</a></code> | <code>string</code> | The path to the directory containing the Dockerfile or source code. |
151
157
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.buildArgs">buildArgs</a></code> | <code>{[ key: string ]: string}</code> | Build arguments to pass to the Docker build process. |
152
158
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.dockerLoginSecretArn">dockerLoginSecretArn</a></code> | <code>string</code> | The ARN of the AWS Secrets Manager secret containing Docker login credentials. |
153
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.installCommands">installCommands</a></code> | <code>string[]</code> | Custom commands to run during the install phase. |
154
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.preBuildCommands">preBuildCommands</a></code> | <code>string[]</code> | Custom commands to run during the pre_build phase. |
159
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.installCommands">installCommands</a></code> | <code>string[]</code> | Custom commands to run during the install phase of CodeBuild. |
160
+ | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.preBuildCommands">preBuildCommands</a></code> | <code>string[]</code> | Custom commands to run during the pre_build phase of CodeBuild. |
155
161
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.securityGroups">securityGroups</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup[]</code> | The security groups to attach to the CodeBuild project. |
156
162
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.subnetSelection">subnetSelection</a></code> | <code>aws-cdk-lib.aws_ec2.SubnetSelection</code> | The subnet selection to specify which subnets to use within the VPC. |
157
163
  | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | The VPC in which the CodeBuild project will be deployed. |
@@ -180,7 +186,7 @@ public readonly buildArgs: {[ key: string ]: string};
180
186
 
181
187
  Build arguments to pass to the Docker build process.
182
188
 
183
- These are transformed into `--build-arg` flags.
189
+ These are transformed into `--build-arg KEY=VALUE` flags.
184
190
 
185
191
  ---
186
192
 
@@ -212,7 +218,8 @@ This secret should store a JSON object with the following structure:
212
218
  }
213
219
  ```
214
220
  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.
221
+
222
+ **Note**: The secret must be in the same region as the stack.
216
223
 
217
224
  ---
218
225
 
@@ -232,20 +239,16 @@ public readonly installCommands: string[];
232
239
  - *Type:* string[]
233
240
  - *Default:* No additional install commands.
234
241
 
235
- Custom commands to run during the install phase.
242
+ Custom commands to run during the install phase of CodeBuild.
236
243
 
237
- **Example Usage:**
238
- ```typescript
239
- new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
240
- path: path.resolve(__dirname, '../app'),
241
- installCommands: [
242
- 'echo "Updating package lists..."',
243
- 'apt-get update -y',
244
- 'echo "Installing required packages..."',
245
- 'apt-get install -y curl dnsutils',
246
- ],
247
- // ... other properties ...
248
- });
244
+ **Example**:
245
+ ```ts
246
+ installCommands: [
247
+ 'echo "Updating package lists..."',
248
+ 'apt-get update -y',
249
+ 'echo "Installing required packages..."',
250
+ 'apt-get install -y curl dnsutils',
251
+ ],
249
252
  ```
250
253
 
251
254
  ---
@@ -259,18 +262,14 @@ public readonly preBuildCommands: string[];
259
262
  - *Type:* string[]
260
263
  - *Default:* No additional pre-build commands.
261
264
 
262
- Custom commands to run during the pre_build phase.
265
+ Custom commands to run during the pre_build phase of CodeBuild.
263
266
 
264
- **Example Usage:**
265
- ```typescript
266
- new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
267
- path: path.resolve(__dirname, '../app'),
268
- preBuildCommands: [
269
- 'echo "Fetching configuration from private API..."',
270
- 'curl -o config.json https://api.example.com/config',
271
- ],
272
- // ... other properties ...
273
- });
267
+ **Example**:
268
+ ```ts
269
+ preBuildCommands: [
270
+ 'echo "Fetching configuration from private API..."',
271
+ 'curl -o config.json https://api.example.com/config',
272
+ ],
274
273
  ```
275
274
 
276
275
  ---
@@ -286,7 +285,7 @@ public readonly securityGroups: ISecurityGroup[];
286
285
 
287
286
  The security groups to attach to the CodeBuild project.
288
287
 
289
- These should define the network access rules for the CodeBuild project.
288
+ These define the network access rules for the CodeBuild project.
290
289
 
291
290
  ---
292
291
 
@@ -1,97 +1,120 @@
1
- const { CodeBuildClient, ListBuildsForProjectCommand, BatchGetBuildsCommand } = require('@aws-sdk/client-codebuild');
2
- const { CloudWatchLogsClient, GetLogEventsCommand } = require('@aws-sdk/client-cloudwatch-logs');
3
-
4
- exports.handler = async (event, context) => {
5
- console.log('isCompleteHandler Event:', JSON.stringify(event, null, 2));
6
-
7
- // Initialize AWS SDK v3 clients
8
- const codebuildClient = new CodeBuildClient({ region: process.env.AWS_REGION });
9
- const cloudwatchlogsClient = new CloudWatchLogsClient({ region: process.env.AWS_REGION });
1
+ const {
2
+ CodeBuildClient,
3
+ ListBuildsForProjectCommand,
4
+ BatchGetBuildsCommand,
5
+ } = require('@aws-sdk/client-codebuild');
6
+ const {
7
+ CloudWatchLogsClient,
8
+ GetLogEventsCommand,
9
+ } = require('@aws-sdk/client-cloudwatch-logs');
10
+
11
+ exports.handler = async (event) => {
12
+ console.log('--- isComplete Handler Invoked ---');
13
+ console.log('AWS_REGION:', process.env.AWS_REGION);
14
+ console.log('Event:', JSON.stringify(event, null, 2));
15
+
16
+ const region = process.env.AWS_REGION;
17
+ const codebuildClient = new CodeBuildClient({ region });
18
+ const logsClient = new CloudWatchLogsClient({ region });
10
19
 
11
20
  try {
12
- const projectName = event.ResourceProperties.ProjectName;
21
+ const projectName = event.ResourceProperties?.ProjectName;
22
+ console.log('ProjectName from ResourceProperties:', projectName);
13
23
 
14
24
  if (!projectName) {
15
- throw new Error('ProjectName is required in ResourceProperties');
25
+ throw new Error('Missing ProjectName in ResourceProperties');
16
26
  }
17
27
 
18
- console.log(`Checking status for CodeBuild project: ${projectName}`);
19
-
20
- // Retrieve the latest build for the given project
21
- const listBuildsCommand = new ListBuildsForProjectCommand({
22
- projectName: projectName,
23
- sortOrder: 'DESCENDING',
24
- maxResults: 1,
25
- });
26
-
27
- const listBuildsResp = await codebuildClient.send(listBuildsCommand);
28
- const buildIds = listBuildsResp.ids;
28
+ // Handle Delete requests gracefully
29
+ if (event.RequestType === 'Delete') {
30
+ console.log('Delete request detected. Marking resource as complete.');
31
+ return { IsComplete: true };
32
+ }
29
33
 
30
- if (!buildIds || buildIds.length === 0) {
34
+ // 1) Retrieve the latest build ID for this project
35
+ console.log('Querying CodeBuild for the most recent build...');
36
+ const listResp = await codebuildClient.send(
37
+ new ListBuildsForProjectCommand({
38
+ projectName,
39
+ sortOrder: 'DESCENDING',
40
+ maxResults: 1,
41
+ })
42
+ );
43
+ console.log('ListBuildsForProjectCommand response:', JSON.stringify(listResp, null, 2));
44
+
45
+ if (!listResp.ids || listResp.ids.length === 0) {
31
46
  throw new Error(`No builds found for project: ${projectName}`);
32
47
  }
33
48
 
34
- const buildId = buildIds[0];
35
- console.log(`Latest Build ID: ${buildId}`);
49
+ const buildId = listResp.ids[0];
50
+ console.log(`Identified latest Build ID: ${buildId}`);
36
51
 
37
- // Get build details
38
- const batchGetBuildsCommand = new BatchGetBuildsCommand({
39
- ids: [buildId],
40
- });
41
-
42
- const buildDetailsResp = await codebuildClient.send(batchGetBuildsCommand);
43
- const build = buildDetailsResp.builds[0];
52
+ // 2) Get details about that specific build
53
+ const batchResp = await codebuildClient.send(
54
+ new BatchGetBuildsCommand({ ids: [buildId] })
55
+ );
56
+ console.log('BatchGetBuildsCommand response:', JSON.stringify(batchResp, null, 2));
44
57
 
58
+ const build = batchResp.builds?.[0];
45
59
  if (!build) {
46
60
  throw new Error(`Build details not found for Build ID: ${buildId}`);
47
61
  }
48
62
 
49
63
  const buildStatus = build.buildStatus;
50
- console.log(`Build Status: ${buildStatus}`);
64
+ console.log(`The build status for ID ${buildId} is: ${buildStatus}`);
51
65
 
66
+ // Check for in-progress status
52
67
  if (buildStatus === 'IN_PROGRESS') {
53
- // Build is still in progress
54
- console.log('Build is still in progress.');
68
+ console.log('Build is still in progress. Requesting more time...');
55
69
  return { IsComplete: false };
56
- } else if (buildStatus === 'SUCCEEDED') {
57
- // Build succeeded
58
- console.log('Build succeeded.');
59
- return { IsComplete: true };
60
- } else if (['FAILED', 'FAULT', 'STOPPED', 'TIMED_OUT'].includes(buildStatus)) {
61
- // Build failed; retrieve last 5 log lines
62
- const logsInfo = build.logs;
63
- if (logsInfo && logsInfo.groupName && logsInfo.streamName) {
64
- console.log(`Retrieving logs from CloudWatch Logs Group: ${logsInfo.groupName}, Stream: ${logsInfo.streamName}`);
65
-
66
- const getLogEventsCommand = new GetLogEventsCommand({
67
- logGroupName: logsInfo.groupName,
68
- logStreamName: logsInfo.streamName,
69
- startFromHead: false, // Start from the end to get latest logs
70
- limit: 5,
71
- });
72
-
73
- const logEventsResp = await cloudwatchlogsClient.send(getLogEventsCommand);
74
- const logEvents = logEventsResp.events;
75
- const lastFiveMessages = logEvents.map((event) => event.message).reverse().join('\n');
70
+ }
76
71
 
77
- const errorMessage = `Build failed with status: ${buildStatus}\nLast 5 build logs:\n${lastFiveMessages}`;
78
- console.error(errorMessage);
72
+ // If build succeeded, retrieve the final artifact with the digest
73
+ if (buildStatus === 'SUCCEEDED') {
74
+ return {
75
+ IsComplete: true,
76
+ Data: {
77
+ ImageTag: process.env.IMAGE_TAG,
78
+ },
79
+ };
80
+ }
79
81
 
80
- // Throw an error to indicate failure to the CDK provider
81
- throw new Error(errorMessage);
82
+ // If the build is in a failed status, retrieve CloudWatch logs
83
+ if (['FAILED', 'FAULT', 'STOPPED', 'TIMED_OUT'].includes(buildStatus)) {
84
+ console.log(`Build ended with status: ${buildStatus}. Attempting to retrieve last log lines...`);
85
+ const logsInfo = build.logs;
86
+ console.log('Logs info:', JSON.stringify(logsInfo, null, 2));
87
+
88
+ if (logsInfo?.groupName && logsInfo?.streamName) {
89
+ console.log(`Retrieving up to 5 log events from CloudWatch Logs in group ${logsInfo.groupName} stream ${logsInfo.streamName}`);
90
+ const logResp = await logsClient.send(
91
+ new GetLogEventsCommand({
92
+ logGroupName: logsInfo.groupName,
93
+ logStreamName: logsInfo.streamName,
94
+ startFromHead: false,
95
+ limit: 5,
96
+ })
97
+ );
98
+ console.log('GetLogEventsCommand response:', JSON.stringify(logResp, null, 2));
99
+
100
+ const logEvents = logResp.events || [];
101
+ const lastFive = logEvents.map(e => e.message).reverse().join('\n');
102
+ console.error('Last 5 build log lines:\n', lastFive);
103
+
104
+ throw new Error(`Build failed with status ${buildStatus}. Last logs:\n${lastFive}`);
82
105
  } else {
83
- const errorMessage = `Build failed with status: ${buildStatus}, but logs are not available.`;
84
- console.error(errorMessage);
85
- throw new Error(errorMessage);
106
+ throw new Error(`Build failed with status: ${buildStatus}, but no logs found.`);
86
107
  }
87
- } else {
88
- const errorMessage = `Unknown build status: ${buildStatus}`;
89
- console.error(errorMessage);
90
- throw new Error(errorMessage);
91
108
  }
109
+
110
+ // If we reach here, it's an unexpected status
111
+ console.log(`Encountered unknown build status: ${buildStatus}`);
112
+ throw new Error(`Unknown build status: ${buildStatus}`);
113
+
92
114
  } catch (error) {
93
- console.error('Error in isCompleteHandler:', error);
94
- // Rethrow the error to inform the CDK provider of the failure
115
+ console.error('--- Caught an error in isComplete handler ---');
116
+ console.error('Error details:', error);
117
+ // re-throw for CloudFormation to see the error
95
118
  throw error;
96
119
  }
97
120
  };
package/lib/index.d.ts CHANGED
@@ -12,7 +12,7 @@ export interface TokenInjectableDockerBuilderProps {
12
12
  readonly path: string;
13
13
  /**
14
14
  * Build arguments to pass to the Docker build process.
15
- * These are transformed into `--build-arg` flags.
15
+ * These are transformed into `--build-arg KEY=VALUE` flags.
16
16
  * @example
17
17
  * {
18
18
  * TOKEN: 'my-secret-token',
@@ -32,7 +32,8 @@ export interface TokenInjectableDockerBuilderProps {
32
32
  * }
33
33
  * ```
34
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
+ * **Note**: The secret must be in the same region as the stack.
36
37
  *
37
38
  * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'
38
39
  */
@@ -40,71 +41,79 @@ export interface TokenInjectableDockerBuilderProps {
40
41
  /**
41
42
  * The VPC in which the CodeBuild project will be deployed.
42
43
  * If provided, the CodeBuild project will be launched within the specified VPC.
43
- * @default No VPC is attached, and the CodeBuild project will use public internet.
44
+ *
45
+ * @default - No VPC is attached, and the CodeBuild project will use public internet.
44
46
  */
45
47
  readonly vpc?: IVpc;
46
48
  /**
47
49
  * The security groups to attach to the CodeBuild project.
48
- * These should define the network access rules for the CodeBuild project.
49
- * @default No security groups are attached.
50
+ * These define the network access rules for the CodeBuild project.
51
+ *
52
+ * @default - No security groups are attached.
50
53
  */
51
54
  readonly securityGroups?: ISecurityGroup[];
52
55
  /**
53
56
  * The subnet selection to specify which subnets to use within the VPC.
54
57
  * Allows the user to select private, public, or isolated subnets.
55
- * @default All subnets in the VPC are used.
58
+ *
59
+ * @default - All subnets in the VPC are used.
56
60
  */
57
61
  readonly subnetSelection?: SubnetSelection;
58
62
  /**
59
- * Custom commands to run during the install phase.
63
+ * Custom commands to run during the install phase of CodeBuild.
60
64
  *
61
- * **Example Usage:**
62
- * ```typescript
63
- * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
64
- * path: path.resolve(__dirname, '../app'),
65
- * installCommands: [
66
- * 'echo "Updating package lists..."',
67
- * 'apt-get update -y',
68
- * 'echo "Installing required packages..."',
69
- * 'apt-get install -y curl dnsutils',
70
- * ],
71
- * // ... other properties ...
72
- * });
65
+ * **Example**:
66
+ * ```ts
67
+ * installCommands: [
68
+ * 'echo "Updating package lists..."',
69
+ * 'apt-get update -y',
70
+ * 'echo "Installing required packages..."',
71
+ * 'apt-get install -y curl dnsutils',
72
+ * ],
73
73
  * ```
74
74
  * @default - No additional install commands.
75
75
  */
76
76
  readonly installCommands?: string[];
77
77
  /**
78
- * Custom commands to run during the pre_build phase.
78
+ * Custom commands to run during the pre_build phase of CodeBuild.
79
79
  *
80
- * **Example Usage:**
81
- * ```typescript
82
- * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
83
- * path: path.resolve(__dirname, '../app'),
84
- * preBuildCommands: [
85
- * 'echo "Fetching configuration from private API..."',
86
- * 'curl -o config.json https://api.example.com/config',
87
- * ],
88
- * // ... other properties ...
89
- * });
80
+ * **Example**:
81
+ * ```ts
82
+ * preBuildCommands: [
83
+ * 'echo "Fetching configuration from private API..."',
84
+ * 'curl -o config.json https://api.example.com/config',
85
+ * ],
90
86
  * ```
91
87
  * @default - No additional pre-build commands.
92
88
  */
93
89
  readonly preBuildCommands?: string[];
94
90
  }
95
91
  /**
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.
92
+ * A CDK construct to build and push Docker images to an ECR repository using
93
+ * CodeBuild and Lambda custom resources, **then** retrieve the final image tag
94
+ * so that ECS/Lambda references use the exact digest.
98
95
  */
99
96
  export declare class TokenInjectableDockerBuilder extends Construct {
97
+ /**
98
+ * The ECR repository that stores the resulting Docker image.
99
+ */
100
100
  private readonly ecrRepository;
101
101
  /**
102
- * An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.
102
+ * An ECS-compatible container image referencing the tag
103
+ * of the built Docker image.
103
104
  */
104
105
  readonly containerImage: ContainerImage;
105
106
  /**
106
- * A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.
107
+ * A Lambda-compatible DockerImageCode referencing the the tag
108
+ * of the built Docker image.
107
109
  */
108
110
  readonly dockerImageCode: DockerImageCode;
111
+ /**
112
+ * Creates a new `TokenInjectableDockerBuilder`.
113
+ *
114
+ * @param scope The scope in which to define this construct.
115
+ * @param id The scoped construct ID.
116
+ * @param props Configuration for building and pushing the Docker image.
117
+ */
109
118
  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
110
119
  }
package/lib/index.js CHANGED
@@ -12,24 +12,33 @@ const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
12
12
  const aws_iam_1 = require("aws-cdk-lib/aws-iam");
13
13
  const aws_kms_1 = require("aws-cdk-lib/aws-kms");
14
14
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
15
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
15
16
  const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
16
17
  const custom_resources_1 = require("aws-cdk-lib/custom-resources");
17
18
  const constructs_1 = require("constructs");
18
19
  /**
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
+ * A CDK construct to build and push Docker images to an ECR repository using
21
+ * CodeBuild and Lambda custom resources, **then** retrieve the final image tag
22
+ * so that ECS/Lambda references use the exact digest.
21
23
  */
22
24
  class TokenInjectableDockerBuilder extends constructs_1.Construct {
25
+ /**
26
+ * Creates a new `TokenInjectableDockerBuilder`.
27
+ *
28
+ * @param scope The scope in which to define this construct.
29
+ * @param id The scoped construct ID.
30
+ * @param props Configuration for building and pushing the Docker image.
31
+ */
23
32
  constructor(scope, id, props) {
24
33
  super(scope, id);
25
34
  const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, } = props;
26
- // Generate a unique tag for this build.
35
+ // Generate an ephemeral tag for CodeBuild
27
36
  const imageTag = crypto.randomUUID();
28
- // KMS key for ECR encryption
37
+ // Define a KMS key for ECR encryption
29
38
  const encryptionKey = new aws_kms_1.Key(this, 'EcrEncryptionKey', {
30
39
  enableKeyRotation: true,
31
40
  });
32
- // ECR repository
41
+ // Create an ECR repository with encryption, lifecycle rules, and image scanning
33
42
  this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository', {
34
43
  lifecycleRules: [
35
44
  {
@@ -43,70 +52,67 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
43
52
  encryptionKey,
44
53
  imageScanOnPush: true,
45
54
  });
46
- // Package source code
55
+ // Wrap the source folder as an S3 asset for CodeBuild to use
47
56
  const sourceAsset = new aws_s3_assets_1.Asset(this, 'SourceAsset', {
48
57
  path: sourcePath,
49
58
  });
50
- // Build args
59
+ // Create an S3 bucket to store the CodeBuild artifacts
60
+ const artifactBucket = new aws_s3_1.Bucket(this, 'ArtifactBucket', {
61
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
62
+ autoDeleteObjects: true,
63
+ blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
64
+ });
65
+ // Convert buildArgs to a CLI-friendly string
51
66
  const buildArgsString = buildArgs
52
67
  ? Object.entries(buildArgs)
53
68
  .map(([k, v]) => `--build-arg ${k}=${v}`)
54
69
  .join(' ')
55
70
  : '';
56
- // Docker Hub login commands
71
+ // Optional DockerHub login, if a secret ARN is provided
57
72
  const dockerLoginCommands = dockerLoginSecretArn
58
73
  ? [
59
- 'echo "Retrieving Docker credentials from Secrets Manager..."',
74
+ 'echo "Retrieving Docker credentials..."',
60
75
  'apt-get update -y && apt-get install -y jq',
61
76
  `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,
62
77
  `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,
63
78
  'echo "Logging in to Docker Hub..."',
64
- // Use non-stdin login to avoid TTY error:
65
79
  'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',
66
80
  ]
67
- : ['echo "No Docker credentials provided. Skipping Docker Hub login."'];
68
- // BuildSpec
81
+ : ['echo "No Docker credentials. Skipping Docker Hub login."'];
69
82
  const buildSpecObj = {
70
83
  version: '0.2',
71
84
  phases: {
72
85
  install: {
73
86
  commands: [
74
87
  'echo "Beginning install phase..."',
75
- ...(installCommands || []),
88
+ ...(installCommands ?? []),
76
89
  ],
77
90
  },
78
91
  pre_build: {
79
92
  commands: [
80
- ...(preBuildCommands || []),
93
+ ...(preBuildCommands ?? []),
81
94
  ...dockerLoginCommands,
82
95
  'echo "Retrieving AWS Account ID..."',
83
96
  'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
84
- 'echo "Logging in to Amazon ECR..."',
97
+ 'echo "Logging into Amazon ECR..."',
85
98
  '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
99
  ],
87
100
  },
88
101
  build: {
89
102
  commands: [
90
- 'echo "Build phase: Building the Docker image..."',
103
+ `echo "Building Docker image with tag ${imageTag}..."`,
91
104
  `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,
92
105
  ],
93
106
  },
94
107
  post_build: {
95
108
  commands: [
96
- `echo "Post-build phase: Pushing the Docker image with tag ${imageTag}..."`,
109
+ `echo "Pushing Docker image with tag ${imageTag}..."`,
97
110
  `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
111
  ],
102
112
  },
103
113
  },
104
- artifacts: {
105
- files: ['imageDetail.json'],
106
- name: 'imageDetail',
107
- },
108
114
  };
109
- // CodeBuild project
115
+ // Create the CodeBuild project
110
116
  const codeBuildProject = new aws_codebuild_1.Project(this, 'CodeBuildProject', {
111
117
  source: aws_codebuild_1.Source.s3({
112
118
  bucket: sourceAsset.bucket,
@@ -119,12 +125,12 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
119
125
  environmentVariables: {
120
126
  ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },
121
127
  },
128
+ buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
122
129
  vpc,
123
130
  securityGroups,
124
131
  subnetSelection,
125
- buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
126
132
  });
127
- // Permissions
133
+ // Grant CodeBuild the ability to interact with ECR
128
134
  this.ecrRepository.grantPullPush(codeBuildProject);
129
135
  codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
130
136
  actions: [
@@ -141,7 +147,7 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
141
147
  }));
142
148
  }
143
149
  encryptionKey.grantEncryptDecrypt(codeBuildProject.role);
144
- // onEvent handler
150
+ // Define the Lambda functions for custom resource event and completion handling
145
151
  const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
146
152
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
147
153
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
@@ -152,10 +158,12 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
152
158
  actions: ['codebuild:StartBuild'],
153
159
  resources: [codeBuildProject.projectArn],
154
160
  }));
155
- // isComplete handler
156
161
  const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
157
162
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
158
163
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
164
+ environment: {
165
+ IMAGE_TAG: imageTag,
166
+ },
159
167
  handler: 'isComplete.handler',
160
168
  timeout: aws_cdk_lib_1.Duration.minutes(15),
161
169
  });
@@ -166,22 +174,21 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
166
174
  'logs:GetLogEvents',
167
175
  'logs:DescribeLogStreams',
168
176
  'logs:DescribeLogGroups',
169
- 's3:GetObject',
170
- 's3:GetBucketLocation',
171
177
  ],
172
178
  resources: ['*'],
173
179
  }));
180
+ artifactBucket.grantReadWrite(isCompleteHandlerFunction);
174
181
  encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
175
182
  encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
176
183
  this.ecrRepository.grantPullPush(onEventHandlerFunction);
177
184
  this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
178
- // Provider
185
+ // Create a custom resource provider that uses the above Lambdas
179
186
  const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
180
187
  onEventHandler: onEventHandlerFunction,
181
188
  isCompleteHandler: isCompleteHandlerFunction,
182
189
  queryInterval: aws_cdk_lib_1.Duration.seconds(30),
183
190
  });
184
- // Custom resource
191
+ // Custom Resource that triggers the CodeBuild and waits for completion
185
192
  const buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
186
193
  serviceToken: provider.serviceToken,
187
194
  properties: {
@@ -191,17 +198,16 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
191
198
  },
192
199
  });
193
200
  buildTriggerResource.node.addDependency(codeBuildProject);
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
201
+ // Retrieve the final Docker image tag from Data.ImageTag
202
+ // This creates a dependency on the Custom Resource...
203
+ const imageTagRef = buildTriggerResource.getAttString('ImageTag');
204
+ this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository, imageTagRef);
199
205
  this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository, {
200
- tagOrDigest: imageDigest,
206
+ tagOrDigest: imageTagRef,
201
207
  });
202
208
  }
203
209
  }
204
210
  exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
205
211
  _a = JSII_RTTI_SYMBOL_1;
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"]}
212
+ TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.3.1" };
213
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;;;AAAA,iCAAiC;AACjC,6BAA6B;AAC7B,6CAAsE;AACtE,6DAAwF;AAExF,iDAAkF;AAClF,iDAAqD;AACrD,iDAAsD;AACtD,iDAA0C;AAC1C,uDAAkF;AAClF,+CAA+D;AAC/D,6DAAkD;AAClD,mEAAwD;AACxD,2CAAuC;AA8FvC;;;;GAIG;AACH,MAAa,4BAA6B,SAAQ,sBAAS;IAkBzD;;;;;;OAMG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAwC;QAChF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,EACJ,IAAI,EAAE,UAAU,EAChB,SAAS,EACT,oBAAoB,EACpB,GAAG,EACH,cAAc,EACd,eAAe,EACf,eAAe,EACf,gBAAgB,GACjB,GAAG,KAAK,CAAC;QAEV,0CAA0C;QAC1C,MAAM,QAAQ,GAAG,MAAM,CAAC,UAAU,EAAE,CAAC;QAErC,sCAAsC;QACtC,MAAM,aAAa,GAAG,IAAI,aAAG,CAAC,IAAI,EAAE,kBAAkB,EAAE;YACtD,iBAAiB,EAAE,IAAI;SACxB,CAAC,CAAC;QAEH,gFAAgF;QAChF,IAAI,CAAC,aAAa,GAAG,IAAI,oBAAU,CAAC,IAAI,EAAE,eAAe,EAAE;YACzD,cAAc,EAAE;gBACd;oBACE,YAAY,EAAE,CAAC;oBACf,WAAW,EAAE,oCAAoC;oBACjD,SAAS,EAAE,mBAAS,CAAC,QAAQ;oBAC7B,WAAW,EAAE,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;iBAC9B;aACF;YACD,UAAU,EAAE,8BAAoB,CAAC,GAAG;YACpC,aAAa;YACb,eAAe,EAAE,IAAI;SACtB,CAAC,CAAC;QAEH,6DAA6D;QAC7D,MAAM,WAAW,GAAG,IAAI,qBAAK,CAAC,IAAI,EAAE,aAAa,EAAE;YACjD,IAAI,EAAE,UAAU;SACjB,CAAC,CAAC;QAEH,uDAAuD;QACvD,MAAM,cAAc,GAAG,IAAI,eAAM,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACxD,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,iBAAiB,EAAE,0BAAiB,CAAC,SAAS;SAC/C,CAAC,CAAC;QAEH,6CAA6C;QAC7C,MAAM,eAAe,GAAG,SAAS;YAC/B,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC;iBACxB,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;iBACxC,IAAI,CAAC,GAAG,CAAC;YACZ,CAAC,CAAC,EAAE,CAAC;QAEP,wDAAwD;QACxD,MAAM,mBAAmB,GAAG,oBAAoB;YAC9C,CAAC,CAAC;gBACA,yCAAyC;gBACzC,4CAA4C;gBAC5C,qEAAqE,oBAAoB,wDAAwD;gBACjJ,qEAAqE,oBAAoB,wDAAwD;gBACjJ,oCAAoC;gBACpC,mFAAmF;aACpF;YACD,CAAC,CAAC,CAAC,0DAA0D,CAAC,CAAC;QAEjE,MAAM,YAAY,GAAG;YACnB,OAAO,EAAE,KAAK;YACd,MAAM,EAAE;gBACN,OAAO,EAAE;oBACP,QAAQ,EAAE;wBACR,mCAAmC;wBACnC,GAAG,CAAC,eAAe,IAAI,EAAE,CAAC;qBAC3B;iBACF;gBACD,SAAS,EAAE;oBACT,QAAQ,EAAE;wBACR,GAAG,CAAC,gBAAgB,IAAI,EAAE,CAAC;wBAC3B,GAAG,mBAAmB;wBACtB,qCAAqC;wBACrC,gFAAgF;wBAChF,mCAAmC;wBACnC,8JAA8J;qBAC/J;iBACF;gBACD,KAAK,EAAE;oBACL,QAAQ,EAAE;wBACR,wCAAwC,QAAQ,MAAM;wBACtD,gBAAgB,eAAe,qBAAqB,QAAQ,qBAAqB;qBAClF;iBACF;gBACD,UAAU,EAAE;oBACV,QAAQ,EAAE;wBACR,uCAAuC,QAAQ,MAAM;wBACrD,6BAA6B,QAAQ,EAAE;qBACxC;iBACF;aACF;SACF,CAAC;QAGF,+BAA+B;QAC/B,MAAM,gBAAgB,GAAG,IAAI,uBAAO,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC7D,MAAM,EAAE,sBAAM,CAAC,EAAE,CAAC;gBAChB,MAAM,EAAE,WAAW,CAAC,MAAM;gBAC1B,IAAI,EAAE,WAAW,CAAC,WAAW;aAC9B,CAAC;YACF,WAAW,EAAE;gBACX,UAAU,EAAE,+BAAe,CAAC,YAAY;gBACxC,UAAU,EAAE,IAAI;aACjB;YACD,oBAAoB,EAAE;gBACpB,YAAY,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,aAAa,CAAC,aAAa,EAAE;aAC1D;YACD,SAAS,EAAE,yBAAS,CAAC,UAAU,CAAC,YAAY,CAAC;YAC7C,GAAG;YACH,cAAc;YACd,eAAe;SAChB,CAAC,CAAC;QAEH,mDAAmD;QACnD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAC;QACnD,gBAAgB,CAAC,IAAI,EAAE,oBAAoB,CAAC,IAAI,yBAAe,CAAC;YAC9D,OAAO,EAAE;gBACP,2BAA2B;gBAC3B,4BAA4B;gBAC5B,iCAAiC;aAClC;YACD,SAAS,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;SAC9C,CAAC,CAAC,CAAC;QACJ,IAAI,oBAAoB,EAAE,CAAC;YACzB,gBAAgB,CAAC,IAAI,EAAE,oBAAoB,CAAC,IAAI,yBAAe,CAAC;gBAC9D,OAAO,EAAE,CAAC,+BAA+B,CAAC;gBAC1C,SAAS,EAAE,CAAC,oBAAoB,CAAC;aAClC,CAAC,CAAC,CAAC;QACN,CAAC;QACD,aAAa,CAAC,mBAAmB,CAAC,gBAAgB,CAAC,IAAK,CAAC,CAAC;QAE1D,gFAAgF;QAChF,MAAM,sBAAsB,GAAG,IAAI,qBAAQ,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC1E,OAAO,EAAE,oBAAO,CAAC,WAAW;YAC5B,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;YAC3D,OAAO,EAAE,iBAAiB;YAC1B,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SAC9B,CAAC,CAAC;QACH,sBAAsB,CAAC,eAAe,CAAC,IAAI,yBAAe,CAAC;YACzD,OAAO,EAAE,CAAC,sBAAsB,CAAC;YACjC,SAAS,EAAE,CAAC,gBAAgB,CAAC,UAAU,CAAC;SACzC,CAAC,CAAC,CAAC;QAEJ,MAAM,yBAAyB,GAAG,IAAI,qBAAQ,CAAC,IAAI,EAAE,2BAA2B,EAAE;YAChF,OAAO,EAAE,oBAAO,CAAC,WAAW;YAC5B,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;YAC9D,WAAW,EAAE;gBACX,SAAS,EAAE,QAAQ;aACpB;YACD,OAAO,EAAE,oBAAoB;YAC7B,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SAC9B,CAAC,CAAC;QAEH,yBAAyB,CAAC,eAAe,CAAC,IAAI,yBAAe,CAAC;YAC5D,OAAO,EAAE;gBACP,0BAA0B;gBAC1B,gCAAgC;gBAChC,mBAAmB;gBACnB,yBAAyB;gBACzB,wBAAwB;aACzB;YACD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CAAC,CAAC;QAEJ,cAAc,CAAC,cAAc,CAAC,yBAAyB,CAAC,CAAC;QACzD,aAAa,CAAC,mBAAmB,CAAC,sBAAsB,CAAC,CAAC;QAC1D,aAAa,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,CAAC;QAC7D,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,sBAAsB,CAAC,CAAC;QACzD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,yBAAyB,CAAC,CAAC;QAE5D,gEAAgE;QAChE,MAAM,QAAQ,GAAG,IAAI,2BAAQ,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC5D,cAAc,EAAE,sBAAsB;YACtC,iBAAiB,EAAE,yBAAyB;YAC5C,aAAa,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SACpC,CAAC,CAAC;QAEH,uEAAuE;QACvE,MAAM,oBAAoB,GAAG,IAAI,4BAAc,CAAC,IAAI,EAAE,sBAAsB,EAAE;YAC5E,YAAY,EAAE,QAAQ,CAAC,YAAY;YACnC,UAAU,EAAE;gBACV,WAAW,EAAE,gBAAgB,CAAC,WAAW;gBACzC,QAAQ,EAAE,QAAQ;gBAClB,OAAO,EAAE,MAAM,CAAC,UAAU,EAAE;aAC7B;SACF,CAAC,CAAC;QACH,oBAAoB,CAAC,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAC;QAE1D,yDAAyD;QACzD,sDAAsD;QACtD,MAAM,WAAW,GAAG,oBAAoB,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;QAClE,IAAI,CAAC,cAAc,GAAG,wBAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC;QACxF,IAAI,CAAC,eAAe,GAAG,4BAAe,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,EAAE;YACjE,WAAW,EAAE,WAAW;SACzB,CAAC,CAAC;IACL,CAAC;;AArOH,oEAsOC","sourcesContent":["import * as crypto from 'crypto';\nimport * as path from 'path';\nimport { CustomResource, Duration, RemovalPolicy } 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 { Bucket, BlockPublicAccess } from 'aws-cdk-lib/aws-s3';\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/**\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 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    } = props;\n\n    // Generate an ephemeral tag for CodeBuild\n    const imageTag = crypto.randomUUID();\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 encryption, lifecycle rules, and image scanning\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    // Wrap the source folder as an S3 asset for CodeBuild to use\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n    });\n\n    // Create an S3 bucket to store the CodeBuild artifacts\n    const artifactBucket = new Bucket(this, 'ArtifactBucket', {\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\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    // 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 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 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            `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,\n          ],\n        },\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      vpc,\n      securityGroups,\n      subnetSelection,\n    });\n\n    // Grant CodeBuild the ability to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.role?.addToPrincipalPolicy(new PolicyStatement({\n      actions: [\n        'ecr:GetAuthorizationToken',\n        'ecr:GetDownloadUrlForLayer',\n        'ecr:BatchCheckLayerAvailability',\n      ],\n      resources: [this.ecrRepository.repositoryArn],\n    }));\n    if (dockerLoginSecretArn) {\n      codeBuildProject.role?.addToPrincipalPolicy(new PolicyStatement({\n        actions: ['secretsmanager:GetSecretValue'],\n        resources: [dockerLoginSecretArn],\n      }));\n    }\n    encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n\n    // Define the Lambda functions for custom resource event and completion handling\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    onEventHandlerFunction.addToRolePolicy(new PolicyStatement({\n      actions: ['codebuild:StartBuild'],\n      resources: [codeBuildProject.projectArn],\n    }));\n\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_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\n    isCompleteHandlerFunction.addToRolePolicy(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    artifactBucket.grantReadWrite(isCompleteHandlerFunction);\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 that uses the above Lambdas\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: 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: crypto.randomUUID(),\n      },\n    });\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Retrieve the final Docker image tag from Data.ImageTag\n    // This creates a dependency on the Custom Resource...\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
@@ -30,7 +30,9 @@
30
30
  "watch": "npx projen watch",
31
31
  "projen": "npx projen",
32
32
  "local-deploy": "cdk deploy --app \"npx ts-node src/integ.default.ts\"",
33
- "local-destroy": "cdk destroy --app \"npx ts-node src/integ.default.ts\""
33
+ "local-deploy-no-rollback": "cdk deploy --no-rollback --app \"npx ts-node src/integ.default.ts\"",
34
+ "local-destroy": "cdk destroy --app \"npx ts-node src/integ.default.ts\"",
35
+ "local-synth": "cdk synth --app \"npx ts-node src/integ.default.ts\""
34
36
  },
35
37
  "author": {
36
38
  "name": "AlexTech314",
@@ -43,7 +45,7 @@
43
45
  "@types/node": "^22.10.2",
44
46
  "@typescript-eslint/eslint-plugin": "^8",
45
47
  "@typescript-eslint/parser": "^8",
46
- "aws-cdk-lib": "2.166.0",
48
+ "aws-cdk-lib": "2.173.2",
47
49
  "commit-and-tag-version": "^12",
48
50
  "constructs": "10.0.5",
49
51
  "eslint": "^9",
@@ -62,7 +64,7 @@
62
64
  "typescript": "^5.7.2"
63
65
  },
64
66
  "peerDependencies": {
65
- "aws-cdk-lib": "^2.166.0",
67
+ "aws-cdk-lib": "^2.173.2",
66
68
  "constructs": "^10.0.5"
67
69
  },
68
70
  "keywords": [
@@ -96,7 +98,7 @@
96
98
  "publishConfig": {
97
99
  "access": "public"
98
100
  },
99
- "version": "1.3.0",
101
+ "version": "1.3.1",
100
102
  "jest": {
101
103
  "coverageProvider": "v8",
102
104
  "testMatch": [