@cloudsnorkel/cdk-github-runners 0.14.16 → 0.14.17

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.
@@ -486,6 +486,14 @@ export declare abstract class BaseProvider extends Construct {
486
486
  */
487
487
  export declare function amiRootDevice(scope: Construct, ami?: string): cdk.CustomResource;
488
488
  /**
489
+ * Creates a shortened state name from a construct's path for use in AWS Step Functions.
490
+ * Step Functions state names are limited to 80 characters. This function generates a name
491
+ * from the construct's path (without the stack name), optionally appends a suffix, and
492
+ * shortens it if necessary by truncating and appending a hash suffix to ensure uniqueness.
493
+ *
494
+ * @param construct The construct to get the path from
495
+ * @param suffix Optional suffix to append to the path (e.g., "data", "rand", "choice")
496
+ * @returns A shortened state name that fits within AWS Step Functions' 80-character limit
489
497
  * @internal
490
498
  */
491
- export declare function nodePathWithoutStack(construct: Construct): string;
499
+ export declare function generateStateName(construct: Construct, suffix?: string): string;
@@ -3,8 +3,9 @@ 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
  exports.amiRootDevice = amiRootDevice;
6
- exports.nodePathWithoutStack = nodePathWithoutStack;
6
+ exports.generateStateName = generateStateName;
7
7
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
8
+ const crypto = require("crypto");
8
9
  const cdk = require("aws-cdk-lib");
9
10
  const aws_cdk_lib_1 = require("aws-cdk-lib");
10
11
  const constructs_1 = require("constructs");
@@ -44,7 +45,7 @@ class RunnerVersion {
44
45
  }
45
46
  exports.RunnerVersion = RunnerVersion;
46
47
  _a = JSII_RTTI_SYMBOL_1;
47
- RunnerVersion[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerVersion", version: "0.14.16" };
48
+ RunnerVersion[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerVersion", version: "0.14.17" };
48
49
  /**
49
50
  * CPU architecture enum for an image.
50
51
  */
@@ -93,7 +94,7 @@ class Architecture {
93
94
  }
94
95
  exports.Architecture = Architecture;
95
96
  _b = JSII_RTTI_SYMBOL_1;
96
- Architecture[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Architecture", version: "0.14.16" };
97
+ Architecture[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Architecture", version: "0.14.17" };
97
98
  /**
98
99
  * ARM64
99
100
  */
@@ -136,7 +137,7 @@ class Os {
136
137
  }
137
138
  exports.Os = Os;
138
139
  _c = JSII_RTTI_SYMBOL_1;
139
- Os[_c] = { fqn: "@cloudsnorkel/cdk-github-runners.Os", version: "0.14.16" };
140
+ Os[_c] = { fqn: "@cloudsnorkel/cdk-github-runners.Os", version: "0.14.17" };
140
141
  /**
141
142
  * Linux
142
143
  *
@@ -237,9 +238,30 @@ function amiRootDevice(scope, ami) {
237
238
  });
238
239
  }
239
240
  /**
241
+ * Creates a shortened state name from a construct's path for use in AWS Step Functions.
242
+ * Step Functions state names are limited to 80 characters. This function generates a name
243
+ * from the construct's path (without the stack name), optionally appends a suffix, and
244
+ * shortens it if necessary by truncating and appending a hash suffix to ensure uniqueness.
245
+ *
246
+ * @param construct The construct to get the path from
247
+ * @param suffix Optional suffix to append to the path (e.g., "data", "rand", "choice")
248
+ * @returns A shortened state name that fits within AWS Step Functions' 80-character limit
240
249
  * @internal
241
250
  */
242
- function nodePathWithoutStack(construct) {
243
- return construct.node.path.split('/').slice(1).join('/');
251
+ function generateStateName(construct, suffix) {
252
+ // Get construct path without stack name
253
+ const basePath = construct.node.path.split('/').slice(1).join('/');
254
+ // Build full name with optional suffix
255
+ const fullName = suffix ? `${basePath} ${suffix}` : basePath;
256
+ // Shorten if necessary
257
+ const maxLength = 80;
258
+ if (fullName.length <= maxLength) {
259
+ return fullName;
260
+ }
261
+ const hashSuffix = crypto.createHash('md5').update(fullName).digest('hex').slice(0, 3);
262
+ const separator = '-';
263
+ const truncatedLength = maxLength - hashSuffix.length - separator.length;
264
+ const truncated = fullName.slice(0, truncatedLength);
265
+ return `${truncated}${separator}${hashSuffix}`;
244
266
  }
245
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/providers/common.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  CustomResource,\n  Duration,\n} from 'aws-cdk-lib';\nimport { EbsDeviceVolumeType } from 'aws-cdk-lib/aws-ec2';\nimport { Construct, IConstruct } from 'constructs';\nimport { AmiRootDeviceFunction } from './ami-root-device-function';\nimport { singletonLambda, singletonLogGroup, SingletonLogType } from '../utils';\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  * @deprecated use {@link LINUX_UBUNTU}, {@link LINUX_UBUNTU_2404}, {@link LINUX_AMAZON_2} or {@link LINUX_AMAZON_2023}\n  */\n  public static readonly LINUX = Os.of('Linux');\n\n  /**\n   * Ubuntu Linux\n   */\n  public static readonly LINUX_UBUNTU = Os.of('Ubuntu Linux');\n\n  /**\n  * Ubuntu Linux 22.04\n  */\n  public static readonly LINUX_UBUNTU_2204 = Os.of('Ubuntu Linux 22.04');\n\n  /**\n   * Ubuntu Linux 24.04\n   */\n  public static readonly LINUX_UBUNTU_2404 = Os.of('Ubuntu Linux 24.04');\n\n  /**\n   * Amazon Linux 2\n   */\n  public static readonly LINUX_AMAZON_2 = Os.of('Amazon Linux 2');\n\n  /**\n   * Amazon Linux 2023\n   */\n  public static readonly LINUX_AMAZON_2023 = Os.of('Amazon Linux 2023');\n\n  /**\n   * @internal\n   */\n  public static readonly _ALL_LINUX_VERSIONS =\n    [Os.LINUX, Os.LINUX_UBUNTU, Os.LINUX_UBUNTU_2204, Os.LINUX_UBUNTU_2404, Os.LINUX_AMAZON_2, Os.LINUX_AMAZON_2023];\n\n  /**\n     * @internal\n     */\n  public static readonly _ALL_LINUX_AMAZON_VERSIONS = [Os.LINUX_AMAZON_2, Os.LINUX_AMAZON_2023];\n\n  /**\n     * @internal\n     */\n  public static readonly _ALL_LINUX_UBUNTU_VERSIONS = [Os.LINUX_UBUNTU, Os.LINUX_UBUNTU_2204, Os.LINUX_UBUNTU_2404];\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 RunnerImageBuilder}.\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   * @deprecated open a ticket if you need this\n   */\n  readonly runnerVersion: RunnerVersion;\n\n  /**\n   * A dependable string that can be waited on to ensure the image is ready.\n   *\n   * @internal\n   */\n  readonly _dependable?: string;\n}\n\n/**\n * Description of a AMI built by {@link RunnerImageBuilder}.\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   * @deprecated open a ticket if you need this\n   */\n  readonly runnerVersion: RunnerVersion;\n}\n\n/**\n * Retry options for providers. The default is to retry 23 times for about 24 hours with increasing interval.\n */\nexport interface ProviderRetryOptions {\n  /**\n   * Set to true to retry provider on supported failures. Which failures generate a retry depends on the specific provider.\n   *\n   * @default true\n   */\n  readonly retry?: boolean;\n\n  /**\n   * How much time to wait after first retryable failure. This interval will be multiplied by {@link backoffRate} each retry.\n   *\n   * @default 1 minute\n   */\n  readonly interval?: Duration;\n\n  /**\n   * How many times to retry.\n   *\n   * @default 23\n   */\n  readonly maxAttempts?: number;\n\n  /**\n   * Multiplication for how much longer the wait interval gets on every retry.\n   *\n   * @default 1.3\n   */\n  readonly backoffRate?: number;\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   * @deprecated use {@link retryOptions} on {@link GitHubRunners} instead\n   */\n  readonly retryOptions?: ProviderRetryOptions;\n\n  /**\n   * Add default labels based on OS and architecture of the runner. This will tell GitHub Runner to add default labels like `self-hosted`, `linux`, `x64`, and `arm64`.\n   *\n   * @default true\n   */\n  readonly defaultLabels?: boolean;\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 \"{REGISTRATION_URL}\" --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 repository owner name.\n   */\n  readonly ownerPath: string;\n\n  /**\n   * Path to repository name.\n   */\n  readonly repoPath: string;\n\n  /**\n   * Repository or organization URL to register runner at.\n   */\n  readonly registrationUrl: string;\n\n  /**\n   * Path to comma-separated labels string to use for runner.\n   */\n  readonly labelsPath: string;\n\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   * CDK construct node path for this provider.\n   */\n  readonly constructPath?: string;\n\n  /**\n   * VPC where runners will be launched.\n   */\n  readonly vpcArn?: string;\n\n  /**\n   * Security groups attached to runners.\n   */\n  readonly securityGroups?: 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   * Log group for runners.\n   */\n  readonly logGroup?: 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, IConstruct {\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   * 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   * List of step functions errors that should be retried.\n   *\n   * @deprecated do not use\n   */\n  readonly retryableErrors: string[];\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 * Interface for composite runner providers that interact with multiple sub-providers.\n * Unlike IRunnerProvider, composite providers do not have connections, grant capabilities,\n * log groups, or retryable errors as they delegate to their sub-providers.\n */\nexport interface ICompositeProvider extends IConstruct {\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   * All sub-providers contained in this composite provider.\n   * This is used to extract providers for metric filters and other operations.\n   */\n  readonly providers: IRunnerProvider[];\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 statuses of all sub-providers to be used in the main status function. Also gives the status function any needed permissions to query the Docker images or AMIs.\n   *\n   * @param statusFunctionRole grantable for the status function\n   */\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[];\n}\n\n/**\n * Storage options for the runner instance.\n */\nexport interface StorageOptions {\n  /**\n   * The EBS volume type\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/EBSVolumeTypes.html\n   *\n   * @default `EbsDeviceVolumeType.GP2`\n   */\n  readonly volumeType?: EbsDeviceVolumeType;\n\n  /**\n   * The number of I/O operations per second (IOPS) to provision for the volume.\n   *\n   * Must only be set for `volumeType`: `EbsDeviceVolumeType.IO1`\n   *\n   * The maximum ratio of IOPS to volume size (in GiB) is 50:1, so for 5,000 provisioned IOPS,\n   * you need at least 100 GiB storage on the volume.\n   *\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/EBSVolumeTypes.html\n   *\n   * @default - none, required for `EbsDeviceVolumeType.IO1`\n   */\n  readonly iops?: number;\n\n  /**\n   * The throughput that the volume supports, in MiB/s\n   * Takes a minimum of 125 and maximum of 1000.\n   * @see https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-ec2-volume.html#cfn-ec2-volume-throughput\n   * @default - 125 MiB/s. Only valid on gp3 volumes.\n   */\n  readonly throughput?: number;\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 constructor(scope: Construct, id: string, _props?: RunnerProviderProps) {\n    super(scope, id);\n\n    cdk.Tags.of(this).add('GitHubRunners:Provider', this.node.path);\n  }\n\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\n/**\n * Use custom resource to determine the root device name of a given AMI, Launch Template, or SSM parameter pointing to AMI.\n *\n * TODO move somewhere more common as it's used by both providers and AMI builder now\n *\n * @internal\n */\nexport function amiRootDevice(scope: Construct, ami?: string) {\n  const crHandler = singletonLambda(AmiRootDeviceFunction, scope, 'AMI Root Device Reader', {\n    description: 'Custom resource handler that discovers the boot drive device name for a given AMI',\n    timeout: cdk.Duration.minutes(1),\n    logGroup: singletonLogGroup(scope, SingletonLogType.RUNNER_IMAGE_BUILD),\n    loggingFormat: lambda.LoggingFormat.JSON,\n    initialPolicy: [\n      new iam.PolicyStatement({\n        actions: [\n          'ssm:GetParameter',\n          'ec2:DescribeImages',\n          'ec2:DescribeLaunchTemplateVersions',\n          'imagebuilder:GetImage',\n        ],\n        resources: ['*'],\n      }),\n    ],\n  });\n\n  return new CustomResource(scope, 'AMI Root Device', {\n    serviceToken: crHandler.functionArn,\n    resourceType: 'Custom::AmiRootDevice',\n    properties: {\n      Ami: ami ?? '',\n    },\n  });\n}\n\n/**\n * @internal\n */\nexport function nodePathWithoutStack(construct: Construct) {\n  return construct.node.path.split('/').slice(1).join('/');\n}\n"]}
267
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/providers/common.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  CustomResource,\n  Duration,\n} from 'aws-cdk-lib';\nimport { EbsDeviceVolumeType } from 'aws-cdk-lib/aws-ec2';\nimport { Construct, IConstruct } from 'constructs';\nimport { AmiRootDeviceFunction } from './ami-root-device-function';\nimport { singletonLambda, singletonLogGroup, SingletonLogType } from '../utils';\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  * @deprecated use {@link LINUX_UBUNTU}, {@link LINUX_UBUNTU_2404}, {@link LINUX_AMAZON_2} or {@link LINUX_AMAZON_2023}\n  */\n  public static readonly LINUX = Os.of('Linux');\n\n  /**\n   * Ubuntu Linux\n   */\n  public static readonly LINUX_UBUNTU = Os.of('Ubuntu Linux');\n\n  /**\n  * Ubuntu Linux 22.04\n  */\n  public static readonly LINUX_UBUNTU_2204 = Os.of('Ubuntu Linux 22.04');\n\n  /**\n   * Ubuntu Linux 24.04\n   */\n  public static readonly LINUX_UBUNTU_2404 = Os.of('Ubuntu Linux 24.04');\n\n  /**\n   * Amazon Linux 2\n   */\n  public static readonly LINUX_AMAZON_2 = Os.of('Amazon Linux 2');\n\n  /**\n   * Amazon Linux 2023\n   */\n  public static readonly LINUX_AMAZON_2023 = Os.of('Amazon Linux 2023');\n\n  /**\n   * @internal\n   */\n  public static readonly _ALL_LINUX_VERSIONS =\n    [Os.LINUX, Os.LINUX_UBUNTU, Os.LINUX_UBUNTU_2204, Os.LINUX_UBUNTU_2404, Os.LINUX_AMAZON_2, Os.LINUX_AMAZON_2023];\n\n  /**\n     * @internal\n     */\n  public static readonly _ALL_LINUX_AMAZON_VERSIONS = [Os.LINUX_AMAZON_2, Os.LINUX_AMAZON_2023];\n\n  /**\n     * @internal\n     */\n  public static readonly _ALL_LINUX_UBUNTU_VERSIONS = [Os.LINUX_UBUNTU, Os.LINUX_UBUNTU_2204, Os.LINUX_UBUNTU_2404];\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 RunnerImageBuilder}.\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   * @deprecated open a ticket if you need this\n   */\n  readonly runnerVersion: RunnerVersion;\n\n  /**\n   * A dependable string that can be waited on to ensure the image is ready.\n   *\n   * @internal\n   */\n  readonly _dependable?: string;\n}\n\n/**\n * Description of a AMI built by {@link RunnerImageBuilder}.\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   * @deprecated open a ticket if you need this\n   */\n  readonly runnerVersion: RunnerVersion;\n}\n\n/**\n * Retry options for providers. The default is to retry 23 times for about 24 hours with increasing interval.\n */\nexport interface ProviderRetryOptions {\n  /**\n   * Set to true to retry provider on supported failures. Which failures generate a retry depends on the specific provider.\n   *\n   * @default true\n   */\n  readonly retry?: boolean;\n\n  /**\n   * How much time to wait after first retryable failure. This interval will be multiplied by {@link backoffRate} each retry.\n   *\n   * @default 1 minute\n   */\n  readonly interval?: Duration;\n\n  /**\n   * How many times to retry.\n   *\n   * @default 23\n   */\n  readonly maxAttempts?: number;\n\n  /**\n   * Multiplication for how much longer the wait interval gets on every retry.\n   *\n   * @default 1.3\n   */\n  readonly backoffRate?: number;\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   * @deprecated use {@link retryOptions} on {@link GitHubRunners} instead\n   */\n  readonly retryOptions?: ProviderRetryOptions;\n\n  /**\n   * Add default labels based on OS and architecture of the runner. This will tell GitHub Runner to add default labels like `self-hosted`, `linux`, `x64`, and `arm64`.\n   *\n   * @default true\n   */\n  readonly defaultLabels?: boolean;\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 \"{REGISTRATION_URL}\" --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 repository owner name.\n   */\n  readonly ownerPath: string;\n\n  /**\n   * Path to repository name.\n   */\n  readonly repoPath: string;\n\n  /**\n   * Repository or organization URL to register runner at.\n   */\n  readonly registrationUrl: string;\n\n  /**\n   * Path to comma-separated labels string to use for runner.\n   */\n  readonly labelsPath: string;\n\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   * CDK construct node path for this provider.\n   */\n  readonly constructPath?: string;\n\n  /**\n   * VPC where runners will be launched.\n   */\n  readonly vpcArn?: string;\n\n  /**\n   * Security groups attached to runners.\n   */\n  readonly securityGroups?: 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   * Log group for runners.\n   */\n  readonly logGroup?: 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, IConstruct {\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   * 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   * List of step functions errors that should be retried.\n   *\n   * @deprecated do not use\n   */\n  readonly retryableErrors: string[];\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 * Interface for composite runner providers that interact with multiple sub-providers.\n * Unlike IRunnerProvider, composite providers do not have connections, grant capabilities,\n * log groups, or retryable errors as they delegate to their sub-providers.\n */\nexport interface ICompositeProvider extends IConstruct {\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   * All sub-providers contained in this composite provider.\n   * This is used to extract providers for metric filters and other operations.\n   */\n  readonly providers: IRunnerProvider[];\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 statuses of all sub-providers to be used in the main status function. Also gives the status function any needed permissions to query the Docker images or AMIs.\n   *\n   * @param statusFunctionRole grantable for the status function\n   */\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[];\n}\n\n/**\n * Storage options for the runner instance.\n */\nexport interface StorageOptions {\n  /**\n   * The EBS volume type\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/EBSVolumeTypes.html\n   *\n   * @default `EbsDeviceVolumeType.GP2`\n   */\n  readonly volumeType?: EbsDeviceVolumeType;\n\n  /**\n   * The number of I/O operations per second (IOPS) to provision for the volume.\n   *\n   * Must only be set for `volumeType`: `EbsDeviceVolumeType.IO1`\n   *\n   * The maximum ratio of IOPS to volume size (in GiB) is 50:1, so for 5,000 provisioned IOPS,\n   * you need at least 100 GiB storage on the volume.\n   *\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/EBSVolumeTypes.html\n   *\n   * @default - none, required for `EbsDeviceVolumeType.IO1`\n   */\n  readonly iops?: number;\n\n  /**\n   * The throughput that the volume supports, in MiB/s\n   * Takes a minimum of 125 and maximum of 1000.\n   * @see https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-ec2-volume.html#cfn-ec2-volume-throughput\n   * @default - 125 MiB/s. Only valid on gp3 volumes.\n   */\n  readonly throughput?: number;\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 constructor(scope: Construct, id: string, _props?: RunnerProviderProps) {\n    super(scope, id);\n\n    cdk.Tags.of(this).add('GitHubRunners:Provider', this.node.path);\n  }\n\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\n/**\n * Use custom resource to determine the root device name of a given AMI, Launch Template, or SSM parameter pointing to AMI.\n *\n * TODO move somewhere more common as it's used by both providers and AMI builder now\n *\n * @internal\n */\nexport function amiRootDevice(scope: Construct, ami?: string) {\n  const crHandler = singletonLambda(AmiRootDeviceFunction, scope, 'AMI Root Device Reader', {\n    description: 'Custom resource handler that discovers the boot drive device name for a given AMI',\n    timeout: cdk.Duration.minutes(1),\n    logGroup: singletonLogGroup(scope, SingletonLogType.RUNNER_IMAGE_BUILD),\n    loggingFormat: lambda.LoggingFormat.JSON,\n    initialPolicy: [\n      new iam.PolicyStatement({\n        actions: [\n          'ssm:GetParameter',\n          'ec2:DescribeImages',\n          'ec2:DescribeLaunchTemplateVersions',\n          'imagebuilder:GetImage',\n        ],\n        resources: ['*'],\n      }),\n    ],\n  });\n\n  return new CustomResource(scope, 'AMI Root Device', {\n    serviceToken: crHandler.functionArn,\n    resourceType: 'Custom::AmiRootDevice',\n    properties: {\n      Ami: ami ?? '',\n    },\n  });\n}\n\n/**\n * Creates a shortened state name from a construct's path for use in AWS Step Functions.\n * Step Functions state names are limited to 80 characters. This function generates a name\n * from the construct's path (without the stack name), optionally appends a suffix, and\n * shortens it if necessary by truncating and appending a hash suffix to ensure uniqueness.\n *\n * @param construct The construct to get the path from\n * @param suffix Optional suffix to append to the path (e.g., \"data\", \"rand\", \"choice\")\n * @returns A shortened state name that fits within AWS Step Functions' 80-character limit\n * @internal\n */\nexport function generateStateName(construct: Construct, suffix?: string): string {\n  // Get construct path without stack name\n  const basePath = construct.node.path.split('/').slice(1).join('/');\n\n  // Build full name with optional suffix\n  const fullName = suffix ? `${basePath} ${suffix}` : basePath;\n\n  // Shorten if necessary\n  const maxLength = 80;\n  if (fullName.length <= maxLength) {\n    return fullName;\n  }\n\n  const hashSuffix = crypto.createHash('md5').update(fullName).digest('hex').slice(0, 3);\n  const separator = '-';\n  const truncatedLength = maxLength - hashSuffix.length - separator.length;\n  const truncated = fullName.slice(0, truncatedLength);\n  return `${truncated}${separator}${hashSuffix}`;\n}\n"]}
@@ -80,7 +80,7 @@ class CompositeProvider {
80
80
  }
81
81
  exports.CompositeProvider = CompositeProvider;
82
82
  _a = JSII_RTTI_SYMBOL_1;
83
- CompositeProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CompositeProvider", version: "0.14.16" };
83
+ CompositeProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CompositeProvider", version: "0.14.17" };
84
84
  /**
85
85
  * Internal implementation of fallback runner provider.
86
86
  *
@@ -135,7 +135,7 @@ class FallbackRunnerProvider extends constructs_1.Construct {
135
135
  // - The provider is not a State instance
136
136
  // - The provider has multiple end states
137
137
  // - The end state doesn't support addCatch directly
138
- const parallel = new aws_cdk_lib_1.aws_stepfunctions.Parallel(this, `${(0, common_1.nodePathWithoutStack)(this)} attempt #${i + 1}`);
138
+ const parallel = new aws_cdk_lib_1.aws_stepfunctions.Parallel(this, (0, common_1.generateStateName)(this, `attempt #${i + 1}`));
139
139
  parallel.branch(currentProvider);
140
140
  parallel.addCatch(nextProvider, {
141
141
  errors: ['States.ALL'],
@@ -200,13 +200,13 @@ class DistributedRunnerProvider extends constructs_1.Construct {
200
200
  */
201
201
  getStepFunctionTask(parameters) {
202
202
  const totalWeight = this.weightedProviders.reduce((sum, wp) => sum + wp.weight, 0);
203
- const rand = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, `${(0, common_1.nodePathWithoutStack)(this)} rand`, {
203
+ const rand = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, (0, common_1.generateStateName)(this, 'rand'), {
204
204
  parameters: {
205
205
  rand: aws_cdk_lib_1.aws_stepfunctions.JsonPath.mathRandom(1, totalWeight + 1),
206
206
  },
207
207
  resultPath: '$.composite',
208
208
  });
209
- const choice = new aws_cdk_lib_1.aws_stepfunctions.Choice(this, `${(0, common_1.nodePathWithoutStack)(this)} choice`);
209
+ const choice = new aws_cdk_lib_1.aws_stepfunctions.Choice(this, (0, common_1.generateStateName)(this, 'choice'));
210
210
  rand.next(choice);
211
211
  // Find provider with the highest weight
212
212
  let rollingWeight = 0;
@@ -226,4 +226,4 @@ class DistributedRunnerProvider extends constructs_1.Construct {
226
226
  return this.providers.map(provider => provider.status(statusFunctionRole));
227
227
  }
228
228
  }
229
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"composite.js","sourceRoot":"","sources":["../../src/providers/composite.ts"],"names":[],"mappings":"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","sourcesContent":["import { aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { ICompositeProvider, IRunnerProvider, IRunnerProviderStatus, nodePathWithoutStack, RunnerRuntimeParameters } from './common';\n\n/**\n * Configuration for weighted distribution of runners.\n */\nexport interface WeightedRunnerProvider {\n  /**\n   * The runner provider to use.\n   */\n  readonly provider: IRunnerProvider;\n\n  /**\n   * Weight for this provider. Higher weights mean higher probability of selection.\n   * Must be a positive number.\n   */\n  readonly weight: number;\n}\n\n/**\n * A composite runner provider that implements fallback and distribution strategies.\n */\nexport class CompositeProvider {\n  /**\n   * Creates a fallback runner provider that tries each provider in order until one succeeds.\n   *\n   * For example, given providers A, B, C:\n   * - Try A first\n   * - If A fails, try B\n   * - If B fails, try C\n   *\n   * You can use this to try spot instance first, and switch to on-demand instances if spot is unavailable.\n   *\n   * Or you can use this to try different instance types in order of preference.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param providers List of runner providers to try in order\n   */\n  public static fallback(scope: Construct, id: string, providers: IRunnerProvider[]): ICompositeProvider {\n    if (providers.length < 2) {\n      throw new Error('At least two providers must be specified for fallback');\n    }\n\n    this.validateLabels(providers);\n\n    return new FallbackRunnerProvider(scope, id, providers);\n  }\n\n  /**\n   * Creates a weighted distribution runner provider that randomly selects a provider based on weights.\n   *\n   * For example, given providers A (weight 10), B (weight 20), C (weight 30):\n   * - Total weight = 60\n   * - Probability of selecting A = 10/60 = 16.67%\n   * - Probability of selecting B = 20/60 = 33.33%\n   * - Probability of selecting C = 30/60 = 50%\n   *\n   * You can use this to distribute load across multiple instance types or availability zones.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param weightedProviders List of weighted runner providers\n   */\n  public static distribute(scope: Construct, id: string, weightedProviders: WeightedRunnerProvider[]): ICompositeProvider {\n    if (weightedProviders.length < 2) {\n      throw new Error('At least two providers must be specified for distribution');\n    }\n\n    // Validate labels\n    this.validateLabels(weightedProviders.map(wp => wp.provider));\n\n    // Validate weights\n    for (const wp of weightedProviders) {\n      if (wp.weight <= 0) {\n        throw new Error('All weights must be positive numbers');\n      }\n    }\n\n    return new DistributedRunnerProvider(scope, id, weightedProviders);\n  }\n\n  /**\n   * Validates that all providers have the exact same labels.\n   * This is required so that any provisioned runner can match the labels requested by the GitHub workflow job.\n   *\n   * @param providers Providers to validate\n   */\n  private static validateLabels(providers: IRunnerProvider[]): void {\n    const firstLabels = new Set(providers[0].labels);\n    for (const provider of providers.slice(1)) {\n      const providerLabels = new Set(provider.labels);\n      if (firstLabels.size !== providerLabels.size || ![...firstLabels].every(label => providerLabels.has(label))) {\n        throw new Error(`All providers must have the exact same labels (${[...firstLabels].join(', ')} != ${[...providerLabels].join(', ')})`);\n      }\n    }\n  }\n}\n\n/**\n * Internal implementation of fallback runner provider.\n *\n * @internal\n */\nclass FallbackRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, providers: IRunnerProvider[]) {\n    super(scope, id);\n    this.labels = providers[0].labels;\n    this.providers = providers;\n  }\n\n  /**\n   * Builds a Step Functions state machine that implements a fallback strategy.\n   *\n   * This method constructs a chain where each provider catches errors and falls back\n   * to the next provider in sequence. We iterate forward through providers, attaching\n   * catch handlers to each one (except the last) that route to the next provider.\n   *\n   * Example with providers [A, B, C]:\n   * - Save firstProvider = A (this will be returned)\n   * - Iteration 1 (i=0, provider A): A catches errors → falls back to B\n   * - Iteration 2 (i=1, provider B): B catches errors → falls back to C\n   * - Result: A → (on error) → B → (on error) → C\n   *\n   * Some providers generate one state while others (like EC2) may generate more complex chains.\n   * We try to avoid creating a complicated state machine, but complex chains may require wrapping in Parallel.\n   *\n   * @param parameters Runtime parameters for the step function task\n   * @returns A Step Functions chainable that implements the fallback logic\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // Get all provider chainables upfront\n    const providerChainables = this.providers.map(p => p.getStepFunctionTask(parameters));\n\n    // Track the entry point - starts as first provider, but may be wrapped\n    let entryPoint = providerChainables[0];\n\n    // Attach catch handlers to each provider (except the last) to fall back to the next provider\n    for (let i = 0; i < this.providers.length - 1; i++) {\n      const currentProvider = providerChainables[i];\n      const nextProvider = providerChainables[i + 1];\n\n      // Try to attach catch handler directly to the provider's end state\n      // This is more efficient than wrapping in a Parallel state when possible\n      if (this.canAddCatchDirectly(currentProvider)) {\n        const endState = (currentProvider as stepfunctions.State).endStates[0] as any;\n        endState.addCatch(nextProvider, {\n          errors: ['States.ALL'],\n          resultPath: `$.fallbackError${i + 1}`,\n        });\n        continue;\n      }\n\n      // Fallback: wrap in Parallel state to add catch capability\n      // This is needed when:\n      // - The provider is not a State instance\n      // - The provider has multiple end states\n      // - The end state doesn't support addCatch directly\n      const parallel = new stepfunctions.Parallel(this, `${nodePathWithoutStack(this)} attempt #${i + 1}`);\n      parallel.branch(currentProvider);\n      parallel.addCatch(nextProvider, {\n        errors: ['States.ALL'],\n        resultPath: `$.fallbackError${i + 1}`,\n      });\n\n      // If this is the first provider, update the entry point to the wrapped version\n      if (i === 0) {\n        entryPoint = parallel;\n      }\n    }\n\n    return entryPoint;\n  }\n\n  /**\n   * Checks if we can add a catch handler directly to the provider's end state.\n   * This avoids wrapping in a Parallel state when possible.\n   */\n  private canAddCatchDirectly(provider: stepfunctions.IChainable): boolean {\n    if (!(provider instanceof stepfunctions.State)) {\n      return false;\n    }\n    const endStates = provider.endStates;\n    if (endStates.length !== 1 || !(endStates[0] instanceof stepfunctions.State)) {\n      return false;\n    }\n    // Use 'any' type assertion because not all State types have addCatch in their type definition,\n    // but Task states and other executable states do support it at runtime\n    const endState = endStates[0] as any;\n    return typeof endState.addCatch === 'function';\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const provider of this.providers) {\n      provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n\n/**\n * Internal implementation of distributed runner provider.\n *\n * @internal\n */\nclass DistributedRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, private readonly weightedProviders: WeightedRunnerProvider[]) {\n    super(scope, id);\n    this.labels = weightedProviders[0].provider.labels;\n    this.providers = weightedProviders.map(wp => wp.provider);\n  }\n\n  /**\n   * Weighted random selection algorithm:\n   * 1. Generate a random number in [1, totalWeight+1)\n   * 2. Build cumulative weight ranges for each provider (e.g., weights [10,20,30] -> ranges [1-10, 11-30, 31-60])\n   * 3. Use Step Functions Choice state to route to the provider whose range contains the random number\n   *    The first matching condition wins, so we check if rand <= cumulativeWeight for each provider in order\n   *\n   * Note: States.MathRandom returns a value in [start, end) where end is exclusive. We use [1, totalWeight+1)\n   * to ensure the random value can be up to totalWeight (inclusive), which allows the last provider to be selected\n   * when rand equals totalWeight.\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    const totalWeight = this.weightedProviders.reduce((sum, wp) => sum + wp.weight, 0);\n    const rand = new stepfunctions.Pass(this, `${nodePathWithoutStack(this)} rand`, {\n      parameters: {\n        rand: stepfunctions.JsonPath.mathRandom(1, totalWeight + 1),\n      },\n      resultPath: '$.composite',\n    });\n    const choice = new stepfunctions.Choice(this, `${nodePathWithoutStack(this)} choice`);\n    rand.next(choice);\n\n    // Find provider with the highest weight\n    let rollingWeight = 0;\n    for (const wp of this.weightedProviders) {\n      rollingWeight += wp.weight;\n      choice.when(\n        stepfunctions.Condition.numberLessThanEquals('$.composite.rand', rollingWeight),\n        wp.provider.getStepFunctionTask(parameters),\n      );\n    }\n\n    return rand;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const wp of this.weightedProviders) {\n      wp.provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"composite.js","sourceRoot":"","sources":["../../src/providers/composite.ts"],"names":[],"mappings":"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","sourcesContent":["import { aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { ICompositeProvider, IRunnerProvider, IRunnerProviderStatus, RunnerRuntimeParameters, generateStateName } from './common';\n\n/**\n * Configuration for weighted distribution of runners.\n */\nexport interface WeightedRunnerProvider {\n  /**\n   * The runner provider to use.\n   */\n  readonly provider: IRunnerProvider;\n\n  /**\n   * Weight for this provider. Higher weights mean higher probability of selection.\n   * Must be a positive number.\n   */\n  readonly weight: number;\n}\n\n/**\n * A composite runner provider that implements fallback and distribution strategies.\n */\nexport class CompositeProvider {\n  /**\n   * Creates a fallback runner provider that tries each provider in order until one succeeds.\n   *\n   * For example, given providers A, B, C:\n   * - Try A first\n   * - If A fails, try B\n   * - If B fails, try C\n   *\n   * You can use this to try spot instance first, and switch to on-demand instances if spot is unavailable.\n   *\n   * Or you can use this to try different instance types in order of preference.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param providers List of runner providers to try in order\n   */\n  public static fallback(scope: Construct, id: string, providers: IRunnerProvider[]): ICompositeProvider {\n    if (providers.length < 2) {\n      throw new Error('At least two providers must be specified for fallback');\n    }\n\n    this.validateLabels(providers);\n\n    return new FallbackRunnerProvider(scope, id, providers);\n  }\n\n  /**\n   * Creates a weighted distribution runner provider that randomly selects a provider based on weights.\n   *\n   * For example, given providers A (weight 10), B (weight 20), C (weight 30):\n   * - Total weight = 60\n   * - Probability of selecting A = 10/60 = 16.67%\n   * - Probability of selecting B = 20/60 = 33.33%\n   * - Probability of selecting C = 30/60 = 50%\n   *\n   * You can use this to distribute load across multiple instance types or availability zones.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param weightedProviders List of weighted runner providers\n   */\n  public static distribute(scope: Construct, id: string, weightedProviders: WeightedRunnerProvider[]): ICompositeProvider {\n    if (weightedProviders.length < 2) {\n      throw new Error('At least two providers must be specified for distribution');\n    }\n\n    // Validate labels\n    this.validateLabels(weightedProviders.map(wp => wp.provider));\n\n    // Validate weights\n    for (const wp of weightedProviders) {\n      if (wp.weight <= 0) {\n        throw new Error('All weights must be positive numbers');\n      }\n    }\n\n    return new DistributedRunnerProvider(scope, id, weightedProviders);\n  }\n\n  /**\n   * Validates that all providers have the exact same labels.\n   * This is required so that any provisioned runner can match the labels requested by the GitHub workflow job.\n   *\n   * @param providers Providers to validate\n   */\n  private static validateLabels(providers: IRunnerProvider[]): void {\n    const firstLabels = new Set(providers[0].labels);\n    for (const provider of providers.slice(1)) {\n      const providerLabels = new Set(provider.labels);\n      if (firstLabels.size !== providerLabels.size || ![...firstLabels].every(label => providerLabels.has(label))) {\n        throw new Error(`All providers must have the exact same labels (${[...firstLabels].join(', ')} != ${[...providerLabels].join(', ')})`);\n      }\n    }\n  }\n}\n\n/**\n * Internal implementation of fallback runner provider.\n *\n * @internal\n */\nclass FallbackRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, providers: IRunnerProvider[]) {\n    super(scope, id);\n    this.labels = providers[0].labels;\n    this.providers = providers;\n  }\n\n  /**\n   * Builds a Step Functions state machine that implements a fallback strategy.\n   *\n   * This method constructs a chain where each provider catches errors and falls back\n   * to the next provider in sequence. We iterate forward through providers, attaching\n   * catch handlers to each one (except the last) that route to the next provider.\n   *\n   * Example with providers [A, B, C]:\n   * - Save firstProvider = A (this will be returned)\n   * - Iteration 1 (i=0, provider A): A catches errors → falls back to B\n   * - Iteration 2 (i=1, provider B): B catches errors → falls back to C\n   * - Result: A → (on error) → B → (on error) → C\n   *\n   * Some providers generate one state while others (like EC2) may generate more complex chains.\n   * We try to avoid creating a complicated state machine, but complex chains may require wrapping in Parallel.\n   *\n   * @param parameters Runtime parameters for the step function task\n   * @returns A Step Functions chainable that implements the fallback logic\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // Get all provider chainables upfront\n    const providerChainables = this.providers.map(p => p.getStepFunctionTask(parameters));\n\n    // Track the entry point - starts as first provider, but may be wrapped\n    let entryPoint = providerChainables[0];\n\n    // Attach catch handlers to each provider (except the last) to fall back to the next provider\n    for (let i = 0; i < this.providers.length - 1; i++) {\n      const currentProvider = providerChainables[i];\n      const nextProvider = providerChainables[i + 1];\n\n      // Try to attach catch handler directly to the provider's end state\n      // This is more efficient than wrapping in a Parallel state when possible\n      if (this.canAddCatchDirectly(currentProvider)) {\n        const endState = (currentProvider as stepfunctions.State).endStates[0] as any;\n        endState.addCatch(nextProvider, {\n          errors: ['States.ALL'],\n          resultPath: `$.fallbackError${i + 1}`,\n        });\n        continue;\n      }\n\n      // Fallback: wrap in Parallel state to add catch capability\n      // This is needed when:\n      // - The provider is not a State instance\n      // - The provider has multiple end states\n      // - The end state doesn't support addCatch directly\n      const parallel = new stepfunctions.Parallel(this, generateStateName(this, `attempt #${i + 1}`));\n      parallel.branch(currentProvider);\n      parallel.addCatch(nextProvider, {\n        errors: ['States.ALL'],\n        resultPath: `$.fallbackError${i + 1}`,\n      });\n\n      // If this is the first provider, update the entry point to the wrapped version\n      if (i === 0) {\n        entryPoint = parallel;\n      }\n    }\n\n    return entryPoint;\n  }\n\n  /**\n   * Checks if we can add a catch handler directly to the provider's end state.\n   * This avoids wrapping in a Parallel state when possible.\n   */\n  private canAddCatchDirectly(provider: stepfunctions.IChainable): boolean {\n    if (!(provider instanceof stepfunctions.State)) {\n      return false;\n    }\n    const endStates = provider.endStates;\n    if (endStates.length !== 1 || !(endStates[0] instanceof stepfunctions.State)) {\n      return false;\n    }\n    // Use 'any' type assertion because not all State types have addCatch in their type definition,\n    // but Task states and other executable states do support it at runtime\n    const endState = endStates[0] as any;\n    return typeof endState.addCatch === 'function';\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const provider of this.providers) {\n      provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n\n/**\n * Internal implementation of distributed runner provider.\n *\n * @internal\n */\nclass DistributedRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, private readonly weightedProviders: WeightedRunnerProvider[]) {\n    super(scope, id);\n    this.labels = weightedProviders[0].provider.labels;\n    this.providers = weightedProviders.map(wp => wp.provider);\n  }\n\n  /**\n   * Weighted random selection algorithm:\n   * 1. Generate a random number in [1, totalWeight+1)\n   * 2. Build cumulative weight ranges for each provider (e.g., weights [10,20,30] -> ranges [1-10, 11-30, 31-60])\n   * 3. Use Step Functions Choice state to route to the provider whose range contains the random number\n   *    The first matching condition wins, so we check if rand <= cumulativeWeight for each provider in order\n   *\n   * Note: States.MathRandom returns a value in [start, end) where end is exclusive. We use [1, totalWeight+1)\n   * to ensure the random value can be up to totalWeight (inclusive), which allows the last provider to be selected\n   * when rand equals totalWeight.\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    const totalWeight = this.weightedProviders.reduce((sum, wp) => sum + wp.weight, 0);\n    const rand = new stepfunctions.Pass(this, generateStateName(this, 'rand'), {\n      parameters: {\n        rand: stepfunctions.JsonPath.mathRandom(1, totalWeight + 1),\n      },\n      resultPath: '$.composite',\n    });\n    const choice = new stepfunctions.Choice(this, generateStateName(this, 'choice'));\n    rand.next(choice);\n\n    // Find provider with the highest weight\n    let rollingWeight = 0;\n    for (const wp of this.weightedProviders) {\n      rollingWeight += wp.weight;\n      choice.when(\n        stepfunctions.Condition.numberLessThanEquals('$.composite.rand', rollingWeight),\n        wp.provider.getStepFunctionTask(parameters),\n      );\n    }\n\n    return rand;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const wp of this.weightedProviders) {\n      wp.provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n"]}
@@ -277,7 +277,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
277
277
  this.group ? this.group : '',
278
278
  this.defaultLabels ? '' : '--no-default-labels',
279
279
  ];
280
- const passUserData = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, `${(0, common_1.nodePathWithoutStack)(this)} data`, {
280
+ const passUserData = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, (0, common_1.generateStateName)(this, 'data'), {
281
281
  parameters: {
282
282
  userdataTemplate: this.ami.os.is(common_1.Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,
283
283
  },
@@ -296,7 +296,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
296
296
  const rootDeviceResource = (0, common_1.amiRootDevice)(this, this.ami.launchTemplate.launchTemplateId);
297
297
  rootDeviceResource.node.addDependency(this.amiBuilder);
298
298
  const subnetRunners = this.subnets.map(subnet => {
299
- return new aws_cdk_lib_1.aws_stepfunctions_tasks.CallAwsService(this, `${(0, common_1.nodePathWithoutStack)(this)} ${subnet.subnetId}`, {
299
+ return new aws_cdk_lib_1.aws_stepfunctions_tasks.CallAwsService(this, (0, common_1.generateStateName)(this, subnet.subnetId), {
300
300
  comment: subnet.availabilityZone,
301
301
  integrationPattern: aws_stepfunctions_1.IntegrationPattern.WAIT_FOR_TASK_TOKEN,
302
302
  service: 'ec2',
@@ -421,7 +421,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
421
421
  }
422
422
  exports.Ec2RunnerProvider = Ec2RunnerProvider;
423
423
  _a = JSII_RTTI_SYMBOL_1;
424
- Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.14.16" };
424
+ Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.14.17" };
425
425
  /**
426
426
  * @deprecated use {@link Ec2RunnerProvider}
427
427
  */
@@ -429,5 +429,5 @@ class Ec2Runner extends Ec2RunnerProvider {
429
429
  }
430
430
  exports.Ec2Runner = Ec2Runner;
431
431
  _b = JSII_RTTI_SYMBOL_1;
432
- Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.14.16" };
433
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":";;;;;AAAA,mCAAmC;AACnC,6CASqB;AACrB,mDAAqD;AACrD,qEAAmE;AAEnE,qCAakB;AAClB,sDAO2B;AAC3B,oCAA4E;AAE5E,6EAA6E;AAC7E,qDAAqD;AACrD,MAAM,qBAAqB,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAwE7B,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;AAErE,6EAA6E;AAC7E,mGAAmG;AACnG,MAAM,uBAAuB,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAwE/B,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;AAuHrE;;;;GAIG;AACH,MAAa,iBAAkB,SAAQ,qBAAY;IACjD;;;;;;;;;;;;;;;;;;OAkBG;IACI,MAAM,CAAC,YAAY,CAAC,KAAgB,EAAE,EAAU,EAAE,KAA+B;QACtF,OAAO,mCAAkB,CAAC,GAAG,CAAC,KAAK,EAAE,EAAE,EAAE;YACvC,EAAE,EAAE,WAAE,CAAC,YAAY;YACnB,YAAY,EAAE,qBAAY,CAAC,MAAM;YACjC,WAAW,EAAE,uCAAsB,CAAC,iBAAiB;YACrD,UAAU,EAAE;gBACV,qCAAoB,CAAC,gBAAgB,EAAE;gBACvC,qCAAoB,CAAC,eAAe,EAAE;gBACtC,qCAAoB,CAAC,UAAU,EAAE;gBACjC,qCAAoB,CAAC,GAAG,EAAE;gBAC1B,qCAAoB,CAAC,SAAS,EAAE;gBAChC,qCAAoB,CAAC,MAAM,EAAE;gBAC7B,qCAAoB,CAAC,MAAM,EAAE;gBAC7B,qCAAoB,CAAC,YAAY,CAAC,KAAK,EAAE,aAAa,IAAI,sBAAa,CAAC,MAAM,EAAE,CAAC;aAClF;YACD,GAAG,KAAK;SACT,CAAC,CAAC;IACL,CAAC;IAsCD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA8B;QACtE,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QApBjB,oBAAe,GAAG;YACzB,kBAAkB;YAClB,gBAAgB;SACjB,CAAC;QAmBA,IAAI,CAAC,MAAM,GAAG,KAAK,EAAE,MAAM,IAAI,CAAC,KAAK,CAAC,CAAC;QACvC,IAAI,CAAC,KAAK,GAAG,KAAK,EAAE,KAAK,CAAC;QAC1B,IAAI,CAAC,GAAG,GAAG,KAAK,EAAE,GAAG,IAAI,qBAAG,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,aAAa,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACtF,IAAI,CAAC,cAAc,GAAG,KAAK,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,cAAc,IAAI,CAAC,IAAI,qBAAG,CAAC,aAAa,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QACvJ,IAAI,CAAC,OAAO,GAAG,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,KAAK,EAAE,eAAe,CAAC,CAAC,OAAO,CAAC;QACvG,IAAI,CAAC,YAAY,GAAG,KAAK,EAAE,YAAY,IAAI,qBAAG,CAAC,YAAY,CAAC,EAAE,CAAC,qBAAG,CAAC,aAAa,CAAC,GAAG,EAAE,qBAAG,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC9G,IAAI,CAAC,WAAW,GAAG,KAAK,EAAE,WAAW,IAAI,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,gCAAgC;QACjG,IAAI,CAAC,cAAc,GAAG,KAAK,EAAE,cAAc,CAAC;QAC5C,IAAI,CAAC,IAAI,GAAG,KAAK,EAAE,IAAI,IAAI,KAAK,CAAC;QACjC,IAAI,CAAC,YAAY,GAAG,KAAK,EAAE,YAAY,CAAC;QACxC,IAAI,CAAC,aAAa,GAAG,KAAK,EAAE,aAAa,IAAI,IAAI,CAAC;QAElD,IAAI,CAAC,UAAU,GAAG,KAAK,EAAE,YAAY,IAAI,KAAK,EAAE,UAAU,IAAI,iBAAiB,CAAC,YAAY,CAAC,IAAI,EAAE,aAAa,EAAE;YAChH,GAAG,EAAE,KAAK,EAAE,GAAG;YACf,eAAe,EAAE,KAAK,EAAE,eAAe;YACvC,cAAc,EAAE,IAAI,CAAC,cAAc;SACpC,CAAC,CAAC;QACH,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;QAErC,IAAI,IAAI,CAAC,UAAU,YAAY,kDAAiC,EAAE,CAAC;YACjE,IAAI,IAAI,CAAC,UAAU,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,OAAO,EAAE,EAAE,CAAC;gBACtG,MAAM,IAAI,KAAK,CAAC,wBAAwB,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE,sEAAsE,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YAChM,CAAC;QACH,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC;YAChE,MAAM,IAAI,KAAK,CAAC,qBAAqB,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,yCAAyC,IAAI,CAAC,YAAY,MAAM,IAAI,CAAC,YAAY,CAAC,YAAY,GAAG,CAAC,CAAC;QACpK,CAAC;QAED,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,IAAI,GAAG,IAAI,qBAAG,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE;YAC3D,SAAS,EAAE,IAAI,qBAAG,CAAC,gBAAgB,CAAC,mBAAmB,CAAC;SACzD,CAAC,CAAC;QACH,IAAI,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YAC/D,OAAO,EAAE,CAAC,wBAAwB,EAAE,wBAAwB,EAAE,0BAA0B,CAAC;YACzF,SAAS,EAAE,CAAC,GAAG,CAAC,EAAE,iDAAiD;YACnE,UAAU,EAAE;gBACV,YAAY,EAAE;oBACZ,6CAA6C,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO;iBAC1E;aACF;SACF,CAAC,CAAC,CAAC;QACJ,IAAI,CAAC,cAAc,CAAC,oBAAoB,CAAC,wDAAgD,CAAC,CAAC;QAE3F,IAAI,CAAC,QAAQ,GAAG,IAAI,sBAAI,CAAC,QAAQ,CAC/B,IAAI,EACJ,MAAM,EACN;YACE,SAAS,EAAE,KAAK,EAAE,YAAY,IAAI,wBAAa,CAAC,SAAS;YACzD,aAAa,EAAE,2BAAa,CAAC,OAAO;SACrC,CACF,CAAC;QACF,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IACjC,CAAC;IAED;;;;;;OAMG;IACH,mBAAmB,CAAC,UAAmC;QACrD,+IAA+I;QAE/I,MAAM,MAAM,GAAG;YACb,+BAAa,CAAC,QAAQ,CAAC,SAAS;YAChC,IAAI,CAAC,QAAQ,CAAC,YAAY;YAC1B,UAAU,CAAC,cAAc;YACzB,UAAU,CAAC,gBAAgB;YAC3B,UAAU,CAAC,SAAS;YACpB,UAAU,CAAC,QAAQ;YACnB,UAAU,CAAC,eAAe;YAC1B,UAAU,CAAC,UAAU;YACrB,UAAU,CAAC,eAAe;YAC1B,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,EAAE;YACjC,+HAA+H;YAC/H,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;YAC5B,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,qBAAqB;SAChD,CAAC;QAEF,MAAM,YAAY,GAAG,IAAI,+BAAa,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,IAAA,6BAAoB,EAAC,IAAI,CAAC,OAAO,EAAE;YACtF,UAAU,EAAE;gBACV,gBAAgB,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,WAAE,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,uBAAuB,CAAC,CAAC,CAAC,qBAAqB;aAC/F;YACD,UAAU,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC;SACrD,CAAC,CAAC;QAEH,0CAA0C;QAC1C,sGAAsG;QACtG,qGAAqG;QACrG,oHAAoH;QAEpH,yGAAyG;QACzG,6EAA6E;QAE7E,+CAA+C;QAC/C,MAAM,eAAe,GAAG,IAAI,qBAAG,CAAC,kBAAkB,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC3E,KAAK,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC;SAC5B,CAAC,CAAC;QACH,MAAM,kBAAkB,GAAG,IAAA,sBAAa,EAAC,IAAI,EAAE,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;QACzF,kBAAkB,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YAC9C,OAAO,IAAI,qCAAmB,CAAC,cAAc,CAAC,IAAI,EAAE,GAAG,IAAA,6BAAoB,EAAC,IAAI,CAAC,IAAI,MAAM,CAAC,QAAQ,EAAE,EAAE;gBACtG,OAAO,EAAE,MAAM,CAAC,gBAAgB;gBAChC,kBAAkB,EAAE,sCAAkB,CAAC,mBAAmB;gBAC1D,OAAO,EAAE,KAAK;gBACd,MAAM,EAAE,cAAc;gBACtB,gBAAgB,EAAE,+BAAa,CAAC,OAAO,CAAC,QAAQ,CAAC,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACtE,UAAU,EAAE;oBACV,cAAc,EAAE;wBACd,gBAAgB,EAAE,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC,gBAAgB;qBAC3D;oBACD,QAAQ,EAAE,CAAC;oBACX,QAAQ,EAAE,CAAC;oBACX,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE;oBAC1C,QAAQ,EAAE,+BAAa,CAAC,QAAQ,CAAC,YAAY,CAC3C,+BAAa,CAAC,QAAQ,CAAC,MAAM,CAC3B,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,wBAAwB,CAAC,EACzD,GAAG,MAAM,CACV,CACF;oBACD,iCAAiC,EAAE,qBAAG,CAAC,iCAAiC,CAAC,SAAS;oBAClF,kBAAkB,EAAE;wBAClB,GAAG,EAAE,eAAe,CAAC,OAAO;qBAC7B;oBACD,eAAe,EAAE;wBACf,UAAU,EAAE,UAAU;qBACvB;oBACD,gBAAgB,EAAE,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,eAAe,CAAC;oBACnE,QAAQ,EAAE,MAAM,CAAC,QAAQ;oBACzB,mBAAmB,EAAE,CAAC;4BACpB,UAAU,EAAE,kBAAkB,CAAC,GAAG;4BAClC,GAAG,EAAE;gCACH,mBAAmB,EAAE,IAAI;gCACzB,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;gCAC1C,UAAU,EAAE,IAAI,CAAC,cAAc,EAAE,UAAU;gCAC3C,IAAI,EAAE,IAAI,CAAC,cAAc,EAAE,IAAI;gCAC/B,UAAU,EAAE,IAAI,CAAC,cAAc,EAAE,UAAU;6BAC5C;yBACF,CAAC;oBACF,qBAAqB,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;wBACjC,UAAU,EAAE,MAAM;wBAClB,WAAW,EAAE;4BACX,QAAQ,EAAE,IAAI,CAAC,YAAY;4BAC3B,gBAAgB,EAAE,UAAU;yBAC7B;qBACF,CAAC,CAAC,CAAC,SAAS;oBACb,iBAAiB,EAAE;wBACjB;4BACE,YAAY,EAAE,UAAU;4BACxB,IAAI,EAAE,CAAC;oCACL,GAAG,EAAE,wBAAwB;oCAC7B,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI;iCACtB,CAAC;yBACH;wBACD;4BACE,YAAY,EAAE,QAAQ;4BACtB,IAAI,EAAE,CAAC;oCACL,GAAG,EAAE,wBAAwB;oCAC7B,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI;iCACtB,CAAC;yBACH;qBACF;iBACF;gBACD,YAAY,EAAE,CAAC,GAAG,CAAC;aACpB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,8BAA8B;QAC9B,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QAEpC,mCAAmC;QACnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAC9C,aAAa,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE;gBAC9C,MAAM,EAAE,CAAC,kBAAkB,EAAE,gBAAgB,CAAC;gBAC9C,UAAU,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,CAAC;aACjE,CAAC,CAAC;QACL,CAAC;QAED,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,iBAAiB,CAAC,gBAAgC;QAChD,gBAAgB,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YAC3E,OAAO,EAAE,CAAC,cAAc,CAAC;YACzB,SAAS,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC;YAC9B,UAAU,EAAE;gBACV,YAAY,EAAE;oBACZ,qBAAqB,EAAE,mBAAmB;iBAC3C;aACF;SACF,CAAC,CAAC,CAAC;QAEJ,gBAAgB,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YAC3E,OAAO,EAAE,CAAC,gBAAgB,CAAC;YAC3B,SAAS,EAAE,CAAC,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC;oBACnC,OAAO,EAAE,KAAK;oBACd,QAAQ,EAAE,GAAG;iBACd,CAAC,CAAC;SACJ,CAAC,CAAC,CAAC;QAEJ,gBAAgB,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YAC3E,OAAO,EAAE,CAAC,6BAA6B,CAAC;YACxC,SAAS,EAAE,CAAC,GAAG,CAAC;YAChB,UAAU,EAAE;gBACV,YAAY,EAAE;oBACZ,oBAAoB,EAAE,oBAAoB;iBAC3C;aACF;SACF,CAAC,CAAC,CAAC;IACN,CAAC;IAED,MAAM,CAAC,kBAAkC;QACvC,kBAAkB,CAAC,cAAc,CAAC,oBAAoB,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YAC7E,OAAO,EAAE,CAAC,oCAAoC,CAAC;YAC/C,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CAAC,CAAC;QAEJ,OAAO;YACL,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI;YAC3B,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,aAAa,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI;YAC7B,cAAc,EAAE,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,eAAe,CAAC;YACjE,OAAO,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO;YAC1B,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,YAAY;YACpC,GAAG,EAAE;gBACH,cAAc,EAAE,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC,gBAAgB,IAAI,SAAS;gBACrE,kBAAkB,EAAE,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,YAAY;aACpD;SACF,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,IAAW,WAAW;QACpB,OAAO,IAAI,qBAAG,CAAC,WAAW,CAAC,EAAE,cAAc,EAAE,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;IACtE,CAAC;;AA3TH,8CA4TC;;;AAED;;GAEG;AACH,MAAa,SAAU,SAAQ,iBAAiB;;AAAhD,8BACC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  amiRootDevice,\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  nodePathWithoutStack,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n  StorageOptions,\n} from './common';\nimport {\n  AwsImageBuilderRunnerImageBuilder,\n  IRunnerImageBuilder,\n  RunnerImageBuilder,\n  RunnerImageBuilderProps,\n  RunnerImageBuilderType,\n  RunnerImageComponent,\n} from '../image-builders';\nimport { MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT } from '../utils';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nlogGroupName=\"{}\"\nrunnerNamePath=\"{}\"\ngithubDomainPath=\"{}\"\nownerPath=\"{}\"\nrepoPath=\"{}\"\nrunnerTokenPath=\"{}\"\nlabels=\"{}\"\nregistrationURL=\"{}\"\nrunnerGroup1=\"{}\"\nrunnerGroup2=\"{}\"\ndefaultLabels=\"{}\"\n\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"$logGroupName\",\n              \"log_stream_name\": \"$runnerNamePath\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  # Determine the value of RUNNER_FLAGS\n  if [ \"$(< /home/runner/RUNNER_VERSION)\" = \"latest\" ]; then\n    RUNNER_FLAGS=\"\"\n  else\n    RUNNER_FLAGS=\"--disableupdate\"\n  fi\n\n  labelsTemplate=\"$labels,cdkghr:started:$(date +%s)\"\n\n  # Execute the configuration command for runner registration\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"$registrationURL\" --token \"$runnerTokenPath\" --ephemeral --work _work --labels \"$labelsTemplate\" $RUNNER_FLAGS --name \"$runnerNamePath\" $runnerGroup1 $runnerGroup2 $defaultLabels || exit 1\n\n  # Execute the run command\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n\n  # Retrieve the status\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\K.*\" /home/runner/_diag/ | tail -n1)\n\n  # Check and print the job status\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$labels\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\n$logGroupName=\"{}\"\n$runnerNamePath=\"{}\"\n$githubDomainPath=\"{}\"\n$ownerPath=\"{}\"\n$repoPath=\"{}\"\n$runnerTokenPath=\"{}\"\n$labels=\"{}\"\n$registrationURL=\"{}\"\n$runnerGroup1=\"{}\"\n$runnerGroup2=\"{}\"\n$defaultLabels=\"{}\"\n\n# EC2Launch only starts ssm agent after user data is done, so we need to start it ourselves (it is disabled by default)\nSet-Service -StartupType Manual AmazonSSMAgent\nStart-Service AmazonSSMAgent\n\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo \"{\n    \\`\"logs\\`\": {\n      \\`\"log_stream_name\\`\": \\`\"unknown\\`\",\n      \\`\"logs_collected\\`\": {\n        \\`\"files\\`\": {\n         \\`\"collect_list\\`\": [\n            {\n              \\`\"file_path\\`\": \\`\"/actions/runner.log\\`\",\n              \\`\"log_group_name\\`\": \\`\"$logGroupName\\`\",\n              \\`\"log_stream_name\\`\": \\`\"$runnerNamePath\\`\",\n              \\`\"timezone\\`\": \\`\"UTC\\`\"\n            }\n          ]\n        }\n      }\n    }\n  }\" | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  $RunnerVersion = Get-Content /actions/RUNNER_VERSION -Raw\n  if ($RunnerVersion -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" }\n  ./config.cmd --unattended --url \"\\${registrationUrl}\" --token \"\\${runnerTokenPath}\" --ephemeral --work _work --labels \"\\${labels},cdkghr:started:$(Get-Date -UFormat +%s)\" $RunnerFlags --name \"\\${runnerNamePath}\" \\${runnerGroup1} \\${runnerGroup2} \\${defaultLabels} 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\n\n  $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1\n\n  if ($STATUS) {\n      echo \"CDKGHA JOB DONE \\${labels} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log\n  }\n\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build AMI containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default Ec2RunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * @deprecated use imageBuilder\n   */\n  readonly amiBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * 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   * Instance type for launched runner instances.\n   *\n   * @default m6i.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Options for runner instance storage volume.\n   */\n  readonly storageOptions?: StorageOptions;\n\n  /**\n   * Security Group to assign to launched runner instances.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds EC2 specific runner images.\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.cloudWatchAgent()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      builderType: RunnerImageBuilderType.AWS_IMAGE_BUILDER,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.cloudWatchAgent(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  readonly retryableErrors = [\n    'Ec2.Ec2Exception',\n    'States.Timeout',\n  ];\n\n  private readonly group?: string;\n  private readonly amiBuilder: IRunnerImageBuilder;\n  private readonly ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly storageOptions?: StorageOptions;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly defaultLabels: boolean;\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.group = props?.group;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M6I, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.storageOptions = props?.storageOptions;\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n    this.defaultLabels = props?.defaultLabels ?? true;\n\n    this.amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = this.amiBuilder.bindAmi();\n\n    if (this.amiBuilder instanceof AwsImageBuilderRunnerImageBuilder) {\n      if (this.amiBuilder.storageSize && this.storageSize.toBytes() < this.amiBuilder.storageSize.toBytes()) {\n        throw new Error(`Runner storage size (${this.storageSize.toGibibytes()} GiB) must be at least the same as the image builder storage size (${this.amiBuilder.storageSize.toGibibytes()} GiB)`);\n      }\n    }\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n      conditions: {\n        StringEquals: {\n          'aws:ResourceTag/aws:cloudformation:stack-id': cdk.Stack.of(this).stackId,\n        },\n      },\n    }));\n    this.grantPrincipal.addToPrincipalPolicy(MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT);\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      parameters.labelsPath,\n      parameters.registrationUrl,\n      this.group ? '--runnergroup' : '',\n      // this is split into 2 for powershell otherwise it will pass \"--runnergroup name\" as a single argument and config.sh will fail\n      this.group ? this.group : '',\n      this.defaultLabels ? '' : '--no-default-labels',\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, `${nodePathWithoutStack(this)} data`, {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const rootDeviceResource = amiRootDevice(this, this.ami.launchTemplate.launchTemplateId);\n    rootDeviceResource.node.addDependency(this.amiBuilder);\n    const subnetRunners = this.subnets.map(subnet => {\n      return new stepfunctions_tasks.CallAwsService(this, `${nodePathWithoutStack(this)} ${subnet.subnetId}`, {\n        comment: subnet.availabilityZone,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeatTimeout: stepfunctions.Timeout.duration(Duration.minutes(10)),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: rootDeviceResource.ref,\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n              VolumeType: this.storageOptions?.volumeType,\n              Iops: this.storageOptions?.iops,\n              Throughput: this.storageOptions?.throughput,\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n          TagSpecifications: [ // manually propagate tags\n            {\n              ResourceType: 'instance',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n            {\n              ResourceType: 'volume',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n          ],\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // start with the first subnet\n    passUserData.next(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i - 1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    return passUserData;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      constructPath: this.node.path,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n}\n\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}
432
+ Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.14.17" };
433
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  amiRootDevice,\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n  generateStateName,\n  StorageOptions,\n} from './common';\nimport {\n  AwsImageBuilderRunnerImageBuilder,\n  IRunnerImageBuilder,\n  RunnerImageBuilder,\n  RunnerImageBuilderProps,\n  RunnerImageBuilderType,\n  RunnerImageComponent,\n} from '../image-builders';\nimport { MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT } from '../utils';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nlogGroupName=\"{}\"\nrunnerNamePath=\"{}\"\ngithubDomainPath=\"{}\"\nownerPath=\"{}\"\nrepoPath=\"{}\"\nrunnerTokenPath=\"{}\"\nlabels=\"{}\"\nregistrationURL=\"{}\"\nrunnerGroup1=\"{}\"\nrunnerGroup2=\"{}\"\ndefaultLabels=\"{}\"\n\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"$logGroupName\",\n              \"log_stream_name\": \"$runnerNamePath\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  # Determine the value of RUNNER_FLAGS\n  if [ \"$(< /home/runner/RUNNER_VERSION)\" = \"latest\" ]; then\n    RUNNER_FLAGS=\"\"\n  else\n    RUNNER_FLAGS=\"--disableupdate\"\n  fi\n\n  labelsTemplate=\"$labels,cdkghr:started:$(date +%s)\"\n\n  # Execute the configuration command for runner registration\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"$registrationURL\" --token \"$runnerTokenPath\" --ephemeral --work _work --labels \"$labelsTemplate\" $RUNNER_FLAGS --name \"$runnerNamePath\" $runnerGroup1 $runnerGroup2 $defaultLabels || exit 1\n\n  # Execute the run command\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n\n  # Retrieve the status\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\K.*\" /home/runner/_diag/ | tail -n1)\n\n  # Check and print the job status\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$labels\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\n$logGroupName=\"{}\"\n$runnerNamePath=\"{}\"\n$githubDomainPath=\"{}\"\n$ownerPath=\"{}\"\n$repoPath=\"{}\"\n$runnerTokenPath=\"{}\"\n$labels=\"{}\"\n$registrationURL=\"{}\"\n$runnerGroup1=\"{}\"\n$runnerGroup2=\"{}\"\n$defaultLabels=\"{}\"\n\n# EC2Launch only starts ssm agent after user data is done, so we need to start it ourselves (it is disabled by default)\nSet-Service -StartupType Manual AmazonSSMAgent\nStart-Service AmazonSSMAgent\n\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo \"{\n    \\`\"logs\\`\": {\n      \\`\"log_stream_name\\`\": \\`\"unknown\\`\",\n      \\`\"logs_collected\\`\": {\n        \\`\"files\\`\": {\n         \\`\"collect_list\\`\": [\n            {\n              \\`\"file_path\\`\": \\`\"/actions/runner.log\\`\",\n              \\`\"log_group_name\\`\": \\`\"$logGroupName\\`\",\n              \\`\"log_stream_name\\`\": \\`\"$runnerNamePath\\`\",\n              \\`\"timezone\\`\": \\`\"UTC\\`\"\n            }\n          ]\n        }\n      }\n    }\n  }\" | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  $RunnerVersion = Get-Content /actions/RUNNER_VERSION -Raw\n  if ($RunnerVersion -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" }\n  ./config.cmd --unattended --url \"\\${registrationUrl}\" --token \"\\${runnerTokenPath}\" --ephemeral --work _work --labels \"\\${labels},cdkghr:started:$(Get-Date -UFormat +%s)\" $RunnerFlags --name \"\\${runnerNamePath}\" \\${runnerGroup1} \\${runnerGroup2} \\${defaultLabels} 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\n\n  $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1\n\n  if ($STATUS) {\n      echo \"CDKGHA JOB DONE \\${labels} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log\n  }\n\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build AMI containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default Ec2RunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * @deprecated use imageBuilder\n   */\n  readonly amiBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * 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   * Instance type for launched runner instances.\n   *\n   * @default m6i.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Options for runner instance storage volume.\n   */\n  readonly storageOptions?: StorageOptions;\n\n  /**\n   * Security Group to assign to launched runner instances.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds EC2 specific runner images.\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.cloudWatchAgent()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      builderType: RunnerImageBuilderType.AWS_IMAGE_BUILDER,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.cloudWatchAgent(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  readonly retryableErrors = [\n    'Ec2.Ec2Exception',\n    'States.Timeout',\n  ];\n\n  private readonly group?: string;\n  private readonly amiBuilder: IRunnerImageBuilder;\n  private readonly ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly storageOptions?: StorageOptions;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly defaultLabels: boolean;\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.group = props?.group;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M6I, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.storageOptions = props?.storageOptions;\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n    this.defaultLabels = props?.defaultLabels ?? true;\n\n    this.amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = this.amiBuilder.bindAmi();\n\n    if (this.amiBuilder instanceof AwsImageBuilderRunnerImageBuilder) {\n      if (this.amiBuilder.storageSize && this.storageSize.toBytes() < this.amiBuilder.storageSize.toBytes()) {\n        throw new Error(`Runner storage size (${this.storageSize.toGibibytes()} GiB) must be at least the same as the image builder storage size (${this.amiBuilder.storageSize.toGibibytes()} GiB)`);\n      }\n    }\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n      conditions: {\n        StringEquals: {\n          'aws:ResourceTag/aws:cloudformation:stack-id': cdk.Stack.of(this).stackId,\n        },\n      },\n    }));\n    this.grantPrincipal.addToPrincipalPolicy(MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT);\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      parameters.labelsPath,\n      parameters.registrationUrl,\n      this.group ? '--runnergroup' : '',\n      // this is split into 2 for powershell otherwise it will pass \"--runnergroup name\" as a single argument and config.sh will fail\n      this.group ? this.group : '',\n      this.defaultLabels ? '' : '--no-default-labels',\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, generateStateName(this, 'data'), {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const rootDeviceResource = amiRootDevice(this, this.ami.launchTemplate.launchTemplateId);\n    rootDeviceResource.node.addDependency(this.amiBuilder);\n    const subnetRunners = this.subnets.map(subnet => {\n      return new stepfunctions_tasks.CallAwsService(this, generateStateName(this, subnet.subnetId), {\n        comment: subnet.availabilityZone,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeatTimeout: stepfunctions.Timeout.duration(Duration.minutes(10)),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: rootDeviceResource.ref,\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n              VolumeType: this.storageOptions?.volumeType,\n              Iops: this.storageOptions?.iops,\n              Throughput: this.storageOptions?.throughput,\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n          TagSpecifications: [ // manually propagate tags\n            {\n              ResourceType: 'instance',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n            {\n              ResourceType: 'volume',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n          ],\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // start with the first subnet\n    passUserData.next(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i - 1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    return passUserData;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      constructPath: this.node.path,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n}\n\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}