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