@cloudsnorkel/cdk-github-runners 0.5.8 → 0.6.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 (33) hide show
  1. package/.jsii +1500 -326
  2. package/API.md +836 -186
  3. package/README.md +11 -11
  4. package/lib/index.d.ts +5 -2
  5. package/lib/index.js +7 -2
  6. package/lib/lambdas/delete-ami/index.js +130 -0
  7. package/lib/lambdas/status/index.js +11 -1
  8. package/lib/lambdas/update-lambda/index.js +165 -107
  9. package/lib/providers/codebuild.d.ts +6 -4
  10. package/lib/providers/codebuild.js +20 -3
  11. package/lib/providers/common.d.ts +137 -9
  12. package/lib/providers/common.js +53 -4
  13. package/lib/providers/ec2.d.ts +106 -0
  14. package/lib/providers/ec2.js +252 -0
  15. package/lib/providers/fargate.d.ts +5 -3
  16. package/lib/providers/fargate.js +26 -5
  17. package/lib/providers/image-builders/ami.d.ts +131 -0
  18. package/lib/providers/image-builders/ami.js +274 -0
  19. package/lib/providers/image-builders/codebuild.js +3 -2
  20. package/lib/providers/image-builders/common.d.ts +196 -0
  21. package/lib/providers/image-builders/common.js +288 -0
  22. package/lib/providers/image-builders/container.d.ts +6 -100
  23. package/lib/providers/image-builders/container.js +41 -304
  24. package/lib/providers/image-builders/linux-components.d.ts +15 -0
  25. package/lib/providers/image-builders/linux-components.js +156 -0
  26. package/lib/providers/image-builders/static.js +3 -2
  27. package/lib/providers/image-builders/windows-components.d.ts +14 -0
  28. package/lib/providers/image-builders/windows-components.js +119 -0
  29. package/lib/providers/lambda.d.ts +5 -3
  30. package/lib/providers/lambda.js +20 -3
  31. package/lib/runner.js +8 -18
  32. package/lib/secrets.js +1 -1
  33. package/package.json +9 -9
@@ -11,7 +11,7 @@ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
11
11
  const common_1 = require("./common");
12
12
  const codebuild_1 = require("./image-builders/codebuild");
13
13
  /**
14
- * GitHub Actions runner provider using CodeBuild to execute the actions.
14
+ * GitHub Actions runner provider using CodeBuild to execute jobs.
15
15
  *
16
16
  * Creates a project that gets started for each job.
17
17
  *
@@ -147,6 +147,23 @@ class CodeBuildRunner extends common_1.BaseProvider {
147
147
  },
148
148
  });
149
149
  }
150
+ grantStateMachine(_) {
151
+ }
152
+ status(statusFunctionRole) {
153
+ this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');
154
+ return {
155
+ type: this.constructor.name,
156
+ labels: this.labels,
157
+ vpcArn: this.vpc?.vpcArn,
158
+ securityGroup: this.securityGroup?.securityGroupId,
159
+ roleArn: this.project.role?.roleArn,
160
+ image: {
161
+ imageRepository: this.image.imageRepository.repositoryUri,
162
+ imageTag: this.image.imageTag,
163
+ imageBuilderLogGroup: this.image.logGroup?.logGroupName,
164
+ },
165
+ };
166
+ }
150
167
  /**
151
168
  * The network connections associated with this resource.
152
169
  */
@@ -156,7 +173,7 @@ class CodeBuildRunner extends common_1.BaseProvider {
156
173
  }
157
174
  exports.CodeBuildRunner = CodeBuildRunner;
158
175
  _a = JSII_RTTI_SYMBOL_1;
159
- CodeBuildRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildRunner", version: "0.5.8" };
176
+ CodeBuildRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildRunner", version: "0.6.0" };
160
177
  /**
161
178
  * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.
162
179
  *
@@ -181,4 +198,4 @@ CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images'
181
198
  * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.
182
199
  */
183
200
  CodeBuildRunner.LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');
184
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/providers/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_codebuild as codebuild,\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} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport { Architecture, BaseProvider, IImageBuilder, IRunnerProvider, Os, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n\nexport interface CodeBuildRunnerProps 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 with access to Docker-in-Docker.\n   *\n   * @default image builder with `CodeBuildRunner.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 ['codebuild']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group to assign to this instance.\n   *\n   * @default public project with no security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * GitHub Actions runner provider using CodeBuild to execute the actions.\n *\n * Creates a project 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 CodeBuildRunner extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild 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   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirements for CodeBuild 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   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');\n\n  /**\n   * CodeBuild project hosting the runner.\n   */\n  readonly project: codebuild.Project;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the project.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Docker image in CodeBuild project.\n   */\n  readonly image: RunnerImage;\n\n  constructor(scope: Construct, id: string, props: CodeBuildRunnerProps) {\n    super(scope, id);\n\n    this.labels = this.labelsFromProperties('codebuild', props.label, props.labels);\n    this.vpc = props.vpc;\n    this.securityGroup = props.securityGroup;\n\n    let buildSpec = {\n      version: '0.2',\n      env: {\n        variables: {\n          RUNNER_TOKEN: 'unspecified',\n          RUNNER_NAME: 'unspecified',\n          RUNNER_LABEL: 'unspecified',\n          OWNER: 'unspecified',\n          REPO: 'unspecified',\n          GITHUB_DOMAIN: 'github.com',\n        },\n      },\n      phases: {\n        install: {\n          commands: [\n            'nohup /usr/local/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2 &',\n            'timeout 15 sh -c \"until docker info; do echo .; sleep 1; done\"',\n            'if [ \"${RUNNER_VERSION}\" = \"latest\" ]; then RUNNER_FLAGS=\"\"; else RUNNER_FLAGS=\"--disableupdate\"; fi',\n            'sudo -Hu runner /home/runner/config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" ${RUNNER_FLAGS} --name \"${RUNNER_NAME}\"',\n          ],\n        },\n        build: {\n          commands: [\n            'sudo --preserve-env=AWS_CONTAINER_CREDENTIALS_RELATIVE_URI,AWS_DEFAULT_REGION,AWS_REGION -Hu runner /home/runner/run.sh',\n          ],\n        },\n      },\n    };\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    if (image.os.is(Os.WINDOWS)) {\n      buildSpec.phases.install.commands = [\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      ];\n      buildSpec.phases.build.commands = [\n        'cd \\\\actions',\n        './run.cmd',\n      ];\n    }\n\n    // choose build image\n    let buildImage: codebuild.IBuildImage | undefined;\n    if (image.os.is(Os.LINUX)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.LinuxBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      } else if (image.architecture.is(Architecture.ARM64)) {\n        buildImage = codebuild.LinuxArmBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      }\n    }\n    if (image.os.is(Os.WINDOWS)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag, codebuild.WindowsImageType.SERVER_2019);\n      }\n    }\n\n    if (buildImage === undefined) {\n      throw new Error(`Unable to find supported CodeBuild image for ${image.os.name}/${image.architecture.name}`);\n    }\n\n    // create project\n    this.project = new codebuild.Project(\n      this,\n      'CodeBuild',\n      {\n        description: `GitHub Actions self-hosted runner for labels ${this.labels}`,\n        buildSpec: codebuild.BuildSpec.fromObject(buildSpec),\n        vpc: this.vpc,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        subnetSelection: props.subnetSelection,\n        timeout: props.timeout ?? Duration.hours(1),\n        environment: {\n          buildImage,\n          computeType: props.computeType ?? ComputeType.SMALL,\n          privileged: image.os.is(Os.LINUX),\n        },\n        logging: {\n          cloudWatch: {\n            logGroup: new logs.LogGroup(\n              this,\n              'Logs',\n              {\n                retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n                removalPolicy: RemovalPolicy.DESTROY,\n              },\n            ),\n          },\n        },\n      },\n    );\n\n    this.grantPrincipal = this.project.grantPrincipal;\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    return new stepfunctions_tasks.CodeBuildStartBuild(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        project: this.project,\n        environmentVariablesOverride: {\n          RUNNER_TOKEN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerTokenPath,\n          },\n          RUNNER_NAME: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerNamePath,\n          },\n          RUNNER_LABEL: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: this.labels.join(','),\n          },\n          GITHUB_DOMAIN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.githubDomainPath,\n          },\n          OWNER: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.ownerPath,\n          },\n          REPO: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.repoPath,\n          },\n        },\n      },\n    );\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return this.project.connections;\n  }\n}\n"]}
201
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/providers/codebuild.ts"],"names":[],"mappings":";;;;;AAAA,6BAA6B;AAC7B,6CASqB;AACrB,6DAAwD;AACxD,mDAAqD;AACrD,qEAAmE;AAEnE,qCAUkB;AAClB,0DAAmE;AAqEnE;;;;;;GAMG;AACH,MAAa,eAAgB,SAAQ,qBAAY;IAyD/C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA2B;QACnE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,oBAAoB,CAAC,WAAW,EAAE,KAAK,CAAC,KAAK,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC;QAChF,IAAI,CAAC,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC;QACrB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,aAAa,CAAC;QAEzC,IAAI,SAAS,GAAG;YACd,OAAO,EAAE,KAAK;YACd,GAAG,EAAE;gBACH,SAAS,EAAE;oBACT,YAAY,EAAE,aAAa;oBAC3B,WAAW,EAAE,aAAa;oBAC1B,YAAY,EAAE,aAAa;oBAC3B,KAAK,EAAE,aAAa;oBACpB,IAAI,EAAE,aAAa;oBACnB,aAAa,EAAE,YAAY;iBAC5B;aACF;YACD,MAAM,EAAE;gBACN,OAAO,EAAE;oBACP,QAAQ,EAAE;wBACR,yHAAyH;wBACzH,gEAAgE;wBAChE,sGAAsG;wBACtG,6NAA6N;qBAC9N;iBACF;gBACD,KAAK,EAAE;oBACL,QAAQ,EAAE;wBACR,yHAAyH;qBAC1H;iBACF;aACF;SACF,CAAC;QAEF,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,IAAI,IAAI,iCAAqB,CAAC,IAAI,EAAE,eAAe,EAAE;YAC1F,cAAc,EAAE,eAAe,CAAC,yBAAyB;SAC1D,CAAC,CAAC;QACH,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,GAAG,YAAY,CAAC,IAAI,EAAE,CAAC;QAE/C,IAAI,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,WAAE,CAAC,OAAO,CAAC,EAAE;YAC3B,SAAS,CAAC,MAAM,CAAC,OAAO,CAAC,QAAQ,GAAG;gBAClC,cAAc;gBACd,yGAAyG;gBACzG,0NAA0N;aAC3N,CAAC;YACF,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,GAAG;gBAChC,cAAc;gBACd,WAAW;aACZ,CAAC;SACH;QAED,qBAAqB;QACrB,IAAI,UAA6C,CAAC;QAClD,IAAI,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,WAAE,CAAC,KAAK,CAAC,EAAE;YACzB,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,qBAAY,CAAC,MAAM,CAAC,EAAE;gBAC9C,UAAU,GAAG,2BAAS,CAAC,eAAe,CAAC,iBAAiB,CAAC,KAAK,CAAC,eAAe,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC;aACjG;iBAAM,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,qBAAY,CAAC,KAAK,CAAC,EAAE;gBACpD,UAAU,GAAG,2BAAS,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,KAAK,CAAC,eAAe,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC;aACpG;SACF;QACD,IAAI,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,WAAE,CAAC,OAAO,CAAC,EAAE;YAC3B,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,qBAAY,CAAC,MAAM,CAAC,EAAE;gBAC9C,UAAU,GAAG,2BAAS,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,KAAK,CAAC,eAAe,EAAE,KAAK,CAAC,QAAQ,EAAE,2BAAS,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAAC;aAC3I;SACF;QAED,IAAI,UAAU,KAAK,SAAS,EAAE;YAC5B,MAAM,IAAI,KAAK,CAAC,gDAAgD,KAAK,CAAC,EAAE,CAAC,IAAI,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;SAC7G;QAED,iBAAiB;QACjB,IAAI,CAAC,OAAO,GAAG,IAAI,2BAAS,CAAC,OAAO,CAClC,IAAI,EACJ,WAAW,EACX;YACE,WAAW,EAAE,gDAAgD,IAAI,CAAC,MAAM,EAAE;YAC1E,SAAS,EAAE,2BAAS,CAAC,SAAS,CAAC,UAAU,CAAC,SAAS,CAAC;YACpD,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,cAAc,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,SAAS;YACrE,eAAe,EAAE,KAAK,CAAC,eAAe;YACtC,OAAO,EAAE,KAAK,CAAC,OAAO,IAAI,sBAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;YAC3C,WAAW,EAAE;gBACX,UAAU;gBACV,WAAW,EAAE,KAAK,CAAC,WAAW,IAAI,2BAAW,CAAC,KAAK;gBACnD,UAAU,EAAE,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,WAAE,CAAC,KAAK,CAAC;aAClC;YACD,OAAO,EAAE;gBACP,UAAU,EAAE;oBACV,QAAQ,EAAE,IAAI,sBAAI,CAAC,QAAQ,CACzB,IAAI,EACJ,MAAM,EACN;wBACE,SAAS,EAAE,KAAK,CAAC,YAAY,IAAI,wBAAa,CAAC,SAAS;wBACxD,aAAa,EAAE,2BAAa,CAAC,OAAO;qBACrC,CACF;iBACF;aACF;SACF,CACF,CAAC;QAEF,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC;IACpD,CAAC;IAED;;;;;;OAMG;IACH,mBAAmB,CAAC,UAAmC;QACrD,OAAO,IAAI,qCAAmB,CAAC,mBAAmB,CAChD,IAAI,EACJ,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EACtB;YACE,kBAAkB,EAAE,sCAAkB,CAAC,OAAO;YAC9C,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,4BAA4B,EAAE;gBAC5B,YAAY,EAAE;oBACZ,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,UAAU,CAAC,eAAe;iBAClC;gBACD,WAAW,EAAE;oBACX,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,UAAU,CAAC,cAAc;iBACjC;gBACD,YAAY,EAAE;oBACZ,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;iBAC7B;gBACD,aAAa,EAAE;oBACb,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,UAAU,CAAC,gBAAgB;iBACnC;gBACD,KAAK,EAAE;oBACL,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,UAAU,CAAC,SAAS;iBAC5B;gBACD,IAAI,EAAE;oBACJ,IAAI,EAAE,2BAAS,CAAC,4BAA4B,CAAC,SAAS;oBACtD,KAAK,EAAE,UAAU,CAAC,QAAQ;iBAC3B;aACF;SACF,CACF,CAAC;IACJ,CAAC;IAED,iBAAiB,CAAC,CAAiB;IACnC,CAAC;IAED,MAAM,CAAC,kBAAkC;QACvC,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,KAAK,CAAC,kBAAkB,EAAE,oBAAoB,CAAC,CAAC;QAE3E,OAAO;YACL,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI;YAC3B,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE,MAAM;YACxB,aAAa,EAAE,IAAI,CAAC,aAAa,EAAE,eAAe;YAClD,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,OAAO;YACnC,KAAK,EAAE;gBACL,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,aAAa;gBACzD,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,QAAQ;gBAC7B,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,YAAY;aACxD;SACF,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,IAAW,WAAW;QACpB,OAAO,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC;IAClC,CAAC;;AAxOH,0CAyOC;;;AAxOC;;;;;;;;;;GAUG;AACoB,yCAAyB,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;AAEnH;;;;;;;;;;GAUG;AACoB,2CAA2B,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,EAAE,WAAW,EAAE,aAAa,CAAC,CAAC","sourcesContent":["import * as path from 'path';\nimport {\n  aws_codebuild as codebuild,\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} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\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} from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n\nexport interface CodeBuildRunnerProps extends RunnerProviderProps {\n  /**\n   * Image builder for CodeBuild image with GitHub runner pre-configured. A user named `runner` is expected to exist with access to Docker-in-Docker.\n   *\n   * @default image builder with `CodeBuildRunner.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 ['codebuild']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group to assign to this instance.\n   *\n   * @default public project with no security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * GitHub Actions runner provider using CodeBuild to execute jobs.\n *\n * Creates a project 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 CodeBuildRunner extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild 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   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirements for CodeBuild 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   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');\n\n  /**\n   * CodeBuild project hosting the runner.\n   */\n  readonly project: codebuild.Project;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the project.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\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 CodeBuild.\n   */\n  readonly image: RunnerImage;\n\n  constructor(scope: Construct, id: string, props: CodeBuildRunnerProps) {\n    super(scope, id);\n\n    this.labels = this.labelsFromProperties('codebuild', props.label, props.labels);\n    this.vpc = props.vpc;\n    this.securityGroup = props.securityGroup;\n\n    let buildSpec = {\n      version: '0.2',\n      env: {\n        variables: {\n          RUNNER_TOKEN: 'unspecified',\n          RUNNER_NAME: 'unspecified',\n          RUNNER_LABEL: 'unspecified',\n          OWNER: 'unspecified',\n          REPO: 'unspecified',\n          GITHUB_DOMAIN: 'github.com',\n        },\n      },\n      phases: {\n        install: {\n          commands: [\n            'nohup /usr/local/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2 &',\n            'timeout 15 sh -c \"until docker info; do echo .; sleep 1; done\"',\n            'if [ \"${RUNNER_VERSION}\" = \"latest\" ]; then RUNNER_FLAGS=\"\"; else RUNNER_FLAGS=\"--disableupdate\"; fi',\n            'sudo -Hu runner /home/runner/config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" ${RUNNER_FLAGS} --name \"${RUNNER_NAME}\"',\n          ],\n        },\n        build: {\n          commands: [\n            'sudo --preserve-env=AWS_CONTAINER_CREDENTIALS_RELATIVE_URI,AWS_DEFAULT_REGION,AWS_REGION -Hu runner /home/runner/run.sh',\n          ],\n        },\n      },\n    };\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    if (image.os.is(Os.WINDOWS)) {\n      buildSpec.phases.install.commands = [\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      ];\n      buildSpec.phases.build.commands = [\n        'cd \\\\actions',\n        './run.cmd',\n      ];\n    }\n\n    // choose build image\n    let buildImage: codebuild.IBuildImage | undefined;\n    if (image.os.is(Os.LINUX)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.LinuxBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      } else if (image.architecture.is(Architecture.ARM64)) {\n        buildImage = codebuild.LinuxArmBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      }\n    }\n    if (image.os.is(Os.WINDOWS)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag, codebuild.WindowsImageType.SERVER_2019);\n      }\n    }\n\n    if (buildImage === undefined) {\n      throw new Error(`Unable to find supported CodeBuild image for ${image.os.name}/${image.architecture.name}`);\n    }\n\n    // create project\n    this.project = new codebuild.Project(\n      this,\n      'CodeBuild',\n      {\n        description: `GitHub Actions self-hosted runner for labels ${this.labels}`,\n        buildSpec: codebuild.BuildSpec.fromObject(buildSpec),\n        vpc: this.vpc,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        subnetSelection: props.subnetSelection,\n        timeout: props.timeout ?? Duration.hours(1),\n        environment: {\n          buildImage,\n          computeType: props.computeType ?? ComputeType.SMALL,\n          privileged: image.os.is(Os.LINUX),\n        },\n        logging: {\n          cloudWatch: {\n            logGroup: new logs.LogGroup(\n              this,\n              'Logs',\n              {\n                retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n                removalPolicy: RemovalPolicy.DESTROY,\n              },\n            ),\n          },\n        },\n      },\n    );\n\n    this.grantPrincipal = this.project.grantPrincipal;\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    return new stepfunctions_tasks.CodeBuildStartBuild(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        project: this.project,\n        environmentVariablesOverride: {\n          RUNNER_TOKEN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerTokenPath,\n          },\n          RUNNER_NAME: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerNamePath,\n          },\n          RUNNER_LABEL: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: this.labels.join(','),\n          },\n          GITHUB_DOMAIN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.githubDomainPath,\n          },\n          OWNER: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.ownerPath,\n          },\n          REPO: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.repoPath,\n          },\n        },\n      },\n    );\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      securityGroup: this.securityGroup?.securityGroupId,\n      roleArn: this.project.role?.roleArn,\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   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return this.project.connections;\n  }\n}\n"]}
@@ -18,6 +18,12 @@ export declare class RunnerVersion {
18
18
  */
19
19
  static specific(version: string): RunnerVersion;
20
20
  protected constructor(version: string);
21
+ /**
22
+ * Check if two versions are the same.
23
+ *
24
+ * @param other version to compare
25
+ */
26
+ is(other: RunnerVersion): boolean;
21
27
  }
22
28
  /**
23
29
  * CPU architecture enum for an image.
@@ -40,6 +46,18 @@ export declare class Architecture {
40
46
  * @param arch architecture to compare
41
47
  */
42
48
  is(arch: Architecture): boolean;
49
+ /**
50
+ * Checks if this architecture is in a given list.
51
+ *
52
+ * @param arches architectures to check
53
+ */
54
+ isIn(arches: Architecture[]): boolean;
55
+ /**
56
+ * Checks if a given EC2 instance type matches this architecture.
57
+ *
58
+ * @param instanceType instance type to check
59
+ */
60
+ instanceTypeMatch(instanceType: ec2.InstanceType): boolean;
43
61
  }
44
62
  /**
45
63
  * OS enum for an image.
@@ -62,7 +80,16 @@ export declare class Os {
62
80
  * @param os OS to compare
63
81
  */
64
82
  is(os: Os): boolean;
83
+ /**
84
+ * Checks if this OS is in a given list.
85
+ *
86
+ * @param oses list of OS to check
87
+ */
88
+ isIn(oses: Os[]): boolean;
65
89
  }
90
+ /**
91
+ * Description of a Docker image built by {@link IImageBuilder}.
92
+ */
66
93
  export interface RunnerImage {
67
94
  /**
68
95
  * ECR repository containing the image.
@@ -84,6 +111,10 @@ export interface RunnerImage {
84
111
  * Log group where image builds are logged.
85
112
  */
86
113
  readonly logGroup?: logs.LogGroup;
114
+ /**
115
+ * Installed runner version.
116
+ */
117
+ readonly runnerVersion: RunnerVersion;
87
118
  }
88
119
  /**
89
120
  * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.
@@ -96,7 +127,7 @@ export interface RunnerImage {
96
127
  */
97
128
  export interface IImageBuilder {
98
129
  /**
99
- * ECR repository containing the image.
130
+ * Finalize and return all required information about the Docker image built by this builder.
100
131
  *
101
132
  * This method can be called multiple times if the image is bound to multiple providers. Make sure you cache the image when implementing or return an error if this builder doesn't support reusing images.
102
133
  *
@@ -104,6 +135,48 @@ export interface IImageBuilder {
104
135
  */
105
136
  bind(): RunnerImage;
106
137
  }
138
+ /**
139
+ * Description of a AMI built by {@link IAmiBuilder}.
140
+ */
141
+ export interface RunnerAmi {
142
+ /**
143
+ * Launch template pointing to the latest AMI.
144
+ */
145
+ readonly launchTemplate: ec2.ILaunchTemplate;
146
+ /**
147
+ * Architecture of the image.
148
+ */
149
+ readonly architecture: Architecture;
150
+ /**
151
+ * OS type of the image.
152
+ */
153
+ readonly os: Os;
154
+ /**
155
+ * Log group where image builds are logged.
156
+ */
157
+ readonly logGroup?: logs.LogGroup;
158
+ /**
159
+ * Installed runner version.
160
+ */
161
+ readonly runnerVersion: RunnerVersion;
162
+ }
163
+ /**
164
+ * Interface for constructs that build an AMI that can be used in {@link IRunnerProvider}.
165
+ *
166
+ * Anything that ends up with a launch template pointing to an AMI that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing AMI and nothing else.
167
+ *
168
+ * The AMI can be further updated over time manually or using a schedule as long as it is always written to the same launch template.
169
+ */
170
+ export interface IAmiBuilder {
171
+ /**
172
+ * Finalize and return all required information about the AMI built by this builder.
173
+ *
174
+ * This method can be called multiple times if the image is bound to multiple providers. Make sure you cache the image when implementing or return an error if this builder doesn't support reusing images.
175
+ *
176
+ * @return ami
177
+ */
178
+ bind(): RunnerAmi;
179
+ }
107
180
  /**
108
181
  * Common properties for all runner providers.
109
182
  */
@@ -149,22 +222,68 @@ export interface RunnerRuntimeParameters {
149
222
  readonly repoPath: string;
150
223
  }
151
224
  /**
152
- * Interface for runner image status used by status.json.
225
+ * Image status returned from runner providers to be displayed in status.json.
153
226
  */
154
227
  export interface IRunnerImageStatus {
155
228
  /**
156
- * Image repository where runner image is pushed.
229
+ * Image repository where image builder pushes runner images.
157
230
  */
158
- readonly imageRepository?: string;
231
+ readonly imageRepository: string;
159
232
  /**
160
233
  * Tag of image that should be used.
161
234
  */
162
- readonly imageTag?: string;
235
+ readonly imageTag: string;
163
236
  /**
164
237
  * Log group name for the image builder where history of image builds can be analyzed.
165
238
  */
166
239
  readonly imageBuilderLogGroup?: string;
167
240
  }
241
+ /**
242
+ * AMI status returned from runner providers to be displayed as output of status function.
243
+ */
244
+ export interface IRunnerAmiStatus {
245
+ /**
246
+ * Id of launch template pointing to the latest AMI built by the AMI builder.
247
+ */
248
+ readonly launchTemplate: string;
249
+ /**
250
+ * Log group name for the AMI builder where history of builds can be analyzed.
251
+ */
252
+ readonly amiBuilderLogGroup?: string;
253
+ }
254
+ /**
255
+ * Interface for runner image status used by status.json.
256
+ */
257
+ export interface IRunnerProviderStatus {
258
+ /**
259
+ * Runner provider type.
260
+ */
261
+ readonly type: string;
262
+ /**
263
+ * Labels associated with provider.
264
+ */
265
+ readonly labels: string[];
266
+ /**
267
+ * VPC where runners will be launched.
268
+ */
269
+ readonly vpcArn?: string;
270
+ /**
271
+ * Security group attached to runners.
272
+ */
273
+ readonly securityGroup?: string;
274
+ /**
275
+ * Role attached to runners.
276
+ */
277
+ readonly roleArn?: string;
278
+ /**
279
+ * Details about Docker image used by this runner provider.
280
+ */
281
+ readonly image?: IRunnerImageStatus;
282
+ /**
283
+ * Details about AMI used by this runner provider.
284
+ */
285
+ readonly ami?: IRunnerAmiStatus;
286
+ }
168
287
  /**
169
288
  * Interface for all runner providers. Implementations create all required resources and return a step function task that starts those resources from {@link getStepFunctionTask}.
170
289
  */
@@ -185,10 +304,6 @@ export interface IRunnerProvider extends ec2.IConnectable, iam.IGrantable {
185
304
  * Security group associated with runners.
186
305
  */
187
306
  readonly securityGroup?: ec2.ISecurityGroup;
188
- /**
189
- * Image used to create a new resource compute. Can be Docker image, AMI, or something else.
190
- */
191
- readonly image: RunnerImage;
192
307
  /**
193
308
  * Generate step function tasks that execute the runner.
194
309
  *
@@ -197,6 +312,19 @@ export interface IRunnerProvider extends ec2.IConnectable, iam.IGrantable {
197
312
  * @param parameters specific build parameters
198
313
  */
199
314
  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;
315
+ /**
316
+ * An optional method that modifies the role of the state machine after all the tasks have been generated. This can be used to add additional policy
317
+ * statements to the state machine role that are not automatically added by the task returned from {@link getStepFunctionTask}.
318
+ *
319
+ * @param stateMachineRole role for the state machine that executes the task returned from {@link getStepFunctionTask}.
320
+ */
321
+ grantStateMachine(stateMachineRole: iam.IGrantable): void;
322
+ /**
323
+ * Return status of the runner provider to be used in the main status function. Also gives the status function any needed permissions to query the Docker image or AMI.
324
+ *
325
+ * @param statusFunctionRole grantable for the status function
326
+ */
327
+ status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus;
200
328
  }
201
329
  /**
202
330
  * Base class for all providers with common methods used by all providers.
@@ -3,6 +3,7 @@ var _a, _b, _c;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
4
  exports.BaseProvider = exports.Os = exports.Architecture = exports.RunnerVersion = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
6
7
  const constructs_1 = require("constructs");
7
8
  /**
8
9
  * Defines desired GitHub Actions runner version.
@@ -27,10 +28,18 @@ class RunnerVersion {
27
28
  static specific(version) {
28
29
  return new RunnerVersion(version);
29
30
  }
31
+ /**
32
+ * Check if two versions are the same.
33
+ *
34
+ * @param other version to compare
35
+ */
36
+ is(other) {
37
+ return this.version == other.version;
38
+ }
30
39
  }
31
40
  exports.RunnerVersion = RunnerVersion;
32
41
  _a = JSII_RTTI_SYMBOL_1;
33
- RunnerVersion[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerVersion", version: "0.5.8" };
42
+ RunnerVersion[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerVersion", version: "0.6.0" };
34
43
  /**
35
44
  * CPU architecture enum for an image.
36
45
  */
@@ -49,10 +58,37 @@ class Architecture {
49
58
  is(arch) {
50
59
  return arch.name == this.name;
51
60
  }
61
+ /**
62
+ * Checks if this architecture is in a given list.
63
+ *
64
+ * @param arches architectures to check
65
+ */
66
+ isIn(arches) {
67
+ for (const arch of arches) {
68
+ if (this.is(arch)) {
69
+ return true;
70
+ }
71
+ }
72
+ return false;
73
+ }
74
+ /**
75
+ * Checks if a given EC2 instance type matches this architecture.
76
+ *
77
+ * @param instanceType instance type to check
78
+ */
79
+ instanceTypeMatch(instanceType) {
80
+ if (instanceType.architecture == aws_cdk_lib_1.aws_ec2.InstanceArchitecture.X86_64) {
81
+ return this.is(Architecture.X86_64);
82
+ }
83
+ if (instanceType.architecture == aws_cdk_lib_1.aws_ec2.InstanceArchitecture.ARM_64) {
84
+ return this.is(Architecture.ARM64);
85
+ }
86
+ throw new Error('Unknown instance type architecture');
87
+ }
52
88
  }
53
89
  exports.Architecture = Architecture;
54
90
  _b = JSII_RTTI_SYMBOL_1;
55
- Architecture[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Architecture", version: "0.5.8" };
91
+ Architecture[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Architecture", version: "0.6.0" };
56
92
  /**
57
93
  * ARM64
58
94
  */
@@ -79,10 +115,23 @@ class Os {
79
115
  is(os) {
80
116
  return os.name == this.name;
81
117
  }
118
+ /**
119
+ * Checks if this OS is in a given list.
120
+ *
121
+ * @param oses list of OS to check
122
+ */
123
+ isIn(oses) {
124
+ for (const os of oses) {
125
+ if (this.is(os)) {
126
+ return true;
127
+ }
128
+ }
129
+ return false;
130
+ }
82
131
  }
83
132
  exports.Os = Os;
84
133
  _c = JSII_RTTI_SYMBOL_1;
85
- Os[_c] = { fqn: "@cloudsnorkel/cdk-github-runners.Os", version: "0.5.8" };
134
+ Os[_c] = { fqn: "@cloudsnorkel/cdk-github-runners.Os", version: "0.6.0" };
86
135
  /**
87
136
  * Linux
88
137
  */
@@ -111,4 +160,4 @@ class BaseProvider extends constructs_1.Construct {
111
160
  }
112
161
  }
113
162
  exports.BaseProvider = BaseProvider;
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/providers/common.ts"],"names":[],"mappings":";;;;;AACA,2CAAuC;AAEvC;;GAEG;AACH,MAAa,aAAa;IAmBxB,YAA+B,OAAe;QAAf,YAAO,GAAP,OAAO,CAAQ;IAC9C,CAAC;IAnBD;;OAEG;IACI,MAAM,CAAC,MAAM;QAClB,OAAO,IAAI,aAAa,CAAC,QAAQ,CAAC,CAAC;IACrC,CAAC;IAED;;;;;;OAMG;IACI,MAAM,CAAC,QAAQ,CAAC,OAAe;QACpC,OAAO,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;;AAjBH,sCAqBC;;;AAED;;GAEG;AACH,MAAa,YAAY;IAevB,YAAoC,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;IAChD,CAAC;IALO,MAAM,CAAC,EAAE,CAAC,YAAoB;QACpC,OAAO,IAAI,YAAY,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAKD;;;;MAIE;IACK,EAAE,CAAC,IAAkB;QAC1B,OAAO,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC;IAChC,CAAC;;AAzBH,oCA0BC;;;AAzBC;;GAEG;AACoB,kBAAK,GAAG,YAAY,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC;AAExD;;GAEG;AACoB,mBAAM,GAAG,YAAY,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;AAmB5D;;GAEG;AACH,MAAa,EAAE;IAeb,YAAoC,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;IAChD,CAAC;IALO,MAAM,CAAC,EAAE,CAAC,EAAU;QAC1B,OAAO,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC;IACpB,CAAC;IAKD;;;;MAIE;IACK,EAAE,CAAC,EAAM;QACd,OAAO,EAAE,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC;IAC9B,CAAC;;AAzBH,gBA0BC;;;AAzBC;;EAEE;AACqB,QAAK,GAAG,EAAE,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC;AAE9C;;EAEE;AACqB,UAAO,GAAG,EAAE,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;AA8KpD;;;;GAIG;AACH,MAAsB,YAAa,SAAQ,sBAAS;IACxC,oBAAoB,CAAC,YAAoB,EAAE,UAA8B,EAAE,WAAiC;QACpH,IAAI,WAAW,IAAI,UAAU,EAAE;YAC7B,MAAM,IAAI,KAAK,CAAC,+GAA+G,CAAC,CAAC;SAClI;QAED,IAAI,WAAW,EAAE;YACf,OAAO,WAAW,CAAC;SACpB;QACD,IAAI,UAAU,EAAE;YACd,OAAO,CAAC,UAAU,CAAC,CAAC;SACrB;QACD,OAAO,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC;CACF;AAdD,oCAcC","sourcesContent":["import { aws_ec2 as ec2, aws_ecr as ecr, aws_iam as iam, aws_logs as logs, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\n/**\n * Defines desired GitHub Actions runner version.\n */\nexport class RunnerVersion {\n  /**\n   * Use the latest version available at the time the runner provider image is built.\n   */\n  public static latest(): RunnerVersion {\n    return new RunnerVersion('latest');\n  }\n\n  /**\n   * Use a specific version.\n   *\n   * @see https://github.com/actions/runner/releases\n   *\n   * @param version GitHub Runner version\n   */\n  public static specific(version: string) {\n    return new RunnerVersion(version);\n  }\n\n  protected constructor(readonly version: string) {\n  }\n}\n\n/**\n * CPU architecture enum for an image.\n */\nexport class Architecture {\n  /**\n   * ARM64\n   */\n  public static readonly ARM64 = Architecture.of('ARM64');\n\n  /**\n   * X86_64\n   */\n  public static readonly X86_64 = Architecture.of('X86_64');\n\n  private static of(architecture: string) {\n    return new Architecture(architecture);\n  }\n\n  private constructor(public readonly name: string) {\n  }\n\n  /**\n  * Checks if the given architecture is the same as this one.\n  *\n  * @param arch architecture to compare\n  */\n  public is(arch: Architecture) {\n    return arch.name == this.name;\n  }\n}\n\n/**\n * OS enum for an image.\n */\nexport class Os {\n  /**\n  * Linux\n  */\n  public static readonly LINUX = Os.of('Linux');\n\n  /**\n  * Windows\n  */\n  public static readonly WINDOWS = Os.of('Windows');\n\n  private static of(os: string) {\n    return new Os(os);\n  }\n\n  private constructor(public readonly name: string) {\n  }\n\n  /**\n  * Checks if the given OS is the same as this one.\n  *\n  * @param os OS to compare\n  */\n  public is(os: Os) {\n    return os.name == this.name;\n  }\n}\n\nexport interface RunnerImage {\n  /**\n   * ECR repository containing the image.\n   */\n  readonly imageRepository: ecr.IRepository;\n\n  /**\n   * Static image tag where the image will be pushed.\n   */\n  readonly imageTag: string;\n\n  /**\n   * Architecture of the image.\n   */\n  readonly architecture: Architecture;\n\n  /**\n   * OS type of the image.\n   */\n  readonly os: Os;\n\n  /**\n   * Log group where image builds are logged.\n   */\n  readonly logGroup?: logs.LogGroup;\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.\n *\n * Anything that ends up with an ECR repository containing a Docker image that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing image and nothing else.\n *\n * It's important that the specified image tag be available at the time the repository is available. Providers usually assume the image is ready and will fail if it's not.\n *\n * The image can be further updated over time manually or using a schedule as long as it is always written to the same tag.\n */\nexport interface IImageBuilder {\n  /**\n   * ECR repository containing the image.\n   *\n   * This method can be called multiple times if the image is bound to multiple providers. Make sure you cache the image when implementing or return an error if this builder doesn't support reusing images.\n   *\n   * @return image\n   */\n  bind(): RunnerImage;\n}\n\n/**\n * Common properties for all runner providers.\n */\nexport interface RunnerProviderProps {\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n}\n\n/**\n * Workflow job parameters as parsed from the webhook event. Pass these into your runner executor and run something like:\n *\n * ```sh\n * ./config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" --name \"${RUNNER_NAME}\" --disableupdate\n * ```\n *\n * All parameters are specified as step function paths and therefore must be used only in step function task parameters.\n */\nexport interface RunnerRuntimeParameters {\n  /**\n   * Path to runner token used to register token.\n   */\n  readonly runnerTokenPath: string;\n\n  /**\n   * Path to desired runner name. We specifically set the name to make troubleshooting easier.\n   */\n  readonly runnerNamePath: string;\n\n  /**\n   * Path to GitHub domain. Most of the time this will be github.com but for self-hosted GitHub instances, this will be different.\n   */\n  readonly githubDomainPath: string;\n\n  /**\n   * Path to repostiroy owner name.\n   */\n  readonly ownerPath: string;\n\n  /**\n   * Path to repository name.\n   */\n  readonly repoPath: string;\n}\n\n/**\n * Interface for runner image status used by status.json.\n */\nexport interface IRunnerImageStatus {\n  /**\n   * Image repository where runner image is pushed.\n   */\n  readonly imageRepository?: string;\n\n  /**\n   * Tag of image that should be used.\n   */\n  readonly imageTag?: string;\n\n  /**\n   * Log group name for the image builder where history of image builds can be analyzed.\n   */\n  readonly imageBuilderLogGroup?: string;\n}\n\n/**\n * Interface for all runner providers. Implementations create all required resources and return a step function task that starts those resources from {@link getStepFunctionTask}.\n */\nexport interface IRunnerProvider extends ec2.IConnectable, iam.IGrantable {\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 use 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  readonly labels: string[];\n\n  /**\n   * VPC network in which runners will be placed.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group associated with runners.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Image used to create a new resource compute. Can be Docker image, AMI, or something else.\n   */\n  readonly image: RunnerImage;\n\n  /**\n   * Generate step function tasks that execute the runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters specific build parameters\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;\n}\n\n/**\n * Base class for all providers with common methods used by all providers.\n *\n * @internal\n */\nexport abstract class BaseProvider extends Construct {\n  protected labelsFromProperties(defaultLabel: string, propsLabel: string | undefined, propsLabels: string[] | undefined): string[] {\n    if (propsLabels && propsLabel) {\n      throw new Error('Must supply either `label` or `labels` in runner properties, but not both. Try removing the `label` property.');\n    }\n\n    if (propsLabels) {\n      return propsLabels;\n    }\n    if (propsLabel) {\n      return [propsLabel];\n    }\n    return [defaultLabel];\n  }\n}\n"]}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/providers/common.ts"],"names":[],"mappings":"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","sourcesContent":["import { aws_ec2 as ec2, aws_ecr as ecr, aws_iam as iam, aws_logs as logs, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\n/**\n * Defines desired GitHub Actions runner version.\n */\nexport class RunnerVersion {\n  /**\n   * Use the latest version available at the time the runner provider image is built.\n   */\n  public static latest(): RunnerVersion {\n    return new RunnerVersion('latest');\n  }\n\n  /**\n   * Use a specific version.\n   *\n   * @see https://github.com/actions/runner/releases\n   *\n   * @param version GitHub Runner version\n   */\n  public static specific(version: string) {\n    return new RunnerVersion(version);\n  }\n\n  protected constructor(readonly version: string) {\n  }\n\n  /**\n   * Check if two versions are the same.\n   *\n   * @param other version to compare\n   */\n  public is(other: RunnerVersion) {\n    return this.version == other.version;\n  }\n}\n\n/**\n * CPU architecture enum for an image.\n */\nexport class Architecture {\n  /**\n   * ARM64\n   */\n  public static readonly ARM64 = Architecture.of('ARM64');\n\n  /**\n   * X86_64\n   */\n  public static readonly X86_64 = Architecture.of('X86_64');\n\n  private static of(architecture: string) {\n    return new Architecture(architecture);\n  }\n\n  private constructor(public readonly name: string) {\n  }\n\n  /**\n  * Checks if the given architecture is the same as this one.\n  *\n  * @param arch architecture to compare\n  */\n  public is(arch: Architecture): boolean {\n    return arch.name == this.name;\n  }\n\n  /**\n   * Checks if this architecture is in a given list.\n   *\n   * @param arches architectures to check\n   */\n  public isIn(arches: Architecture[]): boolean {\n    for (const arch of arches) {\n      if (this.is(arch)) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Checks if a given EC2 instance type matches this architecture.\n   *\n   * @param instanceType instance type to check\n   */\n  public instanceTypeMatch(instanceType: ec2.InstanceType): boolean {\n    if (instanceType.architecture == ec2.InstanceArchitecture.X86_64) {\n      return this.is(Architecture.X86_64);\n    }\n    if (instanceType.architecture == ec2.InstanceArchitecture.ARM_64) {\n      return this.is(Architecture.ARM64);\n    }\n    throw new Error('Unknown instance type architecture');\n  }\n}\n\n/**\n * OS enum for an image.\n */\nexport class Os {\n  /**\n  * Linux\n  */\n  public static readonly LINUX = Os.of('Linux');\n\n  /**\n  * Windows\n  */\n  public static readonly WINDOWS = Os.of('Windows');\n\n  private static of(os: string) {\n    return new Os(os);\n  }\n\n  private constructor(public readonly name: string) {\n  }\n\n  /**\n  * Checks if the given OS is the same as this one.\n  *\n  * @param os OS to compare\n  */\n  public is(os: Os) {\n    return os.name == this.name;\n  }\n\n  /**\n   * Checks if this OS is in a given list.\n   *\n   * @param oses list of OS to check\n   */\n  public isIn(oses: Os[]): boolean {\n    for (const os of oses) {\n      if (this.is(os)) {\n        return true;\n      }\n    }\n    return false;\n  }\n}\n\n/**\n * Description of a Docker image built by {@link IImageBuilder}.\n */\nexport interface RunnerImage {\n  /**\n   * ECR repository containing the image.\n   */\n  readonly imageRepository: ecr.IRepository;\n\n  /**\n   * Static image tag where the image will be pushed.\n   */\n  readonly imageTag: string;\n\n  /**\n   * Architecture of the image.\n   */\n  readonly architecture: Architecture;\n\n  /**\n   * OS type of the image.\n   */\n  readonly os: Os;\n\n  /**\n   * Log group where image builds are logged.\n   */\n  readonly logGroup?: logs.LogGroup;\n\n  /**\n   * Installed runner version.\n   */\n  readonly runnerVersion: RunnerVersion;\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.\n *\n * Anything that ends up with an ECR repository containing a Docker image that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing image and nothing else.\n *\n * It's important that the specified image tag be available at the time the repository is available. Providers usually assume the image is ready and will fail if it's not.\n *\n * The image can be further updated over time manually or using a schedule as long as it is always written to the same tag.\n */\nexport interface IImageBuilder {\n  /**\n   * Finalize and return all required information about the Docker image built by this builder.\n   *\n   * This method can be called multiple times if the image is bound to multiple providers. Make sure you cache the image when implementing or return an error if this builder doesn't support reusing images.\n   *\n   * @return image\n   */\n  bind(): RunnerImage;\n}\n\n/**\n * Description of a AMI built by {@link IAmiBuilder}.\n */\nexport interface RunnerAmi {\n  /**\n   * Launch template pointing to the latest AMI.\n   */\n  readonly launchTemplate: ec2.ILaunchTemplate;\n\n  /**\n   * Architecture of the image.\n   */\n  readonly architecture: Architecture;\n\n  /**\n   * OS type of the image.\n   */\n  readonly os: Os;\n\n  /**\n   * Log group where image builds are logged.\n   */\n  readonly logGroup?: logs.LogGroup;\n\n  /**\n   * Installed runner version.\n   */\n  readonly runnerVersion: RunnerVersion;\n}\n\n/**\n * Interface for constructs that build an AMI that can be used in {@link IRunnerProvider}.\n *\n * Anything that ends up with a launch template pointing to an AMI that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing AMI and nothing else.\n *\n * The AMI can be further updated over time manually or using a schedule as long as it is always written to the same launch template.\n */\nexport interface IAmiBuilder {\n  /**\n   * Finalize and return all required information about the AMI built by this builder.\n   *\n   * This method can be called multiple times if the image is bound to multiple providers. Make sure you cache the image when implementing or return an error if this builder doesn't support reusing images.\n   *\n   * @return ami\n   */\n  bind(): RunnerAmi;\n}\n\n/**\n * Common properties for all runner providers.\n */\nexport interface RunnerProviderProps {\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n}\n\n/**\n * Workflow job parameters as parsed from the webhook event. Pass these into your runner executor and run something like:\n *\n * ```sh\n * ./config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" --name \"${RUNNER_NAME}\" --disableupdate\n * ```\n *\n * All parameters are specified as step function paths and therefore must be used only in step function task parameters.\n */\nexport interface RunnerRuntimeParameters {\n  /**\n   * Path to runner token used to register token.\n   */\n  readonly runnerTokenPath: string;\n\n  /**\n   * Path to desired runner name. We specifically set the name to make troubleshooting easier.\n   */\n  readonly runnerNamePath: string;\n\n  /**\n   * Path to GitHub domain. Most of the time this will be github.com but for self-hosted GitHub instances, this will be different.\n   */\n  readonly githubDomainPath: string;\n\n  /**\n   * Path to repostiroy owner name.\n   */\n  readonly ownerPath: string;\n\n  /**\n   * Path to repository name.\n   */\n  readonly repoPath: string;\n}\n\n/**\n * Image status returned from runner providers to be displayed in status.json.\n */\nexport interface IRunnerImageStatus {\n  /**\n   * Image repository where image builder pushes runner images.\n   */\n  readonly imageRepository: string;\n\n  /**\n   * Tag of image that should be used.\n   */\n  readonly imageTag: string;\n\n  /**\n   * Log group name for the image builder where history of image builds can be analyzed.\n   */\n  readonly imageBuilderLogGroup?: string;\n}\n\n/**\n * AMI status returned from runner providers to be displayed as output of status function.\n */\nexport interface IRunnerAmiStatus {\n  /**\n   * Id of launch template pointing to the latest AMI built by the AMI builder.\n   */\n  readonly launchTemplate: string;\n\n  /**\n   * Log group name for the AMI builder where history of builds can be analyzed.\n   */\n  readonly amiBuilderLogGroup?: string;\n}\n\n/**\n * Interface for runner image status used by status.json.\n */\nexport interface IRunnerProviderStatus {\n  /**\n   * Runner provider type.\n   */\n  readonly type: string;\n\n  /**\n   * Labels associated with provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC where runners will be launched.\n   */\n  readonly vpcArn?: string;\n\n  /**\n   * Security group attached to runners.\n   */\n  readonly securityGroup?: string;\n\n  /**\n   * Role attached to runners.\n   */\n  readonly roleArn?: string;\n\n  /**\n   * Details about Docker image used by this runner provider.\n   */\n  readonly image?: IRunnerImageStatus;\n\n  /**\n   * Details about AMI used by this runner provider.\n   */\n  readonly ami?: IRunnerAmiStatus;\n}\n\n/**\n * Interface for all runner providers. Implementations create all required resources and return a step function task that starts those resources from {@link getStepFunctionTask}.\n */\nexport interface IRunnerProvider extends ec2.IConnectable, iam.IGrantable {\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 use 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  readonly labels: string[];\n\n  /**\n   * VPC network in which runners will be placed.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group associated with runners.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Generate step function tasks that execute the runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters specific build parameters\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;\n\n  /**\n   * An optional method that modifies the role of the state machine after all the tasks have been generated. This can be used to add additional policy\n   * statements to the state machine role that are not automatically added by the task returned from {@link getStepFunctionTask}.\n   *\n   * @param stateMachineRole role for the state machine that executes the task returned from {@link getStepFunctionTask}.\n   */\n  grantStateMachine(stateMachineRole: iam.IGrantable): void;\n\n  /**\n   * Return status of the runner provider to be used in the main status function. Also gives the status function any needed permissions to query the Docker image or AMI.\n   *\n   * @param statusFunctionRole grantable for the status function\n   */\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus;\n}\n\n/**\n * Base class for all providers with common methods used by all providers.\n *\n * @internal\n */\nexport abstract class BaseProvider extends Construct {\n  protected labelsFromProperties(defaultLabel: string, propsLabel: string | undefined, propsLabels: string[] | undefined): string[] {\n    if (propsLabels && propsLabel) {\n      throw new Error('Must supply either `label` or `labels` in runner properties, but not both. Try removing the `label` property.');\n    }\n\n    if (propsLabels) {\n      return propsLabels;\n    }\n    if (propsLabel) {\n      return [propsLabel];\n    }\n    return [defaultLabel];\n  }\n}\n"]}
@@ -0,0 +1,106 @@
1
+ import * as cdk from 'aws-cdk-lib';
2
+ import { aws_ec2 as ec2, aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
3
+ import { Construct } from 'constructs';
4
+ import { BaseProvider, IAmiBuilder, IRunnerProvider, IRunnerProviderStatus, RunnerProviderProps, RunnerRuntimeParameters } from './common';
5
+ /**
6
+ * Properties for {@link Ec2Runner} construct.
7
+ */
8
+ export interface Ec2RunnerProps extends RunnerProviderProps {
9
+ /**
10
+ * AMI builder that creates AMIs with GitHub runner pre-configured. On Linux, a user named `runner` is expected to exist with access to Docker.
11
+ *
12
+ * @default AMI builder for Ubuntu Linux
13
+ */
14
+ readonly amiBuilder?: IAmiBuilder;
15
+ /**
16
+ * GitHub Actions labels used for this provider.
17
+ *
18
+ * 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
19
+ * 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
20
+ * job's labels, this provider will be chosen and spawn a new runner.
21
+ *
22
+ * @default ['ec2']
23
+ */
24
+ readonly labels?: string[];
25
+ /**
26
+ * Instance type for launched runner instances.
27
+ *
28
+ * @default m5.large
29
+ */
30
+ readonly instanceType?: ec2.InstanceType;
31
+ /**
32
+ * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.
33
+ *
34
+ * @default 30GB
35
+ */
36
+ readonly storageSize?: cdk.Size;
37
+ /**
38
+ * Security Group to assign to launched runner instances.
39
+ *
40
+ * @default account's default security group
41
+ */
42
+ readonly securityGroup?: ec2.ISecurityGroup;
43
+ /**
44
+ * Subnet where the runner instances will be launched.
45
+ *
46
+ * @default default subnet of account's default VPC
47
+ */
48
+ readonly subnet?: ec2.ISubnet;
49
+ /**
50
+ * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.
51
+ *
52
+ * @default false
53
+ */
54
+ readonly spot?: boolean;
55
+ /**
56
+ * Set a maximum price for spot instances.
57
+ *
58
+ * @default no max price (you will pay current spot price)
59
+ */
60
+ readonly spotMaxPrice?: string;
61
+ }
62
+ /**
63
+ * GitHub Actions runner provider using EC2 to execute jobs.
64
+ *
65
+ * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
66
+ */
67
+ export declare class Ec2Runner extends BaseProvider implements IRunnerProvider {
68
+ /**
69
+ * Labels associated with this provider.
70
+ */
71
+ readonly labels: string[];
72
+ /**
73
+ * VPC subnet used for hosting launched instances.
74
+ */
75
+ readonly subnet?: ec2.ISubnet;
76
+ /**
77
+ * Security group attached to launched instances.
78
+ */
79
+ readonly securityGroup?: ec2.ISecurityGroup;
80
+ /**
81
+ * Grant principal used to add permissions to the runner role.
82
+ */
83
+ readonly grantPrincipal: iam.IPrincipal;
84
+ private readonly ami;
85
+ private readonly logGroup;
86
+ private readonly role;
87
+ private readonly instanceType;
88
+ private readonly storageSize;
89
+ private readonly spot;
90
+ private readonly spotMaxPrice;
91
+ constructor(scope: Construct, id: string, props: Ec2RunnerProps);
92
+ /**
93
+ * Generate step function task(s) to start a new runner.
94
+ *
95
+ * Called by GithubRunners and shouldn't be called manually.
96
+ *
97
+ * @param parameters workflow job details
98
+ */
99
+ getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;
100
+ grantStateMachine(stateMachineRole: iam.IGrantable): void;
101
+ status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus;
102
+ /**
103
+ * The network connections associated with this resource.
104
+ */
105
+ get connections(): ec2.Connections;
106
+ }