@cloudsnorkel/cdk-github-runners 0.14.20 → 0.14.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/.jsii +27 -27
  2. package/README.md +1 -0
  3. package/assets/delete-failed-runner.lambda/index.js +3248 -11227
  4. package/assets/idle-runner-repear.lambda/index.js +3224 -11221
  5. package/assets/image-builders/aws-image-builder/delete-resources.lambda/index.js +12 -2
  6. package/assets/image-builders/aws-image-builder/filter-failed-builds.lambda/index.js +4 -1
  7. package/assets/image-builders/build-image.lambda/index.js +21 -4
  8. package/assets/providers/ami-root-device.lambda/index.js +45 -10
  9. package/assets/providers/update-lambda.lambda/index.js +6 -1
  10. package/assets/setup.lambda/index.js +1762 -4645
  11. package/assets/status.lambda/index.js +2412 -10615
  12. package/assets/token-retriever.lambda/index.js +3228 -11221
  13. package/assets/webhook-handler.lambda/index.js +3264 -11236
  14. package/assets/webhook-redelivery.lambda/index.js +3220 -11221
  15. package/lib/access.js +1 -1
  16. package/lib/delete-failed-runner.lambda.js +22 -4
  17. package/lib/idle-runner-repear.lambda.js +1 -1
  18. package/lib/image-builders/api.js +1 -1
  19. package/lib/image-builders/aws-image-builder/base-image.js +2 -2
  20. package/lib/image-builders/aws-image-builder/builder.js +3 -3
  21. package/lib/image-builders/aws-image-builder/delete-resources.lambda.js +6 -2
  22. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  23. package/lib/image-builders/aws-image-builder/deprecated/container.js +1 -1
  24. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  25. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  26. package/lib/image-builders/aws-image-builder/filter-failed-builds.lambda.js +5 -2
  27. package/lib/image-builders/build-image.lambda.js +15 -4
  28. package/lib/image-builders/codebuild-deprecated.js +1 -1
  29. package/lib/image-builders/components.d.ts +12 -1
  30. package/lib/image-builders/components.js +48 -10
  31. package/lib/image-builders/static.js +1 -1
  32. package/lib/lambda-github.d.ts +14 -7
  33. package/lib/lambda-github.js +34 -8
  34. package/lib/lambda-helpers.js +8 -2
  35. package/lib/providers/ami-root-device.lambda.js +39 -10
  36. package/lib/providers/codebuild.js +2 -2
  37. package/lib/providers/common.js +3 -3
  38. package/lib/providers/composite.js +1 -1
  39. package/lib/providers/ec2.js +2 -2
  40. package/lib/providers/ecs.js +1 -1
  41. package/lib/providers/fargate.js +2 -2
  42. package/lib/providers/lambda.js +2 -2
  43. package/lib/providers/update-lambda.lambda.js +7 -2
  44. package/lib/runner.js +1 -1
  45. package/lib/secrets.js +1 -1
  46. package/lib/setup.lambda.js +7 -4
  47. package/lib/status.lambda.js +17 -8
  48. package/lib/token-retriever.lambda.js +8 -2
  49. package/lib/webhook-handler.lambda.js +39 -12
  50. package/lib/webhook-redelivery.lambda.js +1 -1
  51. package/package.json +5 -5
@@ -3,7 +3,9 @@ var _a;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
4
  exports.RunnerImageComponent = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const crypto = require("crypto");
6
7
  const path = require("path");
8
+ const cdk = require("aws-cdk-lib");
7
9
  const aws_cdk_lib_1 = require("aws-cdk-lib");
8
10
  const utils_1 = require("../utils");
9
11
  const aws_image_builder_1 = require("./aws-image-builder");
@@ -621,9 +623,12 @@ class RunnerImageComponent {
621
623
  /**
622
624
  * Convert component to an AWS Image Builder component.
623
625
  *
626
+ * Components are cached and reused when the same component is requested with the same
627
+ * OS and architecture, reducing stack template size and number of resources.
628
+ *
624
629
  * @internal
625
630
  */
626
- _asAwsImageBuilderComponent(scope, id, os, architecture) {
631
+ _asAwsImageBuilderComponent(scope, os, architecture) {
627
632
  let platform;
628
633
  if (os.isIn(providers_1.Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(providers_1.Os._ALL_LINUX_AMAZON_VERSIONS)) {
629
634
  platform = 'Linux';
@@ -634,22 +639,55 @@ class RunnerImageComponent {
634
639
  else {
635
640
  throw new Error(`Unknown os/architecture combo for image builder component: ${os.name}/${architecture.name}`);
636
641
  }
637
- return new aws_image_builder_1.ImageBuilderComponent(scope, id, {
642
+ // Get component properties to create a cache key
643
+ const commands = this.getCommands(os, architecture);
644
+ const assets = this.getAssets(os, architecture);
645
+ const reboot = this.shouldReboot(os, architecture);
646
+ // Create a cache key based on component identity and properties
647
+ const stack = cdk.Stack.of(scope);
648
+ const cacheKey = this._getCacheKey(os, architecture, commands, assets, reboot);
649
+ // Create a consistent ID based on the cache key to ensure the same component
650
+ // always gets the same ID, regardless of the passed-in id parameter
651
+ // The cache key is already a hash, so we can use it directly
652
+ // Prefix with GHRInternal/ to avoid conflicts with user-defined constructs
653
+ const consistentId = `GHRInternal/Component-${this.name}-${os.name}-${architecture.name}-${cacheKey.substring(0, 10)}`.replace(/[^a-zA-Z0-9-/]/g, '-');
654
+ // Use the construct tree as the cache - check if component already exists in the stack
655
+ const existing = stack.node.tryFindChild(consistentId);
656
+ if (existing) {
657
+ // Component already exists in this stack, reuse it
658
+ return existing;
659
+ }
660
+ // Create new component in the stack scope so it can be shared across all scopes in the same stack
661
+ const component = new aws_image_builder_1.ImageBuilderComponent(stack, consistentId, {
638
662
  platform: platform,
639
- commands: this.getCommands(os, architecture),
640
- assets: this.getAssets(os, architecture).map((asset, index) => {
663
+ commands: commands,
664
+ assets: assets.map((asset, index) => {
641
665
  return {
642
- asset: new aws_cdk_lib_1.aws_s3_assets.Asset(scope, `${id} asset ${index}`, { path: asset.source }),
666
+ asset: new aws_cdk_lib_1.aws_s3_assets.Asset(stack, `GHRInternal/${consistentId}/Asset${index}`, { path: asset.source }),
643
667
  path: asset.target,
644
668
  };
645
669
  }),
646
- displayName: id,
647
- description: id,
648
- reboot: this.shouldReboot(os, architecture),
670
+ displayName: `${this.name} (${os.name}/${architecture.name})`,
671
+ description: `${this.name} component for ${os.name}/${architecture.name}`,
672
+ reboot: reboot,
649
673
  });
674
+ return component;
675
+ }
676
+ /**
677
+ * Generate a cache key for component reuse.
678
+ * Components with the same name, OS, architecture, commands, assets, and reboot flag will share the same key.
679
+ * Returns a hash of all component properties to ensure uniqueness.
680
+ *
681
+ * @internal
682
+ */
683
+ _getCacheKey(os, architecture, commands, assets, reboot) {
684
+ // Create a hash of the component properties
685
+ const assetKeys = assets.map(a => `${a.source}:${a.target}`).sort().join('|');
686
+ const keyData = `${this.name}:${os.name}:${architecture.name}:${commands.join('\n')}:${assetKeys}:${reboot}`;
687
+ return crypto.createHash('md5').update(keyData).digest('hex');
650
688
  }
651
689
  }
652
690
  exports.RunnerImageComponent = RunnerImageComponent;
653
691
  _a = JSII_RTTI_SYMBOL_1;
654
- RunnerImageComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerImageComponent", version: "0.14.20" };
655
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"components.js","sourceRoot":"","sources":["../../src/image-builders/components.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { aws_s3_assets as s3_assets } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { discoverCertificateFiles } from '../utils';\nimport { ImageBuilderComponent } from './aws-image-builder';\nimport { RunnerImageAsset } from './common';\nimport { Architecture, Os, RunnerVersion } from '../providers';\n\nexport interface RunnerImageComponentCustomProps {\n  /**\n   * Component name used for (1) image build logging and (2) identifier for {@link IConfigurableRunnerImageBuilder.removeComponent}.\n   *\n   * Name must only contain alphanumeric characters and dashes.\n   */\n  readonly name?: string;\n\n  /**\n   * Commands to run in the built image.\n   */\n  readonly commands?: string[];\n\n  /**\n   * Assets to copy into the built image.\n   */\n  readonly assets?: RunnerImageAsset[];\n\n  /**\n   * Docker commands to run in the built image.\n   *\n   * For example: `['ENV foo=bar', 'RUN echo $foo']`\n   *\n   * These commands are ignored when building AMIs.\n   */\n  readonly dockerCommands?: string[];\n}\n\n/**\n * Components are used to build runner images. They can run commands in the image, copy files into the image, and run some Docker commands.\n */\nexport abstract class RunnerImageComponent {\n  /**\n   * Define a custom component that can run commands in the image, copy files into the image, and run some Docker commands.\n   *\n   * The order of operations is (1) assets (2) commands (3) docker commands.\n   *\n   * Use this to customize the image for the runner.\n   *\n   * **WARNING:** Docker commands are not guaranteed to be included before the next component\n   */\n  static custom(props: RunnerImageComponentCustomProps): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      get name() {\n        if (props.name && !props.name.match(/[a-zA-Z0-9\\-]/)) {\n          throw new Error(`Invalid component name: ${props.name}. Name must only contain alphanumeric characters and dashes.`);\n        }\n        return `Custom-${props.name ?? 'Undefined'}`;\n      }\n\n      getCommands(_os: Os, _architecture: Architecture) {\n        return props.commands ?? [];\n      }\n      getAssets(_os: Os, _architecture: Architecture) {\n        return props.assets ?? [];\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture) {\n        return props.dockerCommands ?? [];\n      }\n    }();\n  }\n\n  /**\n   * A component to install the required packages for the runner.\n   */\n  static requiredPackages(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'RequiredPackages';\n\n      getCommands(os: Os, _architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get upgrade -y',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y curl sudo jq bash zip unzip iptables software-properties-common ca-certificates',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum update -y',\n            'yum install -y jq tar gzip bzip2 which binutils zip unzip sudo shadow-utils',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf upgrade -y',\n            'dnf install -y jq tar gzip bzip2 which binutils zip unzip sudo shadow-utils findutils',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [];\n        }\n\n        throw new Error(`Unsupported OS for required packages: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to install CloudWatch Agent for the runner so we can send logs.\n   */\n  static cloudWatchAgent(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'CloudWatchAgent';\n\n      getCommands(os: Os, architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          let archUrl;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'amd64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'arm64';\n          } else {\n            throw new Error(`Unsupported architecture for required packages: ${architecture.name}`);\n          }\n\n          return [\n            `curl -sfLo /tmp/amazon-cloudwatch-agent.deb https://s3.amazonaws.com/amazoncloudwatch-agent/ubuntu/${archUrl}/latest/amazon-cloudwatch-agent.deb`,\n            'dpkg -i -E /tmp/amazon-cloudwatch-agent.deb',\n            'rm /tmp/amazon-cloudwatch-agent.deb',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum install -y amazon-cloudwatch-agent',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y amazon-cloudwatch-agent',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://s3.amazonaws.com/amazoncloudwatch-agent/windows/amd64/latest/amazon-cloudwatch-agent.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n          ];\n        }\n\n        throw new Error(`Unsupported OS for required packages: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to prepare the required runner user.\n   */\n  static runnerUser(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'RunnerUser';\n\n      getCommands(os: Os, _architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'addgroup runner',\n            'adduser --system --disabled-password --home /home/runner --ingroup runner runner',\n            'echo \"%runner   ALL=(ALL:ALL) NOPASSWD: ALL\" > /etc/sudoers.d/runner',\n          ];\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          return [\n            '/usr/sbin/groupadd runner',\n            '/usr/sbin/useradd --system --shell /usr/sbin/nologin --home-dir /home/runner --gid runner runner',\n            'mkdir -p /home/runner',\n            'chown runner /home/runner',\n            'echo \"%runner   ALL=(ALL:ALL) NOPASSWD: ALL\" > /etc/sudoers.d/runner',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [];\n        }\n\n        throw new Error(`Unsupported OS for runner user: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to install the AWS CLI.\n   */\n  static awsCli(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'AwsCli';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          let archUrl: string;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'x86_64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'aarch64';\n          } else {\n            throw new Error(`Unsupported architecture for awscli: ${architecture.name}`);\n          }\n\n          return [\n            `curl -fsSL \"https://awscli.amazonaws.com/awscli-exe-linux-${archUrl}.zip\" -o awscliv2.zip`,\n            'unzip -q awscliv2.zip',\n            './aws/install',\n            'rm -rf awscliv2.zip aws',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for awscli: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub CLI.\n   */\n  static githubCli(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'GithubCli';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'curl -fsSL https://cli.github.com/packages/githubcli-archive-keyring.gpg | sudo dd of=/usr/share/keyrings/githubcli-archive-keyring.gpg',\n            'echo \"deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/githubcli-archive-keyring.gpg] ' +\n            '  https://cli.github.com/packages stable main\" | sudo tee /etc/apt/sources.list.d/github-cli.list > /dev/null',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y gh',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'curl -fsSSL https://cli.github.com/packages/rpm/gh-cli.repo -o /etc/yum.repos.d/gh-cli.repo',\n            'yum install -y gh',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'curl -fsSSL https://cli.github.com/packages/rpm/gh-cli.repo -o /etc/yum.repos.d/gh-cli.repo',\n            'dnf install -y gh',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/cli/cli/releases/latest > $Env:TEMP\\\\latest-gh',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-gh',\n            '$GH_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/cli/cli/releases/download/v${GH_VERSION}/gh_${GH_VERSION}_windows_amd64.msi\" -OutFile gh.msi',\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i gh.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n            'del gh.msi',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for github cli: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub CLI.\n   */\n  static git(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Git';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'add-apt-repository ppa:git-core/ppa',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y git',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum install -y git',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y git',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/git-for-windows/git/releases/latest > $Env:TEMP\\\\latest-git',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-git',\n            '$GIT_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            '$GIT_VERSION_SHORT = ($GIT_VERSION -Split \\'.windows.\\')[0]',\n            '$GIT_REVISION = ($GIT_VERSION -Split \\'.windows.\\')[1]',\n            'If ($GIT_REVISION -gt 1) {$GIT_VERSION_SHORT = \"$GIT_VERSION_SHORT.$GIT_REVISION\"}',\n            'Invoke-WebRequest -UseBasicParsing -Uri https://github.com/git-for-windows/git/releases/download/v${GIT_VERSION}/Git-${GIT_VERSION_SHORT}-64-bit.exe -OutFile git-setup.exe',\n            '$p = Start-Process git-setup.exe -PassThru -Wait -ArgumentList \\'/VERYSILENT\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n            'del git-setup.exe',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for git: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub Actions Runner. This is the actual executable that connects to GitHub to ask for jobs and then execute them.\n   *\n   * @param runnerVersion The version of the runner to install. Usually you would set this to latest.\n   */\n  static githubRunner(runnerVersion: RunnerVersion): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'GithubRunner';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          let versionCommand: string;\n          if (runnerVersion.is(RunnerVersion.latest())) {\n            versionCommand = 'RUNNER_VERSION=`curl -w \"%{redirect_url}\" -fsS https://github.com/actions/runner/releases/latest | grep -oE \"[^/v]+$\"`';\n          } else {\n            versionCommand = `RUNNER_VERSION='${runnerVersion.version}'`;\n          }\n\n          let archUrl;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'x64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'arm64';\n          } else {\n            throw new Error(`Unsupported architecture for GitHub Runner: ${architecture.name}`);\n          }\n\n          let commands = [\n            versionCommand,\n            `curl -fsSLO \"https://github.com/actions/runner/releases/download/v\\${RUNNER_VERSION}/actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz\"`,\n            `tar -C /home/runner -xzf \"actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz\"`,\n            `rm actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz`,\n            `echo -n ${runnerVersion.version} > /home/runner/RUNNER_VERSION`,\n          ];\n\n          if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n            commands.push('/home/runner/bin/installdependencies.sh');\n          } else if (os.is(Os.LINUX_AMAZON_2)) {\n            commands.push('yum install -y openssl-libs krb5-libs zlib libicu60');\n          } else if (os.is(Os.LINUX_AMAZON_2023)) {\n            commands.push('dnf install -y openssl-libs krb5-libs zlib libicu-67.1');\n          }\n\n          commands.push('mkdir -p /opt/hostedtoolcache', 'chown runner /opt/hostedtoolcache');\n\n          return commands;\n        } else if (os.is(Os.WINDOWS)) {\n          let runnerCommands: string[];\n          if (runnerVersion.is(RunnerVersion.latest())) {\n            runnerCommands = [\n              'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/actions/runner/releases/latest > $Env:TEMP\\\\latest-gha',\n              '$LatestUrl = Get-Content $Env:TEMP\\\\latest-gha',\n              '$RUNNER_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            ];\n          } else {\n            runnerCommands = [`$RUNNER_VERSION = '${runnerVersion.version}'`];\n          }\n\n          runnerCommands = runnerCommands.concat([\n            // create directories\n            'mkdir C:\\\\hostedtoolcache\\\\windows',\n            'mkdir C:\\\\tools',\n            // download zstd and extract to C:\\tools\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/facebook/zstd/releases/latest > $Env:TEMP\\\\latest-zstd',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-zstd',\n            '$ZSTD_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/facebook/zstd/releases/download/v$ZSTD_VERSION/zstd-v$ZSTD_VERSION-win64.zip\" -OutFile zstd.zip',\n            'Expand-Archive zstd.zip -DestinationPath C:\\\\tools',\n            'Move-Item -Path C:\\\\tools\\\\zstd-v$ZSTD_VERSION-win64\\\\zstd.exe C:\\\\tools',\n            'Remove-Item -LiteralPath \"C:\\\\tools\\\\zstd-v$ZSTD_VERSION-win64\" -Force -Recurse',\n            'del zstd.zip',\n            // add C:\\tools to PATH\n            '$persistedPaths = [Environment]::GetEnvironmentVariable(\\'Path\\', [EnvironmentVariableTarget]::Machine)',\n            '[Environment]::SetEnvironmentVariable(\"PATH\", $persistedPaths + \";C:\\\\tools\", [EnvironmentVariableTarget]::Machine)',\n          ]);\n\n          return runnerCommands.concat([\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/actions/runner/releases/download/v${RUNNER_VERSION}/actions-runner-win-x64-${RUNNER_VERSION}.zip\" -OutFile actions.zip',\n            'Expand-Archive actions.zip -DestinationPath C:\\\\actions',\n            'del actions.zip',\n            `echo ${runnerVersion.version} | Out-File -Encoding ASCII -NoNewline C:\\\\actions\\\\RUNNER_VERSION`,\n          ]);\n        }\n\n        throw new Error(`Unknown os/architecture combo for github runner: ${os.name}/${architecture.name}`);\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n        return [\n          `ENV RUNNER_VERSION=${runnerVersion.version}`,\n        ];\n      }\n    }();\n  }\n\n  /**\n   * A component to install Docker.\n   *\n   * On Windows this sets up dockerd for Windows containers without Docker Desktop. If you need Linux containers on Windows, you'll need to install Docker Desktop which doesn't seem to play well with servers (PRs welcome).\n   */\n  static docker(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Docker';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker.gpg',\n            'echo ' +\n            '  \"deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker.gpg] https://download.docker.com/linux/ubuntu ' +\n            '  $(lsb_release -cs) stable\" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y docker-ce docker-ce-cli containerd.io docker-compose-plugin',\n            'usermod -aG docker runner',\n            'ln -s /usr/libexec/docker/cli-plugins/docker-compose /usr/bin/docker-compose',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'amazon-linux-extras install docker',\n            'usermod -a -G docker runner',\n            'curl -sfLo /usr/bin/docker-compose https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s | tr \\'[:upper:]\\' \\'[:lower:]\\')-$(uname -m)',\n            'chmod +x /usr/bin/docker-compose',\n            'ln -s /usr/bin/docker-compose /usr/libexec/docker/cli-plugins/docker-compose',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y docker',\n            'usermod -a -G docker runner',\n            'curl -sfLo /usr/bin/docker-compose https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s | tr \\'[:upper:]\\' \\'[:lower:]\\')-$(uname -m)',\n            'chmod +x /usr/bin/docker-compose',\n            'ln -s /usr/bin/docker-compose /usr/libexec/docker/cli-plugins/docker-compose',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            // figure out latest docker version\n            '$BaseUrl = \"https://download.docker.com/win/static/stable/x86_64/\"',\n            '$html = Invoke-WebRequest -UseBasicParsing -Uri $BaseUrl',\n            '$files = $html.Links.href | Where-Object { $_ -match \\'^docker-[0-9\\\\.]+\\\\.zip$\\' }',\n            'if (-not $files) { Write-Error \"No docker-*.zip files found.\" ; exit 1 }',\n            '$latest = $files | Sort-Object { try { [Version]($_ -replace \\'^docker-|\\\\.zip$\\') } catch { [Version]\"0.0.0\" } } -Descending | Select-Object -First 1',\n            // download static binaries\n            'Invoke-WebRequest -UseBasicParsing -Uri $BaseUrl$latest -OutFile docker.zip',\n            // extract to C:\\Program Files\\Docker\n            'Expand-Archive docker.zip -DestinationPath \"$Env:ProgramFiles\"',\n            'del docker.zip',\n            // add to path\n            '$persistedPaths = [Environment]::GetEnvironmentVariable(\\'Path\\', [EnvironmentVariableTarget]::Machine)',\n            '[Environment]::SetEnvironmentVariable(\"PATH\", $persistedPaths + \";$Env:ProgramFiles\\\\Docker\", [EnvironmentVariableTarget]::Machine)',\n            '$env:PATH = $env:PATH + \";$Env:ProgramFiles\\\\Docker\"',\n            // register docker service\n            'dockerd --register-service',\n            'if ($LASTEXITCODE -ne 0) { throw \"Exit code is $LASTEXITCODE\" }',\n            // enable containers feature\n            'Enable-WindowsOptionalFeature -Online -FeatureName containers -All -NoRestart',\n            // install docker-compose\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/docker/compose/releases/latest > $Env:TEMP\\\\latest-docker-compose',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-docker-compose',\n            '$LatestDockerCompose = ($LatestUrl -Split \\'/\\')[-1]',\n            'Invoke-WebRequest -UseBasicParsing -Uri  \"https://github.com/docker/compose/releases/download/${LatestDockerCompose}/docker-compose-Windows-x86_64.exe\" -OutFile $Env:ProgramFiles\\\\Docker\\\\docker-compose.exe',\n            'New-Item -ItemType directory -Path \"$Env:ProgramFiles\\\\Docker\\\\cli-plugins\"',\n            'Copy-Item -Path \"$Env:ProgramFiles\\\\Docker\\\\docker-compose.exe\" -Destination \"$Env:ProgramFiles\\\\Docker\\\\cli-plugins\\\\docker-compose.exe\"',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for docker: ${os.name}/${architecture.name}`);\n      }\n\n      shouldReboot(os: Os, _architecture: Architecture): boolean {\n        return os.is(Os.WINDOWS);\n      }\n    }();\n  }\n\n  /**\n   * A component to install Docker-in-Docker.\n   *\n   * @deprecated use `docker()`\n   */\n  static dockerInDocker(): RunnerImageComponent {\n    return RunnerImageComponent.docker();\n  }\n\n  /**\n   * A component to add a trusted certificate authority. This can be used to support GitHub Enterprise Server with self-signed certificate.\n   *\n   * @param source path to certificate file in PEM format, or a directory containing certificate files (.pem or .crt)\n   * @param name unique certificate name to be used on runner file system\n   */\n  static extraCertificates(source: string, name: string): RunnerImageComponent {\n    // Sanitize the name to only contain alphanumeric characters, dashes and underscores\n    const sanitizedName = name.replace(/[^a-zA-Z0-9_-]/g, '-');\n\n    // Discover certificate files (supports both file and directory)\n    const certificateFiles = discoverCertificateFiles(source);\n\n    return new class extends RunnerImageComponent {\n      name = `Extra-Certificates-${sanitizedName}`;\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'update-ca-certificates',\n          ];\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          return [\n            'update-ca-trust',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          const commands: string[] = [];\n          for (let i = 0; i < certificateFiles.length; i++) {\n            const certName = `${sanitizedName}-${i}`;\n            commands.push(\n              `Import-Certificate -FilePath C:\\\\${certName}.crt -CertStoreLocation Cert:\\\\LocalMachine\\\\Root`,\n              `Remove-Item C:\\\\${certName}.crt`,\n            );\n          }\n          return commands;\n        }\n\n        throw new Error(`Unknown os/architecture combo for extra certificates: ${os.name}/${architecture.name}`);\n      }\n\n      getAssets(os: Os, _architecture: Architecture): RunnerImageAsset[] {\n        const assets: RunnerImageAsset[] = [];\n\n        let targetDir: string;\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          targetDir = '/usr/local/share/ca-certificates/';\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          targetDir = '/etc/pki/ca-trust/source/anchors/';\n        } else if (os.is(Os.WINDOWS)) {\n          targetDir = 'C:\\\\';\n        } else {\n          throw new Error(`Unsupported OS for extra certificates: ${os.name}`);\n        }\n\n        for (let i = 0; i < certificateFiles.length; i++) {\n          const certName = `${sanitizedName}-${i}`;\n          assets.push({\n            source: certificateFiles[i],\n            target: `${targetDir}${certName}.crt`,\n          });\n        }\n\n        return assets;\n      }\n    }();\n  }\n\n  /**\n   * A component to set up the required Lambda entrypoint for Lambda runners.\n   */\n  static lambdaEntrypoint(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Lambda-Entrypoint';\n\n      getCommands(os: Os, _architecture: Architecture) {\n        if (!os.isIn(Os._ALL_LINUX_VERSIONS)) {\n          throw new Error(`Unsupported OS for Lambda entrypoint: ${os.name}`);\n        }\n\n        return [];\n      }\n\n      getAssets(_os: Os, _architecture: Architecture): RunnerImageAsset[] {\n        return [\n          {\n            source: path.join(__dirname, '..', '..', 'assets', 'providers', 'lambda-bootstrap.sh'),\n            target: '/bootstrap.sh',\n          },\n          {\n            source: path.join(__dirname, '..', '..', 'assets', 'providers', 'lambda-runner.sh'),\n            target: '/runner.sh',\n          },\n        ];\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n        return [\n          'LABEL DISABLE_SOCI=1', // hacky way to disable soci v2 indexing on lambda as lambda will fail to start with an index\n          'ENTRYPOINT [\"bash\", \"/bootstrap.sh\"]',\n        ];\n      }\n    };\n  }\n\n  /**\n   * A component to add environment variables for jobs the runner executes.\n   *\n   * These variables only affect the jobs ran by the runner. They are not global. They do not affect other components.\n   *\n   * It is not recommended to use this component to pass secrets. Instead, use GitHub Secrets or AWS Secrets Manager.\n   *\n   * Must be used after the {@link githubRunner} component.\n   */\n  static environmentVariables(vars: Record<string, string>): RunnerImageComponent {\n    Object.entries(vars).forEach(e => {\n      if (e[0].includes('\\n') || e[1].includes('\\n')) {\n        throw new Error(`Environment variable cannot contain newlines: ${e}`);\n      }\n    });\n\n    return new class extends RunnerImageComponent {\n      name = 'EnvironmentVariables';\n\n      getCommands(os: Os, _architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_VERSIONS)) {\n          return Object.entries(vars).map(e => `echo '${e[0]}=${e[1].replace(/'/g, \"'\\\"'\\\"'\")}' >> /home/runner/.env`);\n        } else if (os.is(Os.WINDOWS)) {\n          return Object.entries(vars).map(e => `Add-Content -Path C:\\\\actions\\\\.env -Value '${e[0]}=${e[1].replace(/'/g, \"''\")}'`);\n        } else {\n          throw new Error(`Unsupported OS for environment variables component: ${os.name}`);\n        }\n      }\n    };\n  }\n\n  /**\n   * Component name.\n   *\n   * Used to identify component in image build logs, and for {@link IConfigurableRunnerImageBuilder.removeComponent}\n   */\n  abstract readonly name: string;\n\n  /**\n   * Returns commands to run to in built image. Can be used to install packages, setup build prerequisites, etc.\n   */\n  abstract getCommands(_os: Os, _architecture: Architecture): string[];\n\n  /**\n   * Returns assets to copy into the built image. Can be used to copy files into the image.\n   */\n  getAssets(_os: Os, _architecture: Architecture): RunnerImageAsset[] {\n    return [];\n  }\n\n  /**\n   * Returns Docker commands to run to in built image. Can be used to add commands like `VOLUME`, `ENTRYPOINT`, `CMD`, etc.\n   *\n   * Docker commands are added after assets and normal commands.\n   */\n  getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n    return [];\n  }\n\n  /**\n   * Returns true if the image builder should be rebooted after this component is installed.\n   */\n  shouldReboot(_os: Os, _architecture: Architecture): boolean {\n    return false;\n  }\n\n  /**\n   * Convert component to an AWS Image Builder component.\n   *\n   * @internal\n   */\n  _asAwsImageBuilderComponent(scope: Construct, id: string, os: Os, architecture: Architecture) {\n    let platform: 'Linux' | 'Windows';\n    if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n      platform = 'Linux';\n    } else if (os.is(Os.WINDOWS)) {\n      platform = 'Windows';\n    } else {\n      throw new Error(`Unknown os/architecture combo for image builder component: ${os.name}/${architecture.name}`);\n    }\n\n    return new ImageBuilderComponent(scope, id, {\n      platform: platform,\n      commands: this.getCommands(os, architecture),\n      assets: this.getAssets(os, architecture).map((asset, index) => {\n        return {\n          asset: new s3_assets.Asset(scope, `${id} asset ${index}`, { path: asset.source }),\n          path: asset.target,\n        };\n      }),\n      displayName: id,\n      description: id,\n      reboot: this.shouldReboot(os, architecture),\n    });\n  }\n}\n\n"]}
692
+ RunnerImageComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerImageComponent", version: "0.14.21" };
693
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"components.js","sourceRoot":"","sources":["../../src/image-builders/components.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as path from 'path';\nimport * as cdk from 'aws-cdk-lib';\nimport { aws_s3_assets as s3_assets } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { discoverCertificateFiles } from '../utils';\nimport { ImageBuilderComponent } from './aws-image-builder';\nimport { RunnerImageAsset } from './common';\nimport { Architecture, Os, RunnerVersion } from '../providers';\n\nexport interface RunnerImageComponentCustomProps {\n  /**\n   * Component name used for (1) image build logging and (2) identifier for {@link IConfigurableRunnerImageBuilder.removeComponent}.\n   *\n   * Name must only contain alphanumeric characters and dashes.\n   */\n  readonly name?: string;\n\n  /**\n   * Commands to run in the built image.\n   */\n  readonly commands?: string[];\n\n  /**\n   * Assets to copy into the built image.\n   */\n  readonly assets?: RunnerImageAsset[];\n\n  /**\n   * Docker commands to run in the built image.\n   *\n   * For example: `['ENV foo=bar', 'RUN echo $foo']`\n   *\n   * These commands are ignored when building AMIs.\n   */\n  readonly dockerCommands?: string[];\n}\n\n/**\n * Components are used to build runner images. They can run commands in the image, copy files into the image, and run some Docker commands.\n */\nexport abstract class RunnerImageComponent {\n  /**\n   * Define a custom component that can run commands in the image, copy files into the image, and run some Docker commands.\n   *\n   * The order of operations is (1) assets (2) commands (3) docker commands.\n   *\n   * Use this to customize the image for the runner.\n   *\n   * **WARNING:** Docker commands are not guaranteed to be included before the next component\n   */\n  static custom(props: RunnerImageComponentCustomProps): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      get name() {\n        if (props.name && !props.name.match(/[a-zA-Z0-9\\-]/)) {\n          throw new Error(`Invalid component name: ${props.name}. Name must only contain alphanumeric characters and dashes.`);\n        }\n        return `Custom-${props.name ?? 'Undefined'}`;\n      }\n\n      getCommands(_os: Os, _architecture: Architecture) {\n        return props.commands ?? [];\n      }\n      getAssets(_os: Os, _architecture: Architecture) {\n        return props.assets ?? [];\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture) {\n        return props.dockerCommands ?? [];\n      }\n    }();\n  }\n\n  /**\n   * A component to install the required packages for the runner.\n   */\n  static requiredPackages(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'RequiredPackages';\n\n      getCommands(os: Os, _architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get upgrade -y',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y curl sudo jq bash zip unzip iptables software-properties-common ca-certificates',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum update -y',\n            'yum install -y jq tar gzip bzip2 which binutils zip unzip sudo shadow-utils',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf upgrade -y',\n            'dnf install -y jq tar gzip bzip2 which binutils zip unzip sudo shadow-utils findutils',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [];\n        }\n\n        throw new Error(`Unsupported OS for required packages: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to install CloudWatch Agent for the runner so we can send logs.\n   */\n  static cloudWatchAgent(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'CloudWatchAgent';\n\n      getCommands(os: Os, architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          let archUrl;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'amd64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'arm64';\n          } else {\n            throw new Error(`Unsupported architecture for required packages: ${architecture.name}`);\n          }\n\n          return [\n            `curl -sfLo /tmp/amazon-cloudwatch-agent.deb https://s3.amazonaws.com/amazoncloudwatch-agent/ubuntu/${archUrl}/latest/amazon-cloudwatch-agent.deb`,\n            'dpkg -i -E /tmp/amazon-cloudwatch-agent.deb',\n            'rm /tmp/amazon-cloudwatch-agent.deb',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum install -y amazon-cloudwatch-agent',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y amazon-cloudwatch-agent',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://s3.amazonaws.com/amazoncloudwatch-agent/windows/amd64/latest/amazon-cloudwatch-agent.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n          ];\n        }\n\n        throw new Error(`Unsupported OS for required packages: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to prepare the required runner user.\n   */\n  static runnerUser(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'RunnerUser';\n\n      getCommands(os: Os, _architecture: Architecture): string[] {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'addgroup runner',\n            'adduser --system --disabled-password --home /home/runner --ingroup runner runner',\n            'echo \"%runner   ALL=(ALL:ALL) NOPASSWD: ALL\" > /etc/sudoers.d/runner',\n          ];\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          return [\n            '/usr/sbin/groupadd runner',\n            '/usr/sbin/useradd --system --shell /usr/sbin/nologin --home-dir /home/runner --gid runner runner',\n            'mkdir -p /home/runner',\n            'chown runner /home/runner',\n            'echo \"%runner   ALL=(ALL:ALL) NOPASSWD: ALL\" > /etc/sudoers.d/runner',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [];\n        }\n\n        throw new Error(`Unsupported OS for runner user: ${os.name}`);\n      }\n    };\n  }\n\n  /**\n   * A component to install the AWS CLI.\n   */\n  static awsCli(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'AwsCli';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          let archUrl: string;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'x86_64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'aarch64';\n          } else {\n            throw new Error(`Unsupported architecture for awscli: ${architecture.name}`);\n          }\n\n          return [\n            `curl -fsSL \"https://awscli.amazonaws.com/awscli-exe-linux-${archUrl}.zip\" -o awscliv2.zip`,\n            'unzip -q awscliv2.zip',\n            './aws/install',\n            'rm -rf awscliv2.zip aws',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for awscli: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub CLI.\n   */\n  static githubCli(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'GithubCli';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'curl -fsSL https://cli.github.com/packages/githubcli-archive-keyring.gpg | sudo dd of=/usr/share/keyrings/githubcli-archive-keyring.gpg',\n            'echo \"deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/githubcli-archive-keyring.gpg] ' +\n            '  https://cli.github.com/packages stable main\" | sudo tee /etc/apt/sources.list.d/github-cli.list > /dev/null',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y gh',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'curl -fsSSL https://cli.github.com/packages/rpm/gh-cli.repo -o /etc/yum.repos.d/gh-cli.repo',\n            'yum install -y gh',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'curl -fsSSL https://cli.github.com/packages/rpm/gh-cli.repo -o /etc/yum.repos.d/gh-cli.repo',\n            'dnf install -y gh',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/cli/cli/releases/latest > $Env:TEMP\\\\latest-gh',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-gh',\n            '$GH_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/cli/cli/releases/download/v${GH_VERSION}/gh_${GH_VERSION}_windows_amd64.msi\" -OutFile gh.msi',\n            '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i gh.msi /qn\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n            'del gh.msi',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for github cli: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub CLI.\n   */\n  static git(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Git';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'add-apt-repository ppa:git-core/ppa',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y git',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'yum install -y git',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y git',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/git-for-windows/git/releases/latest > $Env:TEMP\\\\latest-git',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-git',\n            '$GIT_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            '$GIT_VERSION_SHORT = ($GIT_VERSION -Split \\'.windows.\\')[0]',\n            '$GIT_REVISION = ($GIT_VERSION -Split \\'.windows.\\')[1]',\n            'If ($GIT_REVISION -gt 1) {$GIT_VERSION_SHORT = \"$GIT_VERSION_SHORT.$GIT_REVISION\"}',\n            'Invoke-WebRequest -UseBasicParsing -Uri https://github.com/git-for-windows/git/releases/download/v${GIT_VERSION}/Git-${GIT_VERSION_SHORT}-64-bit.exe -OutFile git-setup.exe',\n            '$p = Start-Process git-setup.exe -PassThru -Wait -ArgumentList \\'/VERYSILENT\\'',\n            'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n            'del git-setup.exe',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for git: ${os.name}/${architecture.name}`);\n      }\n    }();\n  }\n\n  /**\n   * A component to install the GitHub Actions Runner. This is the actual executable that connects to GitHub to ask for jobs and then execute them.\n   *\n   * @param runnerVersion The version of the runner to install. Usually you would set this to latest.\n   */\n  static githubRunner(runnerVersion: RunnerVersion): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'GithubRunner';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          let versionCommand: string;\n          if (runnerVersion.is(RunnerVersion.latest())) {\n            versionCommand = 'RUNNER_VERSION=`curl -w \"%{redirect_url}\" -fsS https://github.com/actions/runner/releases/latest | grep -oE \"[^/v]+$\"`';\n          } else {\n            versionCommand = `RUNNER_VERSION='${runnerVersion.version}'`;\n          }\n\n          let archUrl;\n          if (architecture.is(Architecture.X86_64)) {\n            archUrl = 'x64';\n          } else if (architecture.is(Architecture.ARM64)) {\n            archUrl = 'arm64';\n          } else {\n            throw new Error(`Unsupported architecture for GitHub Runner: ${architecture.name}`);\n          }\n\n          let commands = [\n            versionCommand,\n            `curl -fsSLO \"https://github.com/actions/runner/releases/download/v\\${RUNNER_VERSION}/actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz\"`,\n            `tar -C /home/runner -xzf \"actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz\"`,\n            `rm actions-runner-linux-${archUrl}-\\${RUNNER_VERSION}.tar.gz`,\n            `echo -n ${runnerVersion.version} > /home/runner/RUNNER_VERSION`,\n          ];\n\n          if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n            commands.push('/home/runner/bin/installdependencies.sh');\n          } else if (os.is(Os.LINUX_AMAZON_2)) {\n            commands.push('yum install -y openssl-libs krb5-libs zlib libicu60');\n          } else if (os.is(Os.LINUX_AMAZON_2023)) {\n            commands.push('dnf install -y openssl-libs krb5-libs zlib libicu-67.1');\n          }\n\n          commands.push('mkdir -p /opt/hostedtoolcache', 'chown runner /opt/hostedtoolcache');\n\n          return commands;\n        } else if (os.is(Os.WINDOWS)) {\n          let runnerCommands: string[];\n          if (runnerVersion.is(RunnerVersion.latest())) {\n            runnerCommands = [\n              'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/actions/runner/releases/latest > $Env:TEMP\\\\latest-gha',\n              '$LatestUrl = Get-Content $Env:TEMP\\\\latest-gha',\n              '$RUNNER_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            ];\n          } else {\n            runnerCommands = [`$RUNNER_VERSION = '${runnerVersion.version}'`];\n          }\n\n          runnerCommands = runnerCommands.concat([\n            // create directories\n            'mkdir C:\\\\hostedtoolcache\\\\windows',\n            'mkdir C:\\\\tools',\n            // download zstd and extract to C:\\tools\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/facebook/zstd/releases/latest > $Env:TEMP\\\\latest-zstd',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-zstd',\n            '$ZSTD_VERSION = ($LatestUrl -Split \\'/\\')[-1].substring(1)',\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/facebook/zstd/releases/download/v$ZSTD_VERSION/zstd-v$ZSTD_VERSION-win64.zip\" -OutFile zstd.zip',\n            'Expand-Archive zstd.zip -DestinationPath C:\\\\tools',\n            'Move-Item -Path C:\\\\tools\\\\zstd-v$ZSTD_VERSION-win64\\\\zstd.exe C:\\\\tools',\n            'Remove-Item -LiteralPath \"C:\\\\tools\\\\zstd-v$ZSTD_VERSION-win64\" -Force -Recurse',\n            'del zstd.zip',\n            // add C:\\tools to PATH\n            '$persistedPaths = [Environment]::GetEnvironmentVariable(\\'Path\\', [EnvironmentVariableTarget]::Machine)',\n            '[Environment]::SetEnvironmentVariable(\"PATH\", $persistedPaths + \";C:\\\\tools\", [EnvironmentVariableTarget]::Machine)',\n          ]);\n\n          return runnerCommands.concat([\n            'Invoke-WebRequest -UseBasicParsing -Uri \"https://github.com/actions/runner/releases/download/v${RUNNER_VERSION}/actions-runner-win-x64-${RUNNER_VERSION}.zip\" -OutFile actions.zip',\n            'Expand-Archive actions.zip -DestinationPath C:\\\\actions',\n            'del actions.zip',\n            `echo ${runnerVersion.version} | Out-File -Encoding ASCII -NoNewline C:\\\\actions\\\\RUNNER_VERSION`,\n          ]);\n        }\n\n        throw new Error(`Unknown os/architecture combo for github runner: ${os.name}/${architecture.name}`);\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n        return [\n          `ENV RUNNER_VERSION=${runnerVersion.version}`,\n        ];\n      }\n    }();\n  }\n\n  /**\n   * A component to install Docker.\n   *\n   * On Windows this sets up dockerd for Windows containers without Docker Desktop. If you need Linux containers on Windows, you'll need to install Docker Desktop which doesn't seem to play well with servers (PRs welcome).\n   */\n  static docker(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Docker';\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker.gpg',\n            'echo ' +\n            '  \"deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker.gpg] https://download.docker.com/linux/ubuntu ' +\n            '  $(lsb_release -cs) stable\" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null',\n            'apt-get update',\n            'DEBIAN_FRONTEND=noninteractive apt-get install -y docker-ce docker-ce-cli containerd.io docker-compose-plugin',\n            'usermod -aG docker runner',\n            'ln -s /usr/libexec/docker/cli-plugins/docker-compose /usr/bin/docker-compose',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2)) {\n          return [\n            'amazon-linux-extras install docker',\n            'usermod -a -G docker runner',\n            'curl -sfLo /usr/bin/docker-compose https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s | tr \\'[:upper:]\\' \\'[:lower:]\\')-$(uname -m)',\n            'chmod +x /usr/bin/docker-compose',\n            'ln -s /usr/bin/docker-compose /usr/libexec/docker/cli-plugins/docker-compose',\n          ];\n        } else if (os.is(Os.LINUX_AMAZON_2023)) {\n          return [\n            'dnf install -y docker',\n            'usermod -a -G docker runner',\n            'curl -sfLo /usr/bin/docker-compose https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s | tr \\'[:upper:]\\' \\'[:lower:]\\')-$(uname -m)',\n            'chmod +x /usr/bin/docker-compose',\n            'ln -s /usr/bin/docker-compose /usr/libexec/docker/cli-plugins/docker-compose',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          return [\n            // figure out latest docker version\n            '$BaseUrl = \"https://download.docker.com/win/static/stable/x86_64/\"',\n            '$html = Invoke-WebRequest -UseBasicParsing -Uri $BaseUrl',\n            '$files = $html.Links.href | Where-Object { $_ -match \\'^docker-[0-9\\\\.]+\\\\.zip$\\' }',\n            'if (-not $files) { Write-Error \"No docker-*.zip files found.\" ; exit 1 }',\n            '$latest = $files | Sort-Object { try { [Version]($_ -replace \\'^docker-|\\\\.zip$\\') } catch { [Version]\"0.0.0\" } } -Descending | Select-Object -First 1',\n            // download static binaries\n            'Invoke-WebRequest -UseBasicParsing -Uri $BaseUrl$latest -OutFile docker.zip',\n            // extract to C:\\Program Files\\Docker\n            'Expand-Archive docker.zip -DestinationPath \"$Env:ProgramFiles\"',\n            'del docker.zip',\n            // add to path\n            '$persistedPaths = [Environment]::GetEnvironmentVariable(\\'Path\\', [EnvironmentVariableTarget]::Machine)',\n            '[Environment]::SetEnvironmentVariable(\"PATH\", $persistedPaths + \";$Env:ProgramFiles\\\\Docker\", [EnvironmentVariableTarget]::Machine)',\n            '$env:PATH = $env:PATH + \";$Env:ProgramFiles\\\\Docker\"',\n            // register docker service\n            'dockerd --register-service',\n            'if ($LASTEXITCODE -ne 0) { throw \"Exit code is $LASTEXITCODE\" }',\n            // enable containers feature\n            'Enable-WindowsOptionalFeature -Online -FeatureName containers -All -NoRestart',\n            // install docker-compose\n            'cmd /c curl -w \"%{redirect_url}\" -fsS https://github.com/docker/compose/releases/latest > $Env:TEMP\\\\latest-docker-compose',\n            '$LatestUrl = Get-Content $Env:TEMP\\\\latest-docker-compose',\n            '$LatestDockerCompose = ($LatestUrl -Split \\'/\\')[-1]',\n            'Invoke-WebRequest -UseBasicParsing -Uri  \"https://github.com/docker/compose/releases/download/${LatestDockerCompose}/docker-compose-Windows-x86_64.exe\" -OutFile $Env:ProgramFiles\\\\Docker\\\\docker-compose.exe',\n            'New-Item -ItemType directory -Path \"$Env:ProgramFiles\\\\Docker\\\\cli-plugins\"',\n            'Copy-Item -Path \"$Env:ProgramFiles\\\\Docker\\\\docker-compose.exe\" -Destination \"$Env:ProgramFiles\\\\Docker\\\\cli-plugins\\\\docker-compose.exe\"',\n          ];\n        }\n\n        throw new Error(`Unknown os/architecture combo for docker: ${os.name}/${architecture.name}`);\n      }\n\n      shouldReboot(os: Os, _architecture: Architecture): boolean {\n        return os.is(Os.WINDOWS);\n      }\n    }();\n  }\n\n  /**\n   * A component to install Docker-in-Docker.\n   *\n   * @deprecated use `docker()`\n   */\n  static dockerInDocker(): RunnerImageComponent {\n    return RunnerImageComponent.docker();\n  }\n\n  /**\n   * A component to add a trusted certificate authority. This can be used to support GitHub Enterprise Server with self-signed certificate.\n   *\n   * @param source path to certificate file in PEM format, or a directory containing certificate files (.pem or .crt)\n   * @param name unique certificate name to be used on runner file system\n   */\n  static extraCertificates(source: string, name: string): RunnerImageComponent {\n    // Sanitize the name to only contain alphanumeric characters, dashes and underscores\n    const sanitizedName = name.replace(/[^a-zA-Z0-9_-]/g, '-');\n\n    // Discover certificate files (supports both file and directory)\n    const certificateFiles = discoverCertificateFiles(source);\n\n    return new class extends RunnerImageComponent {\n      name = `Extra-Certificates-${sanitizedName}`;\n\n      getCommands(os: Os, architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          return [\n            'update-ca-certificates',\n          ];\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          return [\n            'update-ca-trust',\n          ];\n        } else if (os.is(Os.WINDOWS)) {\n          const commands: string[] = [];\n          for (let i = 0; i < certificateFiles.length; i++) {\n            const certName = `${sanitizedName}-${i}`;\n            commands.push(\n              `Import-Certificate -FilePath C:\\\\${certName}.crt -CertStoreLocation Cert:\\\\LocalMachine\\\\Root`,\n              `Remove-Item C:\\\\${certName}.crt`,\n            );\n          }\n          return commands;\n        }\n\n        throw new Error(`Unknown os/architecture combo for extra certificates: ${os.name}/${architecture.name}`);\n      }\n\n      getAssets(os: Os, _architecture: Architecture): RunnerImageAsset[] {\n        const assets: RunnerImageAsset[] = [];\n\n        let targetDir: string;\n        if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS)) {\n          targetDir = '/usr/local/share/ca-certificates/';\n        } else if (os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n          targetDir = '/etc/pki/ca-trust/source/anchors/';\n        } else if (os.is(Os.WINDOWS)) {\n          targetDir = 'C:\\\\';\n        } else {\n          throw new Error(`Unsupported OS for extra certificates: ${os.name}`);\n        }\n\n        for (let i = 0; i < certificateFiles.length; i++) {\n          const certName = `${sanitizedName}-${i}`;\n          assets.push({\n            source: certificateFiles[i],\n            target: `${targetDir}${certName}.crt`,\n          });\n        }\n\n        return assets;\n      }\n    }();\n  }\n\n  /**\n   * A component to set up the required Lambda entrypoint for Lambda runners.\n   */\n  static lambdaEntrypoint(): RunnerImageComponent {\n    return new class extends RunnerImageComponent {\n      name = 'Lambda-Entrypoint';\n\n      getCommands(os: Os, _architecture: Architecture) {\n        if (!os.isIn(Os._ALL_LINUX_VERSIONS)) {\n          throw new Error(`Unsupported OS for Lambda entrypoint: ${os.name}`);\n        }\n\n        return [];\n      }\n\n      getAssets(_os: Os, _architecture: Architecture): RunnerImageAsset[] {\n        return [\n          {\n            source: path.join(__dirname, '..', '..', 'assets', 'providers', 'lambda-bootstrap.sh'),\n            target: '/bootstrap.sh',\n          },\n          {\n            source: path.join(__dirname, '..', '..', 'assets', 'providers', 'lambda-runner.sh'),\n            target: '/runner.sh',\n          },\n        ];\n      }\n\n      getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n        return [\n          'LABEL DISABLE_SOCI=1', // hacky way to disable soci v2 indexing on lambda as lambda will fail to start with an index\n          'ENTRYPOINT [\"bash\", \"/bootstrap.sh\"]',\n        ];\n      }\n    };\n  }\n\n  /**\n   * A component to add environment variables for jobs the runner executes.\n   *\n   * These variables only affect the jobs ran by the runner. They are not global. They do not affect other components.\n   *\n   * It is not recommended to use this component to pass secrets. Instead, use GitHub Secrets or AWS Secrets Manager.\n   *\n   * Must be used after the {@link githubRunner} component.\n   */\n  static environmentVariables(vars: Record<string, string>): RunnerImageComponent {\n    Object.entries(vars).forEach(e => {\n      if (e[0].includes('\\n') || e[1].includes('\\n')) {\n        throw new Error(`Environment variable cannot contain newlines: ${e}`);\n      }\n    });\n\n    return new class extends RunnerImageComponent {\n      name = 'EnvironmentVariables';\n\n      getCommands(os: Os, _architecture: Architecture) {\n        if (os.isIn(Os._ALL_LINUX_VERSIONS)) {\n          return Object.entries(vars).map(e => `echo '${e[0]}=${e[1].replace(/'/g, \"'\\\"'\\\"'\")}' >> /home/runner/.env`);\n        } else if (os.is(Os.WINDOWS)) {\n          return Object.entries(vars).map(e => `Add-Content -Path C:\\\\actions\\\\.env -Value '${e[0]}=${e[1].replace(/'/g, \"''\")}'`);\n        } else {\n          throw new Error(`Unsupported OS for environment variables component: ${os.name}`);\n        }\n      }\n    };\n  }\n\n  /**\n   * Component name.\n   *\n   * Used to identify component in image build logs, and for {@link IConfigurableRunnerImageBuilder.removeComponent}\n   */\n  abstract readonly name: string;\n\n  /**\n   * Returns commands to run to in built image. Can be used to install packages, setup build prerequisites, etc.\n   */\n  abstract getCommands(_os: Os, _architecture: Architecture): string[];\n\n  /**\n   * Returns assets to copy into the built image. Can be used to copy files into the image.\n   */\n  getAssets(_os: Os, _architecture: Architecture): RunnerImageAsset[] {\n    return [];\n  }\n\n  /**\n   * Returns Docker commands to run to in built image. Can be used to add commands like `VOLUME`, `ENTRYPOINT`, `CMD`, etc.\n   *\n   * Docker commands are added after assets and normal commands.\n   */\n  getDockerCommands(_os: Os, _architecture: Architecture): string[] {\n    return [];\n  }\n\n  /**\n   * Returns true if the image builder should be rebooted after this component is installed.\n   */\n  shouldReboot(_os: Os, _architecture: Architecture): boolean {\n    return false;\n  }\n\n  /**\n   * Convert component to an AWS Image Builder component.\n   *\n   * Components are cached and reused when the same component is requested with the same\n   * OS and architecture, reducing stack template size and number of resources.\n   *\n   * @internal\n   */\n  _asAwsImageBuilderComponent(scope: Construct, os: Os, architecture: Architecture) {\n    let platform: 'Linux' | 'Windows';\n    if (os.isIn(Os._ALL_LINUX_UBUNTU_VERSIONS) || os.isIn(Os._ALL_LINUX_AMAZON_VERSIONS)) {\n      platform = 'Linux';\n    } else if (os.is(Os.WINDOWS)) {\n      platform = 'Windows';\n    } else {\n      throw new Error(`Unknown os/architecture combo for image builder component: ${os.name}/${architecture.name}`);\n    }\n\n    // Get component properties to create a cache key\n    const commands = this.getCommands(os, architecture);\n    const assets = this.getAssets(os, architecture);\n    const reboot = this.shouldReboot(os, architecture);\n\n    // Create a cache key based on component identity and properties\n    const stack = cdk.Stack.of(scope);\n    const cacheKey = this._getCacheKey(os, architecture, commands, assets, reboot);\n\n    // Create a consistent ID based on the cache key to ensure the same component\n    // always gets the same ID, regardless of the passed-in id parameter\n    // The cache key is already a hash, so we can use it directly\n    // Prefix with GHRInternal/ to avoid conflicts with user-defined constructs\n    const consistentId = `GHRInternal/Component-${this.name}-${os.name}-${architecture.name}-${cacheKey.substring(0, 10)}`.replace(/[^a-zA-Z0-9-/]/g, '-');\n\n    // Use the construct tree as the cache - check if component already exists in the stack\n    const existing = stack.node.tryFindChild(consistentId);\n    if (existing) {\n      // Component already exists in this stack, reuse it\n      return existing as ImageBuilderComponent;\n    }\n\n    // Create new component in the stack scope so it can be shared across all scopes in the same stack\n    const component = new ImageBuilderComponent(stack, consistentId, {\n      platform: platform,\n      commands: commands,\n      assets: assets.map((asset, index) => {\n        return {\n          asset: new s3_assets.Asset(stack, `GHRInternal/${consistentId}/Asset${index}`, { path: asset.source }),\n          path: asset.target,\n        };\n      }),\n      displayName: `${this.name} (${os.name}/${architecture.name})`,\n      description: `${this.name} component for ${os.name}/${architecture.name}`,\n      reboot: reboot,\n    });\n\n    return component;\n  }\n\n  /**\n   * Generate a cache key for component reuse.\n   * Components with the same name, OS, architecture, commands, assets, and reboot flag will share the same key.\n   * Returns a hash of all component properties to ensure uniqueness.\n   *\n   * @internal\n   */\n  private _getCacheKey(os: Os, architecture: Architecture, commands: string[], assets: RunnerImageAsset[], reboot: boolean): string {\n    // Create a hash of the component properties\n    const assetKeys = assets.map(a => `${a.source}:${a.target}`).sort().join('|');\n    const keyData = `${this.name}:${os.name}:${architecture.name}:${commands.join('\\n')}:${assetKeys}:${reboot}`;\n    return crypto.createHash('md5').update(keyData).digest('hex');\n  }\n}\n\n"]}
@@ -56,5 +56,5 @@ class StaticRunnerImage {
56
56
  }
57
57
  exports.StaticRunnerImage = StaticRunnerImage;
58
58
  _a = JSII_RTTI_SYMBOL_1;
59
- StaticRunnerImage[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.StaticRunnerImage", version: "0.14.20" };
59
+ StaticRunnerImage[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.StaticRunnerImage", version: "0.14.21" };
60
60
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGljLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2ltYWdlLWJ1aWxkZXJzL3N0YXRpYy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUVBLCtEQUFvRTtBQUNwRSwyQ0FBMEQ7QUFFMUQsNENBQXVGO0FBRXZGOztHQUVHO0FBQ0gsTUFBYSxpQkFBaUI7SUFDNUI7Ozs7Ozs7T0FPRztJQUNJLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxVQUEyQixFQUFFLE1BQWMsUUFBUSxFQUFFLFlBQVksR0FBRyx3QkFBWSxDQUFDLE1BQU0sRUFBRSxFQUFFLEdBQUcsY0FBRSxDQUFDLEtBQUs7UUFDcEksT0FBTztZQUNMLGVBQWU7Z0JBQ2IsT0FBTztvQkFDTCxlQUFlLEVBQUUsVUFBVTtvQkFDM0IsUUFBUSxFQUFFLEdBQUc7b0JBQ2IsWUFBWTtvQkFDWixFQUFFO29CQUNGLGFBQWEsRUFBRSx5QkFBYSxDQUFDLE1BQU0sRUFBRTtvQkFDckMsV0FBVyxFQUFFLFVBQVUsQ0FBQyxhQUFhO2lCQUN0QyxDQUFDO1lBQ0osQ0FBQztZQUVELE9BQU87Z0JBQ0wsTUFBTSxJQUFJLEtBQUssQ0FBQyxrREFBa0QsQ0FBQyxDQUFDO1lBQ3RFLENBQUM7U0FDRixDQUFDO0lBQ0osQ0FBQztJQUVEOzs7Ozs7Ozs7O09BVUc7SUFDSSxNQUFNLENBQUMsYUFBYSxDQUFDLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWEsRUFBRSxZQUFZLEdBQUcsd0JBQVksQ0FBQyxNQUFNLEVBQUUsRUFBRSxHQUFHLGNBQUUsQ0FBQyxLQUFLO1FBQ3hILE9BQU8sSUFBSSx1Q0FBMkIsQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2hELEVBQUU7WUFDRixZQUFZO1lBQ1osZUFBZSxFQUFFLCtCQUFrQixDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUM7U0FDdEQsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUE3Q0gsOENBOENDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgYXdzX2VjciBhcyBlY3IgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEJhc2VDb250YWluZXJJbWFnZSB9IGZyb20gJy4vYXdzLWltYWdlLWJ1aWxkZXIvYmFzZS1pbWFnZSc7XG5pbXBvcnQgeyBDb2RlQnVpbGRSdW5uZXJJbWFnZUJ1aWxkZXIgfSBmcm9tICcuL2NvZGVidWlsZCc7XG5pbXBvcnQgeyBJUnVubmVySW1hZ2VCdWlsZGVyIH0gZnJvbSAnLi9jb21tb24nO1xuaW1wb3J0IHsgQXJjaGl0ZWN0dXJlLCBPcywgUnVubmVyQW1pLCBSdW5uZXJJbWFnZSwgUnVubmVyVmVyc2lvbiB9IGZyb20gJy4uL3Byb3ZpZGVycyc7XG5cbi8qKlxuICogSGVscGVyIGNsYXNzIHdpdGggbWV0aG9kcyB0byB1c2Ugc3RhdGljIGltYWdlcyB0aGF0IGFyZSBidWlsdCBvdXRzaWRlIHRoZSBjb250ZXh0IG9mIHRoaXMgcHJvamVjdC5cbiAqL1xuZXhwb3J0IGNsYXNzIFN0YXRpY1J1bm5lckltYWdlIHtcbiAgLyoqXG4gICAqIENyZWF0ZSBhIGJ1aWxkZXIgKHRoYXQgZG9lc24ndCBhY3R1YWxseSBidWlsZCBhbnl0aGluZykgZnJvbSBhbiBleGlzdGluZyBpbWFnZSBpbiBhbiBleGlzdGluZyByZXBvc2l0b3J5LiBUaGUgaW1hZ2UgbXVzdCBhbHJlYWR5IGhhdmUgR2l0SHViIEFjdGlvbnMgcnVubmVyIGluc3RhbGxlZC4gWW91IGFyZSByZXNwb25zaWJsZSB0byB1cGRhdGUgaXQgYW5kIHJlbW92ZSBpdCB3aGVuIGRvbmUuXG4gICAqXG4gICAqIEBwYXJhbSByZXBvc2l0b3J5IEVDUiByZXBvc2l0b3J5XG4gICAqIEBwYXJhbSB0YWcgaW1hZ2UgdGFnXG4gICAqIEBwYXJhbSBhcmNoaXRlY3R1cmUgaW1hZ2UgYXJjaGl0ZWN0dXJlXG4gICAqIEBwYXJhbSBvcyBpbWFnZSBPU1xuICAgKi9cbiAgcHVibGljIHN0YXRpYyBmcm9tRWNyUmVwb3NpdG9yeShyZXBvc2l0b3J5OiBlY3IuSVJlcG9zaXRvcnksIHRhZzogc3RyaW5nID0gJ2xhdGVzdCcsIGFyY2hpdGVjdHVyZSA9IEFyY2hpdGVjdHVyZS5YODZfNjQsIG9zID0gT3MuTElOVVgpOiBJUnVubmVySW1hZ2VCdWlsZGVyIHtcbiAgICByZXR1cm4ge1xuICAgICAgYmluZERvY2tlckltYWdlKCk6IFJ1bm5lckltYWdlIHtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBpbWFnZVJlcG9zaXRvcnk6IHJlcG9zaXRvcnksXG4gICAgICAgICAgaW1hZ2VUYWc6IHRhZyxcbiAgICAgICAgICBhcmNoaXRlY3R1cmUsXG4gICAgICAgICAgb3MsXG4gICAgICAgICAgcnVubmVyVmVyc2lvbjogUnVubmVyVmVyc2lvbi5sYXRlc3QoKSxcbiAgICAgICAgICBfZGVwZW5kYWJsZTogcmVwb3NpdG9yeS5yZXBvc2l0b3J5QXJuLFxuICAgICAgICB9O1xuICAgICAgfSxcblxuICAgICAgYmluZEFtaSgpOiBSdW5uZXJBbWkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ2Zyb21FY3JSZXBvc2l0b3J5KCkgY2Fubm90IGJlIHVzZWQgdG8gYnVpbGQgQU1JcycpO1xuICAgICAgfSxcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBhIGJ1aWxkZXIgZnJvbSBhbiBleGlzdGluZyBEb2NrZXIgSHViIGltYWdlLiBUaGUgaW1hZ2UgbXVzdCBhbHJlYWR5IGhhdmUgR2l0SHViIEFjdGlvbnMgcnVubmVyIGluc3RhbGxlZC4gWW91IGFyZSByZXNwb25zaWJsZSB0byB1cGRhdGUgaXQgYW5kIHJlbW92ZSBpdCB3aGVuIGRvbmUuXG4gICAqXG4gICAqIFdlIGNyZWF0ZSBhIENvZGVCdWlsZCBpbWFnZSBidWlsZGVyIGJlaGluZCB0aGUgc2NlbmVzIHRvIGNvcHkgdGhlIGltYWdlIG92ZXIgdG8gRUNSLiBUaGlzIGhlbHBzIGF2b2lkIERvY2tlciBIdWIgcmF0ZSBsaW1pdHMgYW5kIHByZXZlbnQgZmFpbHVyZXMuXG4gICAqXG4gICAqIEBwYXJhbSBzY29wZVxuICAgKiBAcGFyYW0gaWRcbiAgICogQHBhcmFtIGltYWdlIERvY2tlciBIdWIgaW1hZ2Ugd2l0aCBvcHRpb25hbCB0YWdcbiAgICogQHBhcmFtIGFyY2hpdGVjdHVyZSBpbWFnZSBhcmNoaXRlY3R1cmVcbiAgICogQHBhcmFtIG9zIGltYWdlIE9TXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGZyb21Eb2NrZXJIdWIoc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgaW1hZ2U6IHN0cmluZywgYXJjaGl0ZWN0dXJlID0gQXJjaGl0ZWN0dXJlLlg4Nl82NCwgb3MgPSBPcy5MSU5VWCk6IElSdW5uZXJJbWFnZUJ1aWxkZXIge1xuICAgIHJldHVybiBuZXcgQ29kZUJ1aWxkUnVubmVySW1hZ2VCdWlsZGVyKHNjb3BlLCBpZCwge1xuICAgICAgb3MsXG4gICAgICBhcmNoaXRlY3R1cmUsXG4gICAgICBiYXNlRG9ja2VySW1hZ2U6IEJhc2VDb250YWluZXJJbWFnZS5mcm9tU3RyaW5nKGltYWdlKSxcbiAgICB9KTtcbiAgfVxufVxuIl19
@@ -1,4 +1,10 @@
1
- import { Octokit } from '@octokit/rest';
1
+ import type { Octokit as RestOctokit } from '@octokit/rest';
2
+ type OctokitRestModule = typeof import('@octokit/rest');
3
+ type OctokitCoreModule = typeof import('@octokit/core');
4
+ type OctokitAuthAppModule = typeof import('@octokit/auth-app');
5
+ export declare function loadOctokitRest(): Promise<OctokitRestModule>;
6
+ export declare function loadOctokitCore(): Promise<OctokitCoreModule>;
7
+ export declare function loadOctokitAuthApp(): Promise<OctokitAuthAppModule>;
2
8
  export declare function baseUrlFromDomain(domain: string): string;
3
9
  type RunnerLevel = 'repo' | 'org' | undefined;
4
10
  export interface GitHubSecrets {
@@ -8,13 +14,13 @@ export interface GitHubSecrets {
8
14
  runnerLevel: RunnerLevel;
9
15
  }
10
16
  export declare function getOctokit(installationId?: number): Promise<{
11
- octokit: Octokit;
17
+ octokit: RestOctokit;
12
18
  githubSecrets: GitHubSecrets;
13
19
  }>;
14
- export declare function getAppOctokit(): Promise<(import("@octokit/core").Octokit & {
20
+ export declare function getAppOctokit(): Promise<(import("@octokit/core").Octokit & import("@octokit/plugin-rest-endpoint-methods/dist-types/generated/method-types").RestEndpointMethods & import("@octokit/plugin-rest-endpoint-methods").Api & {
15
21
  paginate: import("@octokit/plugin-paginate-rest").PaginateInterface;
16
- } & import("@octokit/plugin-rest-endpoint-methods/dist-types/generated/method-types").RestEndpointMethods & import("@octokit/plugin-rest-endpoint-methods/dist-types/types").Api) | undefined>;
17
- export declare function getRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, name: string): Promise<{
22
+ }) | undefined>;
23
+ export declare function getRunner(octokit: RestOctokit, runnerLevel: RunnerLevel, owner: string, repo: string, name: string): Promise<{
18
24
  id: number;
19
25
  runner_group_id?: number;
20
26
  name: string;
@@ -22,7 +28,8 @@ export declare function getRunner(octokit: Octokit, runnerLevel: RunnerLevel, ow
22
28
  status: string;
23
29
  busy: boolean;
24
30
  labels: import("@octokit/openapi-types").components["schemas"]["runner-label"][];
31
+ ephemeral?: boolean;
25
32
  } | undefined>;
26
- export declare function deleteRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, runnerId: number): Promise<void>;
27
- export declare function redeliver(octokit: Octokit, deliveryId: number): Promise<void>;
33
+ export declare function deleteRunner(octokit: RestOctokit, runnerLevel: RunnerLevel, owner: string, repo: string, runnerId: number): Promise<void>;
34
+ export declare function redeliver(octokit: RestOctokit, deliveryId: number): Promise<void>;
28
35
  export {};
@@ -1,5 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.loadOctokitRest = loadOctokitRest;
4
+ exports.loadOctokitCore = loadOctokitCore;
5
+ exports.loadOctokitAuthApp = loadOctokitAuthApp;
3
6
  exports.baseUrlFromDomain = baseUrlFromDomain;
4
7
  exports.getOctokit = getOctokit;
5
8
  exports.getAppOctokit = getAppOctokit;
@@ -7,9 +10,20 @@ exports.getRunner = getRunner;
7
10
  exports.deleteRunner = deleteRunner;
8
11
  exports.redeliver = redeliver;
9
12
  const crypto_1 = require("crypto");
10
- const auth_app_1 = require("@octokit/auth-app");
11
- const rest_1 = require("@octokit/rest");
12
13
  const lambda_helpers_1 = require("./lambda-helpers");
14
+ let restModulePromise;
15
+ let coreModulePromise;
16
+ let authAppModulePromise;
17
+ function loadOctokitRest() {
18
+ return (restModulePromise ?? (restModulePromise = Promise.resolve().then(() => require('@octokit/rest'))));
19
+ }
20
+ function loadOctokitCore() {
21
+ return (coreModulePromise ?? (coreModulePromise = Promise.resolve().then(() => require('@octokit/core'))));
22
+ }
23
+ function loadOctokitAuthApp() {
24
+ return (authAppModulePromise ?? (authAppModulePromise = Promise.resolve().then(() => require('@octokit/auth-app'))));
25
+ }
26
+ // ---- Other helpers ----
13
27
  function baseUrlFromDomain(domain) {
14
28
  if (domain == 'github.com') {
15
29
  return 'https://api.github.com';
@@ -21,6 +35,10 @@ async function getOctokit(installationId) {
21
35
  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {
22
36
  throw new Error('Missing environment variables');
23
37
  }
38
+ const [{ Octokit }, { createAppAuth }] = await Promise.all([
39
+ loadOctokitRest(),
40
+ loadOctokitAuthApp(),
41
+ ]);
24
42
  const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
25
43
  // Create cache key from installation ID and secrets (hash to avoid exposing sensitive data by accident)
26
44
  const cacheKey = (0, crypto_1.createHash)('sha256').update(`${installationId || 'no-install'}-${githubSecrets.domain}-${githubSecrets.appId}-${githubSecrets.personalAuthToken}`).digest('hex');
@@ -52,9 +70,9 @@ async function getOctokit(installationId) {
52
70
  }
53
71
  else {
54
72
  const privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
55
- const appOctokit = new rest_1.Octokit({
73
+ const appOctokit = new Octokit({
56
74
  baseUrl,
57
- authStrategy: auth_app_1.createAppAuth,
75
+ authStrategy: createAppAuth,
58
76
  auth: {
59
77
  appId: githubSecrets.appId,
60
78
  privateKey: privateKey,
@@ -65,7 +83,7 @@ async function getOctokit(installationId) {
65
83
  installationId: installationId,
66
84
  })).token;
67
85
  }
68
- const octokit = new rest_1.Octokit({
86
+ const octokit = new Octokit({
69
87
  baseUrl,
70
88
  auth: token,
71
89
  });
@@ -82,15 +100,19 @@ async function getAppOctokit() {
82
100
  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {
83
101
  throw new Error('Missing environment variables');
84
102
  }
103
+ const [{ Octokit }, { createAppAuth }] = await Promise.all([
104
+ loadOctokitRest(),
105
+ loadOctokitAuthApp(),
106
+ ]);
85
107
  const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
86
108
  const baseUrl = baseUrlFromDomain(githubSecrets.domain);
87
109
  if (githubSecrets.personalAuthToken || !githubSecrets.appId) {
88
110
  return undefined;
89
111
  }
90
112
  const privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
91
- return new rest_1.Octokit({
113
+ return new Octokit({
92
114
  baseUrl,
93
- authStrategy: auth_app_1.createAppAuth,
115
+ authStrategy: createAppAuth,
94
116
  auth: {
95
117
  appId: githubSecrets.appId,
96
118
  privateKey: privateKey,
@@ -103,6 +125,7 @@ async function getRunner(octokit, runnerLevel, owner, repo, name) {
103
125
  let runners;
104
126
  if ((runnerLevel ?? 'repo') === 'repo') {
105
127
  runners = await octokit.rest.actions.listSelfHostedRunnersForRepo({
128
+ name: name,
106
129
  page: page,
107
130
  owner: owner,
108
131
  repo: repo,
@@ -110,6 +133,7 @@ async function getRunner(octokit, runnerLevel, owner, repo, name) {
110
133
  }
111
134
  else {
112
135
  runners = await octokit.rest.actions.listSelfHostedRunnersForOrg({
136
+ name: name,
113
137
  page: page,
114
138
  org: owner,
115
139
  });
@@ -118,6 +142,8 @@ async function getRunner(octokit, runnerLevel, owner, repo, name) {
118
142
  return;
119
143
  }
120
144
  for (const runner of runners.data.runners) {
145
+ // we filter by name in the API call, but still double-check here
146
+ // this is for backward compatibility with old GHES instances that may not support the name filter
121
147
  if (runner.name == name) {
122
148
  return runner;
123
149
  }
@@ -152,4 +178,4 @@ async function redeliver(octokit, deliveryId) {
152
178
  deliveryId,
153
179
  });
154
180
  }
155
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-github.js","sourceRoot":"","sources":["../src/lambda-github.ts"],"names":[],"mappings":"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","sourcesContent":["import { createHash } from 'crypto';\nimport { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/rest';\nimport { getSecretJsonValue, getSecretValue } from './lambda-helpers';\n\nexport function baseUrlFromDomain(domain: string): string {\n  if (domain == 'github.com') {\n    return 'https://api.github.com';\n  }\n  return `https://${domain}/api/v3`;\n}\n\ntype RunnerLevel = 'repo' | 'org' | undefined; // undefined is for backwards compatibility and should be treated as 'repo'\n\nexport interface GitHubSecrets {\n  domain: string;\n  appId: number;\n  personalAuthToken: string;\n  runnerLevel: RunnerLevel;\n}\n\nconst octokitCache = new Map<string, Octokit>();\n\nexport async function getOctokit(installationId?: number): Promise<{ octokit: Octokit; githubSecrets: GitHubSecrets }> {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n\n  // Create cache key from installation ID and secrets (hash to avoid exposing sensitive data by accident)\n  const cacheKey = createHash('sha256').update(`${installationId || 'no-install'}-${githubSecrets.domain}-${githubSecrets.appId}-${githubSecrets.personalAuthToken}`).digest('hex');\n\n  const cached = octokitCache.get(cacheKey);\n  if (cached) {\n    try {\n      // Test if the cached octokit is still valid\n      await cached.rest.meta.getOctocat();\n      console.log({\n        notice: 'Using cached octokit',\n      });\n      return {\n        octokit: cached,\n        githubSecrets,\n      };\n    } catch (e) {\n      console.log({\n        notice: 'Octokit cache is invalid',\n        error: e,\n      });\n      octokitCache.delete(cacheKey);\n    }\n  }\n\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  let token;\n  if (githubSecrets.personalAuthToken) {\n    token = githubSecrets.personalAuthToken;\n  } else {\n    const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n    const appOctokit = new Octokit({\n      baseUrl,\n      authStrategy: createAppAuth,\n      auth: {\n        appId: githubSecrets.appId,\n        privateKey: privateKey,\n      },\n    });\n\n    token = (await appOctokit.auth({\n      type: 'installation',\n      installationId: installationId,\n    }) as any).token;\n  }\n\n  const octokit = new Octokit({\n    baseUrl,\n    auth: token,\n  });\n\n  // Store in cache\n  octokitCache.set(cacheKey, octokit);\n\n  return {\n    octokit,\n    githubSecrets,\n  };\n}\n\n// This function is used to get the Octokit instance for the app itself, not for a specific installation.\n// With PAT authentication, it returns undefined.\nexport async function getAppOctokit() {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  if (githubSecrets.personalAuthToken || !githubSecrets.appId) {\n    return undefined;\n  }\n\n  const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n  return new Octokit({\n    baseUrl,\n    authStrategy: createAppAuth,\n    auth: {\n      appId: githubSecrets.appId,\n      privateKey: privateKey,\n    },\n  });\n}\n\nexport async function getRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, name: string) {\n  let page = 1;\n  while (true) {\n    let runners;\n\n    if ((runnerLevel ?? 'repo') === 'repo') {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForRepo({\n        page: page,\n        owner: owner,\n        repo: repo,\n      });\n    } else {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForOrg({\n        page: page,\n        org: owner,\n      });\n    }\n\n    if (runners.data.runners.length == 0) {\n      return;\n    }\n\n    for (const runner of runners.data.runners) {\n      if (runner.name == name) {\n        return runner;\n      }\n    }\n\n    page++;\n  }\n}\n\nexport async function deleteRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, runnerId: number) {\n  if ((runnerLevel ?? 'repo') === 'repo') {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromRepo({\n      owner: owner,\n      repo: repo,\n      runner_id: runnerId,\n    });\n  } else {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromOrg({\n      org: owner,\n      runner_id: runnerId,\n    });\n  }\n}\n\nexport async function redeliver(octokit: Octokit, deliveryId: number) {\n  const response = await octokit.rest.apps.redeliverWebhookDelivery({\n    delivery_id: deliveryId,\n  });\n\n  if (response.status !== 202) {\n    throw new Error(`Failed to redeliver webhook delivery with ID ${deliveryId}`);\n  }\n  console.log({\n    notice: 'Successfully redelivered webhook delivery',\n    deliveryId,\n  });\n}\n"]}
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-github.js","sourceRoot":"","sources":["../src/lambda-github.ts"],"names":[],"mappings":"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","sourcesContent":["import { createHash } from 'crypto';\nimport type { Octokit as RestOctokit } from '@octokit/rest';\nimport { getSecretJsonValue, getSecretValue } from './lambda-helpers';\n\n// ---- Octokit ESM loader helpers (inlined) ----\n// Octokit packages are ESM, but our Lambda assets are bundled into CJS.\n// Using dynamic `import()` here lets esbuild include Octokit in the bundle.\ntype OctokitRestModule = typeof import('@octokit/rest');\ntype OctokitCoreModule = typeof import('@octokit/core');\ntype OctokitAuthAppModule = typeof import('@octokit/auth-app');\n\nlet restModulePromise: Promise<OctokitRestModule> | undefined;\nlet coreModulePromise: Promise<OctokitCoreModule> | undefined;\nlet authAppModulePromise: Promise<OctokitAuthAppModule> | undefined;\n\nexport function loadOctokitRest(): Promise<OctokitRestModule> {\n  return (restModulePromise ??= import('@octokit/rest') as Promise<OctokitRestModule>);\n}\n\nexport function loadOctokitCore(): Promise<OctokitCoreModule> {\n  return (coreModulePromise ??= import('@octokit/core') as Promise<OctokitCoreModule>);\n}\n\nexport function loadOctokitAuthApp(): Promise<OctokitAuthAppModule> {\n  return (authAppModulePromise ??= import('@octokit/auth-app') as Promise<OctokitAuthAppModule>);\n}\n\n// ---- Other helpers ----\n\nexport function baseUrlFromDomain(domain: string): string {\n  if (domain == 'github.com') {\n    return 'https://api.github.com';\n  }\n  return `https://${domain}/api/v3`;\n}\n\ntype RunnerLevel = 'repo' | 'org' | undefined; // undefined is for backwards compatibility and should be treated as 'repo'\n\nexport interface GitHubSecrets {\n  domain: string;\n  appId: number;\n  personalAuthToken: string;\n  runnerLevel: RunnerLevel;\n}\n\nconst octokitCache = new Map<string, RestOctokit>();\n\nexport async function getOctokit(installationId?: number): Promise<{ octokit: RestOctokit; githubSecrets: GitHubSecrets }> {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const [{ Octokit }, { createAppAuth }] = await Promise.all([\n    loadOctokitRest(),\n    loadOctokitAuthApp(),\n  ]);\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n\n  // Create cache key from installation ID and secrets (hash to avoid exposing sensitive data by accident)\n  const cacheKey = createHash('sha256').update(`${installationId || 'no-install'}-${githubSecrets.domain}-${githubSecrets.appId}-${githubSecrets.personalAuthToken}`).digest('hex');\n\n  const cached = octokitCache.get(cacheKey);\n  if (cached) {\n    try {\n      // Test if the cached octokit is still valid\n      await cached.rest.meta.getOctocat();\n      console.log({\n        notice: 'Using cached octokit',\n      });\n      return {\n        octokit: cached,\n        githubSecrets,\n      };\n    } catch (e) {\n      console.log({\n        notice: 'Octokit cache is invalid',\n        error: e,\n      });\n      octokitCache.delete(cacheKey);\n    }\n  }\n\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  let token;\n  if (githubSecrets.personalAuthToken) {\n    token = githubSecrets.personalAuthToken;\n  } else {\n    const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n    const appOctokit = new Octokit({\n      baseUrl,\n      authStrategy: createAppAuth,\n      auth: {\n        appId: githubSecrets.appId,\n        privateKey: privateKey,\n      },\n    });\n\n    token = (await appOctokit.auth({\n      type: 'installation',\n      installationId: installationId,\n    }) as any).token;\n  }\n\n  const octokit = new Octokit({\n    baseUrl,\n    auth: token,\n  });\n\n  // Store in cache\n  octokitCache.set(cacheKey, octokit);\n\n  return {\n    octokit,\n    githubSecrets,\n  };\n}\n\n// This function is used to get the Octokit instance for the app itself, not for a specific installation.\n// With PAT authentication, it returns undefined.\nexport async function getAppOctokit() {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const [{ Octokit }, { createAppAuth }] = await Promise.all([\n    loadOctokitRest(),\n    loadOctokitAuthApp(),\n  ]);\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  if (githubSecrets.personalAuthToken || !githubSecrets.appId) {\n    return undefined;\n  }\n\n  const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n  return new Octokit({\n    baseUrl,\n    authStrategy: createAppAuth,\n    auth: {\n      appId: githubSecrets.appId,\n      privateKey: privateKey,\n    },\n  });\n}\n\nexport async function getRunner(octokit: RestOctokit, runnerLevel: RunnerLevel, owner: string, repo: string, name: string) {\n  let page = 1;\n  while (true) {\n    let runners;\n\n    if ((runnerLevel ?? 'repo') === 'repo') {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForRepo({\n        name: name,\n        page: page,\n        owner: owner,\n        repo: repo,\n      });\n    } else {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForOrg({\n        name: name,\n        page: page,\n        org: owner,\n      });\n    }\n\n    if (runners.data.runners.length == 0) {\n      return;\n    }\n\n    for (const runner of runners.data.runners) {\n      // we filter by name in the API call, but still double-check here\n      // this is for backward compatibility with old GHES instances that may not support the name filter\n      if (runner.name == name) {\n        return runner;\n      }\n    }\n\n    page++;\n  }\n}\n\nexport async function deleteRunner(octokit: RestOctokit, runnerLevel: RunnerLevel, owner: string, repo: string, runnerId: number) {\n  if ((runnerLevel ?? 'repo') === 'repo') {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromRepo({\n      owner: owner,\n      repo: repo,\n      runner_id: runnerId,\n    });\n  } else {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromOrg({\n      org: owner,\n      runner_id: runnerId,\n    });\n  }\n}\n\nexport async function redeliver(octokit: RestOctokit, deliveryId: number) {\n  const response = await octokit.rest.apps.redeliverWebhookDelivery({\n    delivery_id: deliveryId,\n  });\n\n  if (response.status !== 202) {\n    throw new Error(`Failed to redeliver webhook delivery with ID ${deliveryId}`);\n  }\n  console.log({\n    notice: 'Successfully redelivered webhook delivery',\n    deliveryId,\n  });\n}\n"]}