UNPKG

@cloudsnorkel/cdk-github-runners

Version:

CDK construct to create GitHub Actions self-hosted runners. Creates ephemeral runners on demand. Easy to deploy and highly customizable.

358 lines 63.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.CodeBuildImageBuilderFailedBuildNotifier = exports.CodeBuildRunnerImageBuilder = void 0; const crypto = require("node:crypto"); const cdk = require("aws-cdk-lib"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild"); const aws_ecr_1 = require("aws-cdk-lib/aws-ecr"); const aws_logs_1 = require("aws-cdk-lib/aws-logs"); const aws_image_builder_1 = require("./aws-image-builder"); const base_image_1 = require("./aws-image-builder/base-image"); const build_image_function_1 = require("./build-image-function"); const common_1 = require("./common"); const providers_1 = require("../providers"); const utils_1 = require("../utils"); /** * @internal */ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase { constructor(scope, id, props) { super(scope, id, props); if (props?.awsImageBuilderOptions) { aws_cdk_lib_1.Annotations.of(this).addWarning('awsImageBuilderOptions are ignored when using CodeBuild runner image builder.'); } this.os = props?.os ?? providers_1.Os.LINUX_UBUNTU; this.architecture = props?.architecture ?? providers_1.Architecture.X86_64; this.rebuildInterval = props?.rebuildInterval ?? aws_cdk_lib_1.Duration.days(7); this.logRetention = props?.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH; this.logRemovalPolicy = props?.logRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.DESTROY; this.vpc = props?.vpc; this.securityGroups = props?.securityGroups; this.subnetSelection = props?.subnetSelection; this.timeout = props?.codeBuildOptions?.timeout ?? aws_cdk_lib_1.Duration.hours(1); this.computeType = props?.codeBuildOptions?.computeType ?? aws_codebuild_1.ComputeType.SMALL; this.buildImage = props?.codeBuildOptions?.buildImage ?? this.getDefaultBuildImage(); this.waitOnDeploy = props?.waitOnDeploy ?? true; this.dockerSetupCommands = props?.dockerSetupCommands ?? []; // normalize BaseContainerImageInput to BaseContainerImage (string support is deprecated, only at public API level) const baseDockerImageInput = props?.baseDockerImage ?? (0, aws_image_builder_1.defaultBaseDockerImage)(this.os); this.baseImage = typeof baseDockerImageInput === 'string' ? base_image_1.BaseContainerImage.fromString(baseDockerImageInput) : baseDockerImageInput; // warn if using deprecated string format (only if user explicitly provided it) if (props?.baseDockerImage && typeof props.baseDockerImage === 'string') { aws_cdk_lib_1.Annotations.of(this).addWarning('Passing baseDockerImage as a string is deprecated. Please use BaseContainerImage static factory methods instead, e.g., BaseContainerImage.fromDockerHub("ubuntu", "22.04") or BaseContainerImage.fromString("public.ecr.aws/lts/ubuntu:22.04")'); } // warn against isolated networks if (props?.subnetSelection?.subnetType == aws_cdk_lib_1.aws_ec2.SubnetType.PRIVATE_ISOLATED) { aws_cdk_lib_1.Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' + 'See https://github.com/aws/containers-roadmap/issues/1160'); } // error out on no-nat networks because the build will hang if (props?.subnetSelection?.subnetType == aws_cdk_lib_1.aws_ec2.SubnetType.PUBLIC) { aws_cdk_lib_1.Annotations.of(this).addError('Public subnets do not work with CodeBuild as it cannot be assigned an IP. ' + 'See https://docs.aws.amazon.com/codebuild/latest/userguide/vpc-support.html#best-practices-for-vpcs'); } // check timeout if (this.timeout.toSeconds() > aws_cdk_lib_1.Duration.hours(8).toSeconds()) { aws_cdk_lib_1.Annotations.of(this).addError('CodeBuild runner image builder timeout must 8 hours or less.'); } // create service role for CodeBuild this.role = new aws_cdk_lib_1.aws_iam.Role(this, 'Role', { assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('codebuild.amazonaws.com'), }); // create repository that only keeps one tag this.repository = new aws_cdk_lib_1.aws_ecr.Repository(this, 'Repository', { imageScanOnPush: true, imageTagMutability: aws_ecr_1.TagMutability.MUTABLE, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, emptyOnDelete: true, lifecycleRules: [ { description: 'Remove soci indexes for replaced images', tagStatus: aws_ecr_1.TagStatus.TAGGED, tagPrefixList: ['sha256-'], maxImageCount: 1, }, { description: 'Remove untagged images that have been replaced by CodeBuild', tagStatus: aws_ecr_1.TagStatus.UNTAGGED, maxImageAge: aws_cdk_lib_1.Duration.days(1), }, ], }); } bindAmi() { throw new Error('CodeBuild image builder cannot be used to build AMI'); } bindDockerImage() { if (this.boundDockerImage) { return this.boundDockerImage; } // log group for the image builds const logGroup = new aws_cdk_lib_1.aws_logs.LogGroup(this, 'Logs', { retention: this.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH, removalPolicy: this.logRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.DESTROY, }); // generate buildSpec const [buildSpec, buildSpecHash] = this.getBuildSpec(this.repository); // create CodeBuild project that builds Dockerfile and pushes to repository const project = new aws_cdk_lib_1.aws_codebuild.Project(this, 'CodeBuild', { description: `Build docker image for self-hosted GitHub runner ${this.node.path} (${this.os.name}/${this.architecture.name})`, buildSpec, vpc: this.vpc, securityGroups: this.securityGroups, subnetSelection: this.subnetSelection, role: this.role, timeout: this.timeout, environment: { buildImage: this.buildImage, computeType: this.computeType, privileged: true, }, logging: { cloudWatch: { logGroup, }, }, }); // permissions this.repository.grantPullPush(project); // Grant pull permissions for base image ECR repository if applicable if (this.baseImage.ecrRepository) { this.baseImage.ecrRepository.grantPull(project); } // call CodeBuild during deployment const completedImage = this.customResource(project, buildSpecHash); // rebuild image on a schedule this.rebuildImageOnSchedule(project, this.rebuildInterval); // return the image this.boundDockerImage = { imageRepository: this.repository, imageTag: 'latest', architecture: this.architecture, os: this.os, logGroup, runnerVersion: providers_1.RunnerVersion.specific('unknown'), _dependable: completedImage, }; return this.boundDockerImage; } getDefaultBuildImage() { if (this.os.isIn(providers_1.Os._ALL_LINUX_VERSIONS)) { // CodeBuild just runs `docker build` so its OS doesn't really matter if (this.architecture.is(providers_1.Architecture.X86_64)) { return aws_cdk_lib_1.aws_codebuild.LinuxBuildImage.AMAZON_LINUX_2_5; } else if (this.architecture.is(providers_1.Architecture.ARM64)) { return aws_cdk_lib_1.aws_codebuild.LinuxArmBuildImage.AMAZON_LINUX_2_STANDARD_3_0; } } if (this.os.is(providers_1.Os.WINDOWS)) { throw new Error('CodeBuild cannot be used to build Windows Docker images https://github.com/docker-library/docker/issues/49'); } throw new Error(`Unable to find CodeBuild image for ${this.os.name}/${this.architecture.name}`); } getDockerfileGenerationCommands() { let hashedComponents = []; let commands = []; let dockerfile = `FROM ${this.baseImage.image}\nVOLUME /var/lib/docker\n`; for (let i = 0; i < this.components.length; i++) { const componentName = this.components[i].name; const safeComponentName = componentName.replace(/[^a-zA-Z0-9-]/g, '_'); const assetDescriptors = this.components[i].getAssets(this.os, this.architecture); for (let j = 0; j < assetDescriptors.length; j++) { if (this.os.is(providers_1.Os.WINDOWS)) { throw new Error("Can't add asset as we can't build Windows Docker images on CodeBuild"); } const asset = new aws_cdk_lib_1.aws_s3_assets.Asset(this, `Component ${i} ${componentName} Asset ${j}`, { path: assetDescriptors[j].source, }); if (asset.isFile) { commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${safeComponentName}-${j}`); } else if (asset.isZipArchive) { commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${safeComponentName}-${j}.zip`); commands.push(`unzip asset${i}-${safeComponentName}-${j}.zip -d asset${i}-${safeComponentName}-${j}`); } else { throw new Error(`Unknown asset type: ${asset}`); } dockerfile += `COPY asset${i}-${safeComponentName}-${j} ${assetDescriptors[j].target}\n`; hashedComponents.push(`__ ASSET FILE ${asset.assetHash} ${i}-${componentName}-${j} ${assetDescriptors[j].target}`); asset.grantRead(this); } const componentCommands = this.components[i].getCommands(this.os, this.architecture); const script = '#!/bin/bash\nset -exuo pipefail\n' + componentCommands.join('\n'); commands.push(`cat > component${i}-${safeComponentName}.sh <<'EOFGITHUBRUNNERSDOCKERFILE'\n${script}\nEOFGITHUBRUNNERSDOCKERFILE`); commands.push(`chmod +x component${i}-${safeComponentName}.sh`); hashedComponents.push(`__ COMMAND ${i} ${componentName} ${script}`); dockerfile += `COPY component${i}-${safeComponentName}.sh /tmp\n`; dockerfile += `RUN /tmp/component${i}-${safeComponentName}.sh\n`; const dockerCommands = this.components[i].getDockerCommands(this.os, this.architecture); dockerfile += dockerCommands.join('\n') + '\n'; hashedComponents.push(`__ DOCKER COMMAND ${i} ${dockerCommands.join('\n')}`); } commands.push(`cat > Dockerfile <<'EOFGITHUBRUNNERSDOCKERFILE'\n${dockerfile}\nEOFGITHUBRUNNERSDOCKERFILE`); return [commands, hashedComponents]; } getBuildSpec(repository) { const thisStack = cdk.Stack.of(this); let archUrl; if (this.architecture.is(providers_1.Architecture.X86_64)) { archUrl = 'x86_64'; } else if (this.architecture.is(providers_1.Architecture.ARM64)) { archUrl = 'arm64'; } else { throw new Error(`Unsupported architecture for required CodeBuild: ${this.architecture.name}`); } const [commands, commandsHashedComponents] = this.getDockerfileGenerationCommands(); const buildSpecVersion = 'v2'; // change this every time the build spec changes const hashedComponents = commandsHashedComponents.concat(buildSpecVersion, this.architecture.name, this.baseImage.image, this.os.name); const hash = crypto.createHash('md5').update(hashedComponents.join('\n')).digest('hex').slice(0, 10); const buildSpec = aws_cdk_lib_1.aws_codebuild.BuildSpec.fromObject({ version: 0.2, env: { variables: { REPO_ARN: repository.repositoryArn, REPO_URI: repository.repositoryUri, WAIT_HANDLE: 'unspecified', BASH_ENV: 'codebuild-log.sh', }, shell: 'bash', }, phases: { // we can't use pre_build. the wait handle will never complete if pre_build fails as post_build won't run. this can cause timeouts during deployment. build: { commands: [ 'echo "exec > >(tee -a /tmp/codebuild.log) 2>&1" > codebuild-log.sh', `aws ecr get-login-password --region "$AWS_DEFAULT_REGION" | docker login --username AWS --password-stdin ${thisStack.account}.dkr.ecr.${thisStack.region}.amazonaws.com`, ...this.dockerSetupCommands, ...commands, 'docker build --progress plain . -t "$REPO_URI"', 'docker push "$REPO_URI"', ], }, post_build: { commands: [ 'rm -f codebuild-log.sh && STATUS="SUCCESS"', 'if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ]; then STATUS="FAILURE"; fi', 'cat <<EOF > /tmp/payload.json\n' + '{\n' + ' "Status": "$STATUS",\n' + ' "UniqueId": "build",\n' + // we remove non-printable characters from the log because CloudFormation doesn't like them // https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1601 ' "Reason": `sed \'s/[^[:print:]]//g\' /tmp/codebuild.log | tail -c 400 | jq -Rsa .`,\n' + // for lambda always get a new value because there is always a new image hash ' "Data": "$RANDOM"\n' + '}\n' + 'EOF', 'if [ "$WAIT_HANDLE" != "unspecified" ]; then jq . /tmp/payload.json; curl -fsSL -X PUT -H "Content-Type:" -d "@/tmp/payload.json" "$WAIT_HANDLE"; fi', // generate and push soci index // we do this after finishing the build, so we don't have to wait. it's also not required, so it's ok if it fails 'if [ `docker inspect --format=\'{{json .Config.Labels.DISABLE_SOCI}}\' "$REPO_URI"` = "null" ]; then\n' + 'docker rmi "$REPO_URI"\n' + // it downloads the image again to /tmp, so save on space 'LATEST_SOCI_VERSION=`curl -w "%{redirect_url}" -fsS https://github.com/CloudSnorkel/standalone-soci-indexer/releases/latest | grep -oE "[^/]+$"`\n' + `curl -fsSL https://github.com/CloudSnorkel/standalone-soci-indexer/releases/download/$\{LATEST_SOCI_VERSION}/standalone-soci-indexer_Linux_${archUrl}.tar.gz | tar xz\n` + './standalone-soci-indexer "$REPO_URI"\n' + 'fi', ], }, }, }); return [buildSpec, hash]; } customResource(project, buildSpecHash) { const crHandler = (0, utils_1.singletonLambda)(build_image_function_1.BuildImageFunction, this, 'build-image', { description: 'Custom resource handler that triggers CodeBuild to build runner images', timeout: cdk.Duration.minutes(3), logGroup: (0, utils_1.singletonLogGroup)(this, utils_1.SingletonLogType.RUNNER_IMAGE_BUILD), loggingFormat: aws_cdk_lib_1.aws_lambda.LoggingFormat.JSON, }); const policy = new aws_cdk_lib_1.aws_iam.Policy(this, 'CR Policy', { statements: [ new aws_cdk_lib_1.aws_iam.PolicyStatement({ actions: ['codebuild:StartBuild'], resources: [project.projectArn], }), ], }); crHandler.role.attachInlinePolicy(policy); let waitHandleRef = 'unspecified'; let waitDependable = ''; if (this.waitOnDeploy) { // Wait handle lets us wait for longer than an hour for the image build to complete. // We generate a new wait handle for build spec changes to guarantee a new image is built. // This also helps make sure the changes are good. If they have a bug, the deployment will fail instead of just the scheduled build. // Finally, it's recommended by CloudFormation docs to not reuse wait handles or old responses may interfere in some cases. const handle = new aws_cdk_lib_1.aws_cloudformation.CfnWaitConditionHandle(this, `Build Wait Handle ${buildSpecHash}`); const wait = new aws_cdk_lib_1.aws_cloudformation.CfnWaitCondition(this, `Build Wait ${buildSpecHash}`, { handle: handle.ref, timeout: this.timeout.toSeconds().toString(), // don't wait longer than the build timeout count: 1, }); waitHandleRef = handle.ref; waitDependable = wait.ref; } const cr = new aws_cdk_lib_1.CustomResource(this, 'Builder', { serviceToken: crHandler.functionArn, resourceType: 'Custom::ImageBuilder', properties: { RepoName: this.repository.repositoryName, ProjectName: project.projectName, WaitHandle: waitHandleRef, }, }); // add dependencies to make sure resources are there when we need them cr.node.addDependency(project); cr.node.addDependency(this.role); cr.node.addDependency(policy); cr.node.addDependency(crHandler.role); cr.node.addDependency(crHandler); return waitDependable; // user needs to wait on wait handle which is triggered when the image is built } rebuildImageOnSchedule(project, rebuildInterval) { rebuildInterval = rebuildInterval ?? aws_cdk_lib_1.Duration.days(7); if (rebuildInterval.toMilliseconds() != 0) { const scheduleRule = new aws_cdk_lib_1.aws_events.Rule(this, 'Build Schedule', { description: `Rebuild runner image for ${this.repository.repositoryName}`, schedule: aws_cdk_lib_1.aws_events.Schedule.rate(rebuildInterval), }); scheduleRule.addTarget(new aws_cdk_lib_1.aws_events_targets.CodeBuildProject(project)); } } get connections() { return new aws_cdk_lib_1.aws_ec2.Connections({ securityGroups: this.securityGroups, }); } get grantPrincipal() { return this.role; } } exports.CodeBuildRunnerImageBuilder = CodeBuildRunnerImageBuilder; /** * @internal */ class CodeBuildImageBuilderFailedBuildNotifier { constructor(topic) { this.topic = topic; } visit(node) { if (node instanceof CodeBuildRunnerImageBuilder) { const builder = node; const projectNode = builder.node.tryFindChild('CodeBuild'); if (projectNode) { const project = projectNode; project.notifyOnBuildFailed('BuildFailed', this.topic); } else { cdk.Annotations.of(builder).addWarning('Unused builder cannot get notifications of failed builds'); } } } } exports.CodeBuildImageBuilderFailedBuildNotifier = CodeBuildImageBuilderFailedBuildNotifier; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/image-builders/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'node:crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_cloudformation as cloudformation,\n  aws_codebuild as codebuild,\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  aws_sns as sns,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct, IConstruct } from 'constructs';\nimport { defaultBaseDockerImage } from './aws-image-builder';\nimport { BaseContainerImage } from './aws-image-builder/base-image';\nimport { BuildImageFunction } from './build-image-function';\nimport { BuildImageFunctionProperties } from './build-image.lambda';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps } from './common';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers';\nimport { singletonLambda, singletonLogGroup, SingletonLogType } from '../utils';\n\n\nexport interface CodeBuildRunnerImageBuilderProps {\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * The compute type determines CPU, memory, and disk space:\n   * - SMALL: 2 vCPU, 3 GB RAM, 64 GB disk\n   * - MEDIUM: 4 vCPU, 7 GB RAM, 128 GB disk\n   * - LARGE: 8 vCPU, 15 GB RAM, 128 GB disk\n   * - X2_LARGE: 72 vCPU, 145 GB RAM, 256 GB disk (Linux) or 824 GB disk (Windows)\n   *\n   * Use a larger compute type when you need more disk space for building larger Docker images.\n   *\n   * For more details, see https://docs.aws.amazon.com/codebuild/latest/userguide/build-env-ref-compute-types.html#environment.types\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * Build image to use in CodeBuild. This is the image that's going to run the code that builds the runner image.\n   *\n   * The only action taken in CodeBuild is running `docker build`. You would therefore not need to change this setting often.\n   *\n   * @default Amazon Linux 2023\n   */\n  readonly buildImage?: codebuild.IBuildImage;\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 * @internal\n */\nexport class CodeBuildRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: BaseContainerImage;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc | undefined;\n  private readonly securityGroups: ec2.ISecurityGroup[] | undefined;\n  private readonly buildImage: codebuild.IBuildImage;\n  private readonly repository: ecr.Repository;\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly timeout: cdk.Duration;\n  private readonly computeType: codebuild.ComputeType;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly role: iam.Role;\n  private readonly waitOnDeploy: boolean;\n  private readonly dockerSetupCommands: string[];\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.awsImageBuilderOptions) {\n      Annotations.of(this).addWarning('awsImageBuilderOptions are ignored when using CodeBuild runner image builder.');\n    }\n\n    this.os = props?.os ?? Os.LINUX_UBUNTU;\n    this.architecture = props?.architecture ?? Architecture.X86_64;\n    this.rebuildInterval = props?.rebuildInterval ?? Duration.days(7);\n    this.logRetention = props?.logRetention ?? RetentionDays.ONE_MONTH;\n    this.logRemovalPolicy = props?.logRemovalPolicy ?? RemovalPolicy.DESTROY;\n    this.vpc = props?.vpc;\n    this.securityGroups = props?.securityGroups;\n    this.subnetSelection = props?.subnetSelection;\n    this.timeout = props?.codeBuildOptions?.timeout ?? Duration.hours(1);\n    this.computeType = props?.codeBuildOptions?.computeType ?? ComputeType.SMALL;\n    this.buildImage = props?.codeBuildOptions?.buildImage ?? this.getDefaultBuildImage();\n    this.waitOnDeploy = props?.waitOnDeploy ?? true;\n    this.dockerSetupCommands = props?.dockerSetupCommands ?? [];\n\n    // normalize BaseContainerImageInput to BaseContainerImage (string support is deprecated, only at public API level)\n    const baseDockerImageInput = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.baseImage = typeof baseDockerImageInput === 'string' ? BaseContainerImage.fromString(baseDockerImageInput) : baseDockerImageInput;\n\n    // warn if using deprecated string format (only if user explicitly provided it)\n    if (props?.baseDockerImage && typeof props.baseDockerImage === 'string') {\n      Annotations.of(this).addWarning(\n        'Passing baseDockerImage as a string is deprecated. Please use BaseContainerImage static factory methods instead, e.g., BaseContainerImage.fromDockerHub(\"ubuntu\", \"22.04\") or BaseContainerImage.fromString(\"public.ecr.aws/lts/ubuntu:22.04\")',\n      );\n    }\n\n    // warn against isolated networks\n    if (props?.subnetSelection?.subnetType == ec2.SubnetType.PRIVATE_ISOLATED) {\n      Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +\n        'See https://github.com/aws/containers-roadmap/issues/1160');\n    }\n\n    // error out on no-nat networks because the build will hang\n    if (props?.subnetSelection?.subnetType == ec2.SubnetType.PUBLIC) {\n      Annotations.of(this).addError('Public subnets do not work with CodeBuild as it cannot be assigned an IP. ' +\n        'See https://docs.aws.amazon.com/codebuild/latest/userguide/vpc-support.html#best-practices-for-vpcs');\n    }\n\n    // check timeout\n    if (this.timeout.toSeconds() > Duration.hours(8).toSeconds()) {\n      Annotations.of(this).addError('CodeBuild runner image builder timeout must 8 hours or less.');\n    }\n\n    // create service role for CodeBuild\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com'),\n    });\n\n    // create repository that only keeps one tag\n    this.repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      emptyOnDelete: true,\n      lifecycleRules: [\n        {\n          description: 'Remove soci indexes for replaced images',\n          tagStatus: TagStatus.TAGGED,\n          tagPrefixList: ['sha256-'],\n          maxImageCount: 1,\n        },\n        {\n          description: 'Remove untagged images that have been replaced by CodeBuild',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n    });\n  }\n\n  bindAmi(): RunnerAmi {\n    throw new Error('CodeBuild image builder cannot be used to build AMI');\n  }\n\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    // log group for the image builds\n    const logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: this.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: this.logRemovalPolicy ?? RemovalPolicy.DESTROY,\n      },\n    );\n\n    // generate buildSpec\n    const [buildSpec, buildSpecHash] = this.getBuildSpec(this.repository);\n\n    // create CodeBuild project that builds Dockerfile and pushes to repository\n    const project = new codebuild.Project(this, 'CodeBuild', {\n      description: `Build docker image for self-hosted GitHub runner ${this.node.path} (${this.os.name}/${this.architecture.name})`,\n      buildSpec,\n      vpc: this.vpc,\n      securityGroups: this.securityGroups,\n      subnetSelection: this.subnetSelection,\n      role: this.role,\n      timeout: this.timeout,\n      environment: {\n        buildImage: this.buildImage,\n        computeType: this.computeType,\n        privileged: true,\n      },\n      logging: {\n        cloudWatch: {\n          logGroup,\n        },\n      },\n    });\n\n    // permissions\n    this.repository.grantPullPush(project);\n\n    // Grant pull permissions for base image ECR repository if applicable\n    if (this.baseImage.ecrRepository) {\n      this.baseImage.ecrRepository.grantPull(project);\n    }\n\n    // call CodeBuild during deployment\n    const completedImage = this.customResource(project, buildSpecHash);\n\n    // rebuild image on a schedule\n    this.rebuildImageOnSchedule(project, this.rebuildInterval);\n\n    // return the image\n    this.boundDockerImage = {\n      imageRepository: this.repository,\n      imageTag: 'latest',\n      architecture: this.architecture,\n      os: this.os,\n      logGroup,\n      runnerVersion: RunnerVersion.specific('unknown'),\n      _dependable: completedImage,\n    };\n    return this.boundDockerImage;\n  }\n\n  private getDefaultBuildImage(): codebuild.IBuildImage {\n    if (this.os.isIn(Os._ALL_LINUX_VERSIONS)) {\n      // CodeBuild just runs `docker build` so its OS doesn't really matter\n      if (this.architecture.is(Architecture.X86_64)) {\n        return codebuild.LinuxBuildImage.AMAZON_LINUX_2_5;\n      } else if (this.architecture.is(Architecture.ARM64)) {\n        return codebuild.LinuxArmBuildImage.AMAZON_LINUX_2_STANDARD_3_0;\n      }\n    }\n    if (this.os.is(Os.WINDOWS)) {\n      throw new Error('CodeBuild cannot be used to build Windows Docker images https://github.com/docker-library/docker/issues/49');\n    }\n\n    throw new Error(`Unable to find CodeBuild image for ${this.os.name}/${this.architecture.name}`);\n  }\n\n  private getDockerfileGenera