@cloudsnorkel/cdk-github-runners 0.14.15 → 0.14.16

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 (59) hide show
  1. package/.jsii +800 -274
  2. package/API.md +480 -6
  3. package/README.md +149 -0
  4. package/SETUP_GITHUB.md +99 -3
  5. package/assets/delete-failed-runner.lambda/index.js +40 -31
  6. package/assets/idle-runner-repear.lambda/index.js +40 -31
  7. package/assets/setup.lambda/index.html +12 -7
  8. package/assets/setup.lambda/index.js +26 -26
  9. package/assets/status.lambda/index.js +40 -31
  10. package/assets/token-retriever.lambda/index.js +40 -31
  11. package/assets/webhook-handler.lambda/index.js +110 -43
  12. package/assets/webhook-redelivery.lambda/index.js +40 -31
  13. package/lib/access.js +1 -1
  14. package/lib/image-builders/api.js +1 -1
  15. package/lib/image-builders/aws-image-builder/ami.d.ts +1 -2
  16. package/lib/image-builders/aws-image-builder/ami.js +6 -13
  17. package/lib/image-builders/aws-image-builder/builder.d.ts +4 -2
  18. package/lib/image-builders/aws-image-builder/builder.js +36 -34
  19. package/lib/image-builders/aws-image-builder/container.d.ts +2 -2
  20. package/lib/image-builders/aws-image-builder/container.js +7 -12
  21. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  22. package/lib/image-builders/aws-image-builder/deprecated/container.js +1 -1
  23. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  24. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  25. package/lib/image-builders/aws-image-builder/index.d.ts +0 -1
  26. package/lib/image-builders/aws-image-builder/index.js +1 -2
  27. package/lib/image-builders/aws-image-builder/workflow.d.ts +4 -4
  28. package/lib/image-builders/aws-image-builder/workflow.js +7 -10
  29. package/lib/image-builders/codebuild-deprecated.js +1 -1
  30. package/lib/image-builders/components.js +1 -1
  31. package/lib/image-builders/static.js +1 -1
  32. package/lib/index.d.ts +1 -0
  33. package/lib/index.js +2 -1
  34. package/lib/providers/codebuild.js +16 -10
  35. package/lib/providers/common.d.ts +53 -0
  36. package/lib/providers/common.js +11 -4
  37. package/lib/providers/composite.d.ts +61 -0
  38. package/lib/providers/composite.js +229 -0
  39. package/lib/providers/ec2.js +9 -8
  40. package/lib/providers/ecs.js +11 -6
  41. package/lib/providers/fargate.js +14 -9
  42. package/lib/providers/index.d.ts +1 -0
  43. package/lib/providers/index.js +2 -1
  44. package/lib/providers/lambda.js +6 -5
  45. package/lib/runner.d.ts +29 -5
  46. package/lib/runner.js +57 -24
  47. package/lib/secrets.js +1 -1
  48. package/lib/webhook-handler.lambda.d.ts +11 -0
  49. package/lib/webhook-handler.lambda.js +81 -14
  50. package/lib/webhook.d.ts +52 -7
  51. package/lib/webhook.js +4 -2
  52. package/package.json +15 -19
  53. package/assets/image-builders/aws-image-builder/versioner.lambda/index.js +0 -2115
  54. package/lib/image-builders/aws-image-builder/common.d.ts +0 -10
  55. package/lib/image-builders/aws-image-builder/common.js +0 -48
  56. package/lib/image-builders/aws-image-builder/versioner-function.d.ts +0 -13
  57. package/lib/image-builders/aws-image-builder/versioner-function.js +0 -23
  58. package/lib/image-builders/aws-image-builder/versioner.lambda.d.ts +0 -7
  59. package/lib/image-builders/aws-image-builder/versioner.lambda.js +0 -115
@@ -1,13 +1,17 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.verifyBody = verifyBody;
4
+ exports.callProviderSelector = callProviderSelector;
5
+ exports.selectProvider = selectProvider;
4
6
  exports.generateExecutionName = generateExecutionName;
5
7
  exports.handler = handler;
6
8
  const crypto = require("crypto");
9
+ const client_lambda_1 = require("@aws-sdk/client-lambda");
7
10
  const client_sfn_1 = require("@aws-sdk/client-sfn");
8
11
  const lambda_github_1 = require("./lambda-github");
9
12
  const lambda_helpers_1 = require("./lambda-helpers");
10
13
  const sf = new client_sfn_1.SFNClient();
14
+ const lambdaClient = new client_lambda_1.LambdaClient();
11
15
  // TODO use @octokit/webhooks?
12
16
  function getHeader(event, header) {
13
17
  // API Gateway doesn't lowercase headers (V1 event) but Lambda URLs do (V2 event) :(
@@ -58,18 +62,80 @@ async function isDeploymentPending(payload) {
58
62
  return false;
59
63
  }
60
64
  }
61
- function matchLabelsToProvider(labels) {
62
- const jobLabelSet = labels.map((label) => label.toLowerCase());
63
- const supportedLabels = JSON.parse(process.env.SUPPORTED_LABELS);
65
+ /**
66
+ * Match job labels to a provider using default label matching logic.
67
+ */
68
+ function matchLabelsToProvider(jobLabels, providers) {
69
+ const jobLabelLowerCase = jobLabels.map((label) => label.toLowerCase());
64
70
  // is every label the job requires available in the runner provider?
65
- for (const supportedLabelSet of supportedLabels) {
66
- const lowerCasedSupportedLabelSet = supportedLabelSet.labels.map((label) => label.toLowerCase());
67
- if (jobLabelSet.every(label => label == 'self-hosted' || lowerCasedSupportedLabelSet.includes(label))) {
68
- return supportedLabelSet.provider;
71
+ for (const provider of Object.keys(providers)) {
72
+ const providerLabelsLowerCase = providers[provider].map((label) => label.toLowerCase());
73
+ if (jobLabelLowerCase.every(label => label == 'self-hosted' || providerLabelsLowerCase.includes(label))) {
74
+ return provider;
69
75
  }
70
76
  }
71
77
  return undefined;
72
78
  }
79
+ /**
80
+ * Call the provider selector Lambda function if configured.
81
+ * @internal
82
+ */
83
+ async function callProviderSelector(payload, providers, defaultSelection) {
84
+ if (!process.env.PROVIDER_SELECTOR_ARN) {
85
+ return undefined;
86
+ }
87
+ const selectorInput = {
88
+ payload: payload,
89
+ providers: providers,
90
+ defaultProvider: defaultSelection.provider,
91
+ defaultLabels: defaultSelection.labels,
92
+ };
93
+ // don't catch errors -- the whole webhook handler will be retried on unhandled errors
94
+ const result = await lambdaClient.send(new client_lambda_1.InvokeCommand({
95
+ FunctionName: process.env.PROVIDER_SELECTOR_ARN,
96
+ Payload: JSON.stringify(selectorInput),
97
+ }));
98
+ if (result.FunctionError) {
99
+ console.error(result.FunctionError);
100
+ if (result.Payload) {
101
+ console.error(Buffer.from(result.Payload).toString());
102
+ }
103
+ throw new Error('Provider selector failed');
104
+ }
105
+ if (!result.Payload) {
106
+ throw new Error('Provider selector returned no payload');
107
+ }
108
+ return JSON.parse(Buffer.from(result.Payload).toString());
109
+ }
110
+ /**
111
+ * Exported for unit testing.
112
+ * @internal
113
+ */
114
+ async function selectProvider(payload, jobLabels, hook = callProviderSelector) {
115
+ const providers = JSON.parse(process.env.PROVIDERS);
116
+ const defaultProvider = matchLabelsToProvider(jobLabels, providers);
117
+ const defaultLabels = defaultProvider ? providers[defaultProvider] : undefined;
118
+ const defaultSelection = { provider: defaultProvider, labels: defaultLabels };
119
+ const selectorResult = await hook(payload, providers, defaultSelection);
120
+ if (selectorResult === undefined) {
121
+ return defaultSelection;
122
+ }
123
+ console.log(`Before provider selector provider=${defaultProvider} labels=${defaultLabels}`);
124
+ console.log(`After provider selector provider=${selectorResult.provider} labels=${selectorResult.labels}`);
125
+ // any error here will fail the webhook and cause a retry so the selector has another chance to get it right
126
+ if (selectorResult.provider !== undefined) {
127
+ if (selectorResult.provider === '') {
128
+ throw new Error('Provider selector returned empty provider');
129
+ }
130
+ if (!providers[selectorResult.provider]) {
131
+ throw new Error(`Provider selector returned unknown provider ${selectorResult.provider}`);
132
+ }
133
+ if (selectorResult.labels === undefined || selectorResult.labels.length === 0) {
134
+ throw new Error('Provider selector must return non-empty labels when provider is set');
135
+ }
136
+ }
137
+ return selectorResult;
138
+ }
73
139
  /**
74
140
  * Generate a unique execution name which is limited to 64 characters (also used as runner name).
75
141
  *
@@ -83,7 +149,7 @@ function generateExecutionName(event, payload) {
83
149
  return `${repoNameTruncated}-${deliveryId}`;
84
150
  }
85
151
  async function handler(event) {
86
- if (!process.env.WEBHOOK_SECRET_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SUPPORTED_LABELS || !process.env.REQUIRE_SELF_HOSTED_LABEL) {
152
+ if (!process.env.WEBHOOK_SECRET_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.PROVIDERS || !process.env.REQUIRE_SELF_HOSTED_LABEL) {
87
153
  throw new Error('Missing environment variables');
88
154
  }
89
155
  const webhookSecret = (await (0, lambda_helpers_1.getSecretJsonValue)(process.env.WEBHOOK_SECRET_ARN)).webhookSecret;
@@ -141,9 +207,9 @@ async function handler(event) {
141
207
  body: 'OK. No runner started (no "self-hosted" label).',
142
208
  };
143
209
  }
144
- // don't start step function unless labels match a runner provider
145
- const provider = matchLabelsToProvider(payload.workflow_job.labels);
146
- if (!provider) {
210
+ // Select provider and labels
211
+ const selection = await selectProvider(payload, payload.workflow_job.labels);
212
+ if (!selection.provider || !selection.labels) {
147
213
  console.log({
148
214
  notice: `Ignoring labels "${payload.workflow_job.labels}", as they don't match a supported runner provider`,
149
215
  job: payload.workflow_job,
@@ -172,8 +238,9 @@ async function handler(event) {
172
238
  jobId: payload.workflow_job.id,
173
239
  jobUrl: payload.workflow_job.html_url,
174
240
  installationId: payload.installation?.id ?? -1, // always pass value because step function can't handle missing input
175
- labels: payload.workflow_job.labels.join(','),
176
- provider: provider,
241
+ jobLabels: payload.workflow_job.labels.join(','), // original labels requested by the job
242
+ provider: selection.provider,
243
+ labels: selection.labels.join(','), // labels to use when registering runner
177
244
  };
178
245
  const execution = await sf.send(new client_sfn_1.StartExecutionCommand({
179
246
  stateMachineArn: process.env.STEP_FUNCTION_ARN,
@@ -192,4 +259,4 @@ async function handler(event) {
192
259
  body: executionName,
193
260
  };
194
261
  }
195
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhook-handler.lambda.js","sourceRoot":"","sources":["../src/webhook-handler.lambda.ts"],"names":[],"mappings":";;AA0BA,gCAyBC;AAyCD,sDAIC;AAED,0BAyHC;AA3ND,iCAAiC;AACjC,oDAAuE;AAEvE,mDAA6C;AAC7C,qDAAsD;AAGtD,MAAM,EAAE,GAAG,IAAI,sBAAS,EAAE,CAAC;AAE3B,8BAA8B;AAE9B,SAAS,SAAS,CAAC,KAAuC,EAAE,MAAc;IACxE,oFAAoF;IACpF,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC;QACpD,IAAI,UAAU,CAAC,WAAW,EAAE,KAAK,MAAM,CAAC,WAAW,EAAE,EAAE,CAAC;YACtD,OAAO,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;GAGG;AACH,SAAgB,UAAU,CAAC,KAAuC,EAAE,MAAW;IAC7E,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,qBAAqB,CAAC,IAAI,EAAE,EAAE,MAAM,CAAC,CAAC;IAE/E,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QAChB,MAAM,IAAI,KAAK,CAAC,SAAS,CAAC,CAAC;IAC7B,CAAC;IAED,IAAI,IAAY,CAAC;IACjB,IAAI,KAAK,CAAC,eAAe,EAAE,CAAC;QAC1B,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;IAC3C,CAAC;SAAM,CAAC;QACN,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,EAAE,EAAE,MAAM,CAAC,CAAC;IAC/C,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IACjD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IAClB,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;IAExE,OAAO,CAAC,GAAG,CAAC,wBAAwB,EAAE,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC;IAE9D,IAAI,GAAG,CAAC,MAAM,KAAK,WAAW,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,GAAG,EAAE,WAAW,CAAC,EAAE,CAAC;QACnF,MAAM,IAAI,KAAK,CAAC,gCAAgC,WAAW,CAAC,QAAQ,EAAE,YAAY,GAAG,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IACtG,CAAC;IAED,OAAO,IAAI,CAAC,QAAQ,EAAE,CAAC;AACzB,CAAC;AAED,KAAK,UAAU,mBAAmB,CAAC,OAAY;IAC7C,MAAM,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,YAAY,CAAC;IACrD,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;QAC9B,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,CAAC;QACH,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,IAAA,0BAAU,EAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC;QAC/D,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAEpD,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,KAAK,KAAK,SAAS,CAAC;IAC/C,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CAAC,oEAAoE,EAAE,CAAC,CAAC,CAAC;QACvF,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED,SAAS,qBAAqB,CAAC,MAAgB;IAC7C,MAAM,WAAW,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;IAC/D,MAAM,eAAe,GAAsB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAiB,CAAC,CAAC;IAErF,oEAAoE;IACpE,KAAK,MAAM,iBAAiB,IAAI,eAAe,EAAE,CAAC;QAChD,MAAM,2BAA2B,GAAG,iBAAiB,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;QACjG,IAAI,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,IAAI,aAAa,IAAI,2BAA2B,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC;YACtG,OAAO,iBAAiB,CAAC,QAAQ,CAAC;QACpC,CAAC;IACH,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;;;GAMG;AACH,SAAgB,qBAAqB,CAAC,KAAU,EAAE,OAAY;IAC5D,MAAM,UAAU,GAAG,SAAS,CAAC,KAAK,EAAE,mBAAmB,CAAC,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;IAC/E,MAAM,iBAAiB,GAAG,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACvF,OAAO,GAAG,iBAAiB,IAAI,UAAU,EAAE,CAAC;AAC9C,CAAC;AAEM,KAAK,UAAU,OAAO,CAAC,KAAuC;IACnE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,yBAAyB,EAAE,CAAC;QACjJ,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;IACnD,CAAC;IAED,MAAM,aAAa,GAAG,CAAC,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAAC,aAAa,CAAC;IAE/F,IAAI,IAAI,CAAC;IACT,IAAI,CAAC;QACH,IAAI,GAAG,UAAU,CAAC,KAAK,EAAE,aAAa,CAAC,CAAC;IAC1C,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACjB,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,eAAe;SACtB,CAAC;IACJ,CAAC;IAED,IAAI,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,KAAK,kBAAkB,EAAE,CAAC;QAC5D,OAAO,CAAC,KAAK,CAAC,gDAAgD,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,EAAE,CAAC,CAAC;QAClG,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,wBAAwB;SAC/B,CAAC;IACJ,CAAC;IAED,IAAI,SAAS,CAAC,KAAK,EAAE,gBAAgB,CAAC,KAAK,MAAM,EAAE,CAAC;QAClD,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,MAAM;SACb,CAAC;IACJ,CAAC;IAED,wHAAwH;IACxH,2HAA2H;IAC3H,IAAI,SAAS,CAAC,KAAK,EAAE,gBAAgB,CAAC,KAAK,cAAc,EAAE,CAAC;QAC1D,OAAO,CAAC,KAAK,CAAC,+CAA+C,SAAS,CAAC,KAAK,EAAE,gBAAgB,CAAC,EAAE,CAAC,CAAC;QACnG,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,wBAAwB;SAC/B,CAAC;IACJ,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAEjC,IAAI,OAAO,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;QAChC,OAAO,CAAC,GAAG,CAAC;YACV,MAAM,EAAE,oBAAoB,OAAO,CAAC,MAAM,uBAAuB;YACjE,GAAG,EAAE,OAAO,CAAC,YAAY;SAC1B,CAAC,CAAC;QACH,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,iDAAiD;SACxD,CAAC;IACJ,CAAC;IAED,IAAI,OAAO,CAAC,GAAG,CAAC,yBAAyB,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,EAAE,CAAC;QAC1G,OAAO,CAAC,GAAG,CAAC;YACV,MAAM,EAAE,oBAAoB,OAAO,CAAC,YAAY,CAAC,MAAM,4BAA4B;YACnF,GAAG,EAAE,OAAO,CAAC,YAAY;SAC1B,CAAC,CAAC;QACH,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,iDAAiD;SACxD,CAAC;IACJ,CAAC;IAED,kEAAkE;IAClE,MAAM,QAAQ,GAAG,qBAAqB,CAAC,OAAO,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;IACpE,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,CAAC,GAAG,CAAC;YACV,MAAM,EAAE,oBAAoB,OAAO,CAAC,YAAY,CAAC,MAAM,oDAAoD;YAC3G,GAAG,EAAE,OAAO,CAAC,YAAY;SAC1B,CAAC,CAAC;QACH,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,2DAA2D;SAClE,CAAC;IACJ,CAAC;IAED,8GAA8G;IAC9G,IAAI,MAAM,mBAAmB,CAAC,OAAO,CAAC,EAAE,CAAC;QACvC,OAAO,CAAC,GAAG,CAAC;YACV,MAAM,EAAE,iDAAiD;YACzD,GAAG,EAAE,OAAO,CAAC,YAAY;SAC1B,CAAC,CAAC;QACH,OAAO;YACL,UAAU,EAAE,GAAG;YACf,IAAI,EAAE,6CAA6C;SACpD,CAAC;IACJ,CAAC;IAED,kBAAkB;IAClB,MAAM,aAAa,GAAG,qBAAqB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;IAC5D,MAAM,KAAK,GAAG;QACZ,KAAK,EAAE,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC,KAAK;QACrC,IAAI,EAAE,OAAO,CAAC,UAAU,CAAC,IAAI;QAC7B,KAAK,EAAE,OAAO,CAAC,YAAY,CAAC,EAAE;QAC9B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,QAAQ;QACrC,cAAc,EAAE,OAAO,CAAC,YAAY,EAAE,EAAE,IAAI,CAAC,CAAC,EAAE,qEAAqE;QACrH,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;QAC7C,QAAQ,EAAE,QAAQ;KACnB,CAAC;IACF,MAAM,SAAS,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,IAAI,kCAAqB,CAAC;QACxD,eAAe,EAAE,OAAO,CAAC,GAAG,CAAC,iBAAiB;QAC9C,KAAK,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;QAC5B,kGAAkG;QAClG,IAAI,EAAE,aAAa;KACpB,CAAC,CAAC,CAAC;IAEJ,OAAO,CAAC,GAAG,CAAC;QACV,MAAM,EAAE,sBAAsB;QAC9B,SAAS,EAAE,SAAS,CAAC,YAAY;QACjC,QAAQ,EAAE,KAAK;QACf,GAAG,EAAE,OAAO,CAAC,YAAY;KAC1B,CAAC,CAAC;IAEH,OAAO;QACL,UAAU,EAAE,GAAG;QACf,IAAI,EAAE,aAAa;KACpB,CAAC;AACJ,CAAC","sourcesContent":["import * as crypto from 'crypto';\nimport { SFNClient, StartExecutionCommand } from '@aws-sdk/client-sfn';\nimport * as AWSLambda from 'aws-lambda';\nimport { getOctokit } from './lambda-github';\nimport { getSecretJsonValue } from './lambda-helpers';\nimport { SupportedLabels } from './webhook';\n\nconst sf = new SFNClient();\n\n// TODO use @octokit/webhooks?\n\nfunction getHeader(event: AWSLambda.APIGatewayProxyEventV2, header: string): string | undefined {\n  // API Gateway doesn't lowercase headers (V1 event) but Lambda URLs do (V2 event) :(\n  for (const headerName of Object.keys(event.headers)) {\n    if (headerName.toLowerCase() === header.toLowerCase()) {\n      return event.headers[headerName];\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Exported for unit testing.\n * @internal\n */\nexport function verifyBody(event: AWSLambda.APIGatewayProxyEventV2, secret: any): string {\n  const sig = Buffer.from(getHeader(event, 'x-hub-signature-256') || '', 'utf8');\n\n  if (!event.body) {\n    throw new Error('No body');\n  }\n\n  let body: Buffer;\n  if (event.isBase64Encoded) {\n    body = Buffer.from(event.body, 'base64');\n  } else {\n    body = Buffer.from(event.body || '', 'utf8');\n  }\n\n  const hmac = crypto.createHmac('sha256', secret);\n  hmac.update(body);\n  const expectedSig = Buffer.from(`sha256=${hmac.digest('hex')}`, 'utf8');\n\n  console.log('Calculated signature: ', expectedSig.toString());\n\n  if (sig.length !== expectedSig.length || !crypto.timingSafeEqual(sig, expectedSig)) {\n    throw new Error(`Signature mismatch. Expected ${expectedSig.toString()} but got ${sig.toString()}`);\n  }\n\n  return body.toString();\n}\n\nasync function isDeploymentPending(payload: any) {\n  const statusesUrl = payload.deployment?.statuses_url;\n  if (statusesUrl === undefined) {\n    return false;\n  }\n\n  try {\n    const { octokit } = await getOctokit(payload.installation?.id);\n    const statuses = await octokit.request(statusesUrl);\n\n    return statuses.data[0]?.state === 'waiting';\n  } catch (e) {\n    console.error('Unable to check deployment. Try adding deployment read permission.', e);\n    return false;\n  }\n}\n\nfunction matchLabelsToProvider(labels: string[]) {\n  const jobLabelSet = labels.map((label) => label.toLowerCase());\n  const supportedLabels: SupportedLabels[] = JSON.parse(process.env.SUPPORTED_LABELS!);\n\n  // is every label the job requires available in the runner provider?\n  for (const supportedLabelSet of supportedLabels) {\n    const lowerCasedSupportedLabelSet = supportedLabelSet.labels.map((label) => label.toLowerCase());\n    if (jobLabelSet.every(label => label == 'self-hosted' || lowerCasedSupportedLabelSet.includes(label))) {\n      return supportedLabelSet.provider;\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Generate a unique execution name which is limited to 64 characters (also used as runner name).\n *\n * Exported for unit testing.\n *\n * @internal\n */\nexport function generateExecutionName(event: any, payload: any): string {\n  const deliveryId = getHeader(event, 'x-github-delivery') ?? `${Math.random()}`;\n  const repoNameTruncated = payload.repository.name.slice(0, 64 - deliveryId.length - 1);\n  return `${repoNameTruncated}-${deliveryId}`;\n}\n\nexport async function handler(event: AWSLambda.APIGatewayProxyEventV2): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  if (!process.env.WEBHOOK_SECRET_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SUPPORTED_LABELS || !process.env.REQUIRE_SELF_HOSTED_LABEL) {\n    throw new Error('Missing environment variables');\n  }\n\n  const webhookSecret = (await getSecretJsonValue(process.env.WEBHOOK_SECRET_ARN)).webhookSecret;\n\n  let body;\n  try {\n    body = verifyBody(event, webhookSecret);\n  } catch (e) {\n    console.error(e);\n    return {\n      statusCode: 403,\n      body: 'Bad signature',\n    };\n  }\n\n  if (getHeader(event, 'content-type') !== 'application/json') {\n    console.error(`This webhook only accepts JSON payloads, got ${getHeader(event, 'content-type')}`);\n    return {\n      statusCode: 400,\n      body: 'Expecting JSON payload',\n    };\n  }\n\n  if (getHeader(event, 'x-github-event') === 'ping') {\n    return {\n      statusCode: 200,\n      body: 'Pong',\n    };\n  }\n\n  // if (getHeader(event, 'x-github-event') !== 'workflow_job' && getHeader(event, 'x-github-event') !== 'workflow_run') {\n  //     console.error(`This webhook only accepts workflow_job and workflow_run, got ${getHeader(event, 'x-github-event')}`);\n  if (getHeader(event, 'x-github-event') !== 'workflow_job') {\n    console.error(`This webhook only accepts workflow_job, got ${getHeader(event, 'x-github-event')}`);\n    return {\n      statusCode: 200,\n      body: 'Expecting workflow_job',\n    };\n  }\n\n  const payload = JSON.parse(body);\n\n  if (payload.action !== 'queued') {\n    console.log({\n      notice: `Ignoring action \"${payload.action}\", expecting \"queued\"`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (action is not \"queued\").',\n    };\n  }\n\n  if (process.env.REQUIRE_SELF_HOSTED_LABEL === '1' && !payload.workflow_job.labels.includes('self-hosted')) {\n    console.log({\n      notice: `Ignoring labels \"${payload.workflow_job.labels}\", expecting \"self-hosted\"`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (no \"self-hosted\" label).',\n    };\n  }\n\n  // don't start step function unless labels match a runner provider\n  const provider = matchLabelsToProvider(payload.workflow_job.labels);\n  if (!provider) {\n    console.log({\n      notice: `Ignoring labels \"${payload.workflow_job.labels}\", as they don't match a supported runner provider`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (no provider with matching labels).',\n    };\n  }\n\n  // don't start runners for a deployment that's still pending as GitHub will send another event when it's ready\n  if (await isDeploymentPending(payload)) {\n    console.log({\n      notice: 'Ignoring job as its deployment is still pending',\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (deployment pending).',\n    };\n  }\n\n  // start execution\n  const executionName = generateExecutionName(event, payload);\n  const input = {\n    owner: payload.repository.owner.login,\n    repo: payload.repository.name,\n    jobId: payload.workflow_job.id,\n    jobUrl: payload.workflow_job.html_url,\n    installationId: payload.installation?.id ?? -1, // always pass value because step function can't handle missing input\n    labels: payload.workflow_job.labels.join(','),\n    provider: provider,\n  };\n  const execution = await sf.send(new StartExecutionCommand({\n    stateMachineArn: process.env.STEP_FUNCTION_ARN,\n    input: JSON.stringify(input),\n    // name is not random so multiple execution of this webhook won't cause multiple builders to start\n    name: executionName,\n  }));\n\n  console.log({\n    notice: 'Started orchestrator',\n    execution: execution.executionArn,\n    sfnInput: input,\n    job: payload.workflow_job,\n  });\n\n  return {\n    statusCode: 202,\n    body: executionName,\n  };\n}\n"]}
262
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhook-handler.lambda.js","sourceRoot":"","sources":["../src/webhook-handler.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { InvokeCommand, LambdaClient } from '@aws-sdk/client-lambda';\nimport { SFNClient, StartExecutionCommand } from '@aws-sdk/client-sfn';\nimport * as AWSLambda from 'aws-lambda';\nimport { getOctokit } from './lambda-github';\nimport { getSecretJsonValue } from './lambda-helpers';\nimport { ProviderSelectorInput, ProviderSelectorResult } from './webhook';\n\nconst sf = new SFNClient();\nconst lambdaClient = new LambdaClient();\n\n// TODO use @octokit/webhooks?\n\nfunction getHeader(event: AWSLambda.APIGatewayProxyEventV2, header: string): string | undefined {\n  // API Gateway doesn't lowercase headers (V1 event) but Lambda URLs do (V2 event) :(\n  for (const headerName of Object.keys(event.headers)) {\n    if (headerName.toLowerCase() === header.toLowerCase()) {\n      return event.headers[headerName];\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Exported for unit testing.\n * @internal\n */\nexport function verifyBody(event: AWSLambda.APIGatewayProxyEventV2, secret: any): string {\n  const sig = Buffer.from(getHeader(event, 'x-hub-signature-256') || '', 'utf8');\n\n  if (!event.body) {\n    throw new Error('No body');\n  }\n\n  let body: Buffer;\n  if (event.isBase64Encoded) {\n    body = Buffer.from(event.body, 'base64');\n  } else {\n    body = Buffer.from(event.body || '', 'utf8');\n  }\n\n  const hmac = crypto.createHmac('sha256', secret);\n  hmac.update(body);\n  const expectedSig = Buffer.from(`sha256=${hmac.digest('hex')}`, 'utf8');\n\n  console.log('Calculated signature: ', expectedSig.toString());\n\n  if (sig.length !== expectedSig.length || !crypto.timingSafeEqual(sig, expectedSig)) {\n    throw new Error(`Signature mismatch. Expected ${expectedSig.toString()} but got ${sig.toString()}`);\n  }\n\n  return body.toString();\n}\n\nasync function isDeploymentPending(payload: any) {\n  const statusesUrl = payload.deployment?.statuses_url;\n  if (statusesUrl === undefined) {\n    return false;\n  }\n\n  try {\n    const { octokit } = await getOctokit(payload.installation?.id);\n    const statuses = await octokit.request(statusesUrl);\n\n    return statuses.data[0]?.state === 'waiting';\n  } catch (e) {\n    console.error('Unable to check deployment. Try adding deployment read permission.', e);\n    return false;\n  }\n}\n\n/**\n * Match job labels to a provider using default label matching logic.\n */\nfunction matchLabelsToProvider(jobLabels: string[], providers: Record<string, string[]>): string | undefined {\n  const jobLabelLowerCase = jobLabels.map((label) => label.toLowerCase());\n\n  // is every label the job requires available in the runner provider?\n  for (const provider of Object.keys(providers)) {\n    const providerLabelsLowerCase = providers[provider].map((label) => label.toLowerCase());\n    if (jobLabelLowerCase.every(label => label == 'self-hosted' || providerLabelsLowerCase.includes(label))) {\n      return provider;\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Call the provider selector Lambda function if configured.\n * @internal\n */\nexport async function callProviderSelector(\n  payload: any,\n  providers: Record<string, string[]>,\n  defaultSelection: ProviderSelectorResult,\n): Promise<ProviderSelectorResult | undefined> {\n  if (!process.env.PROVIDER_SELECTOR_ARN) {\n    return undefined;\n  }\n\n  const selectorInput: ProviderSelectorInput = {\n    payload: payload,\n    providers: providers,\n    defaultProvider: defaultSelection.provider,\n    defaultLabels: defaultSelection.labels,\n  };\n\n  // don't catch errors -- the whole webhook handler will be retried on unhandled errors\n  const result = await lambdaClient.send(new InvokeCommand({\n    FunctionName: process.env.PROVIDER_SELECTOR_ARN,\n    Payload: JSON.stringify(selectorInput),\n  }));\n\n  if (result.FunctionError) {\n    console.error(result.FunctionError);\n    if (result.Payload) {\n      console.error(Buffer.from(result.Payload).toString());\n    }\n    throw new Error('Provider selector failed');\n  }\n\n  if (!result.Payload) {\n    throw new Error('Provider selector returned no payload');\n  }\n\n  return JSON.parse(Buffer.from(result.Payload).toString()) as ProviderSelectorResult;\n}\n\n/**\n * Exported for unit testing.\n * @internal\n */\nexport async function selectProvider(payload: any, jobLabels: string[], hook = callProviderSelector): Promise<ProviderSelectorResult> {\n  const providers = JSON.parse(process.env.PROVIDERS!);\n  const defaultProvider = matchLabelsToProvider(jobLabels, providers);\n  const defaultLabels = defaultProvider ? providers[defaultProvider] : undefined;\n  const defaultSelection = { provider: defaultProvider, labels: defaultLabels };\n  const selectorResult = await hook(payload, providers, defaultSelection);\n\n  if (selectorResult === undefined) {\n    return defaultSelection;\n  }\n\n  console.log(`Before provider selector provider=${defaultProvider} labels=${defaultLabels}`);\n  console.log(`After provider selector provider=${selectorResult.provider} labels=${selectorResult.labels}`);\n\n  // any error here will fail the webhook and cause a retry so the selector has another chance to get it right\n  if (selectorResult.provider !== undefined) {\n    if (selectorResult.provider === '') {\n      throw new Error('Provider selector returned empty provider');\n    }\n    if (!providers[selectorResult.provider]) {\n      throw new Error(`Provider selector returned unknown provider ${selectorResult.provider}`);\n    }\n    if (selectorResult.labels === undefined || selectorResult.labels.length === 0) {\n      throw new Error('Provider selector must return non-empty labels when provider is set');\n    }\n  }\n\n  return selectorResult;\n}\n\n/**\n * Generate a unique execution name which is limited to 64 characters (also used as runner name).\n *\n * Exported for unit testing.\n *\n * @internal\n */\nexport function generateExecutionName(event: any, payload: any): string {\n  const deliveryId = getHeader(event, 'x-github-delivery') ?? `${Math.random()}`;\n  const repoNameTruncated = payload.repository.name.slice(0, 64 - deliveryId.length - 1);\n  return `${repoNameTruncated}-${deliveryId}`;\n}\n\nexport async function handler(event: AWSLambda.APIGatewayProxyEventV2): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  if (!process.env.WEBHOOK_SECRET_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.PROVIDERS || !process.env.REQUIRE_SELF_HOSTED_LABEL) {\n    throw new Error('Missing environment variables');\n  }\n\n  const webhookSecret = (await getSecretJsonValue(process.env.WEBHOOK_SECRET_ARN)).webhookSecret;\n\n  let body;\n  try {\n    body = verifyBody(event, webhookSecret);\n  } catch (e) {\n    console.error(e);\n    return {\n      statusCode: 403,\n      body: 'Bad signature',\n    };\n  }\n\n  if (getHeader(event, 'content-type') !== 'application/json') {\n    console.error(`This webhook only accepts JSON payloads, got ${getHeader(event, 'content-type')}`);\n    return {\n      statusCode: 400,\n      body: 'Expecting JSON payload',\n    };\n  }\n\n  if (getHeader(event, 'x-github-event') === 'ping') {\n    return {\n      statusCode: 200,\n      body: 'Pong',\n    };\n  }\n\n  // if (getHeader(event, 'x-github-event') !== 'workflow_job' && getHeader(event, 'x-github-event') !== 'workflow_run') {\n  //     console.error(`This webhook only accepts workflow_job and workflow_run, got ${getHeader(event, 'x-github-event')}`);\n  if (getHeader(event, 'x-github-event') !== 'workflow_job') {\n    console.error(`This webhook only accepts workflow_job, got ${getHeader(event, 'x-github-event')}`);\n    return {\n      statusCode: 200,\n      body: 'Expecting workflow_job',\n    };\n  }\n\n  const payload = JSON.parse(body);\n\n  if (payload.action !== 'queued') {\n    console.log({\n      notice: `Ignoring action \"${payload.action}\", expecting \"queued\"`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (action is not \"queued\").',\n    };\n  }\n\n  if (process.env.REQUIRE_SELF_HOSTED_LABEL === '1' && !payload.workflow_job.labels.includes('self-hosted')) {\n    console.log({\n      notice: `Ignoring labels \"${payload.workflow_job.labels}\", expecting \"self-hosted\"`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (no \"self-hosted\" label).',\n    };\n  }\n\n  // Select provider and labels\n  const selection = await selectProvider(payload, payload.workflow_job.labels);\n  if (!selection.provider || !selection.labels) {\n    console.log({\n      notice: `Ignoring labels \"${payload.workflow_job.labels}\", as they don't match a supported runner provider`,\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (no provider with matching labels).',\n    };\n  }\n\n  // don't start runners for a deployment that's still pending as GitHub will send another event when it's ready\n  if (await isDeploymentPending(payload)) {\n    console.log({\n      notice: 'Ignoring job as its deployment is still pending',\n      job: payload.workflow_job,\n    });\n    return {\n      statusCode: 200,\n      body: 'OK. No runner started (deployment pending).',\n    };\n  }\n\n  // start execution\n  const executionName = generateExecutionName(event, payload);\n  const input = {\n    owner: payload.repository.owner.login,\n    repo: payload.repository.name,\n    jobId: payload.workflow_job.id,\n    jobUrl: payload.workflow_job.html_url,\n    installationId: payload.installation?.id ?? -1, // always pass value because step function can't handle missing input\n    jobLabels: payload.workflow_job.labels.join(','), // original labels requested by the job\n    provider: selection.provider,\n    labels: selection.labels.join(','), // labels to use when registering runner\n  };\n  const execution = await sf.send(new StartExecutionCommand({\n    stateMachineArn: process.env.STEP_FUNCTION_ARN,\n    input: JSON.stringify(input),\n    // name is not random so multiple execution of this webhook won't cause multiple builders to start\n    name: executionName,\n  }));\n\n  console.log({\n    notice: 'Started orchestrator',\n    execution: execution.executionArn,\n    sfnInput: input,\n    job: payload.workflow_job,\n  });\n\n  return {\n    statusCode: 202,\n    body: executionName,\n  };\n}\n"]}
package/lib/webhook.d.ts CHANGED
@@ -1,14 +1,55 @@
1
- import { aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
1
+ import { aws_lambda as lambda, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
2
2
  import { Construct } from 'constructs';
3
3
  import { LambdaAccess } from './access';
4
4
  import { Secrets } from './secrets';
5
5
  import { WebhookHandlerFunction } from './webhook-handler-function';
6
6
  /**
7
- * @internal
7
+ * Input to the provider selector Lambda function.
8
+ */
9
+ export interface ProviderSelectorInput {
10
+ /**
11
+ * Full GitHub webhook payload (workflow_job event structure with action="queued").
12
+ *
13
+ * * Original labels requested by the workflow job can be found at `payload.workflow_job.labels`.
14
+ * * Repository path (e.g. CloudSnorkel/cdk-github-runners) is at `payload.repository.full_name`.
15
+ * * Commit hash is at `payload.workflow_job.head_sha`.
16
+ *
17
+ * @see https://docs.github.com/en/webhooks/webhook-events-and-payloads?actionType=queued#workflow_job
18
+ */
19
+ readonly payload: any;
20
+ /**
21
+ * Map of available provider node paths to their configured labels.
22
+ * Example: { "MyStack/Small": ["linux", "small"], "MyStack/Large": ["linux", "large"] }
23
+ */
24
+ readonly providers: Record<string, string[]>;
25
+ /**
26
+ * Provider node path that would have been selected by default label matching.
27
+ * Use this to easily return the default selection: `{ provider: input.defaultProvider, labels: input.defaultLabels }`
28
+ * May be undefined if no provider matched by default.
29
+ */
30
+ readonly defaultProvider?: string;
31
+ /**
32
+ * Labels that would have been used by default (the selected provider's labels).
33
+ * May be undefined if no provider matched by default.
34
+ */
35
+ readonly defaultLabels?: string[];
36
+ }
37
+ /**
38
+ * Result from the provider selector Lambda function.
8
39
  */
9
- export interface SupportedLabels {
10
- readonly provider: string;
11
- readonly labels: string[];
40
+ export interface ProviderSelectorResult {
41
+ /**
42
+ * Node path of the provider to use (e.g., "MyStack/MyProvider").
43
+ * Must match one of the configured provider node paths from the input.
44
+ * If not provided, the job will be skipped (no runner created).
45
+ */
46
+ readonly provider?: string;
47
+ /**
48
+ * Labels to use when registering the runner.
49
+ * Must be returned when a provider is selected.
50
+ * Can be used to add, remove, or modify labels.
51
+ */
52
+ readonly labels?: string[];
12
53
  }
13
54
  /**
14
55
  * Properties for GithubWebhookHandler
@@ -29,9 +70,13 @@ export interface GithubWebhookHandlerProps {
29
70
  */
30
71
  readonly access?: LambdaAccess;
31
72
  /**
32
- * List of supported label combinations.
73
+ * Mapping of provider node paths to their supported labels.
74
+ */
75
+ readonly providers: Record<string, string[]>;
76
+ /**
77
+ * Optional Lambda function to customize provider selection.
33
78
  */
34
- readonly supportedLabels: SupportedLabels[];
79
+ readonly providerSelector?: lambda.IFunction;
35
80
  /**
36
81
  * Whether to require the "self-hosted" label.
37
82
  */
package/lib/webhook.js CHANGED
@@ -22,8 +22,9 @@ class GithubWebhookHandler extends constructs_1.Construct {
22
22
  WEBHOOK_SECRET_ARN: props.secrets.webhook.secretArn,
23
23
  GITHUB_SECRET_ARN: props.secrets.github.secretArn,
24
24
  GITHUB_PRIVATE_KEY_SECRET_ARN: props.secrets.githubPrivateKey.secretArn,
25
- SUPPORTED_LABELS: JSON.stringify(props.supportedLabels),
25
+ PROVIDERS: JSON.stringify(props.providers),
26
26
  REQUIRE_SELF_HOSTED_LABEL: props.requireSelfHostedLabel ? '1' : '0',
27
+ PROVIDER_SELECTOR_ARN: props.providerSelector?.functionArn ?? '',
27
28
  },
28
29
  timeout: cdk.Duration.seconds(31),
29
30
  logGroup: (0, utils_1.singletonLogGroup)(this, utils_1.SingletonLogType.ORCHESTRATOR),
@@ -35,7 +36,8 @@ class GithubWebhookHandler extends constructs_1.Construct {
35
36
  props.secrets.github.grantRead(this.handler);
36
37
  props.secrets.githubPrivateKey.grantRead(this.handler);
37
38
  props.orchestrator.grantStartExecution(this.handler);
39
+ props.providerSelector?.grantInvoke(this.handler);
38
40
  }
39
41
  }
40
42
  exports.GithubWebhookHandler = GithubWebhookHandler;
41
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2ViaG9vay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy93ZWJob29rLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1DQUFtQztBQUNuQyw2Q0FBdUY7QUFDdkYsMkNBQXVDO0FBQ3ZDLHFDQUF3QztBQUV4QyxtQ0FBOEQ7QUFDOUQseUVBQW9FO0FBMENwRTs7OztHQUlHO0FBQ0gsTUFBYSxvQkFBcUIsU0FBUSxzQkFBUztJQVlqRCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWdDO1FBQ3hFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsSUFBSSxDQUFDLE9BQU8sR0FBRyxJQUFJLGlEQUFzQixDQUN2QyxJQUFJLEVBQ0osaUJBQWlCLEVBQ2pCO1lBQ0UsV0FBVyxFQUFFLHFEQUFxRDtZQUNsRSxXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFlBQVksQ0FBQyxlQUFlO2dCQUNyRCxrQkFBa0IsRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTO2dCQUNuRCxpQkFBaUIsRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxTQUFTO2dCQUNqRCw2QkFBNkIsRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDLGdCQUFnQixDQUFDLFNBQVM7Z0JBQ3ZFLGdCQUFnQixFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLGVBQWUsQ0FBQztnQkFDdkQseUJBQXlCLEVBQUUsS0FBSyxDQUFDLHNCQUFzQixDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEdBQUc7YUFDcEU7WUFDRCxPQUFPLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQ2pDLFFBQVEsRUFBRSxJQUFBLHlCQUFpQixFQUFDLElBQUksRUFBRSx3QkFBZ0IsQ0FBQyxZQUFZLENBQUM7WUFDaEUsYUFBYSxFQUFFLHdCQUFNLENBQUMsYUFBYSxDQUFDLElBQUk7U0FDekMsQ0FDRixDQUFDO1FBRUYsTUFBTSxNQUFNLEdBQUcsS0FBSyxFQUFFLE1BQU0sSUFBSSxxQkFBWSxDQUFDLFNBQVMsRUFBRSxDQUFDO1FBQ3pELElBQUksQ0FBQyxHQUFHLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsUUFBUSxFQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUVyRCxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQzlDLEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDN0MsS0FBSyxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ3ZELEtBQUssQ0FBQyxZQUFZLENBQUMsbUJBQW1CLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ3ZELENBQUM7Q0FDRjtBQTFDRCxvREEwQ0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgYXdzX2xhbWJkYSBhcyBsYW1iZGEsIGF3c19zdGVwZnVuY3Rpb25zIGFzIHN0ZXBmdW5jdGlvbnMgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IExhbWJkYUFjY2VzcyB9IGZyb20gJy4vYWNjZXNzJztcbmltcG9ydCB7IFNlY3JldHMgfSBmcm9tICcuL3NlY3JldHMnO1xuaW1wb3J0IHsgc2luZ2xldG9uTG9nR3JvdXAsIFNpbmdsZXRvbkxvZ1R5cGUgfSBmcm9tICcuL3V0aWxzJztcbmltcG9ydCB7IFdlYmhvb2tIYW5kbGVyRnVuY3Rpb24gfSBmcm9tICcuL3dlYmhvb2staGFuZGxlci1mdW5jdGlvbic7XG5cbi8qKlxuICogQGludGVybmFsXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3VwcG9ydGVkTGFiZWxzIHtcbiAgcmVhZG9ubHkgcHJvdmlkZXI6IHN0cmluZztcbiAgcmVhZG9ubHkgbGFiZWxzOiBzdHJpbmdbXTtcbn1cblxuLyoqXG4gKiBQcm9wZXJ0aWVzIGZvciBHaXRodWJXZWJob29rSGFuZGxlclxuICpcbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEdpdGh1YldlYmhvb2tIYW5kbGVyUHJvcHMge1xuICAvKipcbiAgICogU3RlcCBmdW5jdGlvbiBpbiBjaGFyZ2Ugb2YgaGFuZGxpbmcgdGhlIHdvcmtmbG93IGpvYiBldmVudHMgYW5kIHN0YXJ0IHRoZSBydW5uZXJzLlxuICAgKi9cbiAgcmVhZG9ubHkgb3JjaGVzdHJhdG9yOiBzdGVwZnVuY3Rpb25zLlN0YXRlTWFjaGluZTtcblxuICAvKipcbiAgICogU2VjcmV0cyB1c2VkIHRvIGNvbW11bmljYXRlIHdpdGggR2l0SHViLlxuICAgKi9cbiAgcmVhZG9ubHkgc2VjcmV0czogU2VjcmV0cztcblxuICAvKipcbiAgICogQ29uZmlndXJlIGFjY2VzcyB0byB3ZWJob29rIGZ1bmN0aW9uLlxuICAgKi9cbiAgcmVhZG9ubHkgYWNjZXNzPzogTGFtYmRhQWNjZXNzO1xuXG4gIC8qKlxuICAgKiBMaXN0IG9mIHN1cHBvcnRlZCBsYWJlbCBjb21iaW5hdGlvbnMuXG4gICAqL1xuICByZWFkb25seSBzdXBwb3J0ZWRMYWJlbHM6IFN1cHBvcnRlZExhYmVsc1tdO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIHJlcXVpcmUgdGhlIFwic2VsZi1ob3N0ZWRcIiBsYWJlbC5cbiAgICovXG4gIHJlYWRvbmx5IHJlcXVpcmVTZWxmSG9zdGVkTGFiZWw6IGJvb2xlYW47XG59XG5cbi8qKlxuICogQ3JlYXRlIGEgTGFtYmRhIHdpdGggYSBwdWJsaWMgVVJMIHRvIGhhbmRsZSBHaXRIdWIgd2ViaG9vayBldmVudHMuIEFmdGVyIHZhbGlkYXRpbmcgdGhlIGV2ZW50IHdpdGggdGhlIGdpdmVuIHNlY3JldCwgdGhlIG9yY2hlc3RyYXRvciBzdGVwIGZ1bmN0aW9uIGlzIGNhbGxlZCB3aXRoIGluZm9ybWF0aW9uIGFib3V0IHRoZSB3b3JrZmxvdyBqb2IuXG4gKlxuICogQGludGVybmFsXG4gKi9cbmV4cG9ydCBjbGFzcyBHaXRodWJXZWJob29rSGFuZGxlciBleHRlbmRzIENvbnN0cnVjdCB7XG5cbiAgLyoqXG4gICAqIFB1YmxpYyBVUkwgb2Ygd2ViaG9vayB0byBiZSB1c2VkIHdpdGggR2l0SHViLlxuICAgKi9cbiAgcmVhZG9ubHkgdXJsOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFdlYmhvb2sgZXZlbnQgaGFuZGxlci5cbiAgICovXG4gIHJlYWRvbmx5IGhhbmRsZXI6IFdlYmhvb2tIYW5kbGVyRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IEdpdGh1YldlYmhvb2tIYW5kbGVyUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgdGhpcy5oYW5kbGVyID0gbmV3IFdlYmhvb2tIYW5kbGVyRnVuY3Rpb24oXG4gICAgICB0aGlzLFxuICAgICAgJ3dlYmhvb2staGFuZGxlcicsXG4gICAgICB7XG4gICAgICAgIGRlc2NyaXB0aW9uOiAnSGFuZGxlIEdpdEh1YiB3ZWJob29rIGFuZCBzdGFydCBydW5uZXIgb3JjaGVzdHJhdG9yJyxcbiAgICAgICAgZW52aXJvbm1lbnQ6IHtcbiAgICAgICAgICBTVEVQX0ZVTkNUSU9OX0FSTjogcHJvcHMub3JjaGVzdHJhdG9yLnN0YXRlTWFjaGluZUFybixcbiAgICAgICAgICBXRUJIT09LX1NFQ1JFVF9BUk46IHByb3BzLnNlY3JldHMud2ViaG9vay5zZWNyZXRBcm4sXG4gICAgICAgICAgR0lUSFVCX1NFQ1JFVF9BUk46IHByb3BzLnNlY3JldHMuZ2l0aHViLnNlY3JldEFybixcbiAgICAgICAgICBHSVRIVUJfUFJJVkFURV9LRVlfU0VDUkVUX0FSTjogcHJvcHMuc2VjcmV0cy5naXRodWJQcml2YXRlS2V5LnNlY3JldEFybixcbiAgICAgICAgICBTVVBQT1JURURfTEFCRUxTOiBKU09OLnN0cmluZ2lmeShwcm9wcy5zdXBwb3J0ZWRMYWJlbHMpLFxuICAgICAgICAgIFJFUVVJUkVfU0VMRl9IT1NURURfTEFCRUw6IHByb3BzLnJlcXVpcmVTZWxmSG9zdGVkTGFiZWwgPyAnMScgOiAnMCcsXG4gICAgICAgIH0sXG4gICAgICAgIHRpbWVvdXQ6IGNkay5EdXJhdGlvbi5zZWNvbmRzKDMxKSxcbiAgICAgICAgbG9nR3JvdXA6IHNpbmdsZXRvbkxvZ0dyb3VwKHRoaXMsIFNpbmdsZXRvbkxvZ1R5cGUuT1JDSEVTVFJBVE9SKSxcbiAgICAgICAgbG9nZ2luZ0Zvcm1hdDogbGFtYmRhLkxvZ2dpbmdGb3JtYXQuSlNPTixcbiAgICAgIH0sXG4gICAgKTtcblxuICAgIGNvbnN0IGFjY2VzcyA9IHByb3BzPy5hY2Nlc3MgPz8gTGFtYmRhQWNjZXNzLmxhbWJkYVVybCgpO1xuICAgIHRoaXMudXJsID0gYWNjZXNzLmJpbmQodGhpcywgJ2FjY2VzcycsIHRoaXMuaGFuZGxlcik7XG5cbiAgICBwcm9wcy5zZWNyZXRzLndlYmhvb2suZ3JhbnRSZWFkKHRoaXMuaGFuZGxlcik7XG4gICAgcHJvcHMuc2VjcmV0cy5naXRodWIuZ3JhbnRSZWFkKHRoaXMuaGFuZGxlcik7XG4gICAgcHJvcHMuc2VjcmV0cy5naXRodWJQcml2YXRlS2V5LmdyYW50UmVhZCh0aGlzLmhhbmRsZXIpO1xuICAgIHByb3BzLm9yY2hlc3RyYXRvci5ncmFudFN0YXJ0RXhlY3V0aW9uKHRoaXMuaGFuZGxlcik7XG4gIH1cbn1cbiJdfQ==
43
+ //# 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;AA6FpE;;;;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;aACjE;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;SACzC,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;AA5CD,oDA4CC","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/**\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        },\n        timeout: cdk.Duration.seconds(31),\n        logGroup: singletonLogGroup(this, SingletonLogType.ORCHESTRATOR),\n        loggingFormat: lambda.LoggingFormat.JSON,\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"]}
package/package.json CHANGED
@@ -17,8 +17,6 @@
17
17
  "bundle:image-builders/aws-image-builder/delete-resources.lambda:watch": "npx projen bundle:image-builders/aws-image-builder/delete-resources.lambda:watch",
18
18
  "bundle:image-builders/aws-image-builder/filter-failed-builds.lambda": "npx projen bundle:image-builders/aws-image-builder/filter-failed-builds.lambda",
19
19
  "bundle:image-builders/aws-image-builder/filter-failed-builds.lambda:watch": "npx projen bundle:image-builders/aws-image-builder/filter-failed-builds.lambda:watch",
20
- "bundle:image-builders/aws-image-builder/versioner.lambda": "npx projen bundle:image-builders/aws-image-builder/versioner.lambda",
21
- "bundle:image-builders/aws-image-builder/versioner.lambda:watch": "npx projen bundle:image-builders/aws-image-builder/versioner.lambda:watch",
22
20
  "bundle:image-builders/build-image.lambda": "npx projen bundle:image-builders/build-image.lambda",
23
21
  "bundle:image-builders/build-image.lambda:watch": "npx projen bundle:image-builders/build-image.lambda:watch",
24
22
  "bundle:providers/ami-root-device.lambda": "npx projen bundle:providers/ami-root-device.lambda",
@@ -72,16 +70,16 @@
72
70
  "organization": false
73
71
  },
74
72
  "devDependencies": {
75
- "@aws-sdk/client-cloudformation": "^3.940.0",
76
- "@aws-sdk/client-codebuild": "^3.940.0",
77
- "@aws-sdk/client-ec2": "^3.940.0",
78
- "@aws-sdk/client-ecr": "^3.940.0",
79
- "@aws-sdk/client-imagebuilder": "^3.940.0",
80
- "@aws-sdk/client-lambda": "^3.940.0",
81
- "@aws-sdk/client-secrets-manager": "^3.940.0",
82
- "@aws-sdk/client-sfn": "^3.940.0",
83
- "@aws-sdk/client-sns": "^3.940.0",
84
- "@aws-sdk/client-ssm": "^3.940.0",
73
+ "@aws-sdk/client-cloudformation": "^3.956.0",
74
+ "@aws-sdk/client-codebuild": "^3.956.0",
75
+ "@aws-sdk/client-ec2": "^3.956.0",
76
+ "@aws-sdk/client-ecr": "^3.956.0",
77
+ "@aws-sdk/client-imagebuilder": "^3.956.0",
78
+ "@aws-sdk/client-lambda": "^3.956.0",
79
+ "@aws-sdk/client-secrets-manager": "^3.956.0",
80
+ "@aws-sdk/client-sfn": "^3.956.0",
81
+ "@aws-sdk/client-sns": "^3.956.0",
82
+ "@aws-sdk/client-ssm": "^3.956.0",
85
83
  "@octokit/auth-app": "^4.0.13",
86
84
  "@octokit/core": "^4.2.4",
87
85
  "@octokit/request-error": "^3.0.3",
@@ -92,7 +90,6 @@
92
90
  "@types/aws-lambda": "^8.10.159",
93
91
  "@types/jest": "^29",
94
92
  "@types/node": "ts5.6",
95
- "@types/semver": "^7.7.1",
96
93
  "@typescript-eslint/eslint-plugin": "^8",
97
94
  "@typescript-eslint/parser": "^8",
98
95
  "aws-cdk": "^2",
@@ -100,7 +97,7 @@
100
97
  "bootstrap": "^5.2.0",
101
98
  "commit-and-tag-version": "^12",
102
99
  "constructs": "10.0.5",
103
- "esbuild": "^0.27.0",
100
+ "esbuild": "^0.27.2",
104
101
  "eslint": "^9",
105
102
  "eslint-import-resolver-typescript": "^2.7.1",
106
103
  "eslint-plugin-import": "^2.32.0",
@@ -108,13 +105,12 @@
108
105
  "jest": "^29",
109
106
  "jest-junit": "^16",
110
107
  "jsii": "5.8.x",
111
- "jsii-diff": "^1.120.0",
108
+ "jsii-diff": "^1.121.0",
112
109
  "jsii-docgen": "^10.5.0",
113
- "jsii-pacmak": "^1.120.0",
110
+ "jsii-pacmak": "^1.121.0",
114
111
  "jsii-rosetta": "5.8.x",
115
- "projen": "^0.98.26",
112
+ "projen": "^0.98.30",
116
113
  "sass": "^1.54.0",
117
- "semver": "^7.7.3",
118
114
  "svelte": "^5",
119
115
  "svelte-check": "^4",
120
116
  "svelte-preprocess": "^6",
@@ -145,7 +141,7 @@
145
141
  "publishConfig": {
146
142
  "access": "public"
147
143
  },
148
- "version": "0.14.15",
144
+ "version": "0.14.16",
149
145
  "jest": {
150
146
  "coverageProvider": "v8",
151
147
  "testMatch": [