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 +80 -29
- package/API.md +39 -37
- package/isComplete/isComplete.js +91 -68
- package/lib/index.d.ts +48 -35
- package/lib/index.js +86 -62
- package/package.json +6 -4
package/.jsii
CHANGED
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
]
|
|
8
8
|
},
|
|
9
9
|
"dependencies": {
|
|
10
|
-
"aws-cdk-lib": "^2.
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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
|
|
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":
|
|
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
|
|
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
|
|
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":
|
|
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
|
|
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":
|
|
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":
|
|
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":
|
|
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.
|
|
4182
|
-
"fingerprint": "
|
|
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> |
|
|
20
|
-
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.Initializer.parameter.id">id</a></code> | <code>string</code> |
|
|
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> |
|
|
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> |
|
|
91
|
-
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode">dockerImageCode</a></code> | <code>aws-cdk-lib.aws_lambda.DockerImageCode</code> |
|
|
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
|
|
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
|
|
233
|
-
```
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
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
|
|
261
|
-
```
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
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
|
|
288
|
+
These define the network access rules for the CodeBuild project.
|
|
287
289
|
|
|
288
290
|
---
|
|
289
291
|
|
package/isComplete/isComplete.js
CHANGED
|
@@ -1,97 +1,120 @@
|
|
|
1
|
-
const {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
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
|
|
21
|
+
const projectName = event.ResourceProperties?.ProjectName;
|
|
22
|
+
console.log('ProjectName from ResourceProperties:', projectName);
|
|
13
23
|
|
|
14
24
|
if (!projectName) {
|
|
15
|
-
throw new Error('ProjectName
|
|
25
|
+
throw new Error('Missing ProjectName in ResourceProperties');
|
|
16
26
|
}
|
|
17
27
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
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
|
-
|
|
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 =
|
|
35
|
-
console.log(`
|
|
49
|
+
const buildId = listResp.ids[0];
|
|
50
|
+
console.log(`Identified latest Build ID: ${buildId}`);
|
|
36
51
|
|
|
37
|
-
// Get build
|
|
38
|
-
const
|
|
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(`
|
|
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
|
-
|
|
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
|
-
}
|
|
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
|
-
|
|
78
|
-
|
|
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
|
-
|
|
81
|
-
|
|
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
|
-
|
|
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('
|
|
94
|
-
|
|
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
|
|
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
|
-
*
|
|
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
|
|
48
|
-
*
|
|
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
|
-
*
|
|
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
|
|
61
|
-
* ```
|
|
62
|
-
*
|
|
63
|
-
*
|
|
64
|
-
*
|
|
65
|
-
*
|
|
66
|
-
*
|
|
67
|
-
*
|
|
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
|
|
82
|
-
* ```
|
|
83
|
-
*
|
|
84
|
-
*
|
|
85
|
-
*
|
|
86
|
-
*
|
|
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
|
|
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
|
|
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,
|
|
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
|
|
52
|
+
encryptionKey,
|
|
41
53
|
imageScanOnPush: true,
|
|
42
54
|
});
|
|
43
|
-
//
|
|
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
|
-
//
|
|
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(([
|
|
68
|
+
.map(([k, v]) => `--build-arg ${k}=${v}`)
|
|
51
69
|
.join(' ')
|
|
52
70
|
: '';
|
|
53
|
-
//
|
|
71
|
+
// Optional DockerHub login, if a secret ARN is provided
|
|
54
72
|
const dockerLoginCommands = dockerLoginSecretArn
|
|
55
73
|
? [
|
|
56
|
-
'echo "Retrieving Docker credentials
|
|
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
|
|
63
|
-
|
|
64
|
-
|
|
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
|
|
133
|
+
// Grant CodeBuild the ability to interact with ECR
|
|
118
134
|
this.ecrRepository.grantPullPush(codeBuildProject);
|
|
119
|
-
codeBuildProject.role
|
|
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
|
|
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
|
-
//
|
|
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],
|
|
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
|
-
|
|
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
|
-
//
|
|
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
|
-
|
|
183
|
-
|
|
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.
|
|
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-
|
|
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.
|
|
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.
|
|
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.
|
|
101
|
+
"version": "1.3.1",
|
|
100
102
|
"jest": {
|
|
101
103
|
"coverageProvider": "v8",
|
|
102
104
|
"testMatch": [
|