@cloudsnorkel/cdk-github-runners 0.8.4 → 0.9.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 +2258 -745
- package/API.md +2006 -559
- package/README.md +44 -42
- package/assets/docker-images/codebuild/linux-arm64/Dockerfile +2 -0
- package/assets/docker-images/codebuild/linux-x64/Dockerfile +2 -0
- package/assets/docker-images/fargate/linux-arm64/Dockerfile +2 -0
- package/assets/docker-images/fargate/linux-x64/Dockerfile +2 -0
- package/assets/docker-images/lambda/linux-arm64/Dockerfile +3 -1
- package/assets/docker-images/lambda/linux-arm64/runner.sh +1 -1
- package/assets/docker-images/lambda/linux-x64/Dockerfile +3 -1
- package/assets/docker-images/lambda/linux-x64/runner.sh +1 -1
- package/assets/lambdas/setup.lambda/index.html +1 -1
- package/assets/lambdas/webhook-handler.lambda/index.js +1 -1
- package/lib/index.js +7 -3
- package/lib/lambdas/aws-image-builder-versioner.lambda.js +6 -6
- package/lib/lambdas/build-image.lambda.js +4 -4
- package/lib/lambdas/delete-ami.lambda.js +4 -4
- package/lib/lambdas/delete-runner.lambda.js +2 -2
- package/lib/lambdas/github.js +3 -3
- package/lib/lambdas/setup.lambda.js +16 -16
- package/lib/lambdas/status.lambda.js +5 -5
- package/lib/lambdas/token-retriever.lambda.js +2 -2
- package/lib/lambdas/webhook-handler.lambda.js +3 -3
- package/lib/providers/codebuild.d.ts +24 -4
- package/lib/providers/codebuild.js +42 -12
- package/lib/providers/common.d.ts +17 -39
- package/lib/providers/common.js +26 -16
- package/lib/providers/ec2.d.ts +23 -5
- package/lib/providers/ec2.js +43 -12
- package/lib/providers/ecs.d.ts +214 -0
- package/lib/providers/ecs.js +258 -0
- package/lib/providers/fargate.d.ts +26 -6
- package/lib/providers/fargate.js +81 -42
- package/lib/providers/image-builders/api.d.ts +15 -0
- package/lib/providers/image-builders/api.js +47 -0
- package/lib/providers/image-builders/aws-image-builder/ami.d.ts +43 -0
- package/lib/providers/image-builders/aws-image-builder/ami.js +81 -0
- package/lib/providers/image-builders/aws-image-builder/builder.d.ts +133 -0
- package/lib/providers/image-builders/aws-image-builder/builder.js +488 -0
- package/lib/providers/image-builders/aws-image-builder/common.d.ts +10 -0
- package/lib/providers/image-builders/aws-image-builder/common.js +46 -0
- package/lib/providers/image-builders/aws-image-builder/container.d.ts +58 -0
- package/lib/providers/image-builders/aws-image-builder/container.js +63 -0
- package/lib/providers/image-builders/{ami.d.ts → aws-image-builder/deprecated/ami.d.ts} +8 -4
- package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +239 -0
- package/lib/providers/image-builders/aws-image-builder/deprecated/common.d.ts +34 -0
- package/lib/providers/image-builders/aws-image-builder/deprecated/common.js +139 -0
- package/lib/providers/image-builders/{container.d.ts → aws-image-builder/deprecated/container.d.ts} +8 -4
- package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +222 -0
- package/lib/providers/image-builders/aws-image-builder/deprecated/index.d.ts +5 -0
- package/lib/providers/image-builders/aws-image-builder/deprecated/index.js +22 -0
- package/lib/providers/image-builders/{linux-components.d.ts → aws-image-builder/deprecated/linux-components.d.ts} +4 -2
- package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +180 -0
- package/lib/providers/image-builders/{windows-components.d.ts → aws-image-builder/deprecated/windows-components.d.ts} +4 -2
- package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +142 -0
- package/lib/providers/image-builders/aws-image-builder/index.d.ts +5 -0
- package/lib/providers/image-builders/aws-image-builder/index.js +22 -0
- package/lib/providers/image-builders/codebuild-deprecated.d.ts +195 -0
- package/lib/providers/image-builders/codebuild-deprecated.js +373 -0
- package/lib/providers/image-builders/codebuild.d.ts +26 -157
- package/lib/providers/image-builders/codebuild.js +118 -210
- package/lib/providers/image-builders/common.d.ts +164 -107
- package/lib/providers/image-builders/common.js +30 -272
- package/lib/providers/image-builders/components.d.ts +114 -0
- package/lib/providers/image-builders/components.js +535 -0
- package/lib/providers/image-builders/index.d.ts +6 -4
- package/lib/providers/image-builders/index.js +13 -7
- package/lib/providers/image-builders/static.d.ts +4 -3
- package/lib/providers/image-builders/static.js +10 -10
- package/lib/providers/index.js +7 -3
- package/lib/providers/lambda.d.ts +25 -6
- package/lib/providers/lambda.js +50 -13
- package/lib/runner.d.ts +3 -5
- package/lib/runner.js +4 -4
- package/lib/secrets.js +3 -3
- package/package.json +7 -11
- package/lib/providers/image-builders/ami.js +0 -280
- package/lib/providers/image-builders/container.js +0 -247
- package/lib/providers/image-builders/linux-components.js +0 -177
- package/lib/providers/image-builders/windows-components.js +0 -139
|
@@ -0,0 +1,488 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.AwsImageBuilderRunnerImageBuilder = exports.ImageBuilderComponent = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk = require("aws-cdk-lib");
|
|
7
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const aws_ecr_1 = require("aws-cdk-lib/aws-ecr");
|
|
9
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
10
|
+
const ami_1 = require("./ami");
|
|
11
|
+
const common_1 = require("./common");
|
|
12
|
+
const container_1 = require("./container");
|
|
13
|
+
const build_image_function_1 = require("../../../lambdas/build-image-function");
|
|
14
|
+
const delete_ami_function_1 = require("../../../lambdas/delete-ami-function");
|
|
15
|
+
const utils_1 = require("../../../utils");
|
|
16
|
+
const common_2 = require("../../common");
|
|
17
|
+
const common_3 = require("../common");
|
|
18
|
+
/**
|
|
19
|
+
* Components are a set of commands to run and optional files to add to an image. Components are the building blocks of images built by Image Builder.
|
|
20
|
+
*
|
|
21
|
+
* Example:
|
|
22
|
+
*
|
|
23
|
+
* ```
|
|
24
|
+
* new ImageBuilderComponent(this, 'AWS CLI', {
|
|
25
|
+
* platform: 'Windows',
|
|
26
|
+
* displayName: 'AWS CLI',
|
|
27
|
+
* description: 'Install latest version of AWS CLI',
|
|
28
|
+
* commands: [
|
|
29
|
+
* '$ErrorActionPreference = \'Stop\'',
|
|
30
|
+
* 'Start-Process msiexec.exe -Wait -ArgumentList \'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\'',
|
|
31
|
+
* ],
|
|
32
|
+
* }
|
|
33
|
+
* ```
|
|
34
|
+
*
|
|
35
|
+
* @deprecated Use `RunnerImageComponent` instead as this be internal soon.
|
|
36
|
+
*/
|
|
37
|
+
class ImageBuilderComponent extends common_1.ImageBuilderObjectBase {
|
|
38
|
+
constructor(scope, id, props) {
|
|
39
|
+
super(scope, id);
|
|
40
|
+
this.assets = [];
|
|
41
|
+
this.platform = props.platform;
|
|
42
|
+
let steps = [];
|
|
43
|
+
if (props.assets) {
|
|
44
|
+
let inputs = [];
|
|
45
|
+
let extractCommands = [];
|
|
46
|
+
for (const asset of props.assets) {
|
|
47
|
+
this.assets.push(asset.asset);
|
|
48
|
+
if (asset.asset.isFile) {
|
|
49
|
+
inputs.push({
|
|
50
|
+
source: asset.asset.s3ObjectUrl,
|
|
51
|
+
destination: asset.path,
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
else if (asset.asset.isZipArchive) {
|
|
55
|
+
inputs.push({
|
|
56
|
+
source: asset.asset.s3ObjectUrl,
|
|
57
|
+
destination: `${asset.path}.zip`,
|
|
58
|
+
});
|
|
59
|
+
if (props.platform === 'Windows') {
|
|
60
|
+
extractCommands.push(`Expand-Archive "${asset.path}.zip" -DestinationPath "${asset.path}"`);
|
|
61
|
+
extractCommands.push(`del "${asset.path}.zip"`);
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
extractCommands.push(`unzip "${asset.path}.zip" -d "${asset.path}"`);
|
|
65
|
+
extractCommands.push(`rm "${asset.path}.zip"`);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
throw new Error(`Unknown asset type: ${asset.asset}`);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
steps.push({
|
|
73
|
+
name: 'Download',
|
|
74
|
+
action: 'S3Download',
|
|
75
|
+
inputs,
|
|
76
|
+
});
|
|
77
|
+
if (extractCommands.length > 0) {
|
|
78
|
+
steps.push({
|
|
79
|
+
name: 'Extract',
|
|
80
|
+
action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',
|
|
81
|
+
inputs: {
|
|
82
|
+
commands: this.prefixCommandsWithErrorHandling(props.platform, extractCommands),
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
if (props.commands.length > 0) {
|
|
88
|
+
steps.push({
|
|
89
|
+
name: 'Run',
|
|
90
|
+
action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',
|
|
91
|
+
inputs: {
|
|
92
|
+
commands: this.prefixCommandsWithErrorHandling(props.platform, props.commands),
|
|
93
|
+
},
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
const data = {
|
|
97
|
+
name: props.displayName,
|
|
98
|
+
schemaVersion: '1.0',
|
|
99
|
+
phases: [
|
|
100
|
+
{
|
|
101
|
+
name: 'build',
|
|
102
|
+
steps,
|
|
103
|
+
},
|
|
104
|
+
],
|
|
105
|
+
};
|
|
106
|
+
const name = (0, common_3.uniqueImageBuilderName)(this);
|
|
107
|
+
const component = new aws_cdk_lib_1.aws_imagebuilder.CfnComponent(this, 'Component', {
|
|
108
|
+
name: name,
|
|
109
|
+
description: props.description,
|
|
110
|
+
platform: props.platform,
|
|
111
|
+
version: this.version('Component', name, {
|
|
112
|
+
platform: props.platform,
|
|
113
|
+
data,
|
|
114
|
+
description: props.description,
|
|
115
|
+
}),
|
|
116
|
+
data: JSON.stringify(data),
|
|
117
|
+
});
|
|
118
|
+
this.arn = component.attrArn;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Grants read permissions to the principal on the assets buckets.
|
|
122
|
+
*
|
|
123
|
+
* @param grantee
|
|
124
|
+
*/
|
|
125
|
+
grantAssetsRead(grantee) {
|
|
126
|
+
for (const asset of this.assets) {
|
|
127
|
+
asset.grantRead(grantee);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
prefixCommandsWithErrorHandling(platform, commands) {
|
|
131
|
+
if (platform == 'Windows') {
|
|
132
|
+
return [
|
|
133
|
+
'$ErrorActionPreference = \'Stop\'',
|
|
134
|
+
'$ProgressPreference = \'SilentlyContinue\'',
|
|
135
|
+
].concat(commands);
|
|
136
|
+
}
|
|
137
|
+
else {
|
|
138
|
+
return [
|
|
139
|
+
'set -ex',
|
|
140
|
+
].concat(commands);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
145
|
+
ImageBuilderComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.ImageBuilderComponent", version: "0.9.1" };
|
|
146
|
+
exports.ImageBuilderComponent = ImageBuilderComponent;
|
|
147
|
+
/**
|
|
148
|
+
* @internal
|
|
149
|
+
*/
|
|
150
|
+
class AwsImageBuilderRunnerImageBuilder extends common_3.RunnerImageBuilderBase {
|
|
151
|
+
constructor(scope, id, props) {
|
|
152
|
+
super(scope, id, props);
|
|
153
|
+
this.boundComponents = [];
|
|
154
|
+
if (props?.codeBuildOptions) {
|
|
155
|
+
aws_cdk_lib_1.Annotations.of(this).addWarning('codeBuildOptions are ignored when using AWS Image Builder to build runner images.');
|
|
156
|
+
}
|
|
157
|
+
this.os = props?.os ?? common_2.Os.LINUX_UBUNTU;
|
|
158
|
+
this.architecture = props?.architecture ?? common_2.Architecture.X86_64;
|
|
159
|
+
this.rebuildInterval = props?.rebuildInterval ?? aws_cdk_lib_1.Duration.days(7);
|
|
160
|
+
this.logRetention = props?.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH;
|
|
161
|
+
this.logRemovalPolicy = props?.logRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.DESTROY;
|
|
162
|
+
this.vpc = props?.vpc ?? aws_cdk_lib_1.aws_ec2.Vpc.fromLookup(this, 'VPC', { isDefault: true });
|
|
163
|
+
this.securityGroups = props?.securityGroups ?? [new aws_cdk_lib_1.aws_ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];
|
|
164
|
+
this.subnetSelection = props?.subnetSelection;
|
|
165
|
+
this.baseImage = props?.baseDockerImage ?? (0, container_1.defaultBaseDockerImage)(this.os);
|
|
166
|
+
this.baseAmi = props?.baseAmi ?? (0, ami_1.defaultBaseAmi)(this.os, this.architecture).getImage(this).imageId;
|
|
167
|
+
this.instanceType = props?.awsImageBuilderOptions?.instanceType ?? aws_cdk_lib_1.aws_ec2.InstanceType.of(aws_cdk_lib_1.aws_ec2.InstanceClass.M5, aws_cdk_lib_1.aws_ec2.InstanceSize.LARGE);
|
|
168
|
+
// confirm instance type
|
|
169
|
+
if (!this.architecture.instanceTypeMatch(this.instanceType)) {
|
|
170
|
+
throw new Error(`Builder architecture (${this.architecture.name}) doesn't match selected instance type (${this.instanceType} / ${this.instanceType.architecture})`);
|
|
171
|
+
}
|
|
172
|
+
// warn against isolated networks
|
|
173
|
+
if (props?.subnetSelection?.subnetType == aws_cdk_lib_1.aws_ec2.SubnetType.PRIVATE_ISOLATED) {
|
|
174
|
+
aws_cdk_lib_1.Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +
|
|
175
|
+
'See https://github.com/aws/containers-roadmap/issues/1160');
|
|
176
|
+
}
|
|
177
|
+
// role to be used by AWS Image Builder
|
|
178
|
+
this.role = new aws_cdk_lib_1.aws_iam.Role(this, 'Role', {
|
|
179
|
+
assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('ec2.amazonaws.com'),
|
|
180
|
+
});
|
|
181
|
+
// create repository that only keeps one tag
|
|
182
|
+
this.repository = new aws_cdk_lib_1.aws_ecr.Repository(this, 'Repository', {
|
|
183
|
+
imageScanOnPush: true,
|
|
184
|
+
imageTagMutability: aws_ecr_1.TagMutability.MUTABLE,
|
|
185
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
186
|
+
lifecycleRules: [
|
|
187
|
+
{
|
|
188
|
+
description: 'Remove untagged images that have been replaced by CodeBuild',
|
|
189
|
+
tagStatus: aws_ecr_1.TagStatus.UNTAGGED,
|
|
190
|
+
maxImageAge: aws_cdk_lib_1.Duration.days(1),
|
|
191
|
+
},
|
|
192
|
+
],
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
platform() {
|
|
196
|
+
if (this.os.is(common_2.Os.WINDOWS)) {
|
|
197
|
+
return 'Windows';
|
|
198
|
+
}
|
|
199
|
+
if (this.os.is(common_2.Os.LINUX_AMAZON_2) || this.os.is(common_2.Os.LINUX_UBUNTU)) {
|
|
200
|
+
return 'Linux';
|
|
201
|
+
}
|
|
202
|
+
throw new Error(`OS ${this.os.name} is not supported by AWS Image Builder`);
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* Called by IRunnerProvider to finalize settings and create the image builder.
|
|
206
|
+
*/
|
|
207
|
+
bindDockerImage() {
|
|
208
|
+
if (this.boundDockerImage) {
|
|
209
|
+
return this.boundDockerImage;
|
|
210
|
+
}
|
|
211
|
+
const dist = new aws_cdk_lib_1.aws_imagebuilder.CfnDistributionConfiguration(this, 'Docker Distribution', {
|
|
212
|
+
name: (0, common_3.uniqueImageBuilderName)(this),
|
|
213
|
+
// description: this.description,
|
|
214
|
+
distributions: [
|
|
215
|
+
{
|
|
216
|
+
region: aws_cdk_lib_1.Stack.of(this).region,
|
|
217
|
+
containerDistributionConfiguration: {
|
|
218
|
+
ContainerTags: ['latest'],
|
|
219
|
+
TargetRepository: {
|
|
220
|
+
Service: 'ECR',
|
|
221
|
+
RepositoryName: this.repository.repositoryName,
|
|
222
|
+
},
|
|
223
|
+
},
|
|
224
|
+
},
|
|
225
|
+
],
|
|
226
|
+
});
|
|
227
|
+
let dockerfileTemplate = `FROM {{{ imagebuilder:parentImage }}}
|
|
228
|
+
{{{ imagebuilder:environments }}}
|
|
229
|
+
{{{ imagebuilder:components }}}`;
|
|
230
|
+
if (this.boundComponents.length == 0) {
|
|
231
|
+
this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i}`, this.os, this.architecture)));
|
|
232
|
+
}
|
|
233
|
+
for (const c of this.components) {
|
|
234
|
+
const commands = c.getDockerCommands(this.os, this.architecture);
|
|
235
|
+
if (commands.length > 0) {
|
|
236
|
+
dockerfileTemplate += '\n' + commands.join('\n') + '\n';
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
const recipe = new container_1.ContainerRecipe(this, 'Container Recipe', {
|
|
240
|
+
platform: this.platform(),
|
|
241
|
+
components: this.boundComponents,
|
|
242
|
+
targetRepository: this.repository,
|
|
243
|
+
dockerfileTemplate: dockerfileTemplate,
|
|
244
|
+
parentImage: this.baseImage,
|
|
245
|
+
});
|
|
246
|
+
const log = this.createLog('Docker Log', recipe.name);
|
|
247
|
+
const infra = this.createInfrastructure([
|
|
248
|
+
aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),
|
|
249
|
+
aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilderECRContainerBuilds'),
|
|
250
|
+
]);
|
|
251
|
+
const image = this.createImage(infra, dist, log, undefined, recipe.arn);
|
|
252
|
+
this.createPipeline(infra, dist, log, undefined, recipe.arn);
|
|
253
|
+
this.imageCleaner(image, recipe.name);
|
|
254
|
+
this.boundDockerImage = {
|
|
255
|
+
// There are simpler ways to get the ARN, but we want an image object that depends on the newly built image.
|
|
256
|
+
// We want whoever is using this image to automatically wait for Image Builder to finish building before using the image.
|
|
257
|
+
imageRepository: aws_cdk_lib_1.aws_ecr.Repository.fromRepositoryName(this, 'Dependable Image',
|
|
258
|
+
// we can't use image.attrName because it comes up with upper case
|
|
259
|
+
cdk.Fn.split(':', cdk.Fn.split('/', image.attrImageUri, 2)[1], 2)[0]),
|
|
260
|
+
imageTag: 'latest',
|
|
261
|
+
os: this.os,
|
|
262
|
+
architecture: this.architecture,
|
|
263
|
+
logGroup: log,
|
|
264
|
+
runnerVersion: common_2.RunnerVersion.specific('unknown'),
|
|
265
|
+
};
|
|
266
|
+
return this.boundDockerImage;
|
|
267
|
+
}
|
|
268
|
+
imageCleaner(image, recipeName) {
|
|
269
|
+
const crHandler = (0, utils_1.singletonLambda)(build_image_function_1.BuildImageFunction, this, 'build-image', {
|
|
270
|
+
description: 'Custom resource handler that triggers CodeBuild to build runner images, and cleans-up images on deletion',
|
|
271
|
+
timeout: cdk.Duration.minutes(3),
|
|
272
|
+
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
|
|
273
|
+
});
|
|
274
|
+
const policy = new aws_cdk_lib_1.aws_iam.Policy(this, 'CR Policy', {
|
|
275
|
+
statements: [
|
|
276
|
+
new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
277
|
+
actions: ['ecr:BatchDeleteImage', 'ecr:ListImages'],
|
|
278
|
+
resources: [this.repository.repositoryArn],
|
|
279
|
+
}),
|
|
280
|
+
new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
281
|
+
actions: ['imagebuilder:ListImages', 'imagebuilder:ListImageBuildVersions', 'imagebuilder:DeleteImage'],
|
|
282
|
+
resources: ['*'], // Image Builder doesn't support scoping this :(
|
|
283
|
+
}),
|
|
284
|
+
],
|
|
285
|
+
});
|
|
286
|
+
crHandler.role?.attachInlinePolicy(policy);
|
|
287
|
+
const cr = new aws_cdk_lib_1.CustomResource(this, 'Deleter', {
|
|
288
|
+
serviceToken: crHandler.functionArn,
|
|
289
|
+
resourceType: 'Custom::ImageDeleter',
|
|
290
|
+
properties: {
|
|
291
|
+
RepoName: this.repository.repositoryName,
|
|
292
|
+
ImageBuilderName: recipeName,
|
|
293
|
+
DeleteOnly: true,
|
|
294
|
+
},
|
|
295
|
+
});
|
|
296
|
+
// add dependencies to make sure resources are there when we need them
|
|
297
|
+
cr.node.addDependency(image);
|
|
298
|
+
cr.node.addDependency(policy);
|
|
299
|
+
cr.node.addDependency(crHandler);
|
|
300
|
+
return cr;
|
|
301
|
+
}
|
|
302
|
+
createLog(id, recipeName) {
|
|
303
|
+
return new aws_cdk_lib_1.aws_logs.LogGroup(this, id, {
|
|
304
|
+
logGroupName: `/aws/imagebuilder/${recipeName}`,
|
|
305
|
+
retention: this.logRetention,
|
|
306
|
+
removalPolicy: this.logRemovalPolicy,
|
|
307
|
+
});
|
|
308
|
+
}
|
|
309
|
+
createInfrastructure(managedPolicies) {
|
|
310
|
+
if (this.infrastructure) {
|
|
311
|
+
return this.infrastructure;
|
|
312
|
+
}
|
|
313
|
+
for (const managedPolicy of managedPolicies) {
|
|
314
|
+
this.role.addManagedPolicy(managedPolicy);
|
|
315
|
+
}
|
|
316
|
+
for (const component of this.boundComponents) {
|
|
317
|
+
component.grantAssetsRead(this.role);
|
|
318
|
+
}
|
|
319
|
+
this.infrastructure = new aws_cdk_lib_1.aws_imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {
|
|
320
|
+
name: (0, common_3.uniqueImageBuilderName)(this),
|
|
321
|
+
// description: this.description,
|
|
322
|
+
subnetId: this.vpc?.selectSubnets(this.subnetSelection).subnetIds[0],
|
|
323
|
+
securityGroupIds: this.securityGroups?.map(sg => sg.securityGroupId),
|
|
324
|
+
instanceTypes: [this.instanceType.toString()],
|
|
325
|
+
instanceProfileName: new aws_cdk_lib_1.aws_iam.CfnInstanceProfile(this, 'Instance Profile', {
|
|
326
|
+
roles: [
|
|
327
|
+
this.role.roleName,
|
|
328
|
+
],
|
|
329
|
+
}).ref,
|
|
330
|
+
});
|
|
331
|
+
return this.infrastructure;
|
|
332
|
+
}
|
|
333
|
+
createImage(infra, dist, log, imageRecipeArn, containerRecipeArn) {
|
|
334
|
+
const image = new aws_cdk_lib_1.aws_imagebuilder.CfnImage(this, this.amiOrContainerId('Image', imageRecipeArn, containerRecipeArn), {
|
|
335
|
+
infrastructureConfigurationArn: infra.attrArn,
|
|
336
|
+
distributionConfigurationArn: dist.attrArn,
|
|
337
|
+
imageRecipeArn,
|
|
338
|
+
containerRecipeArn,
|
|
339
|
+
imageTestsConfiguration: {
|
|
340
|
+
imageTestsEnabled: false,
|
|
341
|
+
},
|
|
342
|
+
});
|
|
343
|
+
image.node.addDependency(infra);
|
|
344
|
+
image.node.addDependency(log);
|
|
345
|
+
return image;
|
|
346
|
+
}
|
|
347
|
+
amiOrContainerId(baseId, imageRecipeArn, containerRecipeArn) {
|
|
348
|
+
if (imageRecipeArn) {
|
|
349
|
+
return `AMI ${baseId}`;
|
|
350
|
+
}
|
|
351
|
+
if (containerRecipeArn) {
|
|
352
|
+
return `Docker ${baseId}`;
|
|
353
|
+
}
|
|
354
|
+
throw new Error('Either imageRecipeArn or containerRecipeArn must be defined');
|
|
355
|
+
}
|
|
356
|
+
createPipeline(infra, dist, log, imageRecipeArn, containerRecipeArn) {
|
|
357
|
+
let scheduleOptions;
|
|
358
|
+
if (this.rebuildInterval.toDays() > 0) {
|
|
359
|
+
scheduleOptions = {
|
|
360
|
+
scheduleExpression: aws_cdk_lib_1.aws_events.Schedule.rate(this.rebuildInterval).expressionString,
|
|
361
|
+
pipelineExecutionStartCondition: 'EXPRESSION_MATCH_ONLY',
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
const pipeline = new aws_cdk_lib_1.aws_imagebuilder.CfnImagePipeline(this, this.amiOrContainerId('Pipeline', imageRecipeArn, containerRecipeArn), {
|
|
365
|
+
name: (0, common_3.uniqueImageBuilderName)(this),
|
|
366
|
+
// description: this.description,
|
|
367
|
+
infrastructureConfigurationArn: infra.attrArn,
|
|
368
|
+
distributionConfigurationArn: dist.attrArn,
|
|
369
|
+
imageRecipeArn,
|
|
370
|
+
containerRecipeArn,
|
|
371
|
+
schedule: scheduleOptions,
|
|
372
|
+
imageTestsConfiguration: {
|
|
373
|
+
imageTestsEnabled: false,
|
|
374
|
+
},
|
|
375
|
+
});
|
|
376
|
+
pipeline.node.addDependency(infra);
|
|
377
|
+
pipeline.node.addDependency(log);
|
|
378
|
+
return pipeline;
|
|
379
|
+
}
|
|
380
|
+
/**
|
|
381
|
+
* The network connections associated with this resource.
|
|
382
|
+
*/
|
|
383
|
+
get connections() {
|
|
384
|
+
return new aws_cdk_lib_1.aws_ec2.Connections({ securityGroups: this.securityGroups });
|
|
385
|
+
}
|
|
386
|
+
get grantPrincipal() {
|
|
387
|
+
return this.role;
|
|
388
|
+
}
|
|
389
|
+
bindAmi() {
|
|
390
|
+
if (this.boundAmi) {
|
|
391
|
+
return this.boundAmi;
|
|
392
|
+
}
|
|
393
|
+
const launchTemplate = new aws_cdk_lib_1.aws_ec2.LaunchTemplate(this, 'Launch template');
|
|
394
|
+
const stackName = cdk.Stack.of(this).stackName;
|
|
395
|
+
const builderName = this.node.path;
|
|
396
|
+
const dist = new aws_cdk_lib_1.aws_imagebuilder.CfnDistributionConfiguration(this, 'AMI Distribution', {
|
|
397
|
+
name: (0, common_3.uniqueImageBuilderName)(this),
|
|
398
|
+
// description: this.description,
|
|
399
|
+
distributions: [
|
|
400
|
+
{
|
|
401
|
+
region: aws_cdk_lib_1.Stack.of(this).region,
|
|
402
|
+
amiDistributionConfiguration: {
|
|
403
|
+
Name: `${cdk.Names.uniqueResourceName(this, {
|
|
404
|
+
maxLength: 100,
|
|
405
|
+
separator: '-',
|
|
406
|
+
allowedSpecialCharacters: '_-',
|
|
407
|
+
})}-{{ imagebuilder:buildDate }}`,
|
|
408
|
+
AmiTags: {
|
|
409
|
+
'Name': this.node.id,
|
|
410
|
+
'GitHubRunners:Stack': stackName,
|
|
411
|
+
'GitHubRunners:Builder': builderName,
|
|
412
|
+
},
|
|
413
|
+
},
|
|
414
|
+
launchTemplateConfigurations: [
|
|
415
|
+
{
|
|
416
|
+
launchTemplateId: launchTemplate.launchTemplateId,
|
|
417
|
+
},
|
|
418
|
+
],
|
|
419
|
+
},
|
|
420
|
+
],
|
|
421
|
+
});
|
|
422
|
+
const recipe = new ami_1.AmiRecipe(this, 'Ami Recipe', {
|
|
423
|
+
platform: this.platform(),
|
|
424
|
+
components: this.bindComponents(),
|
|
425
|
+
architecture: this.architecture,
|
|
426
|
+
baseAmi: this.baseAmi,
|
|
427
|
+
});
|
|
428
|
+
const log = this.createLog('Ami Log', recipe.name);
|
|
429
|
+
const infra = this.createInfrastructure([
|
|
430
|
+
aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),
|
|
431
|
+
aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilder'),
|
|
432
|
+
]);
|
|
433
|
+
this.createImage(infra, dist, log, recipe.arn, undefined);
|
|
434
|
+
this.createPipeline(infra, dist, log, recipe.arn, undefined);
|
|
435
|
+
this.boundAmi = {
|
|
436
|
+
launchTemplate: launchTemplate,
|
|
437
|
+
architecture: this.architecture,
|
|
438
|
+
os: this.os,
|
|
439
|
+
logGroup: log,
|
|
440
|
+
runnerVersion: common_2.RunnerVersion.specific('unknown'),
|
|
441
|
+
};
|
|
442
|
+
this.amiCleaner(launchTemplate, stackName, builderName);
|
|
443
|
+
return this.boundAmi;
|
|
444
|
+
}
|
|
445
|
+
amiCleaner(launchTemplate, stackName, builderName) {
|
|
446
|
+
const deleter = (0, utils_1.singletonLambda)(delete_ami_function_1.DeleteAmiFunction, this, 'delete-ami', {
|
|
447
|
+
description: 'Delete old GitHub Runner AMIs',
|
|
448
|
+
initialPolicy: [
|
|
449
|
+
new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
450
|
+
actions: ['ec2:DescribeLaunchTemplateVersions', 'ec2:DescribeImages', 'ec2:DeregisterImage', 'ec2:DeleteSnapshot'],
|
|
451
|
+
resources: ['*'],
|
|
452
|
+
}),
|
|
453
|
+
],
|
|
454
|
+
timeout: cdk.Duration.minutes(5),
|
|
455
|
+
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
|
|
456
|
+
});
|
|
457
|
+
// delete old AMIs on schedule
|
|
458
|
+
const eventRule = new aws_cdk_lib_1.aws_events.Rule(this, 'Delete AMI Schedule', {
|
|
459
|
+
schedule: aws_cdk_lib_1.aws_events.Schedule.rate(cdk.Duration.days(1)),
|
|
460
|
+
description: `Delete old AMIs for ${builderName}`,
|
|
461
|
+
});
|
|
462
|
+
eventRule.addTarget(new aws_cdk_lib_1.aws_events_targets.LambdaFunction(deleter, {
|
|
463
|
+
event: aws_cdk_lib_1.aws_events.RuleTargetInput.fromObject({
|
|
464
|
+
RequestType: 'Scheduled',
|
|
465
|
+
LaunchTemplateId: launchTemplate.launchTemplateId,
|
|
466
|
+
StackName: stackName,
|
|
467
|
+
BuilderName: builderName,
|
|
468
|
+
}),
|
|
469
|
+
}));
|
|
470
|
+
// delete all AMIs when this construct is removed
|
|
471
|
+
new aws_cdk_lib_1.CustomResource(this, 'AMI Deleter', {
|
|
472
|
+
serviceToken: deleter.functionArn,
|
|
473
|
+
resourceType: 'Custom::AmiDeleter',
|
|
474
|
+
properties: {
|
|
475
|
+
StackName: stackName,
|
|
476
|
+
BuilderName: builderName,
|
|
477
|
+
},
|
|
478
|
+
});
|
|
479
|
+
}
|
|
480
|
+
bindComponents() {
|
|
481
|
+
if (this.boundComponents.length == 0) {
|
|
482
|
+
this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i} ${c.name}`, this.os, this.architecture)));
|
|
483
|
+
}
|
|
484
|
+
return this.boundComponents;
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
exports.AwsImageBuilderRunnerImageBuilder = AwsImageBuilderRunnerImageBuilder;
|
|
488
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../../src/providers/image-builders/aws-image-builder/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_imagebuilder as imagebuilder,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct } from 'constructs';\nimport { AmiRecipe, defaultBaseAmi } from './ami';\nimport { ImageBuilderObjectBase } from './common';\nimport { ContainerRecipe, defaultBaseDockerImage } from './container';\nimport { BuildImageFunction } from '../../../lambdas/build-image-function';\nimport { DeleteAmiFunction } from '../../../lambdas/delete-ami-function';\nimport { singletonLambda } from '../../../utils';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../common';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps, uniqueImageBuilderName } from '../common';\n\nexport interface AwsImageBuilderRunnerImageBuilderProps {\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n}\n\n/**\n * An asset including file or directory to place inside the built image.\n */\nexport interface ImageBuilderAsset {\n  /**\n   * Path to place asset in the image.\n   */\n  readonly path: string;\n\n  /**\n   * Asset to place in the image.\n   */\n  readonly asset: s3_assets.Asset;\n}\n\n/**\n * Properties for ImageBuilderComponent construct.\n */\nexport interface ImageBuilderComponentProperties {\n  /**\n   * Component platform. Must match the builder platform.\n   */\n  readonly platform: 'Linux' | 'Windows';\n\n  /**\n   * Component display name.\n   */\n  readonly displayName: string;\n\n  /**\n   * Component description.\n   */\n  readonly description: string;\n\n  /**\n   * Shell commands to run when adding this component to the image.\n   *\n   * On Linux, these are bash commands. On Windows, there are PowerShell commands.\n   */\n  readonly commands: string[];\n\n  /**\n   * Optional assets to add to the built image.\n   */\n  readonly assets?: ImageBuilderAsset[];\n}\n\n/**\n * Components are a set of commands to run and optional files to add to an image. Components are the building blocks of images built by Image Builder.\n *\n * Example:\n *\n * ```\n * new ImageBuilderComponent(this, 'AWS CLI', {\n *   platform: 'Windows',\n *   displayName: 'AWS CLI',\n *   description: 'Install latest version of AWS CLI',\n *   commands: [\n *     '$ErrorActionPreference = \\'Stop\\'',\n *     'Start-Process msiexec.exe -Wait -ArgumentList \\'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\\'',\n *   ],\n * }\n * ```\n *\n * @deprecated Use `RunnerImageComponent` instead as this be internal soon.\n */\nexport class ImageBuilderComponent extends ImageBuilderObjectBase {\n  /**\n   * Component ARN.\n   */\n  public readonly arn: string;\n\n  /**\n   * Supported platform for the component.\n   */\n  public readonly platform: 'Windows' | 'Linux';\n\n  private readonly assets: s3_assets.Asset[] = [];\n\n  constructor(scope: Construct, id: string, props: ImageBuilderComponentProperties) {\n    super(scope, id);\n\n    this.platform = props.platform;\n\n    let steps: any[] = [];\n\n    if (props.assets) {\n      let inputs: any[] = [];\n      let extractCommands: string[] = [];\n      for (const asset of props.assets) {\n        this.assets.push(asset.asset);\n\n        if (asset.asset.isFile) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: asset.path,\n          });\n        } else if (asset.asset.isZipArchive) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: `${asset.path}.zip`,\n          });\n          if (props.platform === 'Windows') {\n            extractCommands.push(`Expand-Archive \"${asset.path}.zip\" -DestinationPath \"${asset.path}\"`);\n            extractCommands.push(`del \"${asset.path}.zip\"`);\n          } else {\n            extractCommands.push(`unzip \"${asset.path}.zip\" -d \"${asset.path}\"`);\n            extractCommands.push(`rm \"${asset.path}.zip\"`);\n          }\n        } else {\n          throw new Error(`Unknown asset type: ${asset.asset}`);\n        }\n      }\n\n      steps.push({\n        name: 'Download',\n        action: 'S3Download',\n        inputs,\n      });\n\n      if (extractCommands.length > 0) {\n        steps.push({\n          name: 'Extract',\n          action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n          inputs: {\n            commands: this.prefixCommandsWithErrorHandling(props.platform, extractCommands),\n          },\n        });\n      }\n    }\n\n    if (props.commands.length > 0) {\n      steps.push({\n        name: 'Run',\n        action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n        inputs: {\n          commands: this.prefixCommandsWithErrorHandling(props.platform, props.commands),\n        },\n      });\n    }\n\n    const data = {\n      name: props.displayName,\n      schemaVersion: '1.0',\n      phases: [\n        {\n          name: 'build',\n          steps,\n        },\n      ],\n    };\n\n    const name = uniqueImageBuilderName(this);\n    const component = new imagebuilder.CfnComponent(this, 'Component', {\n      name: name,\n      description: props.description,\n      platform: props.platform,\n      version: this.version('Component', name, {\n        platform: props.platform,\n        data,\n        description: props.description,\n      }),\n      data: JSON.stringify(data),\n    });\n\n    this.arn = component.attrArn;\n  }\n\n  /**\n   * Grants read permissions to the principal on the assets buckets.\n   *\n   * @param grantee\n   */\n  grantAssetsRead(grantee: iam.IGrantable) {\n    for (const asset of this.assets) {\n      asset.grantRead(grantee);\n    }\n  }\n\n  prefixCommandsWithErrorHandling(platform: 'Windows' | 'Linux', commands: string[]) {\n    if (platform == 'Windows') {\n      return [\n        '$ErrorActionPreference = \\'Stop\\'',\n        '$ProgressPreference = \\'SilentlyContinue\\'',\n      ].concat(commands);\n    } else {\n      return [\n        'set -ex',\n      ].concat(commands);\n    }\n  }\n}\n\n/**\n * @internal\n */\nexport class AwsImageBuilderRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private boundAmi?: RunnerAmi;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: string;\n  private readonly baseAmi: string;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc;\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly repository: ecr.Repository;\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly boundComponents: ImageBuilderComponent[] = [];\n  private readonly instanceType: ec2.InstanceType;\n  private infrastructure: imagebuilder.CfnInfrastructureConfiguration | undefined;\n  private readonly role: iam.Role;\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.codeBuildOptions) {\n      Annotations.of(this).addWarning('codeBuildOptions are ignored when using AWS Image Builder to build runner images.');\n    }\n\n    this.os = props?.os ?? Os.LINUX_UBUNTU;\n    this.architecture = props?.architecture ?? Architecture.X86_64;\n    this.rebuildInterval = props?.rebuildInterval ?? Duration.days(7);\n    this.logRetention = props?.logRetention ?? RetentionDays.ONE_MONTH;\n    this.logRemovalPolicy = props?.logRemovalPolicy ?? RemovalPolicy.DESTROY;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];\n    this.subnetSelection = props?.subnetSelection;\n    this.baseImage = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.baseAmi = props?.baseAmi ?? defaultBaseAmi(this.os, this.architecture).getImage(this).imageId;\n    this.instanceType = props?.awsImageBuilderOptions?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n\n    // confirm instance type\n    if (!this.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`Builder architecture (${this.architecture.name}) doesn't match selected instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    // warn against isolated networks\n    if (props?.subnetSelection?.subnetType == ec2.SubnetType.PRIVATE_ISOLATED) {\n      Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +\n        'See https://github.com/aws/containers-roadmap/issues/1160');\n    }\n\n    // role to be used by AWS Image Builder\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n\n    // create repository that only keeps one tag\n    this.repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      lifecycleRules: [\n        {\n          description: 'Remove untagged images that have been replaced by CodeBuild',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n    });\n  }\n\n  private platform() {\n    if (this.os.is(Os.WINDOWS)) {\n      return 'Windows';\n    }\n    if (this.os.is(Os.LINUX_AMAZON_2) || this.os.is(Os.LINUX_UBUNTU)) {\n      return 'Linux';\n    }\n    throw new Error(`OS ${this.os.name} is not supported by AWS Image Builder`);\n  }\n\n  /**\n   * Called by IRunnerProvider to finalize settings and create the image builder.\n   */\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'Docker Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          containerDistributionConfiguration: {\n            ContainerTags: ['latest'],\n            TargetRepository: {\n              Service: 'ECR',\n              RepositoryName: this.repository.repositoryName,\n            },\n          },\n        },\n      ],\n    });\n\n    let dockerfileTemplate = `FROM {{{ imagebuilder:parentImage }}}\n{{{ imagebuilder:environments }}}\n{{{ imagebuilder:components }}}`;\n\n    if (this.boundComponents.length == 0) {\n      this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i}`, this.os, this.architecture)));\n    }\n\n    for (const c of this.components) {\n      const commands = c.getDockerCommands(this.os, this.architecture);\n      if (commands.length > 0) {\n        dockerfileTemplate += '\\n' + commands.join('\\n') + '\\n';\n      }\n    }\n\n    const recipe = new ContainerRecipe(this, 'Container Recipe', {\n      platform: this.platform(),\n      components: this.boundComponents,\n      targetRepository: this.repository,\n      dockerfileTemplate: dockerfileTemplate,\n      parentImage: this.baseImage,\n    });\n\n    const log = this.createLog('Docker Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilderECRContainerBuilds'),\n    ]);\n    const image = this.createImage(infra, dist, log, undefined, recipe.arn);\n    this.createPipeline(infra, dist, log, undefined, recipe.arn);\n\n    this.imageCleaner(image, recipe.name);\n\n    this.boundDockerImage = {\n      // There are simpler ways to get the ARN, but we want an image object that depends on the newly built image.\n      // We want whoever is using this image to automatically wait for Image Builder to finish building before using the image.\n      imageRepository: ecr.Repository.fromRepositoryName(\n        this, 'Dependable Image',\n        // we can't use image.attrName because it comes up with upper case\n        cdk.Fn.split(':', cdk.Fn.split('/', image.attrImageUri, 2)[1], 2)[0],\n      ),\n      imageTag: 'latest',\n      os: this.os,\n      architecture: this.architecture,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n    };\n\n    return this.boundDockerImage;\n  }\n\n  private imageCleaner(image: imagebuilder.CfnImage, recipeName: string) {\n    const crHandler = singletonLambda(BuildImageFunction, this, 'build-image', {\n      description: 'Custom resource handler that triggers CodeBuild to build runner images, and cleans-up images on deletion',\n      timeout: cdk.Duration.minutes(3),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    const policy = new iam.Policy(this, 'CR Policy', {\n      statements: [\n        new iam.PolicyStatement({\n          actions: ['ecr:BatchDeleteImage', 'ecr:ListImages'],\n          resources: [this.repository.repositoryArn],\n        }),\n        new iam.PolicyStatement({\n          actions: ['imagebuilder:ListImages', 'imagebuilder:ListImageBuildVersions', 'imagebuilder:DeleteImage'],\n          resources: ['*'], // Image Builder doesn't support scoping this :(\n        }),\n      ],\n    });\n    crHandler.role?.attachInlinePolicy(policy);\n\n    const cr = new CustomResource(this, 'Deleter', {\n      serviceToken: crHandler.functionArn,\n      resourceType: 'Custom::ImageDeleter',\n      properties: {\n        RepoName: this.repository.repositoryName,\n        ImageBuilderName: recipeName, // we don't use image.name because CloudFormation complains if it was deleted already\n        DeleteOnly: true,\n      },\n    });\n\n    // add dependencies to make sure resources are there when we need them\n    cr.node.addDependency(image);\n    cr.node.addDependency(policy);\n    cr.node.addDependency(crHandler);\n\n    return cr;\n  }\n\n  protected createLog(id: string, recipeName: string): logs.LogGroup {\n    return new logs.LogGroup(this, id, {\n      logGroupName: `/aws/imagebuilder/${recipeName}`,\n      retention: this.logRetention,\n      removalPolicy: this.logRemovalPolicy,\n    });\n  }\n\n  protected createInfrastructure(managedPolicies: iam.IManagedPolicy[]): imagebuilder.CfnInfrastructureConfiguration {\n    if (this.infrastructure) {\n      return this.infrastructure;\n    }\n\n    for (const managedPolicy of managedPolicies) {\n      this.role.addManagedPolicy(managedPolicy);\n    }\n\n    for (const component of this.boundComponents) {\n      component.grantAssetsRead(this.role);\n    }\n\n    this.infrastructure = new imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      subnetId: this.vpc?.selectSubnets(this.subnetSelection).subnetIds[0],\n      securityGroupIds: this.securityGroups?.map(sg => sg.securityGroupId),\n      instanceTypes: [this.instanceType.toString()],\n      instanceProfileName: new iam.CfnInstanceProfile(this, 'Instance Profile', {\n        roles: [\n          this.role.roleName,\n        ],\n      }).ref,\n    });\n\n    return this.infrastructure;\n  }\n\n  protected createImage(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImage {\n    const image = new imagebuilder.CfnImage(this, this.amiOrContainerId('Image', imageRecipeArn, containerRecipeArn), {\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    image.node.addDependency(infra);\n    image.node.addDependency(log);\n\n    return image;\n  }\n\n  private amiOrContainerId(baseId: string, imageRecipeArn?: string, containerRecipeArn?: string) {\n    if (imageRecipeArn) {\n      return `AMI ${baseId}`;\n    }\n    if (containerRecipeArn) {\n      return `Docker ${baseId}`;\n    }\n    throw new Error('Either imageRecipeArn or containerRecipeArn must be defined');\n  }\n\n  protected createPipeline(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImagePipeline {\n    let scheduleOptions: imagebuilder.CfnImagePipeline.ScheduleProperty | undefined;\n    if (this.rebuildInterval.toDays() > 0) {\n      scheduleOptions = {\n        scheduleExpression: events.Schedule.rate(this.rebuildInterval).expressionString,\n        pipelineExecutionStartCondition: 'EXPRESSION_MATCH_ONLY',\n      };\n    }\n    const pipeline = new imagebuilder.CfnImagePipeline(this, this.amiOrContainerId('Pipeline', imageRecipeArn, containerRecipeArn), {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      schedule: scheduleOptions,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    pipeline.node.addDependency(infra);\n    pipeline.node.addDependency(log);\n\n    return pipeline;\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n\n  public get grantPrincipal(): iam.IPrincipal {\n    return this.role;\n  }\n\n  bindAmi(): RunnerAmi {\n    if (this.boundAmi) {\n      return this.boundAmi;\n    }\n\n    const launchTemplate = new ec2.LaunchTemplate(this, 'Launch template');\n\n    const stackName = cdk.Stack.of(this).stackName;\n    const builderName = this.node.path;\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'AMI Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          amiDistributionConfiguration: {\n            Name: `${cdk.Names.uniqueResourceName(this, {\n              maxLength: 100,\n              separator: '-',\n              allowedSpecialCharacters: '_-',\n            })}-{{ imagebuilder:buildDate }}`,\n            AmiTags: {\n              'Name': this.node.id,\n              'GitHubRunners:Stack': stackName,\n              'GitHubRunners:Builder': builderName,\n            },\n          },\n          launchTemplateConfigurations: [\n            {\n              launchTemplateId: launchTemplate.launchTemplateId,\n            },\n          ],\n        },\n      ],\n    });\n\n    const recipe = new AmiRecipe(this, 'Ami Recipe', {\n      platform: this.platform(),\n      components: this.bindComponents(),\n      architecture: this.architecture,\n      baseAmi: this.baseAmi,\n    });\n\n    const log = this.createLog('Ami Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilder'),\n    ]);\n    this.createImage(infra, dist, log, recipe.arn, undefined);\n    this.createPipeline(infra, dist, log, recipe.arn, undefined);\n\n    this.boundAmi = {\n      launchTemplate: launchTemplate,\n      architecture: this.architecture,\n      os: this.os,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n    };\n\n    this.amiCleaner(launchTemplate, stackName, builderName);\n\n    return this.boundAmi;\n  }\n\n  private amiCleaner(launchTemplate: ec2.LaunchTemplate, stackName: string, builderName: string) {\n    const deleter = singletonLambda(DeleteAmiFunction, this, 'delete-ami', {\n      description: 'Delete old GitHub Runner AMIs',\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: ['ec2:DescribeLaunchTemplateVersions', 'ec2:DescribeImages', 'ec2:DeregisterImage', 'ec2:DeleteSnapshot'],\n          resources: ['*'],\n        }),\n      ],\n      timeout: cdk.Duration.minutes(5),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    // delete old AMIs on schedule\n    const eventRule = new events.Rule(this, 'Delete AMI Schedule', {\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n      description: `Delete old AMIs for ${builderName}`,\n    });\n    eventRule.addTarget(new events_targets.LambdaFunction(deleter, {\n      event: events.RuleTargetInput.fromObject({\n        RequestType: 'Scheduled',\n        LaunchTemplateId: launchTemplate.launchTemplateId,\n        StackName: stackName,\n        BuilderName: builderName,\n      }),\n    }));\n\n    // delete all AMIs when this construct is removed\n    new CustomResource(this, 'AMI Deleter', {\n      serviceToken: deleter.functionArn,\n      resourceType: 'Custom::AmiDeleter',\n      properties: {\n        StackName: stackName,\n        BuilderName: builderName,\n      },\n    });\n  }\n\n  private bindComponents(): ImageBuilderComponent[] {\n    if (this.boundComponents.length == 0) {\n      this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i} ${c.name}`, this.os, this.architecture)));\n    }\n\n    return this.boundComponents;\n  }\n}\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import * as cdk from 'aws-cdk-lib';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
/**
|
|
4
|
+
* @internal
|
|
5
|
+
*/
|
|
6
|
+
export declare abstract class ImageBuilderObjectBase extends cdk.Resource {
|
|
7
|
+
protected constructor(scope: Construct, id: string);
|
|
8
|
+
protected version(type: 'Component' | 'ImageRecipe' | 'ContainerRecipe', name: string, data: any): string;
|
|
9
|
+
private versionFunction;
|
|
10
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ImageBuilderObjectBase = void 0;
|
|
4
|
+
const cdk = require("aws-cdk-lib");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
const aws_image_builder_versioner_function_1 = require("../../../lambdas/aws-image-builder-versioner-function");
|
|
7
|
+
const utils_1 = require("../../../utils");
|
|
8
|
+
/**
|
|
9
|
+
* @internal
|
|
10
|
+
*/
|
|
11
|
+
class ImageBuilderObjectBase extends cdk.Resource {
|
|
12
|
+
constructor(scope, id) {
|
|
13
|
+
super(scope, id);
|
|
14
|
+
}
|
|
15
|
+
version(type, name, data) {
|
|
16
|
+
return new aws_cdk_lib_1.CustomResource(this, 'Version', {
|
|
17
|
+
serviceToken: this.versionFunction().functionArn,
|
|
18
|
+
resourceType: `Custom::ImageBuilder-${type}-Version`,
|
|
19
|
+
removalPolicy: cdk.RemovalPolicy.RETAIN,
|
|
20
|
+
properties: {
|
|
21
|
+
ObjectType: type,
|
|
22
|
+
ObjectName: name,
|
|
23
|
+
VersionedData: data, // get a new version every time something changes, like Image Builder wants
|
|
24
|
+
},
|
|
25
|
+
}).ref;
|
|
26
|
+
}
|
|
27
|
+
versionFunction() {
|
|
28
|
+
return (0, utils_1.singletonLambda)(aws_image_builder_versioner_function_1.AwsImageBuilderVersionerFunction, this, 'aws-image-builder-versioner', {
|
|
29
|
+
description: 'Custom resource handler that bumps up Image Builder versions',
|
|
30
|
+
initialPolicy: [
|
|
31
|
+
new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
32
|
+
actions: [
|
|
33
|
+
'imagebuilder:ListComponents',
|
|
34
|
+
'imagebuilder:ListContainerRecipes',
|
|
35
|
+
'imagebuilder:ListImageRecipes',
|
|
36
|
+
],
|
|
37
|
+
resources: ['*'],
|
|
38
|
+
}),
|
|
39
|
+
],
|
|
40
|
+
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
|
|
41
|
+
timeout: cdk.Duration.minutes(5),
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
exports.ImageBuilderObjectBase = ImageBuilderObjectBase;
|
|
46
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3Byb3ZpZGVycy9pbWFnZS1idWlsZGVycy9hd3MtaW1hZ2UtYnVpbGRlci9jb21tb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbUNBQW1DO0FBQ25DLDZDQUErRTtBQUUvRSxnSEFBeUc7QUFDekcsMENBQWlEO0FBRWpEOztHQUVHO0FBQ0gsTUFBc0Isc0JBQXVCLFNBQVEsR0FBRyxDQUFDLFFBQVE7SUFDL0QsWUFBc0IsS0FBZ0IsRUFBRSxFQUFVO1FBQ2hELEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDbkIsQ0FBQztJQUVTLE9BQU8sQ0FBQyxJQUFxRCxFQUFFLElBQVksRUFBRSxJQUFTO1FBQzlGLE9BQU8sSUFBSSw0QkFBYyxDQUFDLElBQUksRUFBRSxTQUFTLEVBQUU7WUFDekMsWUFBWSxFQUFFLElBQUksQ0FBQyxlQUFlLEVBQUUsQ0FBQyxXQUFXO1lBQ2hELFlBQVksRUFBRSx3QkFBd0IsSUFBSSxVQUFVO1lBQ3BELGFBQWEsRUFBRSxHQUFHLENBQUMsYUFBYSxDQUFDLE1BQU07WUFDdkMsVUFBVSxFQUFFO2dCQUNWLFVBQVUsRUFBRSxJQUFJO2dCQUNoQixVQUFVLEVBQUUsSUFBSTtnQkFDaEIsYUFBYSxFQUFFLElBQUksRUFBRSwyRUFBMkU7YUFDakc7U0FDRixDQUFDLENBQUMsR0FBRyxDQUFDO0lBQ1QsQ0FBQztJQUVPLGVBQWU7UUFDckIsT0FBTyxJQUFBLHVCQUFlLEVBQUMsdUVBQWdDLEVBQUUsSUFBSSxFQUFFLDZCQUE2QixFQUFFO1lBQzVGLFdBQVcsRUFBRSw4REFBOEQ7WUFDM0UsYUFBYSxFQUFFO2dCQUNiLElBQUkscUJBQUcsQ0FBQyxlQUFlLENBQUM7b0JBQ3RCLE9BQU8sRUFBRTt3QkFDUCw2QkFBNkI7d0JBQzdCLG1DQUFtQzt3QkFDbkMsK0JBQStCO3FCQUNoQztvQkFDRCxTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUM7aUJBQ2pCLENBQUM7YUFDSDtZQUNELFlBQVksRUFBRSxzQkFBSSxDQUFDLGFBQWEsQ0FBQyxTQUFTO1lBQzFDLE9BQU8sRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7U0FDakMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBbkNELHdEQW1DQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGNkayBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBhd3NfaWFtIGFzIGlhbSwgYXdzX2xvZ3MgYXMgbG9ncywgQ3VzdG9tUmVzb3VyY2UgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEF3c0ltYWdlQnVpbGRlclZlcnNpb25lckZ1bmN0aW9uIH0gZnJvbSAnLi4vLi4vLi4vbGFtYmRhcy9hd3MtaW1hZ2UtYnVpbGRlci12ZXJzaW9uZXItZnVuY3Rpb24nO1xuaW1wb3J0IHsgc2luZ2xldG9uTGFtYmRhIH0gZnJvbSAnLi4vLi4vLi4vdXRpbHMnO1xuXG4vKipcbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgSW1hZ2VCdWlsZGVyT2JqZWN0QmFzZSBleHRlbmRzIGNkay5SZXNvdXJjZSB7XG4gIHByb3RlY3RlZCBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcbiAgfVxuXG4gIHByb3RlY3RlZCB2ZXJzaW9uKHR5cGU6ICdDb21wb25lbnQnIHwgJ0ltYWdlUmVjaXBlJyB8ICdDb250YWluZXJSZWNpcGUnLCBuYW1lOiBzdHJpbmcsIGRhdGE6IGFueSk6IHN0cmluZyB7XG4gICAgcmV0dXJuIG5ldyBDdXN0b21SZXNvdXJjZSh0aGlzLCAnVmVyc2lvbicsIHtcbiAgICAgIHNlcnZpY2VUb2tlbjogdGhpcy52ZXJzaW9uRnVuY3Rpb24oKS5mdW5jdGlvbkFybixcbiAgICAgIHJlc291cmNlVHlwZTogYEN1c3RvbTo6SW1hZ2VCdWlsZGVyLSR7dHlwZX0tVmVyc2lvbmAsXG4gICAgICByZW1vdmFsUG9saWN5OiBjZGsuUmVtb3ZhbFBvbGljeS5SRVRBSU4sIC8vIG5vIHBvaW50IGluIGRlbGV0aW5nIGFzIGl0IGRvZXNuJ3QgZXZlbiBjcmVhdGUgYW55dGhpbmdcbiAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgT2JqZWN0VHlwZTogdHlwZSxcbiAgICAgICAgT2JqZWN0TmFtZTogbmFtZSxcbiAgICAgICAgVmVyc2lvbmVkRGF0YTogZGF0YSwgLy8gZ2V0IGEgbmV3IHZlcnNpb24gZXZlcnkgdGltZSBzb21ldGhpbmcgY2hhbmdlcywgbGlrZSBJbWFnZSBCdWlsZGVyIHdhbnRzXG4gICAgICB9LFxuICAgIH0pLnJlZjtcbiAgfVxuXG4gIHByaXZhdGUgdmVyc2lvbkZ1bmN0aW9uKCk6IEF3c0ltYWdlQnVpbGRlclZlcnNpb25lckZ1bmN0aW9uIHtcbiAgICByZXR1cm4gc2luZ2xldG9uTGFtYmRhKEF3c0ltYWdlQnVpbGRlclZlcnNpb25lckZ1bmN0aW9uLCB0aGlzLCAnYXdzLWltYWdlLWJ1aWxkZXItdmVyc2lvbmVyJywge1xuICAgICAgZGVzY3JpcHRpb246ICdDdXN0b20gcmVzb3VyY2UgaGFuZGxlciB0aGF0IGJ1bXBzIHVwIEltYWdlIEJ1aWxkZXIgdmVyc2lvbnMnLFxuICAgICAgaW5pdGlhbFBvbGljeTogW1xuICAgICAgICBuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgICAgYWN0aW9uczogW1xuICAgICAgICAgICAgJ2ltYWdlYnVpbGRlcjpMaXN0Q29tcG9uZW50cycsXG4gICAgICAgICAgICAnaW1hZ2VidWlsZGVyOkxpc3RDb250YWluZXJSZWNpcGVzJyxcbiAgICAgICAgICAgICdpbWFnZWJ1aWxkZXI6TGlzdEltYWdlUmVjaXBlcycsXG4gICAgICAgICAgXSxcbiAgICAgICAgICByZXNvdXJjZXM6IFsnKiddLFxuICAgICAgICB9KSxcbiAgICAgIF0sXG4gICAgICBsb2dSZXRlbnRpb246IGxvZ3MuUmV0ZW50aW9uRGF5cy5PTkVfTU9OVEgsXG4gICAgICB0aW1lb3V0OiBjZGsuRHVyYXRpb24ubWludXRlcyg1KSxcbiAgICB9KTtcbiAgfVxufVxuXG4iXX0=
|