@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
@@ -8,7 +8,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 ami_1 = require("./image-builders/ami");
11
+ const image_builders_1 = require("./image-builders");
12
12
  // this script is specifically made so `poweroff` is absolutely always called
13
13
  // each `{}` is a variable coming from `params` below
14
14
  const linuxUserDataTemplate = `#!/bin/bash -x
@@ -42,7 +42,8 @@ EOF
42
42
  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2
43
43
  }
44
44
  action () {
45
- sudo -Hu runner /home/runner/config.sh --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" {} --name "{}" || exit 1
45
+ if [ "$(< RUNNER_VERSION)" = "latest" ]; then RUNNER_FLAGS=""; else RUNNER_FLAGS="--disableupdate"; fi
46
+ sudo -Hu runner /home/runner/config.sh --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" $RUNNER_FLAGS --name "{}" || exit 1
46
47
  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2
47
48
  STATUS=$(grep -Phors "finish job request for job [0-9a-f\\\\-]+ with result: \\\\K.*" /home/runner/_diag/ | tail -n1)
48
49
  [ -n "$STATUS" ] && echo CDKGHA JOB DONE "{}" "$STATUS"
@@ -88,7 +89,9 @@ function setup_logs () {
88
89
  }
89
90
  function action () {
90
91
  cd /actions
91
- ./config.cmd --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" {} --name "{}" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log
92
+ $RunnerVersion = Get-Content RUNNER_VERSION -Raw
93
+ if ($RunnerVersion -eq "latest") { $RunnerFlags = "" } else { $RunnerFlags = "--disableupdate" }
94
+ ./config.cmd --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" $RunnerFlags --name "{}" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log
92
95
  if ($LASTEXITCODE -ne 0) { return 1 }
93
96
  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log
94
97
  if ($LASTEXITCODE -ne 0) { return 2 }
@@ -113,6 +116,35 @@ Stop-Computer -ComputerName localhost -Force
113
116
  * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
114
117
  */
115
118
  class Ec2RunnerProvider extends common_1.BaseProvider {
119
+ /**
120
+ * Create new image builder that builds EC2 specific runner images using Ubuntu.
121
+ *
122
+ * Included components:
123
+ * * `RunnerImageComponent.requiredPackages()`
124
+ * * `RunnerImageComponent.runnerUser()`
125
+ * * `RunnerImageComponent.git()`
126
+ * * `RunnerImageComponent.githubCli()`
127
+ * * `RunnerImageComponent.awsCli()`
128
+ * * `RunnerImageComponent.docker()`
129
+ * * `RunnerImageComponent.githubRunner()`
130
+ */
131
+ static imageBuilder(scope, id, props) {
132
+ return image_builders_1.RunnerImageBuilder.new(scope, id, {
133
+ os: common_1.Os.LINUX_UBUNTU,
134
+ architecture: common_1.Architecture.X86_64,
135
+ builderType: image_builders_1.RunnerImageBuilderType.AWS_IMAGE_BUILDER,
136
+ components: [
137
+ image_builders_1.RunnerImageComponent.requiredPackages(),
138
+ image_builders_1.RunnerImageComponent.runnerUser(),
139
+ image_builders_1.RunnerImageComponent.git(),
140
+ image_builders_1.RunnerImageComponent.githubCli(),
141
+ image_builders_1.RunnerImageComponent.awsCli(),
142
+ image_builders_1.RunnerImageComponent.docker(),
143
+ image_builders_1.RunnerImageComponent.githubRunner(props?.runnerVersion ?? common_1.RunnerVersion.latest()),
144
+ ],
145
+ ...props,
146
+ });
147
+ }
116
148
  constructor(scope, id, props) {
117
149
  super(scope, id, props);
118
150
  this.labels = props?.labels ?? ['ec2'];
@@ -123,12 +155,12 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
123
155
  this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows
124
156
  this.spot = props?.spot ?? false;
125
157
  this.spotMaxPrice = props?.spotMaxPrice;
126
- const amiBuilder = props?.amiBuilder ?? new ami_1.AmiBuilder(this, 'Image Builder', {
158
+ const amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {
127
159
  vpc: props?.vpc,
128
160
  subnetSelection: props?.subnetSelection,
129
161
  securityGroups: this.securityGroups,
130
162
  });
131
- this.ami = amiBuilder.bind();
163
+ this.ami = amiBuilder.bindAmi();
132
164
  if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {
133
165
  throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);
134
166
  }
@@ -140,7 +172,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
140
172
  });
141
173
  this.grantPrincipal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
142
174
  actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],
143
- resources: ['*'],
175
+ resources: ['*'], // no support for stateMachine.stateMachineArn :(
144
176
  }));
145
177
  this.logGroup = new aws_cdk_lib_1.aws_logs.LogGroup(this, 'Logs', {
146
178
  retention: props?.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH,
@@ -166,7 +198,6 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
166
198
  parameters.repoPath,
167
199
  parameters.runnerTokenPath,
168
200
  this.labels.join(','),
169
- this.ami.runnerVersion.is(common_1.RunnerVersion.latest()) ? '' : '--disableupdate',
170
201
  parameters.runnerNamePath,
171
202
  this.labels.join(','),
172
203
  ];
@@ -295,15 +326,15 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
295
326
  return new aws_cdk_lib_1.aws_ec2.Connections({ securityGroups: this.securityGroups });
296
327
  }
297
328
  }
298
- exports.Ec2RunnerProvider = Ec2RunnerProvider;
299
329
  _a = JSII_RTTI_SYMBOL_1;
300
- Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.8.3" };
330
+ Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.9.0" };
331
+ exports.Ec2RunnerProvider = Ec2RunnerProvider;
301
332
  /**
302
333
  * @deprecated use {@link Ec2RunnerProvider}
303
334
  */
304
335
  class Ec2Runner extends Ec2RunnerProvider {
305
336
  }
306
- exports.Ec2Runner = Ec2Runner;
307
337
  _b = JSII_RTTI_SYMBOL_1;
308
- Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.8.3" };
309
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n  Stack,\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  BaseProvider,\n  IAmiBuilder,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n} from './common';\nimport { AmiBuilder } from './image-builders/ami';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" {} --name \"{}\" || exit 1\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\\\\\-]+ with result: \\\\\\\\K.*\" /home/runner/_diag/ | tail -n1)\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"{}\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo '{\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n         \"collect_list\": [\n            {\n              \"file_path\": \"/actions/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }' | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  ./config.cmd --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" {} --name \"{}\" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\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 {} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log }\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * AMI builder that creates AMIs with GitHub runner pre-configured. On Linux, a user named `runner` is expected to exist with access to Docker.\n   *\n   * The AMI builder determines the OS and architecture of the runner.\n   *\n   * @default AMI builder for Ubuntu Linux on the same subnet as configured by {@link vpc} and {@link subnetSelection}\n   */\n  readonly amiBuilder?: IAmiBuilder;\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 ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * Instance type for launched runner instances.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Security Group to assign to launched runner instances.\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 launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\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 Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\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 ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n\n    const amiBuilder = props?.amiBuilder ?? new AmiBuilder(this, 'Image Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = amiBuilder.bind();\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      ],\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n    }));\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\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    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      this.labels.join(','),\n      this.ami.runnerVersion.is(RunnerVersion.latest()) ? '' : '--disableupdate',\n      parameters.runnerNamePath,\n      this.labels.join(','),\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const subnetRunners = this.subnets.map((subnet, index) => {\n      return new stepfunctions_tasks.CallAwsService(this, `${this.labels.join(', ')} subnet${index+1}`, {\n        comment: subnet.subnetId,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeat: Duration.minutes(10),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: '/dev/sda1',\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // use Parallel, so we can easily retry this whole block on failure (only 1 branch)\n    const subnetIterator = new stepfunctions.Parallel(this, `${this.labels.join(', ')} subnet iterator`);\n\n    // start with the first subnet\n    subnetIterator.branch(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i-1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    // retry the whole Parallel block if (only the last state) failed with an Ec2Exception or timed out\n    this.addRetry(subnetIterator, ['Ec2.Ec2Exception', 'States.Timeout']);\n\n    // return Parallel block\n    return passUserData.next(subnetIterator);\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\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\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}
338
+ Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.9.0" };
339
+ exports.Ec2Runner = Ec2Runner;
340
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n  Stack,\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  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n} from './common';\nimport { IRunnerImageBuilder, RunnerImageBuilder, RunnerImageBuilderProps, RunnerImageBuilderType, RunnerImageComponent } from './image-builders';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  if [ \"$(< RUNNER_VERSION)\" = \"latest\" ]; then RUNNER_FLAGS=\"\"; else RUNNER_FLAGS=\"--disableupdate\"; fi\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" $RUNNER_FLAGS --name \"{}\" || exit 1\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\\\\\-]+ with result: \\\\\\\\K.*\" /home/runner/_diag/ | tail -n1)\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"{}\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo '{\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n         \"collect_list\": [\n            {\n              \"file_path\": \"/actions/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }' | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  $RunnerVersion = Get-Content RUNNER_VERSION -Raw \n  if ($RunnerVersion -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" }\n  ./config.cmd --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" $RunnerFlags --name \"{}\" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\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 {} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log }\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build AMI containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default Ec2ProviderProps.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * @deprecated use imageBuilder\n   */\n  readonly amiBuilder?: IRunnerImageBuilder;\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 ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * Instance type for launched runner instances.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Security Group to assign to launched runner instances.\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 launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\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 Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds EC2 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.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      builderType: RunnerImageBuilderType.AWS_IMAGE_BUILDER,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\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 ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n\n    const amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = amiBuilder.bindAmi();\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      ],\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n    }));\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\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    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      this.labels.join(','),\n      parameters.runnerNamePath,\n      this.labels.join(','),\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const subnetRunners = this.subnets.map((subnet, index) => {\n      return new stepfunctions_tasks.CallAwsService(this, `${this.labels.join(', ')} subnet${index+1}`, {\n        comment: subnet.subnetId,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeat: Duration.minutes(10),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: '/dev/sda1',\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // use Parallel, so we can easily retry this whole block on failure (only 1 branch)\n    const subnetIterator = new stepfunctions.Parallel(this, `${this.labels.join(', ')} subnet iterator`);\n\n    // start with the first subnet\n    subnetIterator.branch(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i-1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    // retry the whole Parallel block if (only the last state) failed with an Ec2Exception or timed out\n    this.addRetry(subnetIterator, ['Ec2.Ec2Exception', 'States.Timeout']);\n\n    // return Parallel block\n    return passUserData.next(subnetIterator);\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\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\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}
@@ -1,18 +1,19 @@
1
1
  import { aws_ec2 as ec2, aws_ecs as ecs, aws_iam as iam, aws_logs as logs, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
2
2
  import { Construct } from 'constructs';
3
- import { BaseProvider, IImageBuilder, IRunnerProvider, IRunnerProviderStatus, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';
3
+ import { BaseProvider, IRunnerProvider, IRunnerProviderStatus, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';
4
+ import { IRunnerImageBuilder, RunnerImageBuilder, RunnerImageBuilderProps } from './image-builders';
4
5
  /**
5
6
  * Properties for FargateRunner.
6
7
  */
7
8
  export interface FargateRunnerProviderProps extends RunnerProviderProps {
8
9
  /**
9
- * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist.
10
+ * Runner image builder used to build Docker images containing GitHub Runner and all requirements.
10
11
  *
11
12
  * The image builder determines the OS and architecture of the runner.
12
13
  *
13
- * @default image builder with `FargateRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile
14
+ * @default FargateRunnerProviderProps.imageBuilder()
14
15
  */
15
- readonly imageBuilder?: IImageBuilder;
16
+ readonly imageBuilder?: IRunnerImageBuilder;
16
17
  /**
17
18
  * GitHub Actions label used for this provider.
18
19
  *
@@ -137,6 +138,8 @@ export declare class FargateRunnerProvider extends BaseProvider implements IRunn
137
138
  * Available build arguments that can be set in the image builder:
138
139
  * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.
139
140
  * * `EXTRA_PACKAGES` can be used to install additional packages.
141
+ *
142
+ * @deprecated Use `imageBuilder()` instead.
140
143
  */
141
144
  static readonly LINUX_X64_DOCKERFILE_PATH: string;
142
145
  /**
@@ -145,8 +148,22 @@ export declare class FargateRunnerProvider extends BaseProvider implements IRunn
145
148
  * Available build arguments that can be set in the image builder:
146
149
  * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.
147
150
  * * `EXTRA_PACKAGES` can be used to install additional packages.
151
+ *
152
+ * @deprecated Use `imageBuilder()` instead.
148
153
  */
149
154
  static readonly LINUX_ARM64_DOCKERFILE_PATH: string;
155
+ /**
156
+ * Create new image builder that builds Fargate specific runner images using Ubuntu.
157
+ *
158
+ * Included components:
159
+ * * `RunnerImageComponent.requiredPackages()`
160
+ * * `RunnerImageComponent.runnerUser()`
161
+ * * `RunnerImageComponent.git()`
162
+ * * `RunnerImageComponent.githubCli()`
163
+ * * `RunnerImageComponent.awsCli()`
164
+ * * `RunnerImageComponent.githubRunner()`
165
+ */
166
+ static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps): RunnerImageBuilder;
150
167
  /**
151
168
  * Cluster hosting the task hosting the runner.
152
169
  */