token-injectable-docker-builder 1.2.7 → 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."
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": 123
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,18 +4004,19 @@
3955
4004
  "kind": "class",
3956
4005
  "locationInModule": {
3957
4006
  "filename": "src/index.ts",
3958
- "line": 118
4007
+ "line": 113
3959
4008
  },
3960
4009
  "name": "TokenInjectableDockerBuilder",
3961
4010
  "properties": [
3962
4011
  {
3963
4012
  "docs": {
3964
- "stability": "stable"
4013
+ "stability": "stable",
4014
+ "summary": "An ECS-compatible container image referencing the tag of the built Docker image."
3965
4015
  },
3966
4016
  "immutable": true,
3967
4017
  "locationInModule": {
3968
4018
  "filename": "src/index.ts",
3969
- "line": 120
4019
+ "line": 123
3970
4020
  },
3971
4021
  "name": "containerImage",
3972
4022
  "type": {
@@ -3975,12 +4025,13 @@
3975
4025
  },
3976
4026
  {
3977
4027
  "docs": {
3978
- "stability": "stable"
4028
+ "stability": "stable",
4029
+ "summary": "A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image."
3979
4030
  },
3980
4031
  "immutable": true,
3981
4032
  "locationInModule": {
3982
4033
  "filename": "src/index.ts",
3983
- "line": 121
4034
+ "line": 129
3984
4035
  },
3985
4036
  "name": "dockerImageCode",
3986
4037
  "type": {
@@ -4001,7 +4052,7 @@
4001
4052
  "kind": "interface",
4002
4053
  "locationInModule": {
4003
4054
  "filename": "src/index.ts",
4004
- "line": 18
4055
+ "line": 19
4005
4056
  },
4006
4057
  "name": "TokenInjectableDockerBuilderProps",
4007
4058
  "properties": [
@@ -4014,7 +4065,7 @@
4014
4065
  "immutable": true,
4015
4066
  "locationInModule": {
4016
4067
  "filename": "src/index.ts",
4017
- "line": 22
4068
+ "line": 23
4018
4069
  },
4019
4070
  "name": "path",
4020
4071
  "type": {
@@ -4025,14 +4076,14 @@
4025
4076
  "abstract": true,
4026
4077
  "docs": {
4027
4078
  "example": "{\n TOKEN: 'my-secret-token',\n ENV: 'production'\n}",
4028
- "remarks": "These are transformed into `--build-arg` flags.",
4079
+ "remarks": "These are transformed into `--build-arg KEY=VALUE` flags.",
4029
4080
  "stability": "stable",
4030
4081
  "summary": "Build arguments to pass to the Docker build process."
4031
4082
  },
4032
4083
  "immutable": true,
4033
4084
  "locationInModule": {
4034
4085
  "filename": "src/index.ts",
4035
- "line": 33
4086
+ "line": 34
4036
4087
  },
4037
4088
  "name": "buildArgs",
4038
4089
  "optional": true,
@@ -4049,14 +4100,14 @@
4049
4100
  "abstract": true,
4050
4101
  "docs": {
4051
4102
  "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.",
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.",
4053
4104
  "stability": "stable",
4054
4105
  "summary": "The ARN of the AWS Secrets Manager secret containing Docker login credentials."
4055
4106
  },
4056
4107
  "immutable": true,
4057
4108
  "locationInModule": {
4058
4109
  "filename": "src/index.ts",
4059
- "line": 48
4110
+ "line": 51
4060
4111
  },
4061
4112
  "name": "dockerLoginSecretArn",
4062
4113
  "optional": true,
@@ -4068,9 +4119,9 @@
4068
4119
  "abstract": true,
4069
4120
  "docs": {
4070
4121
  "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.*",
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```",
4072
4123
  "stability": "stable",
4073
- "summary": "Custom commands to run during the install phase."
4124
+ "summary": "Custom commands to run during the install phase of CodeBuild."
4074
4125
  },
4075
4126
  "immutable": true,
4076
4127
  "locationInModule": {
@@ -4092,14 +4143,14 @@
4092
4143
  "abstract": true,
4093
4144
  "docs": {
4094
4145
  "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.*",
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```",
4096
4147
  "stability": "stable",
4097
- "summary": "Custom commands to run during the pre_build phase."
4148
+ "summary": "Custom commands to run during the pre_build phase of CodeBuild."
4098
4149
  },
4099
4150
  "immutable": true,
4100
4151
  "locationInModule": {
4101
4152
  "filename": "src/index.ts",
4102
- "line": 112
4153
+ "line": 105
4103
4154
  },
4104
4155
  "name": "preBuildCommands",
4105
4156
  "optional": true,
@@ -4115,15 +4166,15 @@
4115
4166
  {
4116
4167
  "abstract": true,
4117
4168
  "docs": {
4118
- "default": "No security groups are attached.",
4119
- "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.",
4120
4171
  "stability": "stable",
4121
4172
  "summary": "The security groups to attach to the CodeBuild project."
4122
4173
  },
4123
4174
  "immutable": true,
4124
4175
  "locationInModule": {
4125
4176
  "filename": "src/index.ts",
4126
- "line": 62
4177
+ "line": 67
4127
4178
  },
4128
4179
  "name": "securityGroups",
4129
4180
  "optional": true,
@@ -4139,7 +4190,7 @@
4139
4190
  {
4140
4191
  "abstract": true,
4141
4192
  "docs": {
4142
- "default": "All subnets in the VPC are used.",
4193
+ "default": "- All subnets in the VPC are used.",
4143
4194
  "remarks": "Allows the user to select private, public, or isolated subnets.",
4144
4195
  "stability": "stable",
4145
4196
  "summary": "The subnet selection to specify which subnets to use within the VPC."
@@ -4147,7 +4198,7 @@
4147
4198
  "immutable": true,
4148
4199
  "locationInModule": {
4149
4200
  "filename": "src/index.ts",
4150
- "line": 69
4201
+ "line": 75
4151
4202
  },
4152
4203
  "name": "subnetSelection",
4153
4204
  "optional": true,
@@ -4158,7 +4209,7 @@
4158
4209
  {
4159
4210
  "abstract": true,
4160
4211
  "docs": {
4161
- "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.",
4162
4213
  "remarks": "If provided, the CodeBuild project will be launched within the specified VPC.",
4163
4214
  "stability": "stable",
4164
4215
  "summary": "The VPC in which the CodeBuild project will be deployed."
@@ -4166,7 +4217,7 @@
4166
4217
  "immutable": true,
4167
4218
  "locationInModule": {
4168
4219
  "filename": "src/index.ts",
4169
- "line": 55
4220
+ "line": 59
4170
4221
  },
4171
4222
  "name": "vpc",
4172
4223
  "optional": true,
@@ -4178,6 +4229,6 @@
4178
4229
  "symbolId": "src/index:TokenInjectableDockerBuilderProps"
4179
4230
  }
4180
4231
  },
4181
- "version": "1.2.7",
4182
- "fingerprint": "9GABMeCUEVrnrA3l3vauIPVodVLL/zlUUeRlfhTLS2w="
4232
+ "version": "1.3.1",
4233
+ "fingerprint": "Q9DfzcrA3DzkFuZ3QghI1ZV3uoLI/0611iFI3ZPoifc="
4183
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.
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> | *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.* |
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,6 +118,8 @@ public readonly containerImage: ContainerImage;
112
118
 
113
119
  - *Type:* aws-cdk-lib.aws_ecs.ContainerImage
114
120
 
121
+ An ECS-compatible container image referencing the tag of the built Docker image.
122
+
115
123
  ---
116
124
 
117
125
  ##### `dockerImageCode`<sup>Required</sup> <a name="dockerImageCode" id="token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode"></a>
@@ -122,6 +130,8 @@ public readonly dockerImageCode: DockerImageCode;
122
130
 
123
131
  - *Type:* aws-cdk-lib.aws_lambda.DockerImageCode
124
132
 
133
+ A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image.
134
+
125
135
  ---
126
136
 
127
137
 
@@ -146,8 +156,8 @@ const tokenInjectableDockerBuilderProps: TokenInjectableDockerBuilderProps = { .
146
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. |
147
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. |
148
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. |
149
- | <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.installCommands">installCommands</a></code> | <code>string[]</code> | Custom commands to run during the install phase. |
150
- | <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. |
151
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. |
152
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. |
153
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. |
@@ -176,7 +186,7 @@ public readonly buildArgs: {[ key: string ]: string};
176
186
 
177
187
  Build arguments to pass to the Docker build process.
178
188
 
179
- These are transformed into `--build-arg` flags.
189
+ These are transformed into `--build-arg KEY=VALUE` flags.
180
190
 
181
191
  ---
182
192
 
@@ -207,7 +217,9 @@ This secret should store a JSON object with the following structure:
207
217
  "password": "my-docker-password"
208
218
  }
209
219
  ```
210
- If not provided, the construct will skip Docker login during the build process.
220
+ If not provided (or not needed), the construct will skip Docker Hub login.
221
+
222
+ **Note**: The secret must be in the same region as the stack.
211
223
 
212
224
  ---
213
225
 
@@ -227,22 +239,17 @@ public readonly installCommands: string[];
227
239
  - *Type:* string[]
228
240
  - *Default:* No additional install commands.
229
241
 
230
- Custom commands to run during the install phase.
242
+ Custom commands to run during the install phase of CodeBuild.
231
243
 
232
- **Example Usage:**
233
- ```typescript
234
- new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
235
- path: path.resolve(__dirname, '../app'),
236
- installCommands: [
237
- 'echo "Updating package lists..."',
238
- 'apt-get update -y',
239
- 'echo "Installing required packages..."',
240
- 'apt-get install -y curl dnsutils',
241
- ],
242
- // ... other properties ...
243
- });
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
+ ],
244
252
  ```
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
253
 
247
254
  ---
248
255
 
@@ -255,20 +262,15 @@ public readonly preBuildCommands: string[];
255
262
  - *Type:* string[]
256
263
  - *Default:* No additional pre-build commands.
257
264
 
258
- Custom commands to run during the pre_build phase.
265
+ Custom commands to run during the pre_build phase of CodeBuild.
259
266
 
260
- **Example Usage:**
261
- ```typescript
262
- new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
263
- path: path.resolve(__dirname, '../app'),
264
- preBuildCommands: [
265
- 'echo "Fetching configuration from private API..."',
266
- 'curl -o config.json https://api.example.com/config',
267
- ],
268
- // ... other properties ...
269
- });
267
+ **Example**:
268
+ ```ts
269
+ preBuildCommands: [
270
+ 'echo "Fetching configuration from private API..."',
271
+ 'curl -o config.json https://api.example.com/config',
272
+ ],
270
273
  ```
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
274
 
273
275
  ---
274
276
 
@@ -283,7 +285,7 @@ public readonly securityGroups: ISecurityGroup[];
283
285
 
284
286
  The security groups to attach to the CodeBuild project.
285
287
 
286
- These should define the network access rules for the CodeBuild project.
288
+ These define the network access rules for the CodeBuild project.
287
289
 
288
290
  ---
289
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',
@@ -31,7 +31,9 @@ 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
+ *
36
+ * **Note**: The secret must be in the same region as the stack.
35
37
  *
36
38
  * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'
37
39
  */
@@ -39,68 +41,79 @@ export interface TokenInjectableDockerBuilderProps {
39
41
  /**
40
42
  * The VPC in which the CodeBuild project will be deployed.
41
43
  * If provided, the CodeBuild project will be launched within the specified VPC.
42
- * @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.
43
46
  */
44
47
  readonly vpc?: IVpc;
45
48
  /**
46
49
  * The security groups to attach to the CodeBuild project.
47
- * These should define the network access rules for the CodeBuild project.
48
- * @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.
49
53
  */
50
54
  readonly securityGroups?: ISecurityGroup[];
51
55
  /**
52
56
  * The subnet selection to specify which subnets to use within the VPC.
53
57
  * Allows the user to select private, public, or isolated subnets.
54
- * @default All subnets in the VPC are used.
58
+ *
59
+ * @default - All subnets in the VPC are used.
55
60
  */
56
61
  readonly subnetSelection?: SubnetSelection;
57
62
  /**
58
- * Custom commands to run during the install phase.
63
+ * Custom commands to run during the install phase of CodeBuild.
59
64
  *
60
- * **Example Usage:**
61
- * ```typescript
62
- * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
63
- * path: path.resolve(__dirname, '../app'),
64
- * installCommands: [
65
- * 'echo "Updating package lists..."',
66
- * 'apt-get update -y',
67
- * 'echo "Installing required packages..."',
68
- * 'apt-get install -y curl dnsutils',
69
- * ],
70
- * // ... other properties ...
71
- * });
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
+ * ],
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[];
78
77
  /**
79
- * Custom commands to run during the pre_build phase.
78
+ * Custom commands to run during the pre_build phase of CodeBuild.
80
79
  *
81
- * **Example Usage:**
82
- * ```typescript
83
- * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {
84
- * path: path.resolve(__dirname, '../app'),
85
- * preBuildCommands: [
86
- * 'echo "Fetching configuration from private API..."',
87
- * 'curl -o config.json https://api.example.com/config',
88
- * ],
89
- * // ... other properties ...
90
- * });
80
+ * **Example**:
81
+ * ```ts
82
+ * preBuildCommands: [
83
+ * 'echo "Fetching configuration from private API..."',
84
+ * 'curl -o config.json https://api.example.com/config',
85
+ * ],
91
86
  * ```
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
87
  * @default - No additional pre-build commands.
95
88
  */
96
89
  readonly preBuildCommands?: string[];
97
90
  }
98
91
  /**
99
- * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources.
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.
100
95
  */
101
96
  export declare class TokenInjectableDockerBuilder extends Construct {
97
+ /**
98
+ * The ECR repository that stores the resulting Docker image.
99
+ */
102
100
  private readonly ecrRepository;
101
+ /**
102
+ * An ECS-compatible container image referencing the tag
103
+ * of the built Docker image.
104
+ */
103
105
  readonly containerImage: ContainerImage;
106
+ /**
107
+ * A Lambda-compatible DockerImageCode referencing the the tag
108
+ * of the built Docker image.
109
+ */
104
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
+ */
105
118
  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
106
119
  }
package/lib/index.js CHANGED
@@ -12,21 +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
+ * 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.
20
23
  */
21
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
+ */
22
32
  constructor(scope, id, props) {
23
33
  super(scope, id);
24
34
  const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, } = props;
35
+ // Generate an ephemeral tag for CodeBuild
36
+ const imageTag = crypto.randomUUID();
25
37
  // Define a KMS key for ECR encryption
26
38
  const encryptionKey = new aws_kms_1.Key(this, 'EcrEncryptionKey', {
27
39
  enableKeyRotation: true,
28
40
  });
29
- // Create an ECR repository with lifecycle rules, encryption, and image scanning enabled
41
+ // Create an ECR repository with encryption, lifecycle rules, and image scanning
30
42
  this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository', {
31
43
  lifecycleRules: [
32
44
  {
@@ -37,31 +49,71 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
37
49
  },
38
50
  ],
39
51
  encryption: aws_ecr_1.RepositoryEncryption.KMS,
40
- encryptionKey: encryptionKey,
52
+ encryptionKey,
41
53
  imageScanOnPush: true,
42
54
  });
43
- // Package the source code as an asset
55
+ // Wrap the source folder as an S3 asset for CodeBuild to use
44
56
  const sourceAsset = new aws_s3_assets_1.Asset(this, 'SourceAsset', {
45
57
  path: sourcePath,
46
58
  });
47
- // Transform buildArgs into a string of --build-arg KEY=VALUE
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
48
66
  const buildArgsString = buildArgs
49
67
  ? Object.entries(buildArgs)
50
- .map(([key, value]) => `--build-arg ${key}=${value}`)
68
+ .map(([k, v]) => `--build-arg ${k}=${v}`)
51
69
  .join(' ')
52
70
  : '';
53
- // Conditional Dockerhub login commands
71
+ // Optional DockerHub login, if a secret ARN is provided
54
72
  const dockerLoginCommands = dockerLoginSecretArn
55
73
  ? [
56
- 'echo "Retrieving Docker credentials from Secrets Manager..."',
74
+ 'echo "Retrieving Docker credentials..."',
75
+ 'apt-get update -y && apt-get install -y jq',
57
76
  `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,
58
77
  `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,
59
- 'echo "Logging in to Docker..."',
78
+ 'echo "Logging in to Docker Hub..."',
60
79
  'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',
61
80
  ]
62
- : ['echo "No Docker credentials provided. Skipping login step."'];
63
- // Create a CodeBuild project
64
- const codeBuildProject = new aws_codebuild_1.Project(this, 'UICodeBuildProject', {
81
+ : ['echo "No Docker credentials. Skipping Docker Hub login."'];
82
+ const buildSpecObj = {
83
+ version: '0.2',
84
+ phases: {
85
+ install: {
86
+ commands: [
87
+ 'echo "Beginning install phase..."',
88
+ ...(installCommands ?? []),
89
+ ],
90
+ },
91
+ pre_build: {
92
+ commands: [
93
+ ...(preBuildCommands ?? []),
94
+ ...dockerLoginCommands,
95
+ 'echo "Retrieving AWS Account ID..."',
96
+ 'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
97
+ 'echo "Logging into Amazon ECR..."',
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',
99
+ ],
100
+ },
101
+ build: {
102
+ commands: [
103
+ `echo "Building Docker image with tag ${imageTag}..."`,
104
+ `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,
105
+ ],
106
+ },
107
+ post_build: {
108
+ commands: [
109
+ `echo "Pushing Docker image with tag ${imageTag}..."`,
110
+ `docker push $ECR_REPO_URI:${imageTag}`,
111
+ ],
112
+ },
113
+ },
114
+ };
115
+ // Create the CodeBuild project
116
+ const codeBuildProject = new aws_codebuild_1.Project(this, 'CodeBuildProject', {
65
117
  source: aws_codebuild_1.Source.s3({
66
118
  bucket: sourceAsset.bucket,
67
119
  path: sourceAsset.s3ObjectKey,
@@ -72,51 +124,15 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
72
124
  },
73
125
  environmentVariables: {
74
126
  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
127
  },
128
+ buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
80
129
  vpc,
81
130
  securityGroups,
82
131
  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
- }),
116
132
  });
117
- // Grant permissions to CodeBuild
133
+ // Grant CodeBuild the ability to interact with ECR
118
134
  this.ecrRepository.grantPullPush(codeBuildProject);
119
- codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
135
+ codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
120
136
  actions: [
121
137
  'ecr:GetAuthorizationToken',
122
138
  'ecr:GetDownloadUrlForLayer',
@@ -125,14 +141,13 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
125
141
  resources: [this.ecrRepository.repositoryArn],
126
142
  }));
127
143
  if (dockerLoginSecretArn) {
128
- codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
144
+ codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
129
145
  actions: ['secretsmanager:GetSecretValue'],
130
146
  resources: [dockerLoginSecretArn],
131
147
  }));
132
148
  }
133
- // Grant CodeBuild access to the KMS key
134
149
  encryptionKey.grantEncryptDecrypt(codeBuildProject.role);
135
- // Create Lambda functions for onEvent and isComplete handlers
150
+ // Define the Lambda functions for custom resource event and completion handling
136
151
  const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
137
152
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
138
153
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
@@ -141,11 +156,14 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
141
156
  });
142
157
  onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
143
158
  actions: ['codebuild:StartBuild'],
144
- resources: [codeBuildProject.projectArn], // Restrict to specific project
159
+ resources: [codeBuildProject.projectArn],
145
160
  }));
146
161
  const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
147
162
  runtime: aws_lambda_1.Runtime.NODEJS_18_X,
148
163
  code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
164
+ environment: {
165
+ IMAGE_TAG: imageTag,
166
+ },
149
167
  handler: 'isComplete.handler',
150
168
  timeout: aws_cdk_lib_1.Duration.minutes(15),
151
169
  });
@@ -159,31 +177,37 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
159
177
  ],
160
178
  resources: ['*'],
161
179
  }));
162
- // Grant Lambda functions access to KMS key and ECR
180
+ artifactBucket.grantReadWrite(isCompleteHandlerFunction);
163
181
  encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
164
182
  encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
165
183
  this.ecrRepository.grantPullPush(onEventHandlerFunction);
166
184
  this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
167
- // Create a custom resource provider
185
+ // Create a custom resource provider that uses the above Lambdas
168
186
  const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
169
187
  onEventHandler: onEventHandlerFunction,
170
188
  isCompleteHandler: isCompleteHandlerFunction,
171
189
  queryInterval: aws_cdk_lib_1.Duration.seconds(30),
172
190
  });
173
- // Define the custom resource
191
+ // Custom Resource that triggers the CodeBuild and waits for completion
174
192
  const buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
175
193
  serviceToken: provider.serviceToken,
176
194
  properties: {
177
195
  ProjectName: codeBuildProject.projectName,
196
+ ImageTag: imageTag,
178
197
  Trigger: crypto.randomUUID(),
179
198
  },
180
199
  });
181
200
  buildTriggerResource.node.addDependency(codeBuildProject);
182
- this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository);
183
- this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository);
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);
205
+ this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository, {
206
+ tagOrDigest: imageTagRef,
207
+ });
184
208
  }
185
209
  }
186
210
  exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
187
211
  _a = JSII_RTTI_SYMBOL_1;
188
- TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.2.7" };
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"]}
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":"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","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.2.7",
101
+ "version": "1.3.1",
100
102
  "jest": {
101
103
  "coverageProvider": "v8",
102
104
  "testMatch": [