@cloudsnorkel/cdk-github-runners
Version:
CDK construct to create GitHub Actions self-hosted runners. Creates ephemeral runners on demand. Easy to deploy and highly customizable.
45 lines • 11.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.GithubWebhookHandler = void 0;
const cdk = require("aws-cdk-lib");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
const access_1 = require("./access");
const utils_1 = require("./utils");
const webhook_handler_function_1 = require("./webhook-handler-function");
/**
* Create a Lambda with a public URL to handle GitHub webhook events. After validating the event with the given secret, the orchestrator step function is called with information about the workflow job.
*
* @internal
*/
class GithubWebhookHandler extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.handler = new webhook_handler_function_1.WebhookHandlerFunction(this, 'webhook-handler', {
description: 'Handle GitHub webhook and start runner orchestrator',
environment: {
STEP_FUNCTION_ARN: props.orchestrator.stateMachineArn,
WEBHOOK_SECRET_ARN: props.secrets.webhook.secretArn,
GITHUB_SECRET_ARN: props.secrets.github.secretArn,
GITHUB_PRIVATE_KEY_SECRET_ARN: props.secrets.githubPrivateKey.secretArn,
PROVIDERS: JSON.stringify(props.providers),
REQUIRE_SELF_HOSTED_LABEL: props.requireSelfHostedLabel ? '1' : '0',
PROVIDER_SELECTOR_ARN: props.providerSelector?.functionArn ?? '',
...props.extraLambdaEnv,
},
timeout: cdk.Duration.seconds(31),
logGroup: (0, utils_1.singletonLogGroup)(this, utils_1.SingletonLogType.ORCHESTRATOR),
loggingFormat: aws_cdk_lib_1.aws_lambda.LoggingFormat.JSON,
...props.extraLambdaProps,
});
const access = props?.access ?? access_1.LambdaAccess.lambdaUrl();
this.url = access.bind(this, 'access', this.handler);
props.secrets.webhook.grantRead(this.handler);
props.secrets.github.grantRead(this.handler);
props.secrets.githubPrivateKey.grantRead(this.handler);
props.orchestrator.grantStartExecution(this.handler);
props.providerSelector?.grantInvoke(this.handler);
}
}
exports.GithubWebhookHandler = GithubWebhookHandler;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhook.js","sourceRoot":"","sources":["../src/webhook.ts"],"names":[],"mappings":";;;AAAA,mCAAmC;AACnC,6CAAuF;AACvF,2CAAuC;AACvC,qCAAwC;AAExC,mCAA8D;AAC9D,yEAAoE;AAuGpE;;;;GAIG;AACH,MAAa,oBAAqB,SAAQ,sBAAS;IAYjD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAgC;QACxE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,OAAO,GAAG,IAAI,iDAAsB,CACvC,IAAI,EACJ,iBAAiB,EACjB;YACE,WAAW,EAAE,qDAAqD;YAClE,WAAW,EAAE;gBACX,iBAAiB,EAAE,KAAK,CAAC,YAAY,CAAC,eAAe;gBACrD,kBAAkB,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS;gBACnD,iBAAiB,EAAE,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS;gBACjD,6BAA6B,EAAE,KAAK,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS;gBACvE,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC;gBAC1C,yBAAyB,EAAE,KAAK,CAAC,sBAAsB,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG;gBACnE,qBAAqB,EAAE,KAAK,CAAC,gBAAgB,EAAE,WAAW,IAAI,EAAE;gBAChE,GAAG,KAAK,CAAC,cAAc;aACxB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,QAAQ,EAAE,IAAA,yBAAiB,EAAC,IAAI,EAAE,wBAAgB,CAAC,YAAY,CAAC;YAChE,aAAa,EAAE,wBAAM,CAAC,aAAa,CAAC,IAAI;YACxC,GAAG,KAAK,CAAC,gBAAgB;SAC1B,CACF,CAAC;QAEF,MAAM,MAAM,GAAG,KAAK,EAAE,MAAM,IAAI,qBAAY,CAAC,SAAS,EAAE,CAAC;QACzD,IAAI,CAAC,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAErD,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC9C,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC7C,KAAK,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACvD,KAAK,CAAC,YAAY,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACrD,KAAK,CAAC,gBAAgB,EAAE,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpD,CAAC;CACF;AA9CD,oDA8CC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { aws_lambda as lambda, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { LambdaAccess } from './access';\nimport { Secrets } from './secrets';\nimport { singletonLogGroup, SingletonLogType } from './utils';\nimport { WebhookHandlerFunction } from './webhook-handler-function';\n\n/**\n * Input to the provider selector Lambda function.\n */\nexport interface ProviderSelectorInput {\n  /**\n   * Full GitHub webhook payload (workflow_job event structure with action=\"queued\").\n   *\n   * * Original labels requested by the workflow job can be found at `payload.workflow_job.labels`.\n   * * Repository path (e.g. CloudSnorkel/cdk-github-runners) is at `payload.repository.full_name`.\n   * * Commit hash is at `payload.workflow_job.head_sha`.\n   *\n   * @see https://docs.github.com/en/webhooks/webhook-events-and-payloads?actionType=queued#workflow_job\n   */\n  readonly payload: any;\n\n  /**\n   * Map of available provider node paths to their configured labels.\n   * Example: { \"MyStack/Small\": [\"linux\", \"small\"], \"MyStack/Large\": [\"linux\", \"large\"] }\n   */\n  readonly providers: Record<string, string[]>;\n\n  /**\n   * Provider node path that would have been selected by default label matching.\n   * Use this to easily return the default selection: `{ provider: input.defaultProvider, labels: input.defaultLabels }`\n   * May be undefined if no provider matched by default.\n   */\n  readonly defaultProvider?: string;\n\n  /**\n   * Labels that would have been used by default (the selected provider's labels).\n   * May be undefined if no provider matched by default.\n   */\n  readonly defaultLabels?: string[];\n}\n\n/**\n * Result from the provider selector Lambda function.\n */\nexport interface ProviderSelectorResult {\n  /**\n   * Node path of the provider to use (e.g., \"MyStack/MyProvider\").\n   * Must match one of the configured provider node paths from the input.\n   * If not provided, the job will be skipped (no runner created).\n   */\n  readonly provider?: string;\n\n  /**\n   * Labels to use when registering the runner.\n   * Must be returned when a provider is selected.\n   * Can be used to add, remove, or modify labels.\n   */\n  readonly labels?: string[];\n}\n\n/**\n * Properties for GithubWebhookHandler\n *\n * @internal\n */\nexport interface GithubWebhookHandlerProps {\n  /**\n   * Step function in charge of handling the workflow job events and start the runners.\n   */\n  readonly orchestrator: stepfunctions.StateMachine;\n\n  /**\n   * Secrets used to communicate with GitHub.\n   */\n  readonly secrets: Secrets;\n\n  /**\n   * Configure access to webhook function.\n   */\n  readonly access?: LambdaAccess;\n\n  /**\n   * Mapping of provider node paths to their supported labels.\n   */\n  readonly providers: Record<string, string[]>;\n\n  /**\n   * Optional Lambda function to customize provider selection.\n   */\n  readonly providerSelector?: lambda.IFunction;\n\n  /**\n   * Whether to require the \"self-hosted\" label.\n   */\n  readonly requireSelfHostedLabel: boolean;\n\n  /**\n   * Additional Lambda function options (VPC, security groups, layers, etc.).\n   */\n  readonly extraLambdaProps?: lambda.FunctionOptions;\n\n  /**\n   * Additional environment variables for the Lambda function.\n   */\n  readonly extraLambdaEnv?: { [key: string]: string };\n}\n\n/**\n * Create a Lambda with a public URL to handle GitHub webhook events. After validating the event with the given secret, the orchestrator step function is called with information about the workflow job.\n *\n * @internal\n */\nexport class GithubWebhookHandler extends Construct {\n\n  /**\n   * Public URL of webhook to be used with GitHub.\n   */\n  readonly url: string;\n\n  /**\n   * Webhook event handler.\n   */\n  readonly handler: WebhookHandlerFunction;\n\n  constructor(scope: Construct, id: string, props: GithubWebhookHandlerProps) {\n    super(scope, id);\n\n    this.handler = new WebhookHandlerFunction(\n      this,\n      'webhook-handler',\n      {\n        description: 'Handle GitHub webhook and start runner orchestrator',\n        environment: {\n          STEP_FUNCTION_ARN: props.orchestrator.stateMachineArn,\n          WEBHOOK_SECRET_ARN: props.secrets.webhook.secretArn,\n          GITHUB_SECRET_ARN: props.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: props.secrets.githubPrivateKey.secretArn,\n          PROVIDERS: JSON.stringify(props.providers),\n          REQUIRE_SELF_HOSTED_LABEL: props.requireSelfHostedLabel ? '1' : '0',\n          PROVIDER_SELECTOR_ARN: props.providerSelector?.functionArn ?? '',\n          ...props.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.seconds(31),\n        logGroup: singletonLogGroup(this, SingletonLogType.ORCHESTRATOR),\n        loggingFormat: lambda.LoggingFormat.JSON,\n        ...props.extraLambdaProps,\n      },\n    );\n\n    const access = props?.access ?? LambdaAccess.lambdaUrl();\n    this.url = access.bind(this, 'access', this.handler);\n\n    props.secrets.webhook.grantRead(this.handler);\n    props.secrets.github.grantRead(this.handler);\n    props.secrets.githubPrivateKey.grantRead(this.handler);\n    props.orchestrator.grantStartExecution(this.handler);\n    props.providerSelector?.grantInvoke(this.handler);\n  }\n}\n"]}