token-injectable-docker-builder 1.3.0 → 1.3.2
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 -31
- package/API.md +36 -37
- package/LICENSE +1 -1
- package/isComplete/isComplete.js +91 -68
- package/lib/index.d.ts +43 -34
- package/lib/index.js +47 -41
- package/package.json +10 -8
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": {
|
|
@@ -3860,7 +3899,7 @@
|
|
|
3860
3899
|
"stability": "stable"
|
|
3861
3900
|
},
|
|
3862
3901
|
"homepage": "https://github.com/AlexTech314/TokenInjectableDockerBuilder.git",
|
|
3863
|
-
"jsiiVersion": "5.5.
|
|
3902
|
+
"jsiiVersion": "5.5.17 (build 2bbd2c1)",
|
|
3864
3903
|
"keywords": [
|
|
3865
3904
|
"aws",
|
|
3866
3905
|
"aws-cdk",
|
|
@@ -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,19 +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
4013
|
"stability": "stable",
|
|
3965
|
-
"summary": "An ECS-compatible
|
|
4014
|
+
"summary": "An ECS-compatible container image referencing the tag of the built Docker image."
|
|
3966
4015
|
},
|
|
3967
4016
|
"immutable": true,
|
|
3968
4017
|
"locationInModule": {
|
|
3969
4018
|
"filename": "src/index.ts",
|
|
3970
|
-
"line":
|
|
4019
|
+
"line": 123
|
|
3971
4020
|
},
|
|
3972
4021
|
"name": "containerImage",
|
|
3973
4022
|
"type": {
|
|
@@ -3977,12 +4026,12 @@
|
|
|
3977
4026
|
{
|
|
3978
4027
|
"docs": {
|
|
3979
4028
|
"stability": "stable",
|
|
3980
|
-
"summary": "A Lambda-compatible DockerImageCode referencing the
|
|
4029
|
+
"summary": "A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image."
|
|
3981
4030
|
},
|
|
3982
4031
|
"immutable": true,
|
|
3983
4032
|
"locationInModule": {
|
|
3984
4033
|
"filename": "src/index.ts",
|
|
3985
|
-
"line":
|
|
4034
|
+
"line": 129
|
|
3986
4035
|
},
|
|
3987
4036
|
"name": "dockerImageCode",
|
|
3988
4037
|
"type": {
|
|
@@ -4003,7 +4052,7 @@
|
|
|
4003
4052
|
"kind": "interface",
|
|
4004
4053
|
"locationInModule": {
|
|
4005
4054
|
"filename": "src/index.ts",
|
|
4006
|
-
"line":
|
|
4055
|
+
"line": 19
|
|
4007
4056
|
},
|
|
4008
4057
|
"name": "TokenInjectableDockerBuilderProps",
|
|
4009
4058
|
"properties": [
|
|
@@ -4016,7 +4065,7 @@
|
|
|
4016
4065
|
"immutable": true,
|
|
4017
4066
|
"locationInModule": {
|
|
4018
4067
|
"filename": "src/index.ts",
|
|
4019
|
-
"line":
|
|
4068
|
+
"line": 23
|
|
4020
4069
|
},
|
|
4021
4070
|
"name": "path",
|
|
4022
4071
|
"type": {
|
|
@@ -4027,14 +4076,14 @@
|
|
|
4027
4076
|
"abstract": true,
|
|
4028
4077
|
"docs": {
|
|
4029
4078
|
"example": "{\n TOKEN: 'my-secret-token',\n ENV: 'production'\n}",
|
|
4030
|
-
"remarks": "These are transformed into `--build-arg` flags.",
|
|
4079
|
+
"remarks": "These are transformed into `--build-arg KEY=VALUE` flags.",
|
|
4031
4080
|
"stability": "stable",
|
|
4032
4081
|
"summary": "Build arguments to pass to the Docker build process."
|
|
4033
4082
|
},
|
|
4034
4083
|
"immutable": true,
|
|
4035
4084
|
"locationInModule": {
|
|
4036
4085
|
"filename": "src/index.ts",
|
|
4037
|
-
"line":
|
|
4086
|
+
"line": 34
|
|
4038
4087
|
},
|
|
4039
4088
|
"name": "buildArgs",
|
|
4040
4089
|
"optional": true,
|
|
@@ -4051,14 +4100,14 @@
|
|
|
4051
4100
|
"abstract": true,
|
|
4052
4101
|
"docs": {
|
|
4053
4102
|
"example": "'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'",
|
|
4054
|
-
"remarks": "This secret should store a JSON object with the following structure:\n```json\n{\n \"username\": \"my-docker-username\",\n \"password\": \"my-docker-password\"\n}\n```\nIf not provided (or not needed), the construct will skip Docker Hub login.\
|
|
4103
|
+
"remarks": "This secret should store a JSON object with the following structure:\n```json\n{\n \"username\": \"my-docker-username\",\n \"password\": \"my-docker-password\"\n}\n```\nIf not provided (or not needed), the construct will skip Docker Hub login.\n\n**Note**: The secret must be in the same region as the stack.",
|
|
4055
4104
|
"stability": "stable",
|
|
4056
4105
|
"summary": "The ARN of the AWS Secrets Manager secret containing Docker login credentials."
|
|
4057
4106
|
},
|
|
4058
4107
|
"immutable": true,
|
|
4059
4108
|
"locationInModule": {
|
|
4060
4109
|
"filename": "src/index.ts",
|
|
4061
|
-
"line":
|
|
4110
|
+
"line": 51
|
|
4062
4111
|
},
|
|
4063
4112
|
"name": "dockerLoginSecretArn",
|
|
4064
4113
|
"optional": true,
|
|
@@ -4070,14 +4119,14 @@
|
|
|
4070
4119
|
"abstract": true,
|
|
4071
4120
|
"docs": {
|
|
4072
4121
|
"default": "- No additional install commands.",
|
|
4073
|
-
"remarks": "**Example
|
|
4122
|
+
"remarks": "**Example**:\n```ts\ninstallCommands: [\n 'echo \"Updating package lists...\"',\n 'apt-get update -y',\n 'echo \"Installing required packages...\"',\n 'apt-get install -y curl dnsutils',\n],\n```",
|
|
4074
4123
|
"stability": "stable",
|
|
4075
|
-
"summary": "Custom commands to run during the install phase."
|
|
4124
|
+
"summary": "Custom commands to run during the install phase of CodeBuild."
|
|
4076
4125
|
},
|
|
4077
4126
|
"immutable": true,
|
|
4078
4127
|
"locationInModule": {
|
|
4079
4128
|
"filename": "src/index.ts",
|
|
4080
|
-
"line":
|
|
4129
|
+
"line": 91
|
|
4081
4130
|
},
|
|
4082
4131
|
"name": "installCommands",
|
|
4083
4132
|
"optional": true,
|
|
@@ -4094,14 +4143,14 @@
|
|
|
4094
4143
|
"abstract": true,
|
|
4095
4144
|
"docs": {
|
|
4096
4145
|
"default": "- No additional pre-build commands.",
|
|
4097
|
-
"remarks": "**Example
|
|
4146
|
+
"remarks": "**Example**:\n```ts\npreBuildCommands: [\n 'echo \"Fetching configuration from private API...\"',\n 'curl -o config.json https://api.example.com/config',\n],\n```",
|
|
4098
4147
|
"stability": "stable",
|
|
4099
|
-
"summary": "Custom commands to run during the pre_build phase."
|
|
4148
|
+
"summary": "Custom commands to run during the pre_build phase of CodeBuild."
|
|
4100
4149
|
},
|
|
4101
4150
|
"immutable": true,
|
|
4102
4151
|
"locationInModule": {
|
|
4103
4152
|
"filename": "src/index.ts",
|
|
4104
|
-
"line":
|
|
4153
|
+
"line": 105
|
|
4105
4154
|
},
|
|
4106
4155
|
"name": "preBuildCommands",
|
|
4107
4156
|
"optional": true,
|
|
@@ -4117,15 +4166,15 @@
|
|
|
4117
4166
|
{
|
|
4118
4167
|
"abstract": true,
|
|
4119
4168
|
"docs": {
|
|
4120
|
-
"default": "No security groups are attached.",
|
|
4121
|
-
"remarks": "These
|
|
4169
|
+
"default": "- No security groups are attached.",
|
|
4170
|
+
"remarks": "These define the network access rules for the CodeBuild project.",
|
|
4122
4171
|
"stability": "stable",
|
|
4123
4172
|
"summary": "The security groups to attach to the CodeBuild project."
|
|
4124
4173
|
},
|
|
4125
4174
|
"immutable": true,
|
|
4126
4175
|
"locationInModule": {
|
|
4127
4176
|
"filename": "src/index.ts",
|
|
4128
|
-
"line":
|
|
4177
|
+
"line": 67
|
|
4129
4178
|
},
|
|
4130
4179
|
"name": "securityGroups",
|
|
4131
4180
|
"optional": true,
|
|
@@ -4141,7 +4190,7 @@
|
|
|
4141
4190
|
{
|
|
4142
4191
|
"abstract": true,
|
|
4143
4192
|
"docs": {
|
|
4144
|
-
"default": "All subnets in the VPC are used.",
|
|
4193
|
+
"default": "- All subnets in the VPC are used.",
|
|
4145
4194
|
"remarks": "Allows the user to select private, public, or isolated subnets.",
|
|
4146
4195
|
"stability": "stable",
|
|
4147
4196
|
"summary": "The subnet selection to specify which subnets to use within the VPC."
|
|
@@ -4149,7 +4198,7 @@
|
|
|
4149
4198
|
"immutable": true,
|
|
4150
4199
|
"locationInModule": {
|
|
4151
4200
|
"filename": "src/index.ts",
|
|
4152
|
-
"line":
|
|
4201
|
+
"line": 75
|
|
4153
4202
|
},
|
|
4154
4203
|
"name": "subnetSelection",
|
|
4155
4204
|
"optional": true,
|
|
@@ -4160,7 +4209,7 @@
|
|
|
4160
4209
|
{
|
|
4161
4210
|
"abstract": true,
|
|
4162
4211
|
"docs": {
|
|
4163
|
-
"default": "No VPC is attached, and the CodeBuild project will use public internet.",
|
|
4212
|
+
"default": "- No VPC is attached, and the CodeBuild project will use public internet.",
|
|
4164
4213
|
"remarks": "If provided, the CodeBuild project will be launched within the specified VPC.",
|
|
4165
4214
|
"stability": "stable",
|
|
4166
4215
|
"summary": "The VPC in which the CodeBuild project will be deployed."
|
|
@@ -4168,7 +4217,7 @@
|
|
|
4168
4217
|
"immutable": true,
|
|
4169
4218
|
"locationInModule": {
|
|
4170
4219
|
"filename": "src/index.ts",
|
|
4171
|
-
"line":
|
|
4220
|
+
"line": 59
|
|
4172
4221
|
},
|
|
4173
4222
|
"name": "vpc",
|
|
4174
4223
|
"optional": true,
|
|
@@ -4180,6 +4229,6 @@
|
|
|
4180
4229
|
"symbolId": "src/index:TokenInjectableDockerBuilderProps"
|
|
4181
4230
|
}
|
|
4182
4231
|
},
|
|
4183
|
-
"version": "1.3.
|
|
4184
|
-
"fingerprint": "
|
|
4232
|
+
"version": "1.3.2",
|
|
4233
|
+
"fingerprint": "wJDs0q3ApIhWraxUB0B+6KYlND78J8+CeIAXA0yZcS0="
|
|
4185
4234
|
}
|
package/API.md
CHANGED
|
@@ -4,7 +4,7 @@
|
|
|
4
4
|
|
|
5
5
|
### TokenInjectableDockerBuilder <a name="TokenInjectableDockerBuilder" id="token-injectable-docker-builder.TokenInjectableDockerBuilder"></a>
|
|
6
6
|
|
|
7
|
-
A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources,
|
|
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> | An ECS-compatible
|
|
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
|
|
96
|
+
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.containerImage">containerImage</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerImage</code> | An ECS-compatible container image referencing the tag of the built Docker image. |
|
|
97
|
+
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilder.property.dockerImageCode">dockerImageCode</a></code> | <code>aws-cdk-lib.aws_lambda.DockerImageCode</code> | A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image. |
|
|
92
98
|
|
|
93
99
|
---
|
|
94
100
|
|
|
@@ -112,7 +118,7 @@ public readonly containerImage: ContainerImage;
|
|
|
112
118
|
|
|
113
119
|
- *Type:* aws-cdk-lib.aws_ecs.ContainerImage
|
|
114
120
|
|
|
115
|
-
An ECS-compatible
|
|
121
|
+
An ECS-compatible container image referencing the tag of the built Docker image.
|
|
116
122
|
|
|
117
123
|
---
|
|
118
124
|
|
|
@@ -124,7 +130,7 @@ public readonly dockerImageCode: DockerImageCode;
|
|
|
124
130
|
|
|
125
131
|
- *Type:* aws-cdk-lib.aws_lambda.DockerImageCode
|
|
126
132
|
|
|
127
|
-
A Lambda-compatible DockerImageCode referencing the
|
|
133
|
+
A Lambda-compatible DockerImageCode referencing the the tag of the built Docker image.
|
|
128
134
|
|
|
129
135
|
---
|
|
130
136
|
|
|
@@ -150,8 +156,8 @@ const tokenInjectableDockerBuilderProps: TokenInjectableDockerBuilderProps = { .
|
|
|
150
156
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.path">path</a></code> | <code>string</code> | The path to the directory containing the Dockerfile or source code. |
|
|
151
157
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.buildArgs">buildArgs</a></code> | <code>{[ key: string ]: string}</code> | Build arguments to pass to the Docker build process. |
|
|
152
158
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.dockerLoginSecretArn">dockerLoginSecretArn</a></code> | <code>string</code> | The ARN of the AWS Secrets Manager secret containing Docker login credentials. |
|
|
153
|
-
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.installCommands">installCommands</a></code> | <code>string[]</code> | Custom commands to run during the install phase. |
|
|
154
|
-
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.preBuildCommands">preBuildCommands</a></code> | <code>string[]</code> | Custom commands to run during the pre_build phase. |
|
|
159
|
+
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.installCommands">installCommands</a></code> | <code>string[]</code> | Custom commands to run during the install phase of CodeBuild. |
|
|
160
|
+
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.preBuildCommands">preBuildCommands</a></code> | <code>string[]</code> | Custom commands to run during the pre_build phase of CodeBuild. |
|
|
155
161
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.securityGroups">securityGroups</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup[]</code> | The security groups to attach to the CodeBuild project. |
|
|
156
162
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.subnetSelection">subnetSelection</a></code> | <code>aws-cdk-lib.aws_ec2.SubnetSelection</code> | The subnet selection to specify which subnets to use within the VPC. |
|
|
157
163
|
| <code><a href="#token-injectable-docker-builder.TokenInjectableDockerBuilderProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | The VPC in which the CodeBuild project will be deployed. |
|
|
@@ -180,7 +186,7 @@ public readonly buildArgs: {[ key: string ]: string};
|
|
|
180
186
|
|
|
181
187
|
Build arguments to pass to the Docker build process.
|
|
182
188
|
|
|
183
|
-
These are transformed into `--build-arg` flags.
|
|
189
|
+
These are transformed into `--build-arg KEY=VALUE` flags.
|
|
184
190
|
|
|
185
191
|
---
|
|
186
192
|
|
|
@@ -212,7 +218,8 @@ This secret should store a JSON object with the following structure:
|
|
|
212
218
|
}
|
|
213
219
|
```
|
|
214
220
|
If not provided (or not needed), the construct will skip Docker Hub login.
|
|
215
|
-
|
|
221
|
+
|
|
222
|
+
**Note**: The secret must be in the same region as the stack.
|
|
216
223
|
|
|
217
224
|
---
|
|
218
225
|
|
|
@@ -232,20 +239,16 @@ public readonly installCommands: string[];
|
|
|
232
239
|
- *Type:* string[]
|
|
233
240
|
- *Default:* No additional install commands.
|
|
234
241
|
|
|
235
|
-
Custom commands to run during the install phase.
|
|
242
|
+
Custom commands to run during the install phase of CodeBuild.
|
|
236
243
|
|
|
237
|
-
**Example
|
|
238
|
-
```
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
'apt-get install -y curl dnsutils',
|
|
246
|
-
],
|
|
247
|
-
// ... other properties ...
|
|
248
|
-
});
|
|
244
|
+
**Example**:
|
|
245
|
+
```ts
|
|
246
|
+
installCommands: [
|
|
247
|
+
'echo "Updating package lists..."',
|
|
248
|
+
'apt-get update -y',
|
|
249
|
+
'echo "Installing required packages..."',
|
|
250
|
+
'apt-get install -y curl dnsutils',
|
|
251
|
+
],
|
|
249
252
|
```
|
|
250
253
|
|
|
251
254
|
---
|
|
@@ -259,18 +262,14 @@ public readonly preBuildCommands: string[];
|
|
|
259
262
|
- *Type:* string[]
|
|
260
263
|
- *Default:* No additional pre-build commands.
|
|
261
264
|
|
|
262
|
-
Custom commands to run during the pre_build phase.
|
|
265
|
+
Custom commands to run during the pre_build phase of CodeBuild.
|
|
263
266
|
|
|
264
|
-
**Example
|
|
265
|
-
```
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
'curl -o config.json https://api.example.com/config',
|
|
271
|
-
],
|
|
272
|
-
// ... other properties ...
|
|
273
|
-
});
|
|
267
|
+
**Example**:
|
|
268
|
+
```ts
|
|
269
|
+
preBuildCommands: [
|
|
270
|
+
'echo "Fetching configuration from private API..."',
|
|
271
|
+
'curl -o config.json https://api.example.com/config',
|
|
272
|
+
],
|
|
274
273
|
```
|
|
275
274
|
|
|
276
275
|
---
|
|
@@ -286,7 +285,7 @@ public readonly securityGroups: ISecurityGroup[];
|
|
|
286
285
|
|
|
287
286
|
The security groups to attach to the CodeBuild project.
|
|
288
287
|
|
|
289
|
-
These
|
|
288
|
+
These define the network access rules for the CodeBuild project.
|
|
290
289
|
|
|
291
290
|
---
|
|
292
291
|
|
package/LICENSE
CHANGED
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',
|
|
@@ -32,7 +32,8 @@ export interface TokenInjectableDockerBuilderProps {
|
|
|
32
32
|
* }
|
|
33
33
|
* ```
|
|
34
34
|
* If not provided (or not needed), the construct will skip Docker Hub login.
|
|
35
|
-
*
|
|
35
|
+
*
|
|
36
|
+
* **Note**: The secret must be in the same region as the stack.
|
|
36
37
|
*
|
|
37
38
|
* @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'
|
|
38
39
|
*/
|
|
@@ -40,71 +41,79 @@ export interface TokenInjectableDockerBuilderProps {
|
|
|
40
41
|
/**
|
|
41
42
|
* The VPC in which the CodeBuild project will be deployed.
|
|
42
43
|
* If provided, the CodeBuild project will be launched within the specified VPC.
|
|
43
|
-
*
|
|
44
|
+
*
|
|
45
|
+
* @default - No VPC is attached, and the CodeBuild project will use public internet.
|
|
44
46
|
*/
|
|
45
47
|
readonly vpc?: IVpc;
|
|
46
48
|
/**
|
|
47
49
|
* The security groups to attach to the CodeBuild project.
|
|
48
|
-
* These
|
|
49
|
-
*
|
|
50
|
+
* These define the network access rules for the CodeBuild project.
|
|
51
|
+
*
|
|
52
|
+
* @default - No security groups are attached.
|
|
50
53
|
*/
|
|
51
54
|
readonly securityGroups?: ISecurityGroup[];
|
|
52
55
|
/**
|
|
53
56
|
* The subnet selection to specify which subnets to use within the VPC.
|
|
54
57
|
* Allows the user to select private, public, or isolated subnets.
|
|
55
|
-
*
|
|
58
|
+
*
|
|
59
|
+
* @default - All subnets in the VPC are used.
|
|
56
60
|
*/
|
|
57
61
|
readonly subnetSelection?: SubnetSelection;
|
|
58
62
|
/**
|
|
59
|
-
* Custom commands to run during the install phase.
|
|
63
|
+
* Custom commands to run during the install phase of CodeBuild.
|
|
60
64
|
*
|
|
61
|
-
* **Example
|
|
62
|
-
* ```
|
|
63
|
-
*
|
|
64
|
-
*
|
|
65
|
-
*
|
|
66
|
-
*
|
|
67
|
-
*
|
|
68
|
-
*
|
|
69
|
-
* 'apt-get install -y curl dnsutils',
|
|
70
|
-
* ],
|
|
71
|
-
* // ... other properties ...
|
|
72
|
-
* });
|
|
65
|
+
* **Example**:
|
|
66
|
+
* ```ts
|
|
67
|
+
* installCommands: [
|
|
68
|
+
* 'echo "Updating package lists..."',
|
|
69
|
+
* 'apt-get update -y',
|
|
70
|
+
* 'echo "Installing required packages..."',
|
|
71
|
+
* 'apt-get install -y curl dnsutils',
|
|
72
|
+
* ],
|
|
73
73
|
* ```
|
|
74
74
|
* @default - No additional install commands.
|
|
75
75
|
*/
|
|
76
76
|
readonly installCommands?: string[];
|
|
77
77
|
/**
|
|
78
|
-
* Custom commands to run during the pre_build phase.
|
|
78
|
+
* Custom commands to run during the pre_build phase of CodeBuild.
|
|
79
79
|
*
|
|
80
|
-
* **Example
|
|
81
|
-
* ```
|
|
82
|
-
*
|
|
83
|
-
*
|
|
84
|
-
*
|
|
85
|
-
*
|
|
86
|
-
* 'curl -o config.json https://api.example.com/config',
|
|
87
|
-
* ],
|
|
88
|
-
* // ... other properties ...
|
|
89
|
-
* });
|
|
80
|
+
* **Example**:
|
|
81
|
+
* ```ts
|
|
82
|
+
* preBuildCommands: [
|
|
83
|
+
* 'echo "Fetching configuration from private API..."',
|
|
84
|
+
* 'curl -o config.json https://api.example.com/config',
|
|
85
|
+
* ],
|
|
90
86
|
* ```
|
|
91
87
|
* @default - No additional pre-build commands.
|
|
92
88
|
*/
|
|
93
89
|
readonly preBuildCommands?: string[];
|
|
94
90
|
}
|
|
95
91
|
/**
|
|
96
|
-
* A CDK construct to build and push Docker images to an ECR repository using
|
|
97
|
-
*
|
|
92
|
+
* A CDK construct to build and push Docker images to an ECR repository using
|
|
93
|
+
* CodeBuild and Lambda custom resources, **then** retrieve the final image tag
|
|
94
|
+
* so that ECS/Lambda references use the exact digest.
|
|
98
95
|
*/
|
|
99
96
|
export declare class TokenInjectableDockerBuilder extends Construct {
|
|
97
|
+
/**
|
|
98
|
+
* The ECR repository that stores the resulting Docker image.
|
|
99
|
+
*/
|
|
100
100
|
private readonly ecrRepository;
|
|
101
101
|
/**
|
|
102
|
-
* An ECS-compatible
|
|
102
|
+
* An ECS-compatible container image referencing the tag
|
|
103
|
+
* of the built Docker image.
|
|
103
104
|
*/
|
|
104
105
|
readonly containerImage: ContainerImage;
|
|
105
106
|
/**
|
|
106
|
-
* A Lambda-compatible DockerImageCode referencing the
|
|
107
|
+
* A Lambda-compatible DockerImageCode referencing the the tag
|
|
108
|
+
* of the built Docker image.
|
|
107
109
|
*/
|
|
108
110
|
readonly dockerImageCode: DockerImageCode;
|
|
111
|
+
/**
|
|
112
|
+
* Creates a new `TokenInjectableDockerBuilder`.
|
|
113
|
+
*
|
|
114
|
+
* @param scope The scope in which to define this construct.
|
|
115
|
+
* @param id The scoped construct ID.
|
|
116
|
+
* @param props Configuration for building and pushing the Docker image.
|
|
117
|
+
*/
|
|
109
118
|
constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
|
|
110
119
|
}
|
package/lib/index.js
CHANGED
|
@@ -12,24 +12,33 @@ const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
|
|
|
12
12
|
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
13
13
|
const aws_kms_1 = require("aws-cdk-lib/aws-kms");
|
|
14
14
|
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
15
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
15
16
|
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
16
17
|
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
|
|
17
18
|
const constructs_1 = require("constructs");
|
|
18
19
|
/**
|
|
19
|
-
* A CDK construct to build and push Docker images to an ECR repository using
|
|
20
|
-
*
|
|
20
|
+
* A CDK construct to build and push Docker images to an ECR repository using
|
|
21
|
+
* CodeBuild and Lambda custom resources, **then** retrieve the final image tag
|
|
22
|
+
* so that ECS/Lambda references use the exact digest.
|
|
21
23
|
*/
|
|
22
24
|
class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
25
|
+
/**
|
|
26
|
+
* Creates a new `TokenInjectableDockerBuilder`.
|
|
27
|
+
*
|
|
28
|
+
* @param scope The scope in which to define this construct.
|
|
29
|
+
* @param id The scoped construct ID.
|
|
30
|
+
* @param props Configuration for building and pushing the Docker image.
|
|
31
|
+
*/
|
|
23
32
|
constructor(scope, id, props) {
|
|
24
33
|
super(scope, id);
|
|
25
34
|
const { path: sourcePath, buildArgs, dockerLoginSecretArn, vpc, securityGroups, subnetSelection, installCommands, preBuildCommands, } = props;
|
|
26
|
-
// Generate
|
|
35
|
+
// Generate an ephemeral tag for CodeBuild
|
|
27
36
|
const imageTag = crypto.randomUUID();
|
|
28
|
-
// KMS key for ECR encryption
|
|
37
|
+
// Define a KMS key for ECR encryption
|
|
29
38
|
const encryptionKey = new aws_kms_1.Key(this, 'EcrEncryptionKey', {
|
|
30
39
|
enableKeyRotation: true,
|
|
31
40
|
});
|
|
32
|
-
// ECR repository
|
|
41
|
+
// Create an ECR repository with encryption, lifecycle rules, and image scanning
|
|
33
42
|
this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository', {
|
|
34
43
|
lifecycleRules: [
|
|
35
44
|
{
|
|
@@ -43,70 +52,67 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
43
52
|
encryptionKey,
|
|
44
53
|
imageScanOnPush: true,
|
|
45
54
|
});
|
|
46
|
-
//
|
|
55
|
+
// Wrap the source folder as an S3 asset for CodeBuild to use
|
|
47
56
|
const sourceAsset = new aws_s3_assets_1.Asset(this, 'SourceAsset', {
|
|
48
57
|
path: sourcePath,
|
|
49
58
|
});
|
|
50
|
-
//
|
|
59
|
+
// Create an S3 bucket to store the CodeBuild artifacts
|
|
60
|
+
const artifactBucket = new aws_s3_1.Bucket(this, 'ArtifactBucket', {
|
|
61
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
62
|
+
autoDeleteObjects: true,
|
|
63
|
+
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
|
|
64
|
+
});
|
|
65
|
+
// Convert buildArgs to a CLI-friendly string
|
|
51
66
|
const buildArgsString = buildArgs
|
|
52
67
|
? Object.entries(buildArgs)
|
|
53
68
|
.map(([k, v]) => `--build-arg ${k}=${v}`)
|
|
54
69
|
.join(' ')
|
|
55
70
|
: '';
|
|
56
|
-
//
|
|
71
|
+
// Optional DockerHub login, if a secret ARN is provided
|
|
57
72
|
const dockerLoginCommands = dockerLoginSecretArn
|
|
58
73
|
? [
|
|
59
|
-
'echo "Retrieving Docker credentials
|
|
74
|
+
'echo "Retrieving Docker credentials..."',
|
|
60
75
|
'apt-get update -y && apt-get install -y jq',
|
|
61
76
|
`DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,
|
|
62
77
|
`DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,
|
|
63
78
|
'echo "Logging in to Docker Hub..."',
|
|
64
|
-
// Use non-stdin login to avoid TTY error:
|
|
65
79
|
'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',
|
|
66
80
|
]
|
|
67
|
-
: ['echo "No Docker credentials
|
|
68
|
-
// BuildSpec
|
|
81
|
+
: ['echo "No Docker credentials. Skipping Docker Hub login."'];
|
|
69
82
|
const buildSpecObj = {
|
|
70
83
|
version: '0.2',
|
|
71
84
|
phases: {
|
|
72
85
|
install: {
|
|
73
86
|
commands: [
|
|
74
87
|
'echo "Beginning install phase..."',
|
|
75
|
-
...(installCommands
|
|
88
|
+
...(installCommands ?? []),
|
|
76
89
|
],
|
|
77
90
|
},
|
|
78
91
|
pre_build: {
|
|
79
92
|
commands: [
|
|
80
|
-
...(preBuildCommands
|
|
93
|
+
...(preBuildCommands ?? []),
|
|
81
94
|
...dockerLoginCommands,
|
|
82
95
|
'echo "Retrieving AWS Account ID..."',
|
|
83
96
|
'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
|
|
84
|
-
'echo "Logging
|
|
97
|
+
'echo "Logging into Amazon ECR..."',
|
|
85
98
|
'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',
|
|
86
99
|
],
|
|
87
100
|
},
|
|
88
101
|
build: {
|
|
89
102
|
commands: [
|
|
90
|
-
|
|
103
|
+
`echo "Building Docker image with tag ${imageTag}..."`,
|
|
91
104
|
`docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,
|
|
92
105
|
],
|
|
93
106
|
},
|
|
94
107
|
post_build: {
|
|
95
108
|
commands: [
|
|
96
|
-
`echo "
|
|
109
|
+
`echo "Pushing Docker image with tag ${imageTag}..."`,
|
|
97
110
|
`docker push $ECR_REPO_URI:${imageTag}`,
|
|
98
|
-
`export IMAGE_DIGEST=$(docker inspect --format='{{index .RepoDigests 0}}' $ECR_REPO_URI:${imageTag})`,
|
|
99
|
-
'echo "Image digest: $IMAGE_DIGEST"',
|
|
100
|
-
'echo "{ \\"ImageDigest\\": \\"$IMAGE_DIGEST\\" }" > imageDetail.json',
|
|
101
111
|
],
|
|
102
112
|
},
|
|
103
113
|
},
|
|
104
|
-
artifacts: {
|
|
105
|
-
files: ['imageDetail.json'],
|
|
106
|
-
name: 'imageDetail',
|
|
107
|
-
},
|
|
108
114
|
};
|
|
109
|
-
// CodeBuild project
|
|
115
|
+
// Create the CodeBuild project
|
|
110
116
|
const codeBuildProject = new aws_codebuild_1.Project(this, 'CodeBuildProject', {
|
|
111
117
|
source: aws_codebuild_1.Source.s3({
|
|
112
118
|
bucket: sourceAsset.bucket,
|
|
@@ -119,12 +125,12 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
119
125
|
environmentVariables: {
|
|
120
126
|
ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },
|
|
121
127
|
},
|
|
128
|
+
buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
|
|
122
129
|
vpc,
|
|
123
130
|
securityGroups,
|
|
124
131
|
subnetSelection,
|
|
125
|
-
buildSpec: aws_codebuild_1.BuildSpec.fromObject(buildSpecObj),
|
|
126
132
|
});
|
|
127
|
-
//
|
|
133
|
+
// Grant CodeBuild the ability to interact with ECR
|
|
128
134
|
this.ecrRepository.grantPullPush(codeBuildProject);
|
|
129
135
|
codeBuildProject.role?.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
130
136
|
actions: [
|
|
@@ -141,7 +147,7 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
141
147
|
}));
|
|
142
148
|
}
|
|
143
149
|
encryptionKey.grantEncryptDecrypt(codeBuildProject.role);
|
|
144
|
-
//
|
|
150
|
+
// Define the Lambda functions for custom resource event and completion handling
|
|
145
151
|
const onEventHandlerFunction = new aws_lambda_1.Function(this, 'OnEventHandlerFunction', {
|
|
146
152
|
runtime: aws_lambda_1.Runtime.NODEJS_18_X,
|
|
147
153
|
code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../onEvent')),
|
|
@@ -152,10 +158,12 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
152
158
|
actions: ['codebuild:StartBuild'],
|
|
153
159
|
resources: [codeBuildProject.projectArn],
|
|
154
160
|
}));
|
|
155
|
-
// isComplete handler
|
|
156
161
|
const isCompleteHandlerFunction = new aws_lambda_1.Function(this, 'IsCompleteHandlerFunction', {
|
|
157
162
|
runtime: aws_lambda_1.Runtime.NODEJS_18_X,
|
|
158
163
|
code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '../isComplete')),
|
|
164
|
+
environment: {
|
|
165
|
+
IMAGE_TAG: imageTag,
|
|
166
|
+
},
|
|
159
167
|
handler: 'isComplete.handler',
|
|
160
168
|
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
161
169
|
});
|
|
@@ -166,22 +174,21 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
166
174
|
'logs:GetLogEvents',
|
|
167
175
|
'logs:DescribeLogStreams',
|
|
168
176
|
'logs:DescribeLogGroups',
|
|
169
|
-
's3:GetObject',
|
|
170
|
-
's3:GetBucketLocation',
|
|
171
177
|
],
|
|
172
178
|
resources: ['*'],
|
|
173
179
|
}));
|
|
180
|
+
artifactBucket.grantReadWrite(isCompleteHandlerFunction);
|
|
174
181
|
encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);
|
|
175
182
|
encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);
|
|
176
183
|
this.ecrRepository.grantPullPush(onEventHandlerFunction);
|
|
177
184
|
this.ecrRepository.grantPullPush(isCompleteHandlerFunction);
|
|
178
|
-
//
|
|
185
|
+
// Create a custom resource provider that uses the above Lambdas
|
|
179
186
|
const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
|
|
180
187
|
onEventHandler: onEventHandlerFunction,
|
|
181
188
|
isCompleteHandler: isCompleteHandlerFunction,
|
|
182
189
|
queryInterval: aws_cdk_lib_1.Duration.seconds(30),
|
|
183
190
|
});
|
|
184
|
-
// Custom
|
|
191
|
+
// Custom Resource that triggers the CodeBuild and waits for completion
|
|
185
192
|
const buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
|
|
186
193
|
serviceToken: provider.serviceToken,
|
|
187
194
|
properties: {
|
|
@@ -191,17 +198,16 @@ class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
|
191
198
|
},
|
|
192
199
|
});
|
|
193
200
|
buildTriggerResource.node.addDependency(codeBuildProject);
|
|
194
|
-
//
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository,
|
|
198
|
-
// Lambda-compatible from ECR by digest
|
|
201
|
+
// Retrieve the final Docker image tag from Data.ImageTag
|
|
202
|
+
// This creates a dependency on the Custom Resource...
|
|
203
|
+
const imageTagRef = buildTriggerResource.getAttString('ImageTag');
|
|
204
|
+
this.containerImage = aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository, imageTagRef);
|
|
199
205
|
this.dockerImageCode = aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository, {
|
|
200
|
-
tagOrDigest:
|
|
206
|
+
tagOrDigest: imageTagRef,
|
|
201
207
|
});
|
|
202
208
|
}
|
|
203
209
|
}
|
|
204
210
|
exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
|
|
205
211
|
_a = JSII_RTTI_SYMBOL_1;
|
|
206
|
-
TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.3.
|
|
207
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as path from 'path';\nimport { CustomResource, Duration } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { IVpc, ISecurityGroup, SubnetSelection } from 'aws-cdk-lib/aws-ec2';\nimport { Repository, RepositoryEncryption, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\n/**\n * Properties for the `TokenInjectableDockerBuilder` construct.\n */\nexport interface TokenInjectableDockerBuilderProps {\n  /**\n   * The path to the directory containing the Dockerfile or source code.\n   */\n  readonly path: string;\n\n  /**\n   * Build arguments to pass to the Docker build process.\n   * These are transformed into `--build-arg` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n\n  /**\n   * The ARN of the AWS Secrets Manager secret containing Docker login credentials.\n   * This secret should store a JSON object with the following structure:\n   * ```json\n   * {\n   *   \"username\": \"my-docker-username\",\n   *   \"password\": \"my-docker-password\"\n   * }\n   * ```\n   * If not provided (or not needed), the construct will skip Docker Hub login.\n   * NOTE: The secret must be in the same region as the stack.\n   *\n   * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'\n   */\n  readonly dockerLoginSecretArn?: string;\n\n  /**\n   * The VPC in which the CodeBuild project will be deployed.\n   * If provided, the CodeBuild project will be launched within the specified VPC.\n   * @default No VPC is attached, and the CodeBuild project will use public internet.\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * The security groups to attach to the CodeBuild project.\n   * These should define the network access rules for the CodeBuild project.\n   * @default No security groups are attached.\n   */\n  readonly securityGroups?: ISecurityGroup[];\n\n  /**\n   * The subnet selection to specify which subnets to use within the VPC.\n   * Allows the user to select private, public, or isolated subnets.\n   * @default All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\n   *   installCommands: [\n   *     'echo \"Updating package lists...\"',\n   *     'apt-get update -y',\n   *     'echo \"Installing required packages...\"',\n   *     'apt-get install -y curl dnsutils',\n   *   ],\n   *   // ... other properties ...\n   * });\n   * ```\n   * @default - No additional install commands.\n   */\n  readonly installCommands?: string[];\n\n  /**\n   * Custom commands to run during the pre_build phase.\n   *\n   * **Example Usage:**\n   * ```typescript\n   * new TokenInjectableDockerBuilder(this, 'MyDockerBuilder', {\n   *   path: path.resolve(__dirname, '../app'),\n   *   preBuildCommands: [\n   *     'echo \"Fetching configuration from private API...\"',\n   *     'curl -o config.json https://api.example.com/config',\n   *   ],\n   *   // ... other properties ...\n   * });\n   * ```\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using CodeBuild and Lambda custom resources,\n * retrieving the final image digest (SHA) and using that exact digest for ECS or Lambda references.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  private readonly ecrRepository: Repository;\n\n  /**\n   * An ECS-compatible ContainerImage referencing the *exact* SHA digest of the built Docker image.\n   */\n  public readonly containerImage: ContainerImage;\n\n  /**\n   * A Lambda-compatible DockerImageCode referencing the *exact* SHA digest of the built Docker image.\n   */\n  public readonly dockerImageCode: DockerImageCode;\n\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const {\n      path: sourcePath,\n      buildArgs,\n      dockerLoginSecretArn,\n      vpc,\n      securityGroups,\n      subnetSelection,\n      installCommands,\n      preBuildCommands,\n    } = props;\n\n    // Generate a unique tag for this build.\n    const imageTag = crypto.randomUUID();\n\n    // KMS key for ECR encryption\n    const encryptionKey = new Key(this, 'EcrEncryptionKey', {\n      enableKeyRotation: true,\n    });\n\n    // ECR repository\n    this.ecrRepository = new Repository(this, 'ECRRepository', {\n      lifecycleRules: [\n        {\n          rulePriority: 1,\n          description: 'Remove untagged images after 1 day',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n      encryption: RepositoryEncryption.KMS,\n      encryptionKey,\n      imageScanOnPush: true,\n    });\n\n    // Package source code\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n    });\n\n    // Build args\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([k, v]) => `--build-arg ${k}=${v}`)\n        .join(' ')\n      : '';\n\n    // Docker Hub login commands\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials from Secrets Manager...\"',\n        'apt-get update -y && apt-get install -y jq',\n        `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,\n        `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,\n        'echo \"Logging in to Docker Hub...\"',\n        // Use non-stdin login to avoid TTY error:\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials provided. Skipping Docker Hub login.\"'];\n\n    // BuildSpec\n    const buildSpecObj = {\n      version: '0.2',\n      phases: {\n        install: {\n          commands: [\n            'echo \"Beginning install phase...\"',\n            ...(installCommands || []),\n          ],\n        },\n        pre_build: {\n          commands: [\n            ...(preBuildCommands || []),\n            ...dockerLoginCommands,\n            'echo \"Retrieving AWS Account ID...\"',\n            'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n            'echo \"Logging in to Amazon ECR...\"',\n            'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',\n          ],\n        },\n        build: {\n          commands: [\n            'echo \"Build phase: Building the Docker image...\"',\n            `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,\n          ],\n        },\n        post_build: {\n          commands: [\n            `echo \"Post-build phase: Pushing the Docker image with tag ${imageTag}...\"`,\n            `docker push $ECR_REPO_URI:${imageTag}`,\n            `export IMAGE_DIGEST=$(docker inspect --format='{{index .RepoDigests 0}}' $ECR_REPO_URI:${imageTag})`,\n            'echo \"Image digest: $IMAGE_DIGEST\"',\n            'echo \"{ \\\\\"ImageDigest\\\\\": \\\\\"$IMAGE_DIGEST\\\\\" }\" > imageDetail.json',\n          ],\n        },\n      },\n      artifacts: {\n        files: ['imageDetail.json'],\n        name: 'imageDetail',\n      },\n    };\n\n    // CodeBuild project\n    const codeBuildProject = new Project(this, 'CodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true,\n      },\n      environmentVariables: {\n        ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      },\n      vpc,\n      securityGroups,\n      subnetSelection,\n      buildSpec: BuildSpec.fromObject(buildSpecObj),\n    });\n\n    // Permissions\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.role?.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: [\n          'ecr:GetAuthorizationToken',\n          'ecr:GetDownloadUrlForLayer',\n          'ecr:BatchCheckLayerAvailability',\n        ],\n        resources: [this.ecrRepository.repositoryArn],\n      }),\n    );\n\n    if (dockerLoginSecretArn) {\n      codeBuildProject.role?.addToPrincipalPolicy(\n        new PolicyStatement({\n          actions: ['secretsmanager:GetSecretValue'],\n          resources: [dockerLoginSecretArn],\n        }),\n      );\n    }\n\n    encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n\n    // onEvent handler\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn],\n      }),\n    );\n\n    // isComplete handler\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      handler: 'isComplete.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    isCompleteHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'codebuild:BatchGetBuilds',\n          'codebuild:ListBuildsForProject',\n          'logs:GetLogEvents',\n          'logs:DescribeLogStreams',\n          'logs:DescribeLogGroups',\n          's3:GetObject',\n          's3:GetBucketLocation',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n    encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n    this.ecrRepository.grantPullPush(onEventHandlerFunction);\n    this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\n\n    // Provider\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.seconds(30),\n    });\n\n    // Custom resource\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        ImageTag: imageTag,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Grab the SHA from the custom resource's response (set in isComplete handler)\n    const imageDigest = buildTriggerResource.getAttString('ImageDigest');\n\n    // ECS-compatible from ECR by digest\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository, imageDigest);\n\n    // Lambda-compatible from ECR by digest\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository, {\n      tagOrDigest: imageDigest,\n    });\n  }\n}\n"]}
|
|
212
|
+
TokenInjectableDockerBuilder[_a] = { fqn: "token-injectable-docker-builder.TokenInjectableDockerBuilder", version: "1.3.2" };
|
|
213
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;;;AAAA,iCAAiC;AACjC,6BAA6B;AAC7B,6CAAsE;AACtE,6DAAwF;AAExF,iDAAkF;AAClF,iDAAqD;AACrD,iDAAsD;AACtD,iDAA0C;AAC1C,uDAAkF;AAClF,+CAA+D;AAC/D,6DAAkD;AAClD,mEAAwD;AACxD,2CAAuC;AA8FvC;;;;GAIG;AACH,MAAa,4BAA6B,SAAQ,sBAAS;IAkBzD;;;;;;OAMG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAwC;QAChF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,EACJ,IAAI,EAAE,UAAU,EAChB,SAAS,EACT,oBAAoB,EACpB,GAAG,EACH,cAAc,EACd,eAAe,EACf,eAAe,EACf,gBAAgB,GACjB,GAAG,KAAK,CAAC;QAEV,0CAA0C;QAC1C,MAAM,QAAQ,GAAG,MAAM,CAAC,UAAU,EAAE,CAAC;QAErC,sCAAsC;QACtC,MAAM,aAAa,GAAG,IAAI,aAAG,CAAC,IAAI,EAAE,kBAAkB,EAAE;YACtD,iBAAiB,EAAE,IAAI;SACxB,CAAC,CAAC;QAEH,gFAAgF;QAChF,IAAI,CAAC,aAAa,GAAG,IAAI,oBAAU,CAAC,IAAI,EAAE,eAAe,EAAE;YACzD,cAAc,EAAE;gBACd;oBACE,YAAY,EAAE,CAAC;oBACf,WAAW,EAAE,oCAAoC;oBACjD,SAAS,EAAE,mBAAS,CAAC,QAAQ;oBAC7B,WAAW,EAAE,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;iBAC9B;aACF;YACD,UAAU,EAAE,8BAAoB,CAAC,GAAG;YACpC,aAAa;YACb,eAAe,EAAE,IAAI;SACtB,CAAC,CAAC;QAEH,6DAA6D;QAC7D,MAAM,WAAW,GAAG,IAAI,qBAAK,CAAC,IAAI,EAAE,aAAa,EAAE;YACjD,IAAI,EAAE,UAAU;SACjB,CAAC,CAAC;QAEH,uDAAuD;QACvD,MAAM,cAAc,GAAG,IAAI,eAAM,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACxD,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,iBAAiB,EAAE,0BAAiB,CAAC,SAAS;SAC/C,CAAC,CAAC;QAEH,6CAA6C;QAC7C,MAAM,eAAe,GAAG,SAAS;YAC/B,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC;iBACxB,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;iBACxC,IAAI,CAAC,GAAG,CAAC;YACZ,CAAC,CAAC,EAAE,CAAC;QAEP,wDAAwD;QACxD,MAAM,mBAAmB,GAAG,oBAAoB;YAC9C,CAAC,CAAC;gBACA,yCAAyC;gBACzC,4CAA4C;gBAC5C,qEAAqE,oBAAoB,wDAAwD;gBACjJ,qEAAqE,oBAAoB,wDAAwD;gBACjJ,oCAAoC;gBACpC,mFAAmF;aACpF;YACD,CAAC,CAAC,CAAC,0DAA0D,CAAC,CAAC;QAEjE,MAAM,YAAY,GAAG;YACnB,OAAO,EAAE,KAAK;YACd,MAAM,EAAE;gBACN,OAAO,EAAE;oBACP,QAAQ,EAAE;wBACR,mCAAmC;wBACnC,GAAG,CAAC,eAAe,IAAI,EAAE,CAAC;qBAC3B;iBACF;gBACD,SAAS,EAAE;oBACT,QAAQ,EAAE;wBACR,GAAG,CAAC,gBAAgB,IAAI,EAAE,CAAC;wBAC3B,GAAG,mBAAmB;wBACtB,qCAAqC;wBACrC,gFAAgF;wBAChF,mCAAmC;wBACnC,8JAA8J;qBAC/J;iBACF;gBACD,KAAK,EAAE;oBACL,QAAQ,EAAE;wBACR,wCAAwC,QAAQ,MAAM;wBACtD,gBAAgB,eAAe,qBAAqB,QAAQ,qBAAqB;qBAClF;iBACF;gBACD,UAAU,EAAE;oBACV,QAAQ,EAAE;wBACR,uCAAuC,QAAQ,MAAM;wBACrD,6BAA6B,QAAQ,EAAE;qBACxC;iBACF;aACF;SACF,CAAC;QAGF,+BAA+B;QAC/B,MAAM,gBAAgB,GAAG,IAAI,uBAAO,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC7D,MAAM,EAAE,sBAAM,CAAC,EAAE,CAAC;gBAChB,MAAM,EAAE,WAAW,CAAC,MAAM;gBAC1B,IAAI,EAAE,WAAW,CAAC,WAAW;aAC9B,CAAC;YACF,WAAW,EAAE;gBACX,UAAU,EAAE,+BAAe,CAAC,YAAY;gBACxC,UAAU,EAAE,IAAI;aACjB;YACD,oBAAoB,EAAE;gBACpB,YAAY,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,aAAa,CAAC,aAAa,EAAE;aAC1D;YACD,SAAS,EAAE,yBAAS,CAAC,UAAU,CAAC,YAAY,CAAC;YAC7C,GAAG;YACH,cAAc;YACd,eAAe;SAChB,CAAC,CAAC;QAEH,mDAAmD;QACnD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAC;QACnD,gBAAgB,CAAC,IAAI,EAAE,oBAAoB,CAAC,IAAI,yBAAe,CAAC;YAC9D,OAAO,EAAE;gBACP,2BAA2B;gBAC3B,4BAA4B;gBAC5B,iCAAiC;aAClC;YACD,SAAS,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;SAC9C,CAAC,CAAC,CAAC;QACJ,IAAI,oBAAoB,EAAE,CAAC;YACzB,gBAAgB,CAAC,IAAI,EAAE,oBAAoB,CAAC,IAAI,yBAAe,CAAC;gBAC9D,OAAO,EAAE,CAAC,+BAA+B,CAAC;gBAC1C,SAAS,EAAE,CAAC,oBAAoB,CAAC;aAClC,CAAC,CAAC,CAAC;QACN,CAAC;QACD,aAAa,CAAC,mBAAmB,CAAC,gBAAgB,CAAC,IAAK,CAAC,CAAC;QAE1D,gFAAgF;QAChF,MAAM,sBAAsB,GAAG,IAAI,qBAAQ,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC1E,OAAO,EAAE,oBAAO,CAAC,WAAW;YAC5B,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;YAC3D,OAAO,EAAE,iBAAiB;YAC1B,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SAC9B,CAAC,CAAC;QACH,sBAAsB,CAAC,eAAe,CAAC,IAAI,yBAAe,CAAC;YACzD,OAAO,EAAE,CAAC,sBAAsB,CAAC;YACjC,SAAS,EAAE,CAAC,gBAAgB,CAAC,UAAU,CAAC;SACzC,CAAC,CAAC,CAAC;QAEJ,MAAM,yBAAyB,GAAG,IAAI,qBAAQ,CAAC,IAAI,EAAE,2BAA2B,EAAE;YAChF,OAAO,EAAE,oBAAO,CAAC,WAAW;YAC5B,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;YAC9D,WAAW,EAAE;gBACX,SAAS,EAAE,QAAQ;aACpB;YACD,OAAO,EAAE,oBAAoB;YAC7B,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SAC9B,CAAC,CAAC;QAEH,yBAAyB,CAAC,eAAe,CAAC,IAAI,yBAAe,CAAC;YAC5D,OAAO,EAAE;gBACP,0BAA0B;gBAC1B,gCAAgC;gBAChC,mBAAmB;gBACnB,yBAAyB;gBACzB,wBAAwB;aACzB;YACD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CAAC,CAAC;QAEJ,cAAc,CAAC,cAAc,CAAC,yBAAyB,CAAC,CAAC;QACzD,aAAa,CAAC,mBAAmB,CAAC,sBAAsB,CAAC,CAAC;QAC1D,aAAa,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,CAAC;QAC7D,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,sBAAsB,CAAC,CAAC;QACzD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,yBAAyB,CAAC,CAAC;QAE5D,gEAAgE;QAChE,MAAM,QAAQ,GAAG,IAAI,2BAAQ,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC5D,cAAc,EAAE,sBAAsB;YACtC,iBAAiB,EAAE,yBAAyB;YAC5C,aAAa,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SACpC,CAAC,CAAC;QAEH,uEAAuE;QACvE,MAAM,oBAAoB,GAAG,IAAI,4BAAc,CAAC,IAAI,EAAE,sBAAsB,EAAE;YAC5E,YAAY,EAAE,QAAQ,CAAC,YAAY;YACnC,UAAU,EAAE;gBACV,WAAW,EAAE,gBAAgB,CAAC,WAAW;gBACzC,QAAQ,EAAE,QAAQ;gBAClB,OAAO,EAAE,MAAM,CAAC,UAAU,EAAE;aAC7B;SACF,CAAC,CAAC;QACH,oBAAoB,CAAC,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAC;QAE1D,yDAAyD;QACzD,sDAAsD;QACtD,MAAM,WAAW,GAAG,oBAAoB,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;QAClE,IAAI,CAAC,cAAc,GAAG,wBAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC;QACxF,IAAI,CAAC,eAAe,GAAG,4BAAe,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,EAAE;YACjE,WAAW,EAAE,WAAW;SACzB,CAAC,CAAC;IACL,CAAC;;AArOH,oEAsOC","sourcesContent":["import * as crypto from 'crypto';\nimport * as path from 'path';\nimport { CustomResource, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { IVpc, ISecurityGroup, SubnetSelection } from 'aws-cdk-lib/aws-ec2';\nimport { Repository, RepositoryEncryption, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { Runtime, Code, DockerImageCode, Function } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket, BlockPublicAccess } from 'aws-cdk-lib/aws-s3';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\n/**\n * Properties for the `TokenInjectableDockerBuilder` construct.\n */\nexport interface TokenInjectableDockerBuilderProps {\n  /**\n   * The path to the directory containing the Dockerfile or source code.\n   */\n  readonly path: string;\n\n  /**\n   * Build arguments to pass to the Docker build process.\n   * These are transformed into `--build-arg KEY=VALUE` flags.\n   * @example\n   * {\n   *   TOKEN: 'my-secret-token',\n   *   ENV: 'production'\n   * }\n   */\n  readonly buildArgs?: { [key: string]: string };\n\n  /**\n   * The ARN of the AWS Secrets Manager secret containing Docker login credentials.\n   * This secret should store a JSON object with the following structure:\n   * ```json\n   * {\n   *   \"username\": \"my-docker-username\",\n   *   \"password\": \"my-docker-password\"\n   * }\n   * ```\n   * If not provided (or not needed), the construct will skip Docker Hub login.\n   *\n   * **Note**: The secret must be in the same region as the stack.\n   *\n   * @example 'arn:aws:secretsmanager:us-east-1:123456789012:secret:DockerLoginSecret'\n   */\n  readonly dockerLoginSecretArn?: string;\n\n  /**\n   * The VPC in which the CodeBuild project will be deployed.\n   * If provided, the CodeBuild project will be launched within the specified VPC.\n   *\n   * @default - No VPC is attached, and the CodeBuild project will use public internet.\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * The security groups to attach to the CodeBuild project.\n   * These define the network access rules for the CodeBuild project.\n   *\n   * @default - No security groups are attached.\n   */\n  readonly securityGroups?: ISecurityGroup[];\n\n  /**\n   * The subnet selection to specify which subnets to use within the VPC.\n   * Allows the user to select private, public, or isolated subnets.\n   *\n   * @default - All subnets in the VPC are used.\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Custom commands to run during the install phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * installCommands: [\n   *   'echo \"Updating package lists...\"',\n   *   'apt-get update -y',\n   *   'echo \"Installing required packages...\"',\n   *   'apt-get install -y curl dnsutils',\n   * ],\n   * ```\n   * @default - No additional install commands.\n   */\n  readonly installCommands?: string[];\n\n  /**\n   * Custom commands to run during the pre_build phase of CodeBuild.\n   *\n   * **Example**:\n   * ```ts\n   * preBuildCommands: [\n   *   'echo \"Fetching configuration from private API...\"',\n   *   'curl -o config.json https://api.example.com/config',\n   * ],\n   * ```\n   * @default - No additional pre-build commands.\n   */\n  readonly preBuildCommands?: string[];\n}\n\n/**\n * A CDK construct to build and push Docker images to an ECR repository using\n * CodeBuild and Lambda custom resources, **then** retrieve the final image tag\n * so that ECS/Lambda references use the exact digest.\n */\nexport class TokenInjectableDockerBuilder extends Construct {\n  /**\n   * The ECR repository that stores the resulting Docker image.\n   */\n  private readonly ecrRepository: Repository;\n\n  /**\n   * An ECS-compatible container image referencing the tag\n   * of the built Docker image.\n   */\n  public readonly containerImage: ContainerImage;\n\n  /**\n   * A Lambda-compatible DockerImageCode referencing the the tag\n   * of the built Docker image.\n   */\n  public readonly dockerImageCode: DockerImageCode;\n\n  /**\n   * Creates a new `TokenInjectableDockerBuilder`.\n   *\n   * @param scope The scope in which to define this construct.\n   * @param id The scoped construct ID.\n   * @param props Configuration for building and pushing the Docker image.\n   */\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const {\n      path: sourcePath,\n      buildArgs,\n      dockerLoginSecretArn,\n      vpc,\n      securityGroups,\n      subnetSelection,\n      installCommands,\n      preBuildCommands,\n    } = props;\n\n    // Generate an ephemeral tag for CodeBuild\n    const imageTag = crypto.randomUUID();\n\n    // Define a KMS key for ECR encryption\n    const encryptionKey = new Key(this, 'EcrEncryptionKey', {\n      enableKeyRotation: true,\n    });\n\n    // Create an ECR repository with encryption, lifecycle rules, and image scanning\n    this.ecrRepository = new Repository(this, 'ECRRepository', {\n      lifecycleRules: [\n        {\n          rulePriority: 1,\n          description: 'Remove untagged images after 1 day',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n      encryption: RepositoryEncryption.KMS,\n      encryptionKey,\n      imageScanOnPush: true,\n    });\n\n    // Wrap the source folder as an S3 asset for CodeBuild to use\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath,\n    });\n\n    // Create an S3 bucket to store the CodeBuild artifacts\n    const artifactBucket = new Bucket(this, 'ArtifactBucket', {\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    });\n\n    // Convert buildArgs to a CLI-friendly string\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n        .map(([k, v]) => `--build-arg ${k}=${v}`)\n        .join(' ')\n      : '';\n\n    // Optional DockerHub login, if a secret ARN is provided\n    const dockerLoginCommands = dockerLoginSecretArn\n      ? [\n        'echo \"Retrieving Docker credentials...\"',\n        'apt-get update -y && apt-get install -y jq',\n        `DOCKER_USERNAME=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .username)`,\n        `DOCKER_PASSWORD=$(aws secretsmanager get-secret-value --secret-id ${dockerLoginSecretArn} --query SecretString --output text | jq -r .password)`,\n        'echo \"Logging in to Docker Hub...\"',\n        'echo $DOCKER_PASSWORD | docker login --username $DOCKER_USERNAME --password-stdin',\n      ]\n      : ['echo \"No Docker credentials. Skipping Docker Hub login.\"'];\n\n    const buildSpecObj = {\n      version: '0.2',\n      phases: {\n        install: {\n          commands: [\n            'echo \"Beginning install phase...\"',\n            ...(installCommands ?? []),\n          ],\n        },\n        pre_build: {\n          commands: [\n            ...(preBuildCommands ?? []),\n            ...dockerLoginCommands,\n            'echo \"Retrieving AWS Account ID...\"',\n            'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n            'echo \"Logging into Amazon ECR...\"',\n            'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',\n          ],\n        },\n        build: {\n          commands: [\n            `echo \"Building Docker image with tag ${imageTag}...\"`,\n            `docker build ${buildArgsString} -t $ECR_REPO_URI:${imageTag} $CODEBUILD_SRC_DIR`,\n          ],\n        },\n        post_build: {\n          commands: [\n            `echo \"Pushing Docker image with tag ${imageTag}...\"`,\n            `docker push $ECR_REPO_URI:${imageTag}`,\n          ],\n        },\n      },\n    };\n\n\n    // Create the CodeBuild project\n    const codeBuildProject = new Project(this, 'CodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true,\n      },\n      environmentVariables: {\n        ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      },\n      buildSpec: BuildSpec.fromObject(buildSpecObj),\n      vpc,\n      securityGroups,\n      subnetSelection,\n    });\n\n    // Grant CodeBuild the ability to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n    codeBuildProject.role?.addToPrincipalPolicy(new PolicyStatement({\n      actions: [\n        'ecr:GetAuthorizationToken',\n        'ecr:GetDownloadUrlForLayer',\n        'ecr:BatchCheckLayerAvailability',\n      ],\n      resources: [this.ecrRepository.repositoryArn],\n    }));\n    if (dockerLoginSecretArn) {\n      codeBuildProject.role?.addToPrincipalPolicy(new PolicyStatement({\n        actions: ['secretsmanager:GetSecretValue'],\n        resources: [dockerLoginSecretArn],\n      }));\n    }\n    encryptionKey.grantEncryptDecrypt(codeBuildProject.role!);\n\n    // Define the Lambda functions for custom resource event and completion handling\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../onEvent')),\n      handler: 'onEvent.handler',\n      timeout: Duration.minutes(15),\n    });\n    onEventHandlerFunction.addToRolePolicy(new PolicyStatement({\n      actions: ['codebuild:StartBuild'],\n      resources: [codeBuildProject.projectArn],\n    }));\n\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(path.resolve(__dirname, '../isComplete')),\n      environment: {\n        IMAGE_TAG: imageTag,\n      },\n      handler: 'isComplete.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    isCompleteHandlerFunction.addToRolePolicy(new PolicyStatement({\n      actions: [\n        'codebuild:BatchGetBuilds',\n        'codebuild:ListBuildsForProject',\n        'logs:GetLogEvents',\n        'logs:DescribeLogStreams',\n        'logs:DescribeLogGroups',\n      ],\n      resources: ['*'],\n    }));\n\n    artifactBucket.grantReadWrite(isCompleteHandlerFunction);\n    encryptionKey.grantEncryptDecrypt(onEventHandlerFunction);\n    encryptionKey.grantEncryptDecrypt(isCompleteHandlerFunction);\n    this.ecrRepository.grantPullPush(onEventHandlerFunction);\n    this.ecrRepository.grantPullPush(isCompleteHandlerFunction);\n\n    // Create a custom resource provider that uses the above Lambdas\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.seconds(30),\n    });\n\n    // Custom Resource that triggers the CodeBuild and waits for completion\n    const buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        ImageTag: imageTag,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n    buildTriggerResource.node.addDependency(codeBuildProject);\n\n    // Retrieve the final Docker image tag from Data.ImageTag\n    // This creates a dependency on the Custom Resource...\n    const imageTagRef = buildTriggerResource.getAttString('ImageTag');\n    this.containerImage = ContainerImage.fromEcrRepository(this.ecrRepository, imageTagRef);\n    this.dockerImageCode = DockerImageCode.fromEcr(this.ecrRepository, {\n      tagOrDigest: imageTagRef,\n    });\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -30,7 +30,9 @@
|
|
|
30
30
|
"watch": "npx projen watch",
|
|
31
31
|
"projen": "npx projen",
|
|
32
32
|
"local-deploy": "cdk deploy --app \"npx ts-node src/integ.default.ts\"",
|
|
33
|
-
"local-
|
|
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",
|
|
@@ -40,29 +42,29 @@
|
|
|
40
42
|
"devDependencies": {
|
|
41
43
|
"@stylistic/eslint-plugin": "^2",
|
|
42
44
|
"@types/jest": "^29.5.14",
|
|
43
|
-
"@types/node": "^22.10.
|
|
45
|
+
"@types/node": "^22.10.6",
|
|
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",
|
|
50
52
|
"eslint-import-resolver-typescript": "^3.7.0",
|
|
51
53
|
"eslint-plugin-import": "^2.31.0",
|
|
52
54
|
"jest": "^29.7.0",
|
|
53
|
-
"jest-junit": "^
|
|
55
|
+
"jest-junit": "^16",
|
|
54
56
|
"jsii": "~5.5.0",
|
|
55
57
|
"jsii-diff": "^1.106.0",
|
|
56
58
|
"jsii-docgen": "^10.5.0",
|
|
57
59
|
"jsii-pacmak": "^1.106.0",
|
|
58
60
|
"jsii-rosetta": "~5.5.0",
|
|
59
|
-
"projen": "^0.91.
|
|
61
|
+
"projen": "^0.91.6",
|
|
60
62
|
"ts-jest": "^29.2.5",
|
|
61
63
|
"ts-node": "^10.9.2",
|
|
62
|
-
"typescript": "^5.7.
|
|
64
|
+
"typescript": "^5.7.3"
|
|
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.3.
|
|
101
|
+
"version": "1.3.2",
|
|
100
102
|
"jest": {
|
|
101
103
|
"coverageProvider": "v8",
|
|
102
104
|
"testMatch": [
|