@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.
Files changed (80) hide show
  1. package/.jsii +2258 -745
  2. package/API.md +2006 -559
  3. package/README.md +44 -42
  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/setup.lambda/index.html +1 -1
  13. package/assets/lambdas/webhook-handler.lambda/index.js +1 -1
  14. package/lib/index.js +7 -3
  15. package/lib/lambdas/aws-image-builder-versioner.lambda.js +6 -6
  16. package/lib/lambdas/build-image.lambda.js +4 -4
  17. package/lib/lambdas/delete-ami.lambda.js +4 -4
  18. package/lib/lambdas/delete-runner.lambda.js +2 -2
  19. package/lib/lambdas/github.js +3 -3
  20. package/lib/lambdas/setup.lambda.js +16 -16
  21. package/lib/lambdas/status.lambda.js +5 -5
  22. package/lib/lambdas/token-retriever.lambda.js +2 -2
  23. package/lib/lambdas/webhook-handler.lambda.js +3 -3
  24. package/lib/providers/codebuild.d.ts +24 -4
  25. package/lib/providers/codebuild.js +42 -12
  26. package/lib/providers/common.d.ts +17 -39
  27. package/lib/providers/common.js +26 -16
  28. package/lib/providers/ec2.d.ts +23 -5
  29. package/lib/providers/ec2.js +43 -12
  30. package/lib/providers/ecs.d.ts +214 -0
  31. package/lib/providers/ecs.js +258 -0
  32. package/lib/providers/fargate.d.ts +26 -6
  33. package/lib/providers/fargate.js +81 -42
  34. package/lib/providers/image-builders/api.d.ts +15 -0
  35. package/lib/providers/image-builders/api.js +47 -0
  36. package/lib/providers/image-builders/aws-image-builder/ami.d.ts +43 -0
  37. package/lib/providers/image-builders/aws-image-builder/ami.js +81 -0
  38. package/lib/providers/image-builders/aws-image-builder/builder.d.ts +133 -0
  39. package/lib/providers/image-builders/aws-image-builder/builder.js +488 -0
  40. package/lib/providers/image-builders/aws-image-builder/common.d.ts +10 -0
  41. package/lib/providers/image-builders/aws-image-builder/common.js +46 -0
  42. package/lib/providers/image-builders/aws-image-builder/container.d.ts +58 -0
  43. package/lib/providers/image-builders/aws-image-builder/container.js +63 -0
  44. package/lib/providers/image-builders/{ami.d.ts → aws-image-builder/deprecated/ami.d.ts} +8 -4
  45. package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +239 -0
  46. package/lib/providers/image-builders/aws-image-builder/deprecated/common.d.ts +34 -0
  47. package/lib/providers/image-builders/aws-image-builder/deprecated/common.js +139 -0
  48. package/lib/providers/image-builders/{container.d.ts → aws-image-builder/deprecated/container.d.ts} +8 -4
  49. package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +222 -0
  50. package/lib/providers/image-builders/aws-image-builder/deprecated/index.d.ts +5 -0
  51. package/lib/providers/image-builders/aws-image-builder/deprecated/index.js +22 -0
  52. package/lib/providers/image-builders/{linux-components.d.ts → aws-image-builder/deprecated/linux-components.d.ts} +4 -2
  53. package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +180 -0
  54. package/lib/providers/image-builders/{windows-components.d.ts → aws-image-builder/deprecated/windows-components.d.ts} +4 -2
  55. package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +142 -0
  56. package/lib/providers/image-builders/aws-image-builder/index.d.ts +5 -0
  57. package/lib/providers/image-builders/aws-image-builder/index.js +22 -0
  58. package/lib/providers/image-builders/codebuild-deprecated.d.ts +195 -0
  59. package/lib/providers/image-builders/codebuild-deprecated.js +373 -0
  60. package/lib/providers/image-builders/codebuild.d.ts +26 -157
  61. package/lib/providers/image-builders/codebuild.js +118 -210
  62. package/lib/providers/image-builders/common.d.ts +164 -107
  63. package/lib/providers/image-builders/common.js +30 -272
  64. package/lib/providers/image-builders/components.d.ts +114 -0
  65. package/lib/providers/image-builders/components.js +535 -0
  66. package/lib/providers/image-builders/index.d.ts +6 -4
  67. package/lib/providers/image-builders/index.js +13 -7
  68. package/lib/providers/image-builders/static.d.ts +4 -3
  69. package/lib/providers/image-builders/static.js +10 -10
  70. package/lib/providers/index.js +7 -3
  71. package/lib/providers/lambda.d.ts +25 -6
  72. package/lib/providers/lambda.js +50 -13
  73. package/lib/runner.d.ts +3 -5
  74. package/lib/runner.js +4 -4
  75. package/lib/secrets.js +3 -3
  76. package/package.json +7 -11
  77. package/lib/providers/image-builders/ami.js +0 -280
  78. package/lib/providers/image-builders/container.js +0 -247
  79. package/lib/providers/image-builders/linux-components.js +0 -177
  80. package/lib/providers/image-builders/windows-components.js +0 -139
@@ -1,14 +1,14 @@
1
1
  "use strict";
2
2
  var _a, _b;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
- exports.FargateRunner = exports.FargateRunnerProvider = void 0;
4
+ exports.FargateRunner = exports.FargateRunnerProvider = exports.ecsRunCommand = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
6
  const path = require("path");
7
7
  const aws_cdk_lib_1 = require("aws-cdk-lib");
8
8
  const aws_logs_1 = require("aws-cdk-lib/aws-logs");
9
9
  const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
10
10
  const common_1 = require("./common");
11
- const codebuild_1 = require("./image-builders/codebuild");
11
+ const image_builders_1 = require("./image-builders");
12
12
  /**
13
13
  * Our special launch target that can use spot instances and set EnableExecuteCommand.
14
14
  */
@@ -25,6 +25,7 @@ class EcsFargateLaunchTarget {
25
25
  }
26
26
  return {
27
27
  parameters: {
28
+ PropagateTags: aws_cdk_lib_1.aws_ecs.PropagatedTagSource.TASK_DEFINITION,
28
29
  EnableExecuteCommand: this.props.enableExecute,
29
30
  CapacityProviderStrategy: [
30
31
  {
@@ -35,6 +36,42 @@ class EcsFargateLaunchTarget {
35
36
  };
36
37
  }
37
38
  }
39
+ /**
40
+ * @internal
41
+ */
42
+ function ecsRunCommand(os, dind) {
43
+ if (os.is(common_1.Os.LINUX) || os.is(common_1.Os.LINUX_UBUNTU) || os.is(common_1.Os.LINUX_AMAZON_2)) {
44
+ let dindCommand = '';
45
+ if (dind) {
46
+ dindCommand = 'sudo socat UNIX-LISTEN:/var/run/docker.sock,group=docker,mode=770,fork UNIX-CONNECT:/var/run/docker.sock.host &';
47
+ }
48
+ return [
49
+ 'sh', '-c',
50
+ `${dindCommand}
51
+ cd /home/runner &&
52
+ if [ "$RUNNER_VERSION" = "latest" ]; then RUNNER_FLAGS=""; else RUNNER_FLAGS="--disableupdate"; fi &&
53
+ ./config.sh --unattended --url "https://$GITHUB_DOMAIN/$OWNER/$REPO" --token "$RUNNER_TOKEN" --ephemeral --work _work --labels "$RUNNER_LABEL" $RUNNER_FLAGS --name "$RUNNER_NAME" &&
54
+ ./run.sh &&
55
+ STATUS=$(grep -Phors "finish job request for job [0-9a-f\\-]+ with result: \\K.*" _diag/ | tail -n1) &&
56
+ [ -n "$STATUS" ] && echo CDKGHA JOB DONE "$RUNNER_LABEL" "$STATUS"`,
57
+ ];
58
+ }
59
+ else if (os.is(common_1.Os.WINDOWS)) {
60
+ return [
61
+ 'powershell', '-Command',
62
+ `cd \\actions ;
63
+ if ($Env:RUNNER_VERSION -eq "latest") { $RunnerFlags = "" } else { $RunnerFlags = "--disableupdate" } ;
64
+ ./config.cmd --unattended --url "https://\${Env:GITHUB_DOMAIN}/\${Env:OWNER}/\${Env:REPO}" --token "\${Env:RUNNER_TOKEN}" --ephemeral --work _work --labels "\${Env:RUNNER_LABEL}" $RunnerFlags --name "\${Env:RUNNER_NAME}" ;
65
+ ./run.cmd ;
66
+ $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1 ;
67
+ if ($STATUS) { echo "CDKGHA JOB DONE $\{Env:RUNNER_LABEL\} $STATUS" }`,
68
+ ];
69
+ }
70
+ else {
71
+ throw new Error(`Fargate runner doesn't support ${os.name}`);
72
+ }
73
+ }
74
+ exports.ecsRunCommand = ecsRunCommand;
38
75
  /**
39
76
  * GitHub Actions runner provider using Fargate to execute jobs.
40
77
  *
@@ -43,6 +80,32 @@ class EcsFargateLaunchTarget {
43
80
  * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
44
81
  */
45
82
  class FargateRunnerProvider extends common_1.BaseProvider {
83
+ /**
84
+ * Create new image builder that builds Fargate specific runner images using Ubuntu.
85
+ *
86
+ * Included components:
87
+ * * `RunnerImageComponent.requiredPackages()`
88
+ * * `RunnerImageComponent.runnerUser()`
89
+ * * `RunnerImageComponent.git()`
90
+ * * `RunnerImageComponent.githubCli()`
91
+ * * `RunnerImageComponent.awsCli()`
92
+ * * `RunnerImageComponent.githubRunner()`
93
+ */
94
+ static imageBuilder(scope, id, props) {
95
+ return image_builders_1.RunnerImageBuilder.new(scope, id, {
96
+ os: common_1.Os.LINUX_UBUNTU,
97
+ architecture: common_1.Architecture.X86_64,
98
+ components: [
99
+ image_builders_1.RunnerImageComponent.requiredPackages(),
100
+ image_builders_1.RunnerImageComponent.runnerUser(),
101
+ image_builders_1.RunnerImageComponent.git(),
102
+ image_builders_1.RunnerImageComponent.githubCli(),
103
+ image_builders_1.RunnerImageComponent.awsCli(),
104
+ image_builders_1.RunnerImageComponent.githubRunner(props?.runnerVersion ?? common_1.RunnerVersion.latest()),
105
+ ],
106
+ ...props,
107
+ });
108
+ }
46
109
  constructor(scope, id, props) {
47
110
  super(scope, id, props);
48
111
  this.labels = this.labelsFromProperties('fargate', props?.label, props?.labels);
@@ -56,10 +119,8 @@ class FargateRunnerProvider extends common_1.BaseProvider {
56
119
  enableFargateCapacityProviders: true,
57
120
  });
58
121
  this.spot = props?.spot ?? false;
59
- const imageBuilder = props?.imageBuilder ?? new codebuild_1.CodeBuildImageBuilder(this, 'Image Builder', {
60
- dockerfilePath: FargateRunnerProvider.LINUX_X64_DOCKERFILE_PATH,
61
- });
62
- const image = this.image = imageBuilder.bind();
122
+ const imageBuilder = props?.imageBuilder ?? FargateRunnerProvider.imageBuilder(this, 'Image Builder');
123
+ const image = this.image = imageBuilder.bindDockerImage();
63
124
  let arch;
64
125
  if (image.architecture.is(common_1.Architecture.ARM64)) {
65
126
  arch = aws_cdk_lib_1.aws_ecs.CpuArchitecture.ARM64;
@@ -71,7 +132,7 @@ class FargateRunnerProvider extends common_1.BaseProvider {
71
132
  throw new Error(`${image.architecture.name} is not supported on Fargate`);
72
133
  }
73
134
  let os;
74
- if (image.os.is(common_1.Os.LINUX)) {
135
+ if (image.os.is(common_1.Os.LINUX) || image.os.is(common_1.Os.LINUX_UBUNTU) || image.os.is(common_1.Os.LINUX_AMAZON_2)) {
75
136
  os = aws_cdk_lib_1.aws_ecs.OperatingSystemFamily.LINUX;
76
137
  }
77
138
  else if (image.os.is(common_1.Os.WINDOWS)) {
@@ -102,9 +163,10 @@ class FargateRunnerProvider extends common_1.BaseProvider {
102
163
  logGroup: this.logGroup,
103
164
  streamPrefix: 'runner',
104
165
  }),
105
- command: this.runCommand(),
166
+ command: ecsRunCommand(this.image.os, false),
167
+ user: image.os.is(common_1.Os.WINDOWS) ? undefined : 'runner',
106
168
  });
107
- this.grantPrincipal = new aws_cdk_lib_1.aws_iam.UnknownPrincipal({ resource: this.task.taskRole });
169
+ this.grantPrincipal = this.task.taskRole;
108
170
  }
109
171
  /**
110
172
  * Generate step function task(s) to start a new runner.
@@ -120,7 +182,7 @@ class FargateRunnerProvider extends common_1.BaseProvider {
120
182
  cluster: this.cluster,
121
183
  launchTarget: new EcsFargateLaunchTarget({
122
184
  spot: this.spot,
123
- enableExecute: this.image.os.is(common_1.Os.LINUX),
185
+ enableExecute: this.image.os.is(common_1.Os.LINUX) || this.image.os.is(common_1.Os.LINUX_UBUNTU) || this.image.os.is(common_1.Os.LINUX_AMAZON_2),
124
186
  }),
125
187
  subnets: this.subnetSelection,
126
188
  assignPublicIp: this.assignPublicIp,
@@ -178,43 +240,17 @@ class FargateRunnerProvider extends common_1.BaseProvider {
178
240
  },
179
241
  };
180
242
  }
181
- runCommand() {
182
- let runnerFlags = '';
183
- if (!this.image.runnerVersion.is(common_1.RunnerVersion.latest())) {
184
- runnerFlags = '--disableupdate';
185
- }
186
- if (this.image.os.is(common_1.Os.LINUX)) {
187
- return [
188
- 'sh', '-c',
189
- `./config.sh --unattended --url "https://$GITHUB_DOMAIN/$OWNER/$REPO" --token "$RUNNER_TOKEN" --ephemeral --work _work --labels "$RUNNER_LABEL" ${runnerFlags} --name "$RUNNER_NAME" &&
190
- ./run.sh &&
191
- STATUS=$(grep -Phors "finish job request for job [0-9a-f\\-]+ with result: \\K.*" _diag/ | tail -n1) &&
192
- [ -n "$STATUS" ] && echo CDKGHA JOB DONE "$RUNNER_LABEL" "$STATUS"`,
193
- ];
194
- }
195
- else if (this.image.os.is(common_1.Os.WINDOWS)) {
196
- return [
197
- 'powershell', '-Command',
198
- `cd \\actions ; ./config.cmd --unattended --url "https://\${Env:GITHUB_DOMAIN}/\${Env:OWNER}/\${Env:REPO}" --token "\${Env:RUNNER_TOKEN}" --ephemeral --work _work --labels "\${Env:RUNNER_LABEL}" ${runnerFlags} --name "\${Env:RUNNER_NAME}" ;
199
- ./run.cmd ;
200
- $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1 ;
201
- if ($STATUS) { echo "CDKGHA JOB DONE $\{Env:RUNNER_LABEL\} $STATUS" }`,
202
- ];
203
- }
204
- else {
205
- throw new Error(`Fargate runner doesn't support ${this.image.os.name}`);
206
- }
207
- }
208
243
  }
209
- exports.FargateRunnerProvider = FargateRunnerProvider;
210
244
  _a = JSII_RTTI_SYMBOL_1;
211
- FargateRunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunnerProvider", version: "0.8.4" };
245
+ FargateRunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunnerProvider", version: "0.9.1" };
212
246
  /**
213
247
  * Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.
214
248
  *
215
249
  * Available build arguments that can be set in the image builder:
216
250
  * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.
217
251
  * * `EXTRA_PACKAGES` can be used to install additional packages.
252
+ *
253
+ * @deprecated Use `imageBuilder()` instead.
218
254
  */
219
255
  FargateRunnerProvider.LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-x64');
220
256
  /**
@@ -223,14 +259,17 @@ FargateRunnerProvider.LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, '..', '..
223
259
  * Available build arguments that can be set in the image builder:
224
260
  * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.
225
261
  * * `EXTRA_PACKAGES` can be used to install additional packages.
262
+ *
263
+ * @deprecated Use `imageBuilder()` instead.
226
264
  */
227
265
  FargateRunnerProvider.LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-arm64');
266
+ exports.FargateRunnerProvider = FargateRunnerProvider;
228
267
  /**
229
268
  * @deprecated use {@link FargateRunnerProvider}
230
269
  */
231
270
  class FargateRunner extends FargateRunnerProvider {
232
271
  }
233
- exports.FargateRunner = FargateRunner;
234
272
  _b = JSII_RTTI_SYMBOL_1;
235
- FargateRunner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunner", version: "0.8.4" };
236
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate.js","sourceRoot":"","sources":["../../src/providers/fargate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_ec2 as ec2,\n  aws_ecs as ecs,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  Architecture,\n  BaseProvider,\n  IImageBuilder,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerImage,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n} from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n/**\n * Properties for FargateRunner.\n */\nexport interface FargateRunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default image builder with `FargateRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile\n   */\n  readonly imageBuilder?: IImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default undefined\n   * @deprecated use {@link labels} instead\n   */\n  readonly label?: string;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['fargate']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets to run the runners in.\n   *\n   * @default Fargate default\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security group to assign to the task.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to the task.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Existing Fargate cluster to use.\n   *\n   * @default a new cluster\n   */\n  readonly cluster?: ecs.Cluster;\n\n  /**\n   * Assign public IP to the runner task.\n   *\n   * Make sure the task will have access to GitHub. A public IP might be required unless you have NAT gateway.\n   *\n   * @default true\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The number of cpu units used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values,\n   * which determines your range of valid values for the memory parameter:\n   *\n   * 256 (.25 vCPU) - Available memory values: 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB)\n   *\n   * 512 (.5 vCPU) - Available memory values: 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB)\n   *\n   * 1024 (1 vCPU) - Available memory values: 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB)\n   *\n   * 2048 (2 vCPU) - Available memory values: Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB)\n   *\n   * 4096 (4 vCPU) - Available memory values: Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB)\n   *\n   * @default 1024\n   */\n  readonly cpu?: number;\n\n  /**\n   * The amount (in MiB) of memory used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values, which determines your range of valid values for the cpu parameter:\n   *\n   * 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB) - Available cpu values: 256 (.25 vCPU)\n   *\n   * 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB) - Available cpu values: 512 (.5 vCPU)\n   *\n   * 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB) - Available cpu values: 1024 (1 vCPU)\n   *\n   * Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB) - Available cpu values: 2048 (2 vCPU)\n   *\n   * Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB) - Available cpu values: 4096 (4 vCPU)\n   *\n   * @default 2048\n   */\n  readonly memoryLimitMiB?: number;\n\n  /**\n   * The amount (in GiB) of ephemeral storage to be allocated to the task. The maximum supported value is 200 GiB.\n   *\n   * NOTE: This parameter is only supported for tasks hosted on AWS Fargate using platform version 1.4.0 or later.\n   *\n   * @default 20\n   */\n  readonly ephemeralStorageGiB?: number;\n\n  /**\n   * Use Fargate spot capacity provider to save money.\n   *\n   * * Runners may fail to start due to missing capacity.\n   * * Runners might be stopped prematurely with spot pricing.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n}\n\n/**\n * Properties for EcsFargateLaunchTarget.\n */\ninterface EcsFargateLaunchTargetProps {\n  readonly spot: boolean;\n  readonly enableExecute: boolean;\n}\n\n/**\n * Our special launch target that can use spot instances and set EnableExecuteCommand.\n */\nclass EcsFargateLaunchTarget implements stepfunctions_tasks.IEcsLaunchTarget {\n  constructor(readonly props: EcsFargateLaunchTargetProps) {\n  }\n\n  /**\n   * Called when the Fargate launch type configured on RunTask\n   */\n  public bind(_task: stepfunctions_tasks.EcsRunTask,\n    launchTargetOptions: stepfunctions_tasks.LaunchTargetBindOptions): stepfunctions_tasks.EcsLaunchTargetConfig {\n    if (!launchTargetOptions.taskDefinition.isFargateCompatible) {\n      throw new Error('Supplied TaskDefinition is not compatible with Fargate');\n    }\n\n    return {\n      parameters: {\n        EnableExecuteCommand: this.props.enableExecute,\n        CapacityProviderStrategy: [\n          {\n            CapacityProvider: this.props.spot ? 'FARGATE_SPOT' : 'FARGATE',\n          },\n        ],\n      },\n    };\n  }\n}\n\n/**\n * GitHub Actions runner provider using Fargate to execute jobs.\n *\n * Creates a task definition with a single container that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class FargateRunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-arm64');\n\n  /**\n   * Cluster hosting the task hosting the runner.\n   */\n  readonly cluster: ecs.Cluster;\n\n  /**\n   * Fargate task hosting the runner.\n   */\n  readonly task: ecs.FargateTaskDefinition;\n\n  /**\n   * Container definition hosting the runner.\n   */\n  readonly container: ecs.ContainerDefinition;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the runner task.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets used for hosting the runner task.\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Whether runner task will have a public IP.\n   */\n  readonly assignPublicIp: boolean;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * The network connections associated with this resource.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Use spot pricing for Fargate tasks.\n   */\n  readonly spot: boolean;\n\n  /**\n   * Docker image loaded with GitHub Actions Runner and its prerequisites. The image is built by an image builder and is specific to Fargate tasks.\n   */\n  readonly image: RunnerImage;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  private readonly securityGroups: ec2.ISecurityGroup[];\n\n  constructor(scope: Construct, id: string, props?: FargateRunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = this.labelsFromProperties('fargate', props?.label, props?.labels);\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });\n    this.subnetSelection = props?.subnetSelection;\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'security group', { vpc: this.vpc })]);\n    this.connections = new ec2.Connections({ securityGroups: this.securityGroups });\n    this.assignPublicIp = props?.assignPublicIp ?? true;\n    this.cluster = props?.cluster ? props.cluster : new ecs.Cluster(\n      this,\n      'cluster',\n      {\n        vpc: this.vpc,\n        enableFargateCapacityProviders: true,\n      },\n    );\n    this.spot = props?.spot ?? false;\n\n    const imageBuilder = props?.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: FargateRunnerProvider.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    let arch: ecs.CpuArchitecture;\n    if (image.architecture.is(Architecture.ARM64)) {\n      arch = ecs.CpuArchitecture.ARM64;\n    } else if (image.architecture.is(Architecture.X86_64)) {\n      arch = ecs.CpuArchitecture.X86_64;\n    } else {\n      throw new Error(`${image.architecture.name} is not supported on Fargate`);\n    }\n\n    let os: ecs.OperatingSystemFamily;\n    if (image.os.is(Os.LINUX)) {\n      os = ecs.OperatingSystemFamily.LINUX;\n    } else if (image.os.is(Os.WINDOWS)) {\n      os = ecs.OperatingSystemFamily.WINDOWS_SERVER_2019_CORE;\n      if (props?.ephemeralStorageGiB) {\n        throw new Error('Ephemeral storage is not supported on Fargate Windows');\n      }\n    } else {\n      throw new Error(`${image.os.name} is not supported on Fargate`);\n    }\n\n    this.logGroup = new logs.LogGroup(this, 'logs', {\n      retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n      removalPolicy: RemovalPolicy.DESTROY,\n    });\n\n    this.task = new ecs.FargateTaskDefinition(\n      this,\n      'task',\n      {\n        cpu: props?.cpu ?? 1024,\n        memoryLimitMiB: props?.memoryLimitMiB ?? 2048,\n        ephemeralStorageGiB: props?.ephemeralStorageGiB ?? !image.os.is(Os.WINDOWS) ? 25 : undefined,\n        runtimePlatform: {\n          operatingSystemFamily: os,\n          cpuArchitecture: arch,\n        },\n      },\n    );\n    this.container = this.task.addContainer(\n      'runner',\n      {\n        image: ecs.AssetImage.fromEcrRepository(image.imageRepository, image.imageTag),\n        logging: ecs.AwsLogDriver.awsLogs({\n          logGroup: this.logGroup,\n          streamPrefix: 'runner',\n        }),\n        command: this.runCommand(),\n      },\n    );\n\n    this.grantPrincipal = new iam.UnknownPrincipal({ resource: this.task.taskRole });\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    const task = new stepfunctions_tasks.EcsRunTask(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        taskDefinition: this.task,\n        cluster: this.cluster,\n        launchTarget: new EcsFargateLaunchTarget({\n          spot: this.spot,\n          enableExecute: this.image.os.is(Os.LINUX),\n        }),\n        subnets: this.subnetSelection,\n        assignPublicIp: this.assignPublicIp,\n        securityGroups: this.securityGroups,\n        containerOverrides: [\n          {\n            containerDefinition: this.container,\n            environment: [\n              {\n                name: 'RUNNER_TOKEN',\n                value: parameters.runnerTokenPath,\n              },\n              {\n                name: 'RUNNER_NAME',\n                value: parameters.runnerNamePath,\n              },\n              {\n                name: 'RUNNER_LABEL',\n                value: this.labels.join(','),\n              },\n              {\n                name: 'GITHUB_DOMAIN',\n                value: parameters.githubDomainPath,\n              },\n              {\n                name: 'OWNER',\n                value: parameters.ownerPath,\n              },\n              {\n                name: 'REPO',\n                value: parameters.repoPath,\n              },\n            ],\n          },\n        ],\n      },\n    );\n\n    this.addRetry(task, ['Ecs.EcsException', 'Ecs.LimitExceededException', 'Ecs.UpdateInProgressException']);\n\n    return task;\n  }\n\n  grantStateMachine(_: iam.IGrantable) {\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      vpcArn: this.vpc?.vpcArn,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.task.taskRole.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      image: {\n        imageRepository: this.image.imageRepository.repositoryUri,\n        imageTag: this.image.imageTag,\n        imageBuilderLogGroup: this.image.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  private runCommand(): string[] {\n    let runnerFlags = '';\n    if (!this.image.runnerVersion.is(RunnerVersion.latest())) {\n      runnerFlags = '--disableupdate';\n    }\n\n    if (this.image.os.is(Os.LINUX)) {\n      return [\n        'sh', '-c',\n        `./config.sh --unattended --url \"https://$GITHUB_DOMAIN/$OWNER/$REPO\" --token \"$RUNNER_TOKEN\" --ephemeral --work _work --labels \"$RUNNER_LABEL\" ${runnerFlags} --name \"$RUNNER_NAME\" && \n        ./run.sh &&\n        STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\\\K.*\" _diag/ | tail -n1) &&\n        [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$RUNNER_LABEL\" \"$STATUS\"`,\n      ];\n    } else if (this.image.os.is(Os.WINDOWS)) {\n      return [\n        'powershell', '-Command',\n        `cd \\\\actions ; ./config.cmd --unattended --url \"https://\\${Env:GITHUB_DOMAIN}/\\${Env:OWNER}/\\${Env:REPO}\" --token \"\\${Env:RUNNER_TOKEN}\" --ephemeral --work _work --labels \"\\${Env:RUNNER_LABEL}\" ${runnerFlags} --name \"\\${Env:RUNNER_NAME}\" ; \n        ./run.cmd ; \n        $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1 ; \n        if ($STATUS) { echo \"CDKGHA JOB DONE $\\{Env:RUNNER_LABEL\\} $STATUS\" }`,\n      ];\n    } else {\n      throw new Error(`Fargate runner doesn't support ${this.image.os.name}`);\n    }\n  }\n}\n\n/**\n * @deprecated use {@link FargateRunnerProvider}\n */\nexport class FargateRunner extends FargateRunnerProvider {\n}\n"]}
273
+ FargateRunner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunner", version: "0.9.1" };
274
+ exports.FargateRunner = FargateRunner;
275
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate.js","sourceRoot":"","sources":["../../src/providers/fargate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_ec2 as ec2,\n  aws_ecs as ecs,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerImage,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n} from './common';\nimport { IRunnerImageBuilder, RunnerImageBuilder, RunnerImageBuilderProps, RunnerImageComponent } from './image-builders';\n\n/**\n * Properties for FargateRunnerProvider.\n */\nexport interface FargateRunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build Docker images containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default FargateRunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default undefined\n   * @deprecated use {@link labels} instead\n   */\n  readonly label?: string;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['fargate']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets to run the runners in.\n   *\n   * @default Fargate default\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security group to assign to the task.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to the task.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Existing Fargate cluster to use.\n   *\n   * @default a new cluster\n   */\n  readonly cluster?: ecs.Cluster;\n\n  /**\n   * Assign public IP to the runner task.\n   *\n   * Make sure the task will have access to GitHub. A public IP might be required unless you have NAT gateway.\n   *\n   * @default true\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The number of cpu units used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values,\n   * which determines your range of valid values for the memory parameter:\n   *\n   * 256 (.25 vCPU) - Available memory values: 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB)\n   *\n   * 512 (.5 vCPU) - Available memory values: 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB)\n   *\n   * 1024 (1 vCPU) - Available memory values: 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB)\n   *\n   * 2048 (2 vCPU) - Available memory values: Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB)\n   *\n   * 4096 (4 vCPU) - Available memory values: Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB)\n   *\n   * @default 1024\n   */\n  readonly cpu?: number;\n\n  /**\n   * The amount (in MiB) of memory used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values, which determines your range of valid values for the cpu parameter:\n   *\n   * 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB) - Available cpu values: 256 (.25 vCPU)\n   *\n   * 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB) - Available cpu values: 512 (.5 vCPU)\n   *\n   * 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB) - Available cpu values: 1024 (1 vCPU)\n   *\n   * Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB) - Available cpu values: 2048 (2 vCPU)\n   *\n   * Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB) - Available cpu values: 4096 (4 vCPU)\n   *\n   * @default 2048\n   */\n  readonly memoryLimitMiB?: number;\n\n  /**\n   * The amount (in GiB) of ephemeral storage to be allocated to the task. The maximum supported value is 200 GiB.\n   *\n   * NOTE: This parameter is only supported for tasks hosted on AWS Fargate using platform version 1.4.0 or later.\n   *\n   * @default 20\n   */\n  readonly ephemeralStorageGiB?: number;\n\n  /**\n   * Use Fargate spot capacity provider to save money.\n   *\n   * * Runners may fail to start due to missing capacity.\n   * * Runners might be stopped prematurely with spot pricing.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n}\n\n/**\n * Properties for EcsFargateLaunchTarget.\n */\ninterface EcsFargateLaunchTargetProps {\n  readonly spot: boolean;\n  readonly enableExecute: boolean;\n}\n\n/**\n * Our special launch target that can use spot instances and set EnableExecuteCommand.\n */\nclass EcsFargateLaunchTarget implements stepfunctions_tasks.IEcsLaunchTarget {\n  constructor(readonly props: EcsFargateLaunchTargetProps) {\n  }\n\n  /**\n   * Called when the Fargate launch type configured on RunTask\n   */\n  public bind(_task: stepfunctions_tasks.EcsRunTask,\n    launchTargetOptions: stepfunctions_tasks.LaunchTargetBindOptions): stepfunctions_tasks.EcsLaunchTargetConfig {\n    if (!launchTargetOptions.taskDefinition.isFargateCompatible) {\n      throw new Error('Supplied TaskDefinition is not compatible with Fargate');\n    }\n\n    return {\n      parameters: {\n        PropagateTags: ecs.PropagatedTagSource.TASK_DEFINITION,\n        EnableExecuteCommand: this.props.enableExecute,\n        CapacityProviderStrategy: [\n          {\n            CapacityProvider: this.props.spot ? 'FARGATE_SPOT' : 'FARGATE',\n          },\n        ],\n      },\n    };\n  }\n}\n\n/**\n * @internal\n */\nexport function ecsRunCommand(os: Os, dind: boolean): string[] {\n  if (os.is(Os.LINUX) || os.is(Os.LINUX_UBUNTU) || os.is(Os.LINUX_AMAZON_2)) {\n    let dindCommand = '';\n    if (dind) {\n      dindCommand = 'sudo socat UNIX-LISTEN:/var/run/docker.sock,group=docker,mode=770,fork UNIX-CONNECT:/var/run/docker.sock.host &';\n    }\n\n    return [\n      'sh', '-c',\n      `${dindCommand}\n        cd /home/runner &&\n        if [ \"$RUNNER_VERSION\" = \"latest\" ]; then RUNNER_FLAGS=\"\"; else RUNNER_FLAGS=\"--disableupdate\"; fi &&\n        ./config.sh --unattended --url \"https://$GITHUB_DOMAIN/$OWNER/$REPO\" --token \"$RUNNER_TOKEN\" --ephemeral --work _work --labels \"$RUNNER_LABEL\" $RUNNER_FLAGS --name \"$RUNNER_NAME\" && \n        ./run.sh &&\n        STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\\\K.*\" _diag/ | tail -n1) &&\n        [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$RUNNER_LABEL\" \"$STATUS\"`,\n    ];\n  } else if (os.is(Os.WINDOWS)) {\n    return [\n      'powershell', '-Command',\n      `cd \\\\actions ;\n        if ($Env:RUNNER_VERSION -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" } ; \n        ./config.cmd --unattended --url \"https://\\${Env:GITHUB_DOMAIN}/\\${Env:OWNER}/\\${Env:REPO}\" --token \"\\${Env:RUNNER_TOKEN}\" --ephemeral --work _work --labels \"\\${Env:RUNNER_LABEL}\" $RunnerFlags --name \"\\${Env:RUNNER_NAME}\" ; \n        ./run.cmd ; \n        $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1 ; \n        if ($STATUS) { echo \"CDKGHA JOB DONE $\\{Env:RUNNER_LABEL\\} $STATUS\" }`,\n    ];\n  } else {\n    throw new Error(`Fargate runner doesn't support ${os.name}`);\n  }\n}\n\n/**\n * GitHub Actions runner provider using Fargate to execute jobs.\n *\n * Creates a task definition with a single container that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class FargateRunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   *\n   * @deprecated Use `imageBuilder()` instead.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   *\n   * @deprecated Use `imageBuilder()` instead.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, '..', '..', 'assets', 'docker-images', 'fargate', 'linux-arm64');\n\n  /**\n   * Create new image builder that builds Fargate specific runner images using Ubuntu.\n   *\n   * Included components:\n   *  * `RunnerImageComponent.requiredPackages()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps): RunnerImageBuilder {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Cluster hosting the task hosting the runner.\n   */\n  readonly cluster: ecs.Cluster;\n\n  /**\n   * Fargate task hosting the runner.\n   */\n  readonly task: ecs.FargateTaskDefinition;\n\n  /**\n   * Container definition hosting the runner.\n   */\n  readonly container: ecs.ContainerDefinition;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the runner task.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets used for hosting the runner task.\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Whether runner task will have a public IP.\n   */\n  readonly assignPublicIp: boolean;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * The network connections associated with this resource.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Use spot pricing for Fargate tasks.\n   */\n  readonly spot: boolean;\n\n  /**\n   * Docker image loaded with GitHub Actions Runner and its prerequisites. The image is built by an image builder and is specific to Fargate tasks.\n   */\n  readonly image: RunnerImage;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  private readonly securityGroups: ec2.ISecurityGroup[];\n\n  constructor(scope: Construct, id: string, props?: FargateRunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = this.labelsFromProperties('fargate', props?.label, props?.labels);\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });\n    this.subnetSelection = props?.subnetSelection;\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'security group', { vpc: this.vpc })]);\n    this.connections = new ec2.Connections({ securityGroups: this.securityGroups });\n    this.assignPublicIp = props?.assignPublicIp ?? true;\n    this.cluster = props?.cluster ? props.cluster : new ecs.Cluster(\n      this,\n      'cluster',\n      {\n        vpc: this.vpc,\n        enableFargateCapacityProviders: true,\n      },\n    );\n    this.spot = props?.spot ?? false;\n\n    const imageBuilder = props?.imageBuilder ?? FargateRunnerProvider.imageBuilder(this, 'Image Builder');\n    const image = this.image = imageBuilder.bindDockerImage();\n\n    let arch: ecs.CpuArchitecture;\n    if (image.architecture.is(Architecture.ARM64)) {\n      arch = ecs.CpuArchitecture.ARM64;\n    } else if (image.architecture.is(Architecture.X86_64)) {\n      arch = ecs.CpuArchitecture.X86_64;\n    } else {\n      throw new Error(`${image.architecture.name} is not supported on Fargate`);\n    }\n\n    let os: ecs.OperatingSystemFamily;\n    if (image.os.is(Os.LINUX) || image.os.is(Os.LINUX_UBUNTU) || image.os.is(Os.LINUX_AMAZON_2)) {\n      os = ecs.OperatingSystemFamily.LINUX;\n    } else if (image.os.is(Os.WINDOWS)) {\n      os = ecs.OperatingSystemFamily.WINDOWS_SERVER_2019_CORE;\n      if (props?.ephemeralStorageGiB) {\n        throw new Error('Ephemeral storage is not supported on Fargate Windows');\n      }\n    } else {\n      throw new Error(`${image.os.name} is not supported on Fargate`);\n    }\n\n    this.logGroup = new logs.LogGroup(this, 'logs', {\n      retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n      removalPolicy: RemovalPolicy.DESTROY,\n    });\n\n    this.task = new ecs.FargateTaskDefinition(\n      this,\n      'task',\n      {\n        cpu: props?.cpu ?? 1024,\n        memoryLimitMiB: props?.memoryLimitMiB ?? 2048,\n        ephemeralStorageGiB: props?.ephemeralStorageGiB ?? !image.os.is(Os.WINDOWS) ? 25 : undefined,\n        runtimePlatform: {\n          operatingSystemFamily: os,\n          cpuArchitecture: arch,\n        },\n      },\n    );\n    this.container = this.task.addContainer(\n      'runner',\n      {\n        image: ecs.AssetImage.fromEcrRepository(image.imageRepository, image.imageTag),\n        logging: ecs.AwsLogDriver.awsLogs({\n          logGroup: this.logGroup,\n          streamPrefix: 'runner',\n        }),\n        command: ecsRunCommand(this.image.os, false),\n        user: image.os.is(Os.WINDOWS) ? undefined : 'runner',\n      },\n    );\n\n    this.grantPrincipal = this.task.taskRole;\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    const task = new stepfunctions_tasks.EcsRunTask(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        taskDefinition: this.task,\n        cluster: this.cluster,\n        launchTarget: new EcsFargateLaunchTarget({\n          spot: this.spot,\n          enableExecute: this.image.os.is(Os.LINUX) || this.image.os.is(Os.LINUX_UBUNTU) || this.image.os.is(Os.LINUX_AMAZON_2),\n        }),\n        subnets: this.subnetSelection,\n        assignPublicIp: this.assignPublicIp,\n        securityGroups: this.securityGroups,\n        containerOverrides: [\n          {\n            containerDefinition: this.container,\n            environment: [\n              {\n                name: 'RUNNER_TOKEN',\n                value: parameters.runnerTokenPath,\n              },\n              {\n                name: 'RUNNER_NAME',\n                value: parameters.runnerNamePath,\n              },\n              {\n                name: 'RUNNER_LABEL',\n                value: this.labels.join(','),\n              },\n              {\n                name: 'GITHUB_DOMAIN',\n                value: parameters.githubDomainPath,\n              },\n              {\n                name: 'OWNER',\n                value: parameters.ownerPath,\n              },\n              {\n                name: 'REPO',\n                value: parameters.repoPath,\n              },\n            ],\n          },\n        ],\n      },\n    );\n\n    this.addRetry(task, ['Ecs.EcsException', 'Ecs.LimitExceededException', 'Ecs.UpdateInProgressException']);\n\n    return task;\n  }\n\n  grantStateMachine(_: iam.IGrantable) {\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      vpcArn: this.vpc?.vpcArn,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.task.taskRole.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      image: {\n        imageRepository: this.image.imageRepository.repositoryUri,\n        imageTag: this.image.imageTag,\n        imageBuilderLogGroup: this.image.logGroup?.logGroupName,\n      },\n    };\n  }\n}\n\n/**\n * @deprecated use {@link FargateRunnerProvider}\n */\nexport class FargateRunner extends FargateRunnerProvider {\n}\n"]}
@@ -0,0 +1,15 @@
1
+ import { Construct } from 'constructs';
2
+ import { RunnerImageBuilderBase, RunnerImageBuilderProps } from './common';
3
+ /**
4
+ * GitHub Runner image builder. Builds a Docker image or AMI with GitHub Runner and other requirements installed.
5
+ *
6
+ * Images can be customized before passed into the provider by adding or removing components to be installed.
7
+ *
8
+ * Images are rebuilt every week by default to ensure that the latest security patches are applied.
9
+ */
10
+ export declare abstract class RunnerImageBuilder extends RunnerImageBuilderBase {
11
+ /**
12
+ * Create a new image builder based on the provided properties. The implementation will differ based on the OS, architecture, and requested builder type.
13
+ */
14
+ static new(scope: Construct, id: string, props?: RunnerImageBuilderProps): RunnerImageBuilder;
15
+ }
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.RunnerImageBuilder = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_image_builder_1 = require("./aws-image-builder");
8
+ const codebuild_1 = require("./codebuild");
9
+ const common_1 = require("./common");
10
+ const common_2 = require("../common");
11
+ /**
12
+ * GitHub Runner image builder. Builds a Docker image or AMI with GitHub Runner and other requirements installed.
13
+ *
14
+ * Images can be customized before passed into the provider by adding or removing components to be installed.
15
+ *
16
+ * Images are rebuilt every week by default to ensure that the latest security patches are applied.
17
+ */
18
+ class RunnerImageBuilder extends common_1.RunnerImageBuilderBase {
19
+ /**
20
+ * Create a new image builder based on the provided properties. The implementation will differ based on the OS, architecture, and requested builder type.
21
+ */
22
+ static new(scope, id, props) {
23
+ if (props?.components && props.runnerVersion) {
24
+ aws_cdk_lib_1.Annotations.of(scope).addWarning('runnerVersion is ignored when components are specified. The runner version will be determined by the components.');
25
+ }
26
+ if (props?.builderType === common_1.RunnerImageBuilderType.CODE_BUILD) {
27
+ return new codebuild_1.CodeBuildRunnerImageBuilder(scope, id, props);
28
+ }
29
+ else if (props?.builderType === common_1.RunnerImageBuilderType.AWS_IMAGE_BUILDER) {
30
+ return new aws_image_builder_1.AwsImageBuilderRunnerImageBuilder(scope, id, props);
31
+ }
32
+ const os = props?.os ?? common_2.Os.LINUX_UBUNTU;
33
+ if (os.is(common_2.Os.LINUX_UBUNTU) || os.is(common_2.Os.LINUX_AMAZON_2)) {
34
+ return new codebuild_1.CodeBuildRunnerImageBuilder(scope, id, props);
35
+ }
36
+ else if (os.is(common_2.Os.WINDOWS)) {
37
+ return new aws_image_builder_1.AwsImageBuilderRunnerImageBuilder(scope, id, props);
38
+ }
39
+ else {
40
+ throw new Error(`Unable to find runner image builder implementation for ${os.name}`);
41
+ }
42
+ }
43
+ }
44
+ _a = JSII_RTTI_SYMBOL_1;
45
+ RunnerImageBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerImageBuilder", version: "0.9.1" };
46
+ exports.RunnerImageBuilder = RunnerImageBuilder;
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3Byb3ZpZGVycy9pbWFnZS1idWlsZGVycy9hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSw2Q0FBMEM7QUFFMUMsMkRBQXdFO0FBQ3hFLDJDQUEwRDtBQUMxRCxxQ0FBbUc7QUFDbkcsc0NBQStCO0FBRS9COzs7Ozs7R0FNRztBQUNILE1BQXNCLGtCQUFtQixTQUFRLCtCQUFzQjtJQUNyRTs7T0FFRztJQUNILE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBK0I7UUFDdEUsSUFBSSxLQUFLLEVBQUUsVUFBVSxJQUFJLEtBQUssQ0FBQyxhQUFhLEVBQUU7WUFDNUMseUJBQVcsQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUMsVUFBVSxDQUFDLGtIQUFrSCxDQUFDLENBQUM7U0FDdEo7UUFFRCxJQUFJLEtBQUssRUFBRSxXQUFXLEtBQUssK0JBQXNCLENBQUMsVUFBVSxFQUFFO1lBQzVELE9BQU8sSUFBSSx1Q0FBMkIsQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFLEtBQUssQ0FBQyxDQUFDO1NBQzFEO2FBQU0sSUFBSSxLQUFLLEVBQUUsV0FBVyxLQUFLLCtCQUFzQixDQUFDLGlCQUFpQixFQUFFO1lBQzFFLE9BQU8sSUFBSSxxREFBaUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFLEtBQUssQ0FBQyxDQUFDO1NBQ2hFO1FBRUQsTUFBTSxFQUFFLEdBQUcsS0FBSyxFQUFFLEVBQUUsSUFBSSxXQUFFLENBQUMsWUFBWSxDQUFDO1FBQ3hDLElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsWUFBWSxDQUFDLElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsY0FBYyxDQUFDLEVBQUU7WUFDdEQsT0FBTyxJQUFJLHVDQUEyQixDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7U0FDMUQ7YUFBTSxJQUFJLEVBQUUsQ0FBQyxFQUFFLENBQUMsV0FBRSxDQUFDLE9BQU8sQ0FBQyxFQUFFO1lBQzVCLE9BQU8sSUFBSSxxREFBaUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFLEtBQUssQ0FBQyxDQUFDO1NBQ2hFO2FBQU07WUFDTCxNQUFNLElBQUksS0FBSyxDQUFDLDBEQUEwRCxFQUFFLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztTQUN0RjtJQUNILENBQUM7Ozs7QUF2Qm1CLGdEQUFrQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFubm90YXRpb25zIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBBd3NJbWFnZUJ1aWxkZXJSdW5uZXJJbWFnZUJ1aWxkZXIgfSBmcm9tICcuL2F3cy1pbWFnZS1idWlsZGVyJztcbmltcG9ydCB7IENvZGVCdWlsZFJ1bm5lckltYWdlQnVpbGRlciB9IGZyb20gJy4vY29kZWJ1aWxkJztcbmltcG9ydCB7IFJ1bm5lckltYWdlQnVpbGRlckJhc2UsIFJ1bm5lckltYWdlQnVpbGRlclByb3BzLCBSdW5uZXJJbWFnZUJ1aWxkZXJUeXBlIH0gZnJvbSAnLi9jb21tb24nO1xuaW1wb3J0IHsgT3MgfSBmcm9tICcuLi9jb21tb24nO1xuXG4vKipcbiAqIEdpdEh1YiBSdW5uZXIgaW1hZ2UgYnVpbGRlci4gQnVpbGRzIGEgRG9ja2VyIGltYWdlIG9yIEFNSSB3aXRoIEdpdEh1YiBSdW5uZXIgYW5kIG90aGVyIHJlcXVpcmVtZW50cyBpbnN0YWxsZWQuXG4gKlxuICogSW1hZ2VzIGNhbiBiZSBjdXN0b21pemVkIGJlZm9yZSBwYXNzZWQgaW50byB0aGUgcHJvdmlkZXIgYnkgYWRkaW5nIG9yIHJlbW92aW5nIGNvbXBvbmVudHMgdG8gYmUgaW5zdGFsbGVkLlxuICpcbiAqIEltYWdlcyBhcmUgcmVidWlsdCBldmVyeSB3ZWVrIGJ5IGRlZmF1bHQgdG8gZW5zdXJlIHRoYXQgdGhlIGxhdGVzdCBzZWN1cml0eSBwYXRjaGVzIGFyZSBhcHBsaWVkLlxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgUnVubmVySW1hZ2VCdWlsZGVyIGV4dGVuZHMgUnVubmVySW1hZ2VCdWlsZGVyQmFzZSB7XG4gIC8qKlxuICAgKiBDcmVhdGUgYSBuZXcgaW1hZ2UgYnVpbGRlciBiYXNlZCBvbiB0aGUgcHJvdmlkZWQgcHJvcGVydGllcy4gVGhlIGltcGxlbWVudGF0aW9uIHdpbGwgZGlmZmVyIGJhc2VkIG9uIHRoZSBPUywgYXJjaGl0ZWN0dXJlLCBhbmQgcmVxdWVzdGVkIGJ1aWxkZXIgdHlwZS5cbiAgICovXG4gIHN0YXRpYyBuZXcoc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBSdW5uZXJJbWFnZUJ1aWxkZXJQcm9wcyk6IFJ1bm5lckltYWdlQnVpbGRlciB7XG4gICAgaWYgKHByb3BzPy5jb21wb25lbnRzICYmIHByb3BzLnJ1bm5lclZlcnNpb24pIHtcbiAgICAgIEFubm90YXRpb25zLm9mKHNjb3BlKS5hZGRXYXJuaW5nKCdydW5uZXJWZXJzaW9uIGlzIGlnbm9yZWQgd2hlbiBjb21wb25lbnRzIGFyZSBzcGVjaWZpZWQuIFRoZSBydW5uZXIgdmVyc2lvbiB3aWxsIGJlIGRldGVybWluZWQgYnkgdGhlIGNvbXBvbmVudHMuJyk7XG4gICAgfVxuXG4gICAgaWYgKHByb3BzPy5idWlsZGVyVHlwZSA9PT0gUnVubmVySW1hZ2VCdWlsZGVyVHlwZS5DT0RFX0JVSUxEKSB7XG4gICAgICByZXR1cm4gbmV3IENvZGVCdWlsZFJ1bm5lckltYWdlQnVpbGRlcihzY29wZSwgaWQsIHByb3BzKTtcbiAgICB9IGVsc2UgaWYgKHByb3BzPy5idWlsZGVyVHlwZSA9PT0gUnVubmVySW1hZ2VCdWlsZGVyVHlwZS5BV1NfSU1BR0VfQlVJTERFUikge1xuICAgICAgcmV0dXJuIG5ldyBBd3NJbWFnZUJ1aWxkZXJSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfVxuXG4gICAgY29uc3Qgb3MgPSBwcm9wcz8ub3MgPz8gT3MuTElOVVhfVUJVTlRVO1xuICAgIGlmIChvcy5pcyhPcy5MSU5VWF9VQlVOVFUpIHx8IG9zLmlzKE9zLkxJTlVYX0FNQVpPTl8yKSkge1xuICAgICAgcmV0dXJuIG5ldyBDb2RlQnVpbGRSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfSBlbHNlIGlmIChvcy5pcyhPcy5XSU5ET1dTKSkge1xuICAgICAgcmV0dXJuIG5ldyBBd3NJbWFnZUJ1aWxkZXJSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVW5hYmxlIHRvIGZpbmQgcnVubmVyIGltYWdlIGJ1aWxkZXIgaW1wbGVtZW50YXRpb24gZm9yICR7b3MubmFtZX1gKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,43 @@
1
+ import { aws_ec2 as ec2 } from 'aws-cdk-lib';
2
+ import { Construct } from 'constructs';
3
+ import { ImageBuilderComponent } from './builder';
4
+ import { ImageBuilderObjectBase } from './common';
5
+ import { Architecture, Os } from '../../common';
6
+ /**
7
+ * Properties for AmiRecipe construct.
8
+ */
9
+ interface AmiRecipeProperties {
10
+ /**
11
+ * Target platform. Must match builder platform.
12
+ */
13
+ readonly platform: 'Linux' | 'Windows';
14
+ /**
15
+ * Target architecture. Must match builder platform.
16
+ */
17
+ readonly architecture: Architecture;
18
+ /**
19
+ * Base AMI to use for the new runner AMI.
20
+ */
21
+ readonly baseAmi: string;
22
+ /**
23
+ * Components to add to target container image.
24
+ */
25
+ readonly components: ImageBuilderComponent[];
26
+ }
27
+ /**
28
+ * Image builder recipe for Amazon Machine Image (AMI).
29
+ *
30
+ * @internal
31
+ */
32
+ export declare class AmiRecipe extends ImageBuilderObjectBase {
33
+ readonly arn: string;
34
+ readonly name: string;
35
+ constructor(scope: Construct, id: string, props: AmiRecipeProperties);
36
+ }
37
+ /**
38
+ * Default base AMI for given OS and architecture.
39
+ *
40
+ * @internal
41
+ */
42
+ export declare function defaultBaseAmi(os: Os, architecture: Architecture): ec2.IMachineImage;
43
+ export {};
@@ -0,0 +1,81 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.defaultBaseAmi = exports.AmiRecipe = void 0;
4
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
5
+ const common_1 = require("./common");
6
+ const common_2 = require("../../common");
7
+ const common_3 = require("../common");
8
+ /**
9
+ * Image builder recipe for Amazon Machine Image (AMI).
10
+ *
11
+ * @internal
12
+ */
13
+ class AmiRecipe extends common_1.ImageBuilderObjectBase {
14
+ constructor(scope, id, props) {
15
+ super(scope, id);
16
+ const name = (0, common_3.uniqueImageBuilderName)(this);
17
+ let components = props.components.map(component => {
18
+ return {
19
+ componentArn: component.arn,
20
+ };
21
+ });
22
+ let workingDirectory;
23
+ if (props.platform == 'Linux') {
24
+ workingDirectory = '/home/runner';
25
+ }
26
+ else if (props.platform == 'Windows') {
27
+ workingDirectory = 'C:/'; // must exist or Image Builder fails and must not be empty or git will stall installing from the default windows\system32
28
+ }
29
+ else {
30
+ throw new Error(`Unsupported AMI recipe platform: ${props.platform}`);
31
+ }
32
+ const recipe = new aws_cdk_lib_1.aws_imagebuilder.CfnImageRecipe(this, 'Recipe', {
33
+ name: name,
34
+ version: this.version('ImageRecipe', name, {
35
+ platform: props.platform,
36
+ components,
37
+ parentAmi: props.baseAmi,
38
+ }),
39
+ parentImage: props.baseAmi,
40
+ components,
41
+ workingDirectory,
42
+ });
43
+ this.arn = recipe.attrArn;
44
+ this.name = name;
45
+ }
46
+ }
47
+ exports.AmiRecipe = AmiRecipe;
48
+ /**
49
+ * Default base AMI for given OS and architecture.
50
+ *
51
+ * @internal
52
+ */
53
+ function defaultBaseAmi(os, architecture) {
54
+ let archUrl;
55
+ let cpuType;
56
+ if (architecture.is(common_2.Architecture.X86_64)) {
57
+ archUrl = 'amd64';
58
+ cpuType = aws_cdk_lib_1.aws_ec2.AmazonLinuxCpuType.X86_64;
59
+ }
60
+ else if (architecture.is(common_2.Architecture.ARM64)) {
61
+ archUrl = 'arm64';
62
+ cpuType = aws_cdk_lib_1.aws_ec2.AmazonLinuxCpuType.ARM_64;
63
+ }
64
+ else {
65
+ throw new Error(`Unsupported architecture for base AMI: ${architecture.name}`);
66
+ }
67
+ if (os.is(common_2.Os.LINUX_UBUNTU) || os.is(common_2.Os.LINUX)) {
68
+ return aws_cdk_lib_1.aws_ec2.MachineImage.fromSsmParameter(`/aws/service/canonical/ubuntu/server/focal/stable/current/${archUrl}/hvm/ebs-gp2/ami-id`, {
69
+ os: aws_cdk_lib_1.aws_ec2.OperatingSystemType.LINUX,
70
+ });
71
+ }
72
+ if (os.is(common_2.Os.LINUX_AMAZON_2)) {
73
+ return aws_cdk_lib_1.aws_ec2.MachineImage.latestAmazonLinux({ cpuType });
74
+ }
75
+ if (os.is(common_2.Os.WINDOWS)) {
76
+ return aws_cdk_lib_1.aws_ec2.MachineImage.latestWindows(aws_cdk_lib_1.aws_ec2.WindowsVersion.WINDOWS_SERVER_2022_ENGLISH_FULL_CONTAINERSLATEST);
77
+ }
78
+ throw new Error(`OS ${os.name} not supported for AMI runner image`);
79
+ }
80
+ exports.defaultBaseAmi = defaultBaseAmi;
81
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW1pLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3Byb3ZpZGVycy9pbWFnZS1idWlsZGVycy9hd3MtaW1hZ2UtYnVpbGRlci9hbWkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQStFO0FBRy9FLHFDQUFrRDtBQUNsRCx5Q0FBZ0Q7QUFDaEQsc0NBQW1EO0FBMkJuRDs7OztHQUlHO0FBQ0gsTUFBYSxTQUFVLFNBQVEsK0JBQXNCO0lBSW5ELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMEI7UUFDbEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixNQUFNLElBQUksR0FBRyxJQUFBLCtCQUFzQixFQUFDLElBQUksQ0FBQyxDQUFDO1FBRTFDLElBQUksVUFBVSxHQUFHLEtBQUssQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxFQUFFO1lBQ2hELE9BQU87Z0JBQ0wsWUFBWSxFQUFFLFNBQVMsQ0FBQyxHQUFHO2FBQzVCLENBQUM7UUFDSixDQUFDLENBQUMsQ0FBQztRQUVILElBQUksZ0JBQWdCLENBQUM7UUFDckIsSUFBSSxLQUFLLENBQUMsUUFBUSxJQUFJLE9BQU8sRUFBRTtZQUM3QixnQkFBZ0IsR0FBRyxjQUFjLENBQUM7U0FDbkM7YUFBTSxJQUFJLEtBQUssQ0FBQyxRQUFRLElBQUksU0FBUyxFQUFFO1lBQ3RDLGdCQUFnQixHQUFHLEtBQUssQ0FBQyxDQUFDLHlIQUF5SDtTQUNwSjthQUFNO1lBQ0wsTUFBTSxJQUFJLEtBQUssQ0FBQyxvQ0FBb0MsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7U0FDdkU7UUFFRCxNQUFNLE1BQU0sR0FBRyxJQUFJLDhCQUFZLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxRQUFRLEVBQUU7WUFDN0QsSUFBSSxFQUFFLElBQUk7WUFDVixPQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsSUFBSSxFQUFFO2dCQUN6QyxRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVE7Z0JBQ3hCLFVBQVU7Z0JBQ1YsU0FBUyxFQUFFLEtBQUssQ0FBQyxPQUFPO2FBQ3pCLENBQUM7WUFDRixXQUFXLEVBQUUsS0FBSyxDQUFDLE9BQU87WUFDMUIsVUFBVTtZQUNWLGdCQUFnQjtTQUNqQixDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsR0FBRyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUM7UUFDMUIsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUM7SUFDbkIsQ0FBQztDQUNGO0FBdkNELDhCQXVDQztBQUVEOzs7O0dBSUc7QUFDSCxTQUFnQixjQUFjLENBQUMsRUFBTSxFQUFFLFlBQTBCO0lBQy9ELElBQUksT0FBTyxDQUFDO0lBQ1osSUFBSSxPQUFPLENBQUM7SUFDWixJQUFJLFlBQVksQ0FBQyxFQUFFLENBQUMscUJBQVksQ0FBQyxNQUFNLENBQUMsRUFBRTtRQUN4QyxPQUFPLEdBQUcsT0FBTyxDQUFDO1FBQ2xCLE9BQU8sR0FBRyxxQkFBRyxDQUFDLGtCQUFrQixDQUFDLE1BQU0sQ0FBQztLQUN6QztTQUFNLElBQUksWUFBWSxDQUFDLEVBQUUsQ0FBQyxxQkFBWSxDQUFDLEtBQUssQ0FBQyxFQUFFO1FBQzlDLE9BQU8sR0FBRyxPQUFPLENBQUM7UUFDbEIsT0FBTyxHQUFHLHFCQUFHLENBQUMsa0JBQWtCLENBQUMsTUFBTSxDQUFDO0tBQ3pDO1NBQU07UUFDTCxNQUFNLElBQUksS0FBSyxDQUFDLDBDQUEwQyxZQUFZLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztLQUNoRjtJQUVELElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsWUFBWSxDQUFDLElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsS0FBSyxDQUFDLEVBQUU7UUFDN0MsT0FBTyxxQkFBRyxDQUFDLFlBQVksQ0FBQyxnQkFBZ0IsQ0FDdEMsNkRBQTZELE9BQU8scUJBQXFCLEVBQ3pGO1lBQ0UsRUFBRSxFQUFFLHFCQUFHLENBQUMsbUJBQW1CLENBQUMsS0FBSztTQUNsQyxDQUNGLENBQUM7S0FDSDtJQUVELElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsY0FBYyxDQUFDLEVBQUU7UUFDNUIsT0FBTyxxQkFBRyxDQUFDLFlBQVksQ0FBQyxpQkFBaUIsQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7S0FDeEQ7SUFFRCxJQUFJLEVBQUUsQ0FBQyxFQUFFLENBQUMsV0FBRSxDQUFDLE9BQU8sQ0FBQyxFQUFFO1FBQ3JCLE9BQU8scUJBQUcsQ0FBQyxZQUFZLENBQUMsYUFBYSxDQUFDLHFCQUFHLENBQUMsY0FBYyxDQUFDLGlEQUFpRCxDQUFDLENBQUM7S0FDN0c7SUFFRCxNQUFNLElBQUksS0FBSyxDQUFDLE1BQU0sRUFBRSxDQUFDLElBQUkscUNBQXFDLENBQUMsQ0FBQztBQUN0RSxDQUFDO0FBL0JELHdDQStCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGF3c19lYzIgYXMgZWMyLCBhd3NfaW1hZ2VidWlsZGVyIGFzIGltYWdlYnVpbGRlciB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgSW1hZ2VCdWlsZGVyQ29tcG9uZW50IH0gZnJvbSAnLi9idWlsZGVyJztcbmltcG9ydCB7IEltYWdlQnVpbGRlck9iamVjdEJhc2UgfSBmcm9tICcuL2NvbW1vbic7XG5pbXBvcnQgeyBBcmNoaXRlY3R1cmUsIE9zIH0gZnJvbSAnLi4vLi4vY29tbW9uJztcbmltcG9ydCB7IHVuaXF1ZUltYWdlQnVpbGRlck5hbWUgfSBmcm9tICcuLi9jb21tb24nO1xuXG4vKipcbiAqIFByb3BlcnRpZXMgZm9yIEFtaVJlY2lwZSBjb25zdHJ1Y3QuXG4gKi9cbmludGVyZmFjZSBBbWlSZWNpcGVQcm9wZXJ0aWVzIHtcbiAgLyoqXG4gICAqIFRhcmdldCBwbGF0Zm9ybS4gTXVzdCBtYXRjaCBidWlsZGVyIHBsYXRmb3JtLlxuICAgKi9cbiAgcmVhZG9ubHkgcGxhdGZvcm06ICdMaW51eCcgfCAnV2luZG93cyc7XG5cbiAgLyoqXG4gICAqIFRhcmdldCBhcmNoaXRlY3R1cmUuIE11c3QgbWF0Y2ggYnVpbGRlciBwbGF0Zm9ybS5cbiAgICovXG4gIHJlYWRvbmx5IGFyY2hpdGVjdHVyZTogQXJjaGl0ZWN0dXJlO1xuXG4gIC8qKlxuICAgKiBCYXNlIEFNSSB0byB1c2UgZm9yIHRoZSBuZXcgcnVubmVyIEFNSS5cbiAgICovXG4gIHJlYWRvbmx5IGJhc2VBbWk6IHN0cmluZztcblxuICAvKipcbiAgICogQ29tcG9uZW50cyB0byBhZGQgdG8gdGFyZ2V0IGNvbnRhaW5lciBpbWFnZS5cbiAgICovXG4gIHJlYWRvbmx5IGNvbXBvbmVudHM6IEltYWdlQnVpbGRlckNvbXBvbmVudFtdO1xufVxuXG4vKipcbiAqIEltYWdlIGJ1aWxkZXIgcmVjaXBlIGZvciBBbWF6b24gTWFjaGluZSBJbWFnZSAoQU1JKS5cbiAqXG4gKiBAaW50ZXJuYWxcbiAqL1xuZXhwb3J0IGNsYXNzIEFtaVJlY2lwZSBleHRlbmRzIEltYWdlQnVpbGRlck9iamVjdEJhc2Uge1xuICBwdWJsaWMgcmVhZG9ubHkgYXJuOiBzdHJpbmc7XG4gIHB1YmxpYyByZWFkb25seSBuYW1lOiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IEFtaVJlY2lwZVByb3BlcnRpZXMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgY29uc3QgbmFtZSA9IHVuaXF1ZUltYWdlQnVpbGRlck5hbWUodGhpcyk7XG5cbiAgICBsZXQgY29tcG9uZW50cyA9IHByb3BzLmNvbXBvbmVudHMubWFwKGNvbXBvbmVudCA9PiB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBjb21wb25lbnRBcm46IGNvbXBvbmVudC5hcm4sXG4gICAgICB9O1xuICAgIH0pO1xuXG4gICAgbGV0IHdvcmtpbmdEaXJlY3Rvcnk7XG4gICAgaWYgKHByb3BzLnBsYXRmb3JtID09ICdMaW51eCcpIHtcbiAgICAgIHdvcmtpbmdEaXJlY3RvcnkgPSAnL2hvbWUvcnVubmVyJztcbiAgICB9IGVsc2UgaWYgKHByb3BzLnBsYXRmb3JtID09ICdXaW5kb3dzJykge1xuICAgICAgd29ya2luZ0RpcmVjdG9yeSA9ICdDOi8nOyAvLyBtdXN0IGV4aXN0IG9yIEltYWdlIEJ1aWxkZXIgZmFpbHMgYW5kIG11c3Qgbm90IGJlIGVtcHR5IG9yIGdpdCB3aWxsIHN0YWxsIGluc3RhbGxpbmcgZnJvbSB0aGUgZGVmYXVsdCB3aW5kb3dzXFxzeXN0ZW0zMlxuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFVuc3VwcG9ydGVkIEFNSSByZWNpcGUgcGxhdGZvcm06ICR7cHJvcHMucGxhdGZvcm19YCk7XG4gICAgfVxuXG4gICAgY29uc3QgcmVjaXBlID0gbmV3IGltYWdlYnVpbGRlci5DZm5JbWFnZVJlY2lwZSh0aGlzLCAnUmVjaXBlJywge1xuICAgICAgbmFtZTogbmFtZSxcbiAgICAgIHZlcnNpb246IHRoaXMudmVyc2lvbignSW1hZ2VSZWNpcGUnLCBuYW1lLCB7XG4gICAgICAgIHBsYXRmb3JtOiBwcm9wcy5wbGF0Zm9ybSxcbiAgICAgICAgY29tcG9uZW50cyxcbiAgICAgICAgcGFyZW50QW1pOiBwcm9wcy5iYXNlQW1pLFxuICAgICAgfSksXG4gICAgICBwYXJlbnRJbWFnZTogcHJvcHMuYmFzZUFtaSxcbiAgICAgIGNvbXBvbmVudHMsXG4gICAgICB3b3JraW5nRGlyZWN0b3J5LFxuICAgIH0pO1xuXG4gICAgdGhpcy5hcm4gPSByZWNpcGUuYXR0ckFybjtcbiAgICB0aGlzLm5hbWUgPSBuYW1lO1xuICB9XG59XG5cbi8qKlxuICogRGVmYXVsdCBiYXNlIEFNSSBmb3IgZ2l2ZW4gT1MgYW5kIGFyY2hpdGVjdHVyZS5cbiAqXG4gKiBAaW50ZXJuYWxcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGRlZmF1bHRCYXNlQW1pKG9zOiBPcywgYXJjaGl0ZWN0dXJlOiBBcmNoaXRlY3R1cmUpIHtcbiAgbGV0IGFyY2hVcmw7XG4gIGxldCBjcHVUeXBlO1xuICBpZiAoYXJjaGl0ZWN0dXJlLmlzKEFyY2hpdGVjdHVyZS5YODZfNjQpKSB7XG4gICAgYXJjaFVybCA9ICdhbWQ2NCc7XG4gICAgY3B1VHlwZSA9IGVjMi5BbWF6b25MaW51eENwdVR5cGUuWDg2XzY0O1xuICB9IGVsc2UgaWYgKGFyY2hpdGVjdHVyZS5pcyhBcmNoaXRlY3R1cmUuQVJNNjQpKSB7XG4gICAgYXJjaFVybCA9ICdhcm02NCc7XG4gICAgY3B1VHlwZSA9IGVjMi5BbWF6b25MaW51eENwdVR5cGUuQVJNXzY0O1xuICB9IGVsc2Uge1xuICAgIHRocm93IG5ldyBFcnJvcihgVW5zdXBwb3J0ZWQgYXJjaGl0ZWN0dXJlIGZvciBiYXNlIEFNSTogJHthcmNoaXRlY3R1cmUubmFtZX1gKTtcbiAgfVxuXG4gIGlmIChvcy5pcyhPcy5MSU5VWF9VQlVOVFUpIHx8IG9zLmlzKE9zLkxJTlVYKSkge1xuICAgIHJldHVybiBlYzIuTWFjaGluZUltYWdlLmZyb21Tc21QYXJhbWV0ZXIoXG4gICAgICBgL2F3cy9zZXJ2aWNlL2Nhbm9uaWNhbC91YnVudHUvc2VydmVyL2ZvY2FsL3N0YWJsZS9jdXJyZW50LyR7YXJjaFVybH0vaHZtL2Vicy1ncDIvYW1pLWlkYCxcbiAgICAgIHtcbiAgICAgICAgb3M6IGVjMi5PcGVyYXRpbmdTeXN0ZW1UeXBlLkxJTlVYLFxuICAgICAgfSxcbiAgICApO1xuICB9XG5cbiAgaWYgKG9zLmlzKE9zLkxJTlVYX0FNQVpPTl8yKSkge1xuICAgIHJldHVybiBlYzIuTWFjaGluZUltYWdlLmxhdGVzdEFtYXpvbkxpbnV4KHsgY3B1VHlwZSB9KTtcbiAgfVxuXG4gIGlmIChvcy5pcyhPcy5XSU5ET1dTKSkge1xuICAgIHJldHVybiBlYzIuTWFjaGluZUltYWdlLmxhdGVzdFdpbmRvd3MoZWMyLldpbmRvd3NWZXJzaW9uLldJTkRPV1NfU0VSVkVSXzIwMjJfRU5HTElTSF9GVUxMX0NPTlRBSU5FUlNMQVRFU1QpO1xuICB9XG5cbiAgdGhyb3cgbmV3IEVycm9yKGBPUyAke29zLm5hbWV9IG5vdCBzdXBwb3J0ZWQgZm9yIEFNSSBydW5uZXIgaW1hZ2VgKTtcbn1cblxuIl19