@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.
340 lines • 60.8 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.EcsRunnerProvider = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cdk = require("aws-cdk-lib");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const autoscaling = require("aws-cdk-lib/aws-autoscaling");
const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
const common_1 = require("./common");
const fargate_1 = require("./fargate");
const image_builders_1 = require("../image-builders");
const utils_1 = require("../utils");
class EcsEc2LaunchTarget {
constructor(props) {
this.props = props;
}
/**
* Called when the ECS launch type configured on RunTask
*/
bind(_task, _launchTargetOptions) {
return {
parameters: {
PropagateTags: aws_cdk_lib_1.aws_ecs.PropagatedTagSource.TASK_DEFINITION,
CapacityProviderStrategy: [
{
CapacityProvider: this.props.capacityProvider,
},
],
},
};
}
}
/**
* GitHub Actions runner provider using ECS on EC2 to execute jobs.
*
* ECS can be useful when you want more control of the infrastructure running the GitHub Actions Docker containers. You can control the autoscaling
* group to scale down to zero during the night and scale up during work hours. This way you can still save money, but have to wait less for
* infrastructure to spin up.
*
* This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
*/
class EcsRunnerProvider extends common_1.BaseProvider {
/**
* Create new image builder that builds ECS specific runner images.
*
* You can customize the OS, architecture, VPC, subnet, security groups, etc. by passing in props.
*
* You can add components to the image builder by calling `imageBuilder.addComponent()`.
*
* The default OS is Ubuntu running on x64 architecture.
*
* Included components:
* * `RunnerImageComponent.requiredPackages()`
* * `RunnerImageComponent.runnerUser()`
* * `RunnerImageComponent.git()`
* * `RunnerImageComponent.githubCli()`
* * `RunnerImageComponent.awsCli()`
* * `RunnerImageComponent.docker()`
* * `RunnerImageComponent.githubRunner()`
*/
static imageBuilder(scope, id, props) {
return image_builders_1.RunnerImageBuilder.new(scope, id, {
os: common_1.Os.LINUX_UBUNTU,
architecture: common_1.Architecture.X86_64,
components: [
image_builders_1.RunnerImageComponent.requiredPackages(),
image_builders_1.RunnerImageComponent.runnerUser(),
image_builders_1.RunnerImageComponent.git(),
image_builders_1.RunnerImageComponent.githubCli(),
image_builders_1.RunnerImageComponent.awsCli(),
image_builders_1.RunnerImageComponent.docker(),
image_builders_1.RunnerImageComponent.githubRunner(props?.runnerVersion ?? common_1.RunnerVersion.latest()),
],
...props,
});
}
constructor(scope, id, props) {
super(scope, id, props);
this.retryableErrors = [
'Ecs.EcsException',
'ECS.AmazonECSException',
'Ecs.LimitExceededException',
'Ecs.UpdateInProgressException',
];
this.labels = props?.labels ?? ['ecs'];
this.group = props?.group;
this.vpc = props?.vpc ?? aws_cdk_lib_1.aws_ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });
this.subnetSelection = props?.subnetSelection;
this.securityGroups = props?.securityGroups ?? [new aws_cdk_lib_1.aws_ec2.SecurityGroup(this, 'security group', { vpc: this.vpc })];
this.connections = new aws_cdk_lib_1.aws_ec2.Connections({ securityGroups: this.securityGroups });
this.assignPublicIp = props?.assignPublicIp ?? true;
this.cluster = props?.cluster ? props.cluster : new aws_cdk_lib_1.aws_ecs.Cluster(this, 'cluster', {
vpc: this.vpc,
enableFargateCapacityProviders: false,
});
if (props?.storageOptions && !props?.storageSize) {
throw new Error('storageSize is required when storageOptions are specified');
}
const imageBuilder = props?.imageBuilder ?? EcsRunnerProvider.imageBuilder(this, 'Image Builder');
const image = this.image = imageBuilder.bindDockerImage();
if (props?.capacityProvider) {
if (props?.minInstances || props?.maxInstances || props?.instanceType || props?.storageSize || props?.spot || props?.spotMaxPrice) {
cdk.Annotations.of(this).addWarning('When using a custom capacity provider, minInstances, maxInstances, instanceType, storageSize, spot, and spotMaxPrice will be ignored.');
}
this.capacityProvider = props.capacityProvider;
}
else {
const spot = props?.spot ?? props?.spotMaxPrice !== undefined;
const launchTemplate = new aws_cdk_lib_1.aws_ec2.LaunchTemplate(this, 'Launch Template', {
machineImage: this.defaultClusterInstanceAmi(),
instanceType: props?.instanceType ?? this.defaultClusterInstanceType(),
blockDevices: props?.storageSize ? [
{
deviceName: (0, common_1.amiRootDevice)(this, this.defaultClusterInstanceAmi().getImage(this).imageId).ref,
volume: {
ebsDevice: {
deleteOnTermination: true,
volumeSize: props.storageSize.toGibibytes(),
volumeType: props.storageOptions?.volumeType,
iops: props.storageOptions?.iops,
throughput: props.storageOptions?.throughput,
},
},
},
] : undefined,
spotOptions: spot ? {
requestType: aws_cdk_lib_1.aws_ec2.SpotRequestType.ONE_TIME,
maxPrice: props?.spotMaxPrice ? parseFloat(props?.spotMaxPrice) : undefined,
} : undefined,
requireImdsv2: true,
securityGroup: this.securityGroups[0],
role: new aws_cdk_lib_1.aws_iam.Role(this, 'Launch Template Role', {
assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('ec2.amazonaws.com'),
}),
userData: aws_cdk_lib_1.aws_ec2.UserData.forOperatingSystem(image.os.is(common_1.Os.WINDOWS) ? aws_cdk_lib_1.aws_ec2.OperatingSystemType.WINDOWS : aws_cdk_lib_1.aws_ec2.OperatingSystemType.LINUX),
});
this.securityGroups.slice(1).map(sg => launchTemplate.connections.addSecurityGroup(sg));
const autoScalingGroup = new autoscaling.AutoScalingGroup(this, 'Auto Scaling Group', {
vpc: this.vpc,
launchTemplate,
vpcSubnets: this.subnetSelection,
minCapacity: props?.minInstances ?? 0,
maxCapacity: props?.maxInstances ?? 5,
});
this.capacityProvider = props?.capacityProvider ?? new aws_cdk_lib_1.aws_ecs.AsgCapacityProvider(this, 'Capacity Provider', {
autoScalingGroup,
spotInstanceDraining: false, // waste of money to restart jobs as the restarted job won't have a token
});
}
this.capacityProvider.autoScalingGroup.addUserData(
// we don't exit on errors because all of these commands are optional
...this.loginCommands(), this.pullCommand(), ...this.ecsSettingsCommands());
this.capacityProvider.autoScalingGroup.role.addToPrincipalPolicy(utils_1.MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT);
image.imageRepository.grantPull(this.capacityProvider.autoScalingGroup);
this.cluster.addAsgCapacityProvider(this.capacityProvider, {
spotInstanceDraining: false,
machineImageType: aws_ecs_1.MachineImageType.AMAZON_LINUX_2,
});
this.logGroup = new aws_cdk_lib_1.aws_logs.LogGroup(this, 'logs', {
retention: props?.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH,
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
});
this.dind = (props?.dockerInDocker ?? true) && !image.os.is(common_1.Os.WINDOWS);
this.task = new aws_cdk_lib_1.aws_ecs.Ec2TaskDefinition(this, 'task');
this.container = this.task.addContainer('runner', {
image: aws_cdk_lib_1.aws_ecs.AssetImage.fromEcrRepository(image.imageRepository, image.imageTag),
cpu: props?.cpu ?? 1024,
memoryLimitMiB: props?.memoryLimitMiB ?? (props?.memoryReservationMiB ? undefined : 3500),
memoryReservationMiB: props?.memoryReservationMiB,
logging: aws_cdk_lib_1.aws_ecs.AwsLogDriver.awsLogs({
logGroup: this.logGroup,
streamPrefix: 'runner',
}),
command: (0, fargate_1.ecsRunCommand)(this.image.os, this.dind),
user: image.os.is(common_1.Os.WINDOWS) ? undefined : 'runner',
privileged: this.dind,
});
this.grantPrincipal = this.task.taskRole;
// permissions for SSM Session Manager
this.task.taskRole.addToPrincipalPolicy(utils_1.MINIMAL_ECS_SSM_SESSION_MANAGER_POLICY_STATEMENT);
}
defaultClusterInstanceType() {
if (this.image.architecture.is(common_1.Architecture.X86_64)) {
return aws_cdk_lib_1.aws_ec2.InstanceType.of(aws_cdk_lib_1.aws_ec2.InstanceClass.M6I, aws_cdk_lib_1.aws_ec2.InstanceSize.LARGE);
}
if (this.image.architecture.is(common_1.Architecture.ARM64)) {
return aws_cdk_lib_1.aws_ec2.InstanceType.of(aws_cdk_lib_1.aws_ec2.InstanceClass.M6G, aws_cdk_lib_1.aws_ec2.InstanceSize.LARGE);
}
throw new Error(`Unable to find instance type for ECS instances for ${this.image.architecture.name}`);
}
defaultClusterInstanceAmi() {
let baseImage;
let ssmPath;
let found = false;
if (this.image.os.isIn(common_1.Os._ALL_LINUX_VERSIONS)) {
if (this.image.architecture.is(common_1.Architecture.X86_64)) {
baseImage = aws_cdk_lib_1.aws_ecs.EcsOptimizedImage.amazonLinux2(aws_cdk_lib_1.aws_ecs.AmiHardwareType.STANDARD);
ssmPath = '/aws/service/ecs/optimized-ami/amazon-linux-2023/recommended/image_id';
found = true;
}
if (this.image.architecture.is(common_1.Architecture.ARM64)) {
baseImage = aws_cdk_lib_1.aws_ecs.EcsOptimizedImage.amazonLinux2(aws_cdk_lib_1.aws_ecs.AmiHardwareType.ARM);
ssmPath = '/aws/service/ecs/optimized-ami/amazon-linux-2023/arm64/recommended/image_id';
found = true;
}
}
if (this.image.os.is(common_1.Os.WINDOWS)) {
baseImage = aws_cdk_lib_1.aws_ecs.EcsOptimizedImage.windows(aws_cdk_lib_1.aws_ecs.WindowsOptimizedVersion.SERVER_2019);
ssmPath = '/aws/service/ami-windows-latest/Windows_Server-2019-English-Full-ECS_Optimized/image_id';
found = true;
}
if (!found) {
throw new Error(`Unable to find AMI for ECS instances for ${this.image.os.name}/${this.image.architecture.name}`);
}
const image = {
getImage(scope) {
const baseImageRes = baseImage.getImage(scope);
return {
imageId: `resolve:ssm:${ssmPath}`,
userData: baseImageRes.userData,
osType: baseImageRes.osType,
};
},
};
return image;
}
pullCommand() {
if (this.image.os.is(common_1.Os.WINDOWS)) {
return `Start-Job -ScriptBlock { docker pull ${this.image.imageRepository.repositoryUri}:${this.image.imageTag} }`;
}
return `docker pull ${this.image.imageRepository.repositoryUri}:${this.image.imageTag} &`;
}
loginCommands() {
const thisStack = aws_cdk_lib_1.Stack.of(this);
if (this.image.os.is(common_1.Os.WINDOWS)) {
return [`(Get-ECRLoginCommand).Password | docker login --username AWS --password-stdin ${thisStack.account}.dkr.ecr.${thisStack.region}.amazonaws.com`];
}
return [
'yum install -y awscli || dnf install -y awscli',
`aws ecr get-login-password --region ${thisStack.region} | docker login --username AWS --password-stdin ${thisStack.account}.dkr.ecr.${thisStack.region}.amazonaws.com`,
];
}
ecsSettingsCommands() {
// don't let ECS accumulate too many stopped tasks that can end up very big in our case
// the default is 10m duration with 1h jitter which can end up with 1h10m delay for cleaning up stopped tasks
if (this.image.os.is(common_1.Os.WINDOWS)) {
return [
'[Environment]::SetEnvironmentVariable("ECS_ENGINE_TASK_CLEANUP_WAIT_DURATION", "5s", "Machine")',
'[Environment]::SetEnvironmentVariable("ECS_ENGINE_TASK_CLEANUP_WAIT_DURATION_JITTER", "5s", "Machine")',
];
}
return [
'echo ECS_ENGINE_TASK_CLEANUP_WAIT_DURATION=5s >> /etc/ecs/ecs.config',
'echo ECS_ENGINE_TASK_CLEANUP_WAIT_DURATION_JITTER=5s >> /etc/ecs/ecs.config',
];
}
/**
* Generate step function task(s) to start a new runner.
*
* Called by GithubRunners and shouldn't be called manually.
*
* @param parameters workflow job details
*/
getStepFunctionTask(parameters) {
return new aws_cdk_lib_1.aws_stepfunctions_tasks.EcsRunTask(this, this.labels.join(', '), {
integrationPattern: aws_stepfunctions_1.IntegrationPattern.RUN_JOB, // sync
taskDefinition: this.task,
cluster: this.cluster,
launchTarget: new EcsEc2LaunchTarget({
capacityProvider: this.capacityProvider.capacityProviderName,
}),
enableExecuteCommand: this.image.os.isIn(common_1.Os._ALL_LINUX_VERSIONS),
assignPublicIp: this.assignPublicIp,
containerOverrides: [
{
containerDefinition: this.container,
environment: [
{
name: 'RUNNER_TOKEN',
value: parameters.runnerTokenPath,
},
{
name: 'RUNNER_NAME',
value: parameters.runnerNamePath,
},
{
name: 'RUNNER_LABEL',
value: this.labels.join(','),
},
{
name: 'RUNNER_GROUP',
value: this.group ? `--runnergroup ${this.group}` : '',
},
{
name: 'GITHUB_DOMAIN',
value: parameters.githubDomainPath,
},
{
name: 'OWNER',
value: parameters.ownerPath,
},
{
name: 'REPO',
value: parameters.repoPath,
},
{
name: 'REGISTRATION_URL',
value: parameters.registrationUrl,
},
],
},
],
});
}
grantStateMachine(_) {
}
status(statusFunctionRole) {
this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');
return {
type: this.constructor.name,
labels: this.labels,
vpcArn: this.vpc?.vpcArn,
securityGroups: this.securityGroups.map(sg => sg.securityGroupId),
roleArn: this.task.taskRole.roleArn,
logGroup: this.logGroup.logGroupName,
image: {
imageRepository: this.image.imageRepository.repositoryUri,
imageTag: this.image.imageTag,
imageBuilderLogGroup: this.image.logGroup?.logGroupName,
},
};
}
}
exports.EcsRunnerProvider = EcsRunnerProvider;
_a = JSII_RTTI_SYMBOL_1;
EcsRunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.EcsRunnerProvider", version: "0.14.11" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs.js","sourceRoot":"","sources":["../../src/providers/ecs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_ecs as ecs,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport * as autoscaling from 'aws-cdk-lib/aws-autoscaling';\nimport { MachineImageType } from 'aws-cdk-lib/aws-ecs';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  amiRootDevice,\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerImage,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n  StorageOptions,\n} from './common';\nimport { ecsRunCommand } from './fargate';\nimport { IRunnerImageBuilder, RunnerImageBuilder, RunnerImageBuilderProps, RunnerImageComponent } from '../image-builders';\nimport { MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT, MINIMAL_ECS_SSM_SESSION_MANAGER_POLICY_STATEMENT } from '../utils';\n\n/**\n * Properties for EcsRunnerProvider.\n */\nexport interface EcsRunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build Docker images containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default EcsRunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ecs']\n   */\n  readonly labels?: string[];\n\n  /**\n   * GitHub Actions runner group name.\n   *\n   * If specified, the runner will be registered with this group name. Setting a runner group can help managing access to self-hosted runners. It\n   * requires a paid GitHub account.\n   *\n   * The group must exist or the runner will not start.\n   *\n   * Users will still be able to trigger this runner with the correct labels. But the runner will only be able to run jobs from repos allowed to use the group.\n   *\n   * @default undefined\n   */\n  readonly group?: string;\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets to run the runners in.\n   *\n   * @default ECS default\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security groups to assign to the task.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Existing ECS cluster to use.\n   *\n   * @default a new cluster\n   */\n  readonly cluster?: ecs.Cluster;\n\n  /**\n   * Existing capacity provider to use.\n   *\n   * Make sure the AMI used by the capacity provider is compatible with ECS.\n   *\n   * @default new capacity provider\n   */\n  readonly capacityProvider?: ecs.AsgCapacityProvider;\n\n  /**\n   * Assign public IP to the runner task.\n   *\n   * Make sure the task will have access to GitHub. A public IP might be required unless you have NAT gateway.\n   *\n   * @default true\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The number of cpu units used by the task. 1024 units is 1 vCPU. Fractions of a vCPU are supported.\n   *\n   * @default 1024\n   */\n  readonly cpu?: number;\n\n  /**\n   * The amount (in MiB) of memory used by the task.\n   *\n   * @default 3500, unless `memoryReservationMiB` is used and then it's undefined\n   */\n  readonly memoryLimitMiB?: number;\n\n  /**\n   * The soft limit (in MiB) of memory to reserve for the container.\n   *\n   * @default undefined\n   */\n  readonly memoryReservationMiB?: number;\n\n  /**\n   * Instance type of ECS cluster instances. Only used when creating a new cluster.\n   *\n   * @default m6i.large or m6g.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * The minimum number of instances to run in the cluster. Only used when creating a new cluster.\n   *\n   * @default 0\n   */\n  readonly minInstances?: number;\n\n  /**\n   * The maximum number of instances to run in the cluster. Only used when creating a new cluster.\n   *\n   * @default 5\n   */\n  readonly maxInstances?: number;\n\n  /**\n   * Size of volume available for launched cluster instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * Each instance can be used by multiple runners, so make sure there is enough space for all of them.\n   *\n   * @default default size for AMI (usually 30GB for Linux and 50GB for Windows)\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Options for runner instance storage volume.\n   */\n  readonly storageOptions?: StorageOptions;\n\n  /**\n   * Support building and running Docker images by enabling Docker-in-Docker (dind) and the required CodeBuild privileged mode. Disabling this can\n   * speed up provisioning of CodeBuild runners. If you don't intend on running or building Docker images, disable this for faster start-up times.\n   *\n   * @default true\n   */\n  readonly dockerInDocker?: boolean;\n\n  /**\n   * Use spot capacity.\n   *\n   * @default false (true if spotMaxPrice is specified)\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Maximum price for spot instances.\n   */\n  readonly spotMaxPrice?: string;\n}\n\ninterface EcsEc2LaunchTargetProps {\n  readonly capacityProvider: string;\n}\n\nclass EcsEc2LaunchTarget implements stepfunctions_tasks.IEcsLaunchTarget {\n  constructor(readonly props: EcsEc2LaunchTargetProps) {\n  }\n\n  /**\n   * Called when the ECS launch type configured on RunTask\n   */\n  public bind(_task: stepfunctions_tasks.EcsRunTask,\n    _launchTargetOptions: stepfunctions_tasks.LaunchTargetBindOptions): stepfunctions_tasks.EcsLaunchTargetConfig {\n    return {\n      parameters: {\n        PropagateTags: ecs.PropagatedTagSource.TASK_DEFINITION,\n        CapacityProviderStrategy: [\n          {\n            CapacityProvider: this.props.capacityProvider,\n          },\n        ],\n      },\n    };\n  }\n}\n\n/**\n * GitHub Actions runner provider using ECS on EC2 to execute jobs.\n *\n * ECS can be useful when you want more control of the infrastructure running the GitHub Actions Docker containers. You can control the autoscaling\n * group to scale down to zero during the night and scale up during work hours. This way you can still save money, but have to wait less for\n * infrastructure to spin up.\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 EcsRunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds ECS specific runner images.\n   *\n   * You can customize the OS, architecture, VPC, subnet, security groups, etc. by passing in props.\n   *\n   * You can add components to the image builder by calling `imageBuilder.addComponent()`.\n   *\n   * The default OS is Ubuntu running on x64 architecture.\n   *\n   * Included components:\n   *  * `RunnerImageComponent.requiredPackages()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Cluster hosting the task hosting the runner.\n   */\n  private readonly cluster: ecs.Cluster;\n\n  /**\n   * Capacity provider used to scale the cluster.\n   */\n  private readonly capacityProvider: ecs.AsgCapacityProvider;\n\n  /**\n   * ECS task hosting the runner.\n   */\n  private readonly task: ecs.Ec2TaskDefinition;\n\n  /**\n   * Container definition hosting the runner.\n   */\n  private readonly container: ecs.ContainerDefinition;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the runner task.\n   */\n  private readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets used for hosting the runner task.\n   */\n  private readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Whether runner task will have a public IP.\n   */\n  private readonly assignPublicIp: boolean;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * The network connections associated with this resource.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Docker image loaded with GitHub Actions Runner and its prerequisites. The image is built by an image builder and is specific to ECS tasks.\n   */\n  private readonly image: RunnerImage;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  /**\n   * Security groups associated with this provider.\n   */\n  private readonly securityGroups: ec2.ISecurityGroup[];\n\n  /**\n   * Run docker in docker.\n   */\n  private readonly dind: boolean;\n\n  /**\n   * Runner group name.\n   */\n  private readonly group?: string;\n\n  readonly retryableErrors = [\n    'Ecs.EcsException',\n    'ECS.AmazonECSException',\n    'Ecs.LimitExceededException',\n    'Ecs.UpdateInProgressException',\n  ];\n\n  constructor(scope: Construct, id: string, props?: EcsRunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ecs'];\n    this.group = props?.group;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });\n    this.subnetSelection = props?.subnetSelection;\n    this.securityGroups = props?.securityGroups ?? [new ec2.SecurityGroup(this, 'security group', { vpc: this.vpc })];\n    this.connections = new ec2.Connections({ securityGroups: this.securityGroups });\n    this.assignPublicIp = props?.assignPublicIp ?? true;\n    this.cluster = props?.cluster ? props.cluster : new ecs.Cluster(\n      this,\n      'cluster',\n      {\n        vpc: this.vpc,\n        enableFargateCapacityProviders: false,\n      },\n    );\n\n    if (props?.storageOptions && !props?.storageSize) {\n      throw new Error('storageSize is required when storageOptions are specified');\n    }\n\n    const imageBuilder = props?.imageBuilder ?? EcsRunnerProvider.imageBuilder(this, 'Image Builder');\n    const image = this.image = imageBuilder.bindDockerImage();\n\n    if (props?.capacityProvider) {\n      if (props?.minInstances || props?.maxInstances || props?.instanceType || props?.storageSize || props?.spot || props?.spotMaxPrice) {\n        cdk.Annotations.of(this).addWarning('When using a custom capacity provider, minInstances, maxInstances, instanceType, storageSize, spot, and spotMaxPrice will be ignored.');\n      }\n\n      this.capacityProvider = props.capacityProvider;\n    } else {\n      const spot = props?.spot ?? props?.spotMaxPrice !== undefined;\n\n      const launchTemplate = new ec2.LaunchTemplate(this, 'Launch Template', {\n        machineImage: this.defaultClusterInstanceAmi(),\n        instanceType: props?.instanceType ?? this.defaultClusterInstanceType(),\n        blockDevices: props?.storageSize ? [\n          {\n            deviceName: amiRootDevice(this, this.defaultClusterInstanceAmi().getImage(this).imageId).ref,\n            volume: {\n              ebsDevice: {\n                deleteOnTermination: true,\n                volumeSize: props.storageSize.toGibibytes(),\n                volumeType: props.storageOptions?.volumeType,\n                iops: props.storageOptions?.iops,\n                throughput: props.storageOptions?.throughput,\n              },\n            },\n          },\n        ] : undefined,\n        spotOptions: spot ? {\n          requestType: ec2.SpotRequestType.ONE_TIME,\n          maxPrice: props?.spotMaxPrice ? parseFloat(props?.spotMaxPrice) : undefined,\n        } : undefined,\n        requireImdsv2: true,\n        securityGroup: this.securityGroups[0],\n        role: new iam.Role(this, 'Launch Template Role', {\n          assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n        }),\n        userData: ec2.UserData.forOperatingSystem(image.os.is(Os.WINDOWS) ? ec2.OperatingSystemType.WINDOWS : ec2.OperatingSystemType.LINUX),\n      });\n      this.securityGroups.slice(1).map(sg => launchTemplate.connections.addSecurityGroup(sg));\n\n      const autoScalingGroup = new autoscaling.AutoScalingGroup(this, 'Auto Scaling Group', {\n        vpc: this.vpc,\n        launchTemplate,\n        vpcSubnets: this.subnetSelection,\n        minCapacity: props?.minInstances ?? 0,\n        maxCapacity: props?.maxInstances ?? 5,\n      });\n\n      this.capacityProvider = props?.capacityProvider ?? new ecs.AsgCapacityProvider(this, 'Capacity Provider', {\n        autoScalingGroup,\n        spotInstanceDraining: false, // waste of money to restart jobs as the restarted job 