@cloudsnorkel/cdk-github-runners 0.8.3 → 0.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. package/.jsii +2257 -744
  2. package/API.md +2005 -558
  3. package/README.md +32 -34
  4. package/assets/docker-images/codebuild/linux-arm64/Dockerfile +2 -0
  5. package/assets/docker-images/codebuild/linux-x64/Dockerfile +2 -0
  6. package/assets/docker-images/fargate/linux-arm64/Dockerfile +2 -0
  7. package/assets/docker-images/fargate/linux-x64/Dockerfile +2 -0
  8. package/assets/docker-images/lambda/linux-arm64/Dockerfile +3 -1
  9. package/assets/docker-images/lambda/linux-arm64/runner.sh +1 -1
  10. package/assets/docker-images/lambda/linux-x64/Dockerfile +3 -1
  11. package/assets/docker-images/lambda/linux-x64/runner.sh +1 -1
  12. package/assets/lambdas/aws-image-builder-versioner.lambda/index.js +42 -27
  13. package/assets/lambdas/setup.lambda/index.html +12 -12
  14. package/lib/index.js +7 -3
  15. package/lib/lambdas/aws-image-builder-versioner-function.js +2 -2
  16. package/lib/lambdas/aws-image-builder-versioner.lambda.js +43 -27
  17. package/lib/lambdas/build-image-function.js +2 -2
  18. package/lib/lambdas/build-image.lambda.js +4 -4
  19. package/lib/lambdas/delete-ami-function.js +2 -2
  20. package/lib/lambdas/delete-ami.lambda.js +4 -4
  21. package/lib/lambdas/delete-runner-function.js +2 -2
  22. package/lib/lambdas/delete-runner.lambda.js +2 -2
  23. package/lib/lambdas/github.js +3 -3
  24. package/lib/lambdas/setup-function.js +2 -2
  25. package/lib/lambdas/setup.lambda.js +16 -16
  26. package/lib/lambdas/status-function.js +2 -2
  27. package/lib/lambdas/status.lambda.js +5 -5
  28. package/lib/lambdas/token-retriever-function.js +2 -2
  29. package/lib/lambdas/token-retriever.lambda.js +2 -2
  30. package/lib/lambdas/update-lambda-function.js +2 -2
  31. package/lib/lambdas/webhook-handler-function.js +2 -2
  32. package/lib/lambdas/webhook-handler.lambda.js +2 -2
  33. package/lib/providers/codebuild.d.ts +24 -4
  34. package/lib/providers/codebuild.js +43 -13
  35. package/lib/providers/common.d.ts +17 -39
  36. package/lib/providers/common.js +26 -16
  37. package/lib/providers/ec2.d.ts +23 -5
  38. package/lib/providers/ec2.js +43 -12
  39. package/lib/providers/fargate.d.ts +21 -4
  40. package/lib/providers/fargate.js +50 -20
  41. package/lib/providers/image-builders/api.d.ts +15 -0
  42. package/lib/providers/image-builders/api.js +47 -0
  43. package/lib/providers/image-builders/aws-image-builder/ami.d.ts +43 -0
  44. package/lib/providers/image-builders/aws-image-builder/ami.js +81 -0
  45. package/lib/providers/image-builders/aws-image-builder/builder.d.ts +133 -0
  46. package/lib/providers/image-builders/aws-image-builder/builder.js +488 -0
  47. package/lib/providers/image-builders/aws-image-builder/common.d.ts +10 -0
  48. package/lib/providers/image-builders/aws-image-builder/common.js +46 -0
  49. package/lib/providers/image-builders/aws-image-builder/container.d.ts +58 -0
  50. package/lib/providers/image-builders/aws-image-builder/container.js +63 -0
  51. package/lib/providers/image-builders/{ami.d.ts → aws-image-builder/deprecated/ami.d.ts} +8 -4
  52. package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +239 -0
  53. package/lib/providers/image-builders/aws-image-builder/deprecated/common.d.ts +34 -0
  54. package/lib/providers/image-builders/aws-image-builder/deprecated/common.js +139 -0
  55. package/lib/providers/image-builders/{container.d.ts → aws-image-builder/deprecated/container.d.ts} +8 -4
  56. package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +222 -0
  57. package/lib/providers/image-builders/aws-image-builder/deprecated/index.d.ts +5 -0
  58. package/lib/providers/image-builders/aws-image-builder/deprecated/index.js +22 -0
  59. package/lib/providers/image-builders/{linux-components.d.ts → aws-image-builder/deprecated/linux-components.d.ts} +4 -2
  60. package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +180 -0
  61. package/lib/providers/image-builders/{windows-components.d.ts → aws-image-builder/deprecated/windows-components.d.ts} +4 -2
  62. package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +142 -0
  63. package/lib/providers/image-builders/aws-image-builder/index.d.ts +5 -0
  64. package/lib/providers/image-builders/aws-image-builder/index.js +22 -0
  65. package/lib/providers/image-builders/codebuild-deprecated.d.ts +195 -0
  66. package/lib/providers/image-builders/codebuild-deprecated.js +373 -0
  67. package/lib/providers/image-builders/codebuild.d.ts +26 -157
  68. package/lib/providers/image-builders/codebuild.js +119 -211
  69. package/lib/providers/image-builders/common.d.ts +164 -107
  70. package/lib/providers/image-builders/common.js +30 -272
  71. package/lib/providers/image-builders/components.d.ts +114 -0
  72. package/lib/providers/image-builders/components.js +534 -0
  73. package/lib/providers/image-builders/index.d.ts +6 -4
  74. package/lib/providers/image-builders/index.js +13 -7
  75. package/lib/providers/image-builders/static.d.ts +4 -3
  76. package/lib/providers/image-builders/static.js +10 -10
  77. package/lib/providers/index.js +7 -3
  78. package/lib/providers/lambda.d.ts +25 -6
  79. package/lib/providers/lambda.js +50 -13
  80. package/lib/runner.d.ts +3 -5
  81. package/lib/runner.js +3 -3
  82. package/lib/secrets.js +3 -3
  83. package/package.json +12 -16
  84. package/lib/providers/image-builders/ami.js +0 -280
  85. package/lib/providers/image-builders/container.js +0 -247
  86. package/lib/providers/image-builders/linux-components.js +0 -177
  87. package/lib/providers/image-builders/windows-components.js +0 -139
@@ -1,14 +1,8 @@
1
1
  "use strict";
2
- var _a;
3
2
  Object.defineProperty(exports, "__esModule", { value: true });
4
- exports.ImageBuilderBase = exports.ImageBuilderComponent = exports.ImageBuilderObjectBase = exports.uniqueImageBuilderName = void 0;
5
- const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
3
+ exports.RunnerImageBuilderBase = exports.RunnerImageBuilderType = exports.uniqueImageBuilderName = void 0;
6
4
  const cdk = require("aws-cdk-lib");
7
- const aws_cdk_lib_1 = require("aws-cdk-lib");
8
5
  const constructs_1 = require("constructs");
9
- const aws_image_builder_versioner_function_1 = require("../../lambdas/aws-image-builder-versioner-function");
10
- const utils_1 = require("../../utils");
11
- const common_1 = require("../common");
12
6
  /**
13
7
  * @internal
14
8
  */
@@ -20,284 +14,48 @@ function uniqueImageBuilderName(scope) {
20
14
  });
21
15
  }
22
16
  exports.uniqueImageBuilderName = uniqueImageBuilderName;
23
- /**
24
- * @internal
25
- */
26
- class ImageBuilderObjectBase extends cdk.Resource {
27
- constructor(scope, id) {
28
- super(scope, id);
29
- }
30
- version(type, name, data) {
31
- return new aws_cdk_lib_1.CustomResource(this, 'Version', {
32
- serviceToken: this.versionFunction().functionArn,
33
- resourceType: `Custom::ImageBuilder-${type}-Version`,
34
- removalPolicy: cdk.RemovalPolicy.RETAIN,
35
- properties: {
36
- ObjectType: type,
37
- ObjectName: name,
38
- VersionedData: data,
39
- },
40
- }).ref;
41
- }
42
- versionFunction() {
43
- return utils_1.singletonLambda(aws_image_builder_versioner_function_1.AwsImageBuilderVersionerFunction, this, 'aws-image-builder-versioner', {
44
- description: 'Custom resource handler that bumps up Image Builder versions',
45
- initialPolicy: [
46
- new aws_cdk_lib_1.aws_iam.PolicyStatement({
47
- actions: [
48
- 'imagebuilder:ListComponents',
49
- 'imagebuilder:ListContainerRecipes',
50
- 'imagebuilder:ListImageRecipes',
51
- ],
52
- resources: ['*'],
53
- }),
54
- ],
55
- logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
56
- });
57
- }
58
- }
59
- exports.ImageBuilderObjectBase = ImageBuilderObjectBase;
60
- /**
61
- * 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.
62
- *
63
- * Example:
64
- *
65
- * ```
66
- * new ImageBuilderComponent(this, 'AWS CLI', {
67
- * platform: 'Windows',
68
- * displayName: 'AWS CLI',
69
- * description: 'Install latest version of AWS CLI',
70
- * commands: [
71
- * '$ErrorActionPreference = \'Stop\'',
72
- * 'Start-Process msiexec.exe -Wait -ArgumentList \'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\'',
73
- * ],
74
- * }
75
- * ```
76
- */
77
- class ImageBuilderComponent extends ImageBuilderObjectBase {
78
- constructor(scope, id, props) {
79
- super(scope, id);
80
- this.assets = [];
81
- this.platform = props.platform;
82
- let steps = [];
83
- if (props.assets) {
84
- let inputs = [];
85
- let extractCommands = [];
86
- for (const asset of props.assets) {
87
- this.assets.push(asset.asset);
88
- if (asset.asset.isFile) {
89
- inputs.push({
90
- source: asset.asset.s3ObjectUrl,
91
- destination: asset.path,
92
- });
93
- }
94
- else if (asset.asset.isZipArchive) {
95
- inputs.push({
96
- source: asset.asset.s3ObjectUrl,
97
- destination: `${asset.path}.zip`,
98
- });
99
- if (props.platform === 'Windows') {
100
- extractCommands.push('$ErrorActionPreference = \'Stop\'');
101
- extractCommands.push(`Expand-Archive "${asset.path}.zip" -DestinationPath "${asset.path}"`);
102
- extractCommands.push(`del "${asset.path}.zip"`);
103
- }
104
- else {
105
- extractCommands.push(`unzip "${asset.path}.zip" -d "${asset.path}"`);
106
- extractCommands.push(`rm "${asset.path}.zip"`);
107
- }
108
- }
109
- else {
110
- throw new Error(`Unknown asset type: ${asset.asset}`);
111
- }
112
- }
113
- steps.push({
114
- name: 'Download',
115
- action: 'S3Download',
116
- inputs,
117
- });
118
- if (extractCommands.length > 0) {
119
- steps.push({
120
- name: 'Extract',
121
- action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',
122
- inputs: {
123
- commands: extractCommands,
124
- },
125
- });
126
- }
127
- }
128
- steps.push({
129
- name: 'Run',
130
- action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',
131
- inputs: {
132
- commands: props.commands,
133
- },
134
- });
135
- const data = {
136
- name: props.displayName,
137
- schemaVersion: '1.0',
138
- phases: [
139
- {
140
- name: 'build',
141
- steps,
142
- },
143
- ],
144
- };
145
- const name = uniqueImageBuilderName(this);
146
- const component = new aws_cdk_lib_1.aws_imagebuilder.CfnComponent(this, 'Component', {
147
- name: name,
148
- description: props.description,
149
- platform: props.platform,
150
- version: this.version('Component', name, {
151
- platform: props.platform,
152
- data,
153
- description: props.description,
154
- }),
155
- data: JSON.stringify(data),
156
- });
157
- this.arn = component.attrArn;
158
- }
17
+ var RunnerImageBuilderType;
18
+ (function (RunnerImageBuilderType) {
159
19
  /**
160
- * Grants read permissions to the principal on the assets buckets.
20
+ * Build runner images using AWS CodeBuild.
161
21
  *
162
- * @param grantee
22
+ * Faster than AWS Image Builder, but can only be used to build Linux Docker images.
163
23
  */
164
- grantAssetsRead(grantee) {
165
- for (const asset of this.assets) {
166
- asset.grantRead(grantee);
167
- }
168
- }
169
- }
170
- exports.ImageBuilderComponent = ImageBuilderComponent;
171
- _a = JSII_RTTI_SYMBOL_1;
172
- ImageBuilderComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.ImageBuilderComponent", version: "0.8.3" };
24
+ RunnerImageBuilderType["CODE_BUILD"] = "CodeBuild";
25
+ /**
26
+ * Build runner images using AWS Image Builder.
27
+ *
28
+ * Slower than CodeBuild, but can be used to build any type of image including AMIs and Windows images.
29
+ */
30
+ RunnerImageBuilderType["AWS_IMAGE_BUILDER"] = "AwsImageBuilder";
31
+ })(RunnerImageBuilderType = exports.RunnerImageBuilderType || (exports.RunnerImageBuilderType = {}));
173
32
  /**
174
33
  * @internal
175
34
  */
176
- class ImageBuilderBase extends constructs_1.Construct {
35
+ class RunnerImageBuilderBase extends constructs_1.Construct {
177
36
  constructor(scope, id, props) {
178
37
  super(scope, id);
179
38
  this.components = [];
180
- // arch
181
- this.architecture = props?.architecture ?? common_1.Architecture.X86_64;
182
- if (!this.architecture.isIn(props.supportedArchitectures)) {
183
- throw new Error(`Unsupported architecture: ${this.architecture.name}. Consider CodeBuild for faster image builds.`);
184
- }
185
- // os
186
- this.os = props?.os ?? common_1.Os.LINUX;
187
- if (!this.os.isIn(props.supportedOs)) {
188
- throw new Error(`Unsupported OS: ${this.os.name}.`);
189
- }
190
- // platform
191
- if (this.os.is(common_1.Os.WINDOWS)) {
192
- this.platform = 'Windows';
193
- }
194
- else if (this.os.is(common_1.Os.LINUX)) {
195
- this.platform = 'Linux';
196
- }
197
- else {
198
- throw new Error(`Unsupported OS: ${this.os.name}.`);
199
- }
200
- // builder options
201
- this.rebuildInterval = props?.rebuildInterval ?? aws_cdk_lib_1.Duration.days(7);
202
- // vpc settings
203
- if (props?.vpc) {
204
- this.vpc = props.vpc;
205
- this.subnetId = props.vpc.selectSubnets(props.subnetSelection).subnetIds[0];
206
- }
207
- else {
208
- this.vpc = aws_cdk_lib_1.aws_ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });
209
- }
210
- if (props?.securityGroups) {
211
- this.securityGroups = props.securityGroups;
212
- }
213
- else {
214
- this.securityGroups = [new aws_cdk_lib_1.aws_ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];
215
- }
216
- // instance type
217
- this.instanceType = props?.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);
218
- if (!this.architecture.instanceTypeMatch(this.instanceType)) {
219
- throw new Error(`Builder architecture (${this.architecture.name}) doesn't match selected instance type (${this.instanceType} / ${this.instanceType.architecture})`);
39
+ if (props?.components) {
40
+ this.components.push(...props.components);
220
41
  }
221
- // log settings
222
- this.logRetention = props?.logRetention ?? aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH;
223
- this.logRemovalPolicy = props?.logRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.DESTROY;
224
- // runner version
225
- this.runnerVersion = props?.runnerVersion ?? common_1.RunnerVersion.latest();
226
- // description
227
- this.description = `Build ${props.imageTypeName} for GitHub Actions runner ${this.node.path} (${this.os.name}/${this.architecture.name})`;
228
42
  }
229
- createLog(recipeName) {
230
- return new aws_cdk_lib_1.aws_logs.LogGroup(this, 'Log', {
231
- logGroupName: `/aws/imagebuilder/${recipeName}`,
232
- retention: this.logRetention,
233
- removalPolicy: this.logRemovalPolicy,
234
- });
235
- }
236
- createInfrastructure(managedPolicies) {
237
- let role = new aws_cdk_lib_1.aws_iam.Role(this, 'Role', {
238
- assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('ec2.amazonaws.com'),
239
- managedPolicies: managedPolicies,
240
- });
241
- for (const component of this.components) {
242
- component.grantAssetsRead(role);
243
- }
244
- return new aws_cdk_lib_1.aws_imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {
245
- name: uniqueImageBuilderName(this),
246
- description: this.description,
247
- subnetId: this.subnetId,
248
- securityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),
249
- instanceTypes: [this.instanceType.toString()],
250
- instanceProfileName: new aws_cdk_lib_1.aws_iam.CfnInstanceProfile(this, 'Instance Profile', {
251
- roles: [
252
- role.roleName,
253
- ],
254
- }).ref,
255
- });
256
- }
257
- createImage(infra, dist, log, imageRecipeArn, containerRecipeArn) {
258
- const image = new aws_cdk_lib_1.aws_imagebuilder.CfnImage(this, 'Image', {
259
- infrastructureConfigurationArn: infra.attrArn,
260
- distributionConfigurationArn: dist.attrArn,
261
- imageRecipeArn,
262
- containerRecipeArn,
263
- imageTestsConfiguration: {
264
- imageTestsEnabled: false,
265
- },
266
- });
267
- image.node.addDependency(infra);
268
- image.node.addDependency(log);
269
- return image;
270
- }
271
- createPipeline(infra, dist, log, imageRecipeArn, containerRecipeArn) {
272
- let scheduleOptions;
273
- if (this.rebuildInterval.toDays() > 0) {
274
- scheduleOptions = {
275
- scheduleExpression: aws_cdk_lib_1.aws_events.Schedule.rate(this.rebuildInterval).expressionString,
276
- pipelineExecutionStartCondition: 'EXPRESSION_MATCH_ONLY',
277
- };
278
- }
279
- const pipeline = new aws_cdk_lib_1.aws_imagebuilder.CfnImagePipeline(this, 'Pipeline', {
280
- name: uniqueImageBuilderName(this),
281
- description: this.description,
282
- infrastructureConfigurationArn: infra.attrArn,
283
- distributionConfigurationArn: dist.attrArn,
284
- imageRecipeArn,
285
- containerRecipeArn,
286
- schedule: scheduleOptions,
287
- imageTestsConfiguration: {
288
- imageTestsEnabled: false,
289
- },
290
- });
291
- pipeline.node.addDependency(infra);
292
- pipeline.node.addDependency(log);
293
- return pipeline;
43
+ /**
44
+ * Add a component to the image builder. The component will be added to the end of the list of components.
45
+ *
46
+ * @param component component to add
47
+ */
48
+ addComponent(component) {
49
+ this.components.push(component);
294
50
  }
295
51
  /**
296
- * The network connections associated with this resource.
52
+ * Remove a component from the image builder. Removal is done by component name. Multiple components with the same name will all be removed.
53
+ *
54
+ * @param component component to remove
297
55
  */
298
- get connections() {
299
- return new aws_cdk_lib_1.aws_ec2.Connections({ securityGroups: this.securityGroups });
56
+ removeComponent(component) {
57
+ this.components = this.components.filter(c => c.name !== component.name);
300
58
  }
301
59
  }
302
- exports.ImageBuilderBase = ImageBuilderBase;
303
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../../src/providers/image-builders/common.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_events as events,\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} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { AwsImageBuilderVersionerFunction } from '../../lambdas/aws-image-builder-versioner-function';\nimport { singletonLambda } from '../../utils';\nimport { Architecture, Os, RunnerVersion } from '../common';\n\n/**\n * @internal\n */\nexport function uniqueImageBuilderName(scope: Construct): string {\n  return cdk.Names.uniqueResourceName(scope, {\n    maxLength: 126,\n    separator: '-',\n    allowedSpecialCharacters: '_-',\n  });\n}\n\n/**\n * @internal\n */\nexport abstract class ImageBuilderObjectBase extends cdk.Resource {\n  protected constructor(scope: Construct, id: string) {\n    super(scope, id);\n  }\n\n  protected version(type: 'Component' | 'ImageRecipe' | 'ContainerRecipe', name: string, data: any): string {\n    return new CustomResource(this, 'Version', {\n      serviceToken: this.versionFunction().functionArn,\n      resourceType: `Custom::ImageBuilder-${type}-Version`,\n      removalPolicy: cdk.RemovalPolicy.RETAIN, // no point in deleting as it doesn't even create anything\n      properties: {\n        ObjectType: type,\n        ObjectName: name,\n        VersionedData: data, // get a new version every time something changes, like Image Builder wants\n      },\n    }).ref;\n  }\n\n  private versionFunction(): AwsImageBuilderVersionerFunction {\n    return singletonLambda(AwsImageBuilderVersionerFunction, this, 'aws-image-builder-versioner', {\n      description: 'Custom resource handler that bumps up Image Builder versions',\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: [\n            'imagebuilder:ListComponents',\n            'imagebuilder:ListContainerRecipes',\n            'imagebuilder:ListImageRecipes',\n          ],\n          resources: ['*'],\n        }),\n      ],\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n  }\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 */\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('$ErrorActionPreference = \\'Stop\\'');\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: extractCommands,\n          },\n        });\n      }\n    }\n\n    steps.push({\n      name: 'Run',\n      action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n      inputs: {\n        commands: props.commands,\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\n/**\n * @internal\n */\nexport interface ImageBuilderBaseProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * List of supported architectures to be checked against {@link architecture}.\n   */\n  readonly supportedArchitectures: Architecture[];\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX\n   */\n  readonly os?: Os;\n\n  /**\n   * List of supported OS to be checked against {@link os}.\n   */\n  readonly supportedOs: Os[];\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Schedule the AMI to be rebuilt every given interval. Useful for keeping the AMI up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC where builder instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security groups to assign to launched builder instances.\n   *\n   * @default new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * Pipeline and infrastructure description.\n   */\n  readonly imageTypeName: string;\n}\n\n/**\n * @internal\n */\nexport abstract class ImageBuilderBase extends Construct implements ec2.IConnectable {\n  protected readonly architecture: Architecture;\n  protected readonly os: Os;\n  protected readonly platform: 'Windows' | 'Linux';\n\n  protected readonly description: string;\n\n  protected readonly runnerVersion: RunnerVersion;\n\n  protected components: ImageBuilderComponent[] = [];\n\n  private readonly vpc: ec2.IVpc;\n  private readonly subnetId: string | undefined;\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly instanceType: ec2.InstanceType;\n\n  private readonly rebuildInterval: Duration;\n  private readonly logRetention: logs.RetentionDays;\n  private readonly logRemovalPolicy: cdk.RemovalPolicy;\n\n  protected constructor(scope: Construct, id: string, props: ImageBuilderBaseProps) {\n    super(scope, id);\n\n    // arch\n    this.architecture = props?.architecture ?? Architecture.X86_64;\n    if (!this.architecture.isIn(props.supportedArchitectures)) {\n      throw new Error(`Unsupported architecture: ${this.architecture.name}. Consider CodeBuild for faster image builds.`);\n    }\n\n    // os\n    this.os = props?.os ?? Os.LINUX;\n    if (!this.os.isIn(props.supportedOs)) {\n      throw new Error(`Unsupported OS: ${this.os.name}.`);\n    }\n\n    // platform\n    if (this.os.is(Os.WINDOWS)) {\n      this.platform = 'Windows';\n    } else if (this.os.is(Os.LINUX)) {\n      this.platform = 'Linux';\n    } else {\n      throw new Error(`Unsupported OS: ${this.os.name}.`);\n    }\n\n    // builder options\n    this.rebuildInterval = props?.rebuildInterval ?? Duration.days(7);\n\n    // vpc settings\n    if (props?.vpc) {\n      this.vpc = props.vpc;\n      this.subnetId = props.vpc.selectSubnets(props.subnetSelection).subnetIds[0];\n    } else {\n      this.vpc = ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    }\n\n    if (props?.securityGroups) {\n      this.securityGroups = props.securityGroups;\n    } else {\n      this.securityGroups = [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];\n    }\n\n    // instance type\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\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    // log settings\n    this.logRetention = props?.logRetention ?? logs.RetentionDays.ONE_MONTH;\n    this.logRemovalPolicy = props?.logRemovalPolicy ?? RemovalPolicy.DESTROY;\n\n    // runner version\n    this.runnerVersion = props?.runnerVersion ?? RunnerVersion.latest();\n\n    // description\n    this.description = `Build ${props.imageTypeName} for GitHub Actions runner ${this.node.path} (${this.os.name}/${this.architecture.name})`;\n  }\n\n  protected createLog(recipeName: string): logs.LogGroup {\n    return new logs.LogGroup(this, 'Log', {\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    let role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n      managedPolicies: managedPolicies,\n    });\n\n    for (const component of this.components) {\n      component.grantAssetsRead(role);\n    }\n\n    return new imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {\n      name: uniqueImageBuilderName(this),\n      description: this.description,\n      subnetId: this.subnetId,\n      securityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n      instanceTypes: [this.instanceType.toString()],\n      instanceProfileName: new iam.CfnInstanceProfile(this, 'Instance Profile', {\n        roles: [\n          role.roleName,\n        ],\n      }).ref,\n    });\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, 'Image', {\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  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, 'Pipeline', {\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"]}
60
+ exports.RunnerImageBuilderBase = RunnerImageBuilderBase;
61
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../../src/providers/image-builders/common.ts"],"names":[],"mappings":";;;AAAA,mCAAmC;AAEnC,2CAAuC;AAMvC;;GAEG;AACH,SAAgB,sBAAsB,CAAC,KAAgB;IACrD,OAAO,GAAG,CAAC,KAAK,CAAC,kBAAkB,CAAC,KAAK,EAAE;QACzC,SAAS,EAAE,GAAG;QACd,SAAS,EAAE,GAAG;QACd,wBAAwB,EAAE,IAAI;KAC/B,CAAC,CAAC;AACL,CAAC;AAND,wDAMC;AA0ND,IAAY,sBAcX;AAdD,WAAY,sBAAsB;IAChC;;;;OAIG;IACH,kDAAwB,CAAA;IAExB;;;;OAIG;IACH,+DAAqC,CAAA;AACvC,CAAC,EAdW,sBAAsB,GAAtB,8BAAsB,KAAtB,8BAAsB,QAcjC;AA6BD;;GAEG;AACH,MAAsB,sBAAuB,SAAQ,sBAAS;IAG5D,YAAsB,KAAgB,EAAE,EAAU,EAAE,KAA+B;QACjF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAHT,eAAU,GAA2B,EAAE,CAAC;QAKhD,IAAI,KAAK,EAAE,UAAU,EAAE;YACrB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,UAAU,CAAC,CAAC;SAC3C;IACH,CAAC;IASD;;;;OAIG;IACI,YAAY,CAAC,SAA+B;QACjD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAED;;;;OAIG;IACI,eAAe,CAAC,SAA+B;QACpD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI,CAAC,CAAC;IAC3E,CAAC;CACF;AAnCD,wDAmCC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { aws_ec2 as ec2, aws_iam as iam, aws_logs as logs, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { AwsImageBuilderRunnerImageBuilderProps } from './aws-image-builder';\nimport { CodeBuildRunnerImageBuilderProps } from './codebuild';\nimport { RunnerImageComponent } from './components';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../common';\n\n/**\n * @internal\n */\nexport function uniqueImageBuilderName(scope: Construct): string {\n  return cdk.Names.uniqueResourceName(scope, {\n    maxLength: 126,\n    separator: '-',\n    allowedSpecialCharacters: '_-',\n  });\n}\n\n/**\n * @internal\n */\nexport interface ImageBuilderBaseProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * List of supported architectures to be checked against {@link architecture}.\n   */\n  readonly supportedArchitectures: Architecture[];\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX\n   */\n  readonly os?: Os;\n\n  /**\n   * List of supported OS to be checked against {@link os}.\n   */\n  readonly supportedOs: Os[];\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Schedule the AMI to be rebuilt every given interval. Useful for keeping the AMI up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC where builder instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security groups to assign to launched builder instances.\n   *\n   * @default new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * Pipeline and infrastructure description.\n   */\n  readonly imageTypeName: string;\n}\n\n/**\n * Asset to copy into a built image.\n */\nexport interface RunnerImageAsset {\n  /**\n   * Path on local system to copy into the image. Can be a file or a directory.\n   */\n  readonly source: string;\n\n  /**\n   * Target path in the built image.\n   */\n  readonly target: string;\n}\n\nexport interface RunnerImageBuilderProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX\n   */\n  readonly os?: Os;\n\n  /**\n   * Base image from which Docker runner images will be built.\n   *\n   * @default public.ecr.aws/lts/ubuntu:22.04 for Os.LINUX_UBUNTU, public.ecr.aws/amazonlinux/amazonlinux:2 for Os.LINUX_AMAZON_2, mcr.microsoft.com/windows/servercore:ltsc2019-amd64 for Os.WINDOWS\n   */\n  readonly baseDockerImage?: string;\n\n  /**\n   * Base AMI from which runner AMIs will be built.\n   *\n   * @default latest Ubuntu 20.04 AMI for Os.LINUX_UBUNTU, latest Amazon Linux 2 AMI for Os.LINUX_AMAZON_2, latest Windows Server 2022 AMI for Os.WINDOWS\n   */\n  readonly baseAmi?: string;\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Components to install on the image.\n   *\n   * @default none\n   */\n  readonly components?: RunnerImageComponent[];\n\n  /**\n   * Schedule the image to be rebuilt every given interval. Useful for keeping the image up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC to build the image in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Groups to assign to this instance.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the CodeBuild logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * @default CodeBuild for Linux Docker image, AWS Image Builder for Windows Docker image and any AMI\n   */\n  readonly builderType?: RunnerImageBuilderType;\n\n  /**\n   * Options specific to CodeBuild image builder. Only used when builderType is RunnerImageBuilderType.CODE_BUILD.\n   */\n  readonly codeBuildOptions?: CodeBuildRunnerImageBuilderProps;\n\n  /**\n   * Options specific to AWS Image Builder. Only used when builderType is RunnerImageBuilderType.AWS_IMAGE_BUILDER.\n   */\n  readonly awsImageBuilderOptions?: AwsImageBuilderRunnerImageBuilderProps;\n}\n\nexport enum RunnerImageBuilderType {\n  /**\n   * Build runner images using AWS CodeBuild.\n   *\n   * Faster than AWS Image Builder, but can only be used to build Linux Docker images.\n   */\n  CODE_BUILD = 'CodeBuild',\n\n  /**\n   * Build runner images using AWS Image Builder.\n   *\n   * Slower than CodeBuild, but can be used to build any type of image including AMIs and Windows images.\n   */\n  AWS_IMAGE_BUILDER = 'AwsImageBuilder',\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.\n *\n * An image can be a Docker image or AMI.\n */\nexport interface IRunnerImageBuilder {\n  /**\n   * Build and return a Docker image with GitHub Runner installed in it.\n   *\n   * Anything that ends up with an ECR repository containing a Docker image that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing image and nothing else.\n   *\n   * It's important that the specified image tag be available at the time the repository is available. Providers usually assume the image is ready and will fail if it's not.\n   *\n   * The image can be further updated over time manually or using a schedule as long as it is always written to the same tag.\n   */\n  bindDockerImage(): RunnerImage;\n\n  /**\n   * Build and return an AMI with GitHub Runner installed in it.\n   *\n   * Anything that ends up with a launch template pointing to an AMI that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing AMI and nothing else.\n   *\n   * The AMI can be further updated over time manually or using a schedule as long as it is always written to the same launch template.\n   */\n  bindAmi(): RunnerAmi;\n}\n\n/**\n * @internal\n */\nexport abstract class RunnerImageBuilderBase extends Construct implements ec2.IConnectable, iam.IGrantable, IRunnerImageBuilder {\n  protected components: RunnerImageComponent[] = [];\n\n  protected constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id);\n\n    if (props?.components) {\n      this.components.push(...props.components);\n    }\n  }\n\n  abstract bindDockerImage(): RunnerImage;\n\n  abstract bindAmi(): RunnerAmi;\n\n  abstract get connections(): ec2.Connections;\n  abstract get grantPrincipal(): iam.IPrincipal;\n\n  /**\n   * Add a component to the image builder. The component will be added to the end of the list of components.\n   *\n   * @param component component to add\n   */\n  public addComponent(component: RunnerImageComponent) {\n    this.components.push(component);\n  }\n\n  /**\n   * Remove a component from the image builder. Removal is done by component name. Multiple components with the same name will all be removed.\n   *\n   * @param component component to remove\n   */\n  public removeComponent(component: RunnerImageComponent) {\n    this.components = this.components.filter(c => c.name !== component.name);\n  }\n}\n\n"]}
@@ -0,0 +1,114 @@
1
+ import { Construct } from 'constructs';
2
+ import { ImageBuilderComponent } from './aws-image-builder';
3
+ import { RunnerImageAsset } from './common';
4
+ import { Architecture, Os, RunnerVersion } from '../common';
5
+ export interface RunnerImageComponentCustomProps {
6
+ /**
7
+ * Component name used for (1) image build logging and (2) identifier for {@link ImageRunnerBuilder.removeComponent}.
8
+ *
9
+ * Name must only contain alphanumeric characters and dashes.
10
+ */
11
+ readonly name?: string;
12
+ /**
13
+ * Commands to run in the built image.
14
+ */
15
+ readonly commands?: string[];
16
+ /**
17
+ * Assets to copy into the built image.
18
+ */
19
+ readonly assets?: RunnerImageAsset[];
20
+ /**
21
+ * Docker commands to run in the built image.
22
+ *
23
+ * For example: `['ENV foo=bar', 'RUN echo $foo']`
24
+ *
25
+ * These commands are ignored when building AMIs.
26
+ */
27
+ readonly dockerCommands?: string[];
28
+ }
29
+ /**
30
+ * Components are used to build runner images. They can run commands in the image, copy files into the image, and run some Docker commands.
31
+ */
32
+ export declare abstract class RunnerImageComponent {
33
+ /**
34
+ * Define a custom component that can run commands in the image, copy files into the image, and run some Docker commands.
35
+ *
36
+ * The order of operations is (1) assets (2) commands (3) docker commands.
37
+ *
38
+ * Use this to customize the image for the runner.
39
+ *
40
+ * **WARNING:** Docker commands are not guaranteed to be included before the next component
41
+ */
42
+ static custom(props: RunnerImageComponentCustomProps): RunnerImageComponent;
43
+ /**
44
+ * A component to install the required packages for the runner.
45
+ */
46
+ static requiredPackages(): RunnerImageComponent;
47
+ /**
48
+ * A component to prepare the required runner user.
49
+ */
50
+ static runnerUser(): RunnerImageComponent;
51
+ /**
52
+ * A component to install the AWS CLI.
53
+ */
54
+ static awsCli(): RunnerImageComponent;
55
+ /**
56
+ * A component to install the GitHub CLI.
57
+ */
58
+ static githubCli(): RunnerImageComponent;
59
+ /**
60
+ * A component to install the GitHub CLI.
61
+ */
62
+ static git(): RunnerImageComponent;
63
+ /**
64
+ * A component to install the GitHub Actions Runner. This is the actual executable that connects to GitHub to ask for jobs and then execute them.
65
+ *
66
+ * @param runnerVersion The version of the runner to install. Usually you would set this to latest.
67
+ */
68
+ static githubRunner(runnerVersion: RunnerVersion): RunnerImageComponent;
69
+ /**
70
+ * A component to install Docker. On Windows this installs Docker Desktop.
71
+ */
72
+ static docker(): RunnerImageComponent;
73
+ /**
74
+ * A component to install Docker-in-Docker.
75
+ */
76
+ static dockerInDocker(): RunnerImageComponent;
77
+ /**
78
+ * A component to add a trusted certificate authority. This can be used to support GitHub Enterprise Server with self-signed certificate.
79
+ *
80
+ * @param source path to certificate file in PEM format
81
+ * @param name unique certificate name to be used on runner file system
82
+ */
83
+ static extraCertificates(source: string, name: string): RunnerImageComponent;
84
+ /**
85
+ * A component to set up the required Lambda entrypoint for Lambda runners.
86
+ */
87
+ static lambdaEntrypoint(): RunnerImageComponent;
88
+ /**
89
+ * Component name.
90
+ *
91
+ * Used to identify component in image build logs, and for {@link RunnerImageBuilder.removeComponent}
92
+ */
93
+ abstract readonly name: string;
94
+ /**
95
+ * Returns commands to run to in built image. Can be used to install packages, setup build prerequisites, etc.
96
+ */
97
+ abstract getCommands(_os: Os, _architecture: Architecture): string[];
98
+ /**
99
+ * Returns assets to copy into the built image. Can be used to copy files into the image.
100
+ */
101
+ getAssets(_os: Os, _architecture: Architecture): RunnerImageAsset[];
102
+ /**
103
+ * Returns Docker commands to run to in built image. Can be used to add commands like `VOLUME`, `ENTRYPOINT`, `CMD`, etc.
104
+ *
105
+ * Docker commands are added after assets and normal commands.
106
+ */
107
+ getDockerCommands(_os: Os, _architecture: Architecture): string[];
108
+ /**
109
+ * Convert component to an AWS Image Builder component.
110
+ *
111
+ * @internal
112
+ */
113
+ _asAwsImageBuilderComponent(scope: Construct, id: string, os: Os, architecture: Architecture): ImageBuilderComponent;
114
+ }