@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.
- package/.jsii +800 -274
- package/API.md +480 -6
- package/README.md +149 -0
- package/SETUP_GITHUB.md +99 -3
- package/assets/delete-failed-runner.lambda/index.js +40 -31
- package/assets/idle-runner-repear.lambda/index.js +40 -31
- package/assets/setup.lambda/index.html +12 -7
- package/assets/setup.lambda/index.js +26 -26
- package/assets/status.lambda/index.js +40 -31
- package/assets/token-retriever.lambda/index.js +40 -31
- package/assets/webhook-handler.lambda/index.js +110 -43
- package/assets/webhook-redelivery.lambda/index.js +40 -31
- package/lib/access.js +1 -1
- package/lib/image-builders/api.js +1 -1
- package/lib/image-builders/aws-image-builder/ami.d.ts +1 -2
- package/lib/image-builders/aws-image-builder/ami.js +6 -13
- package/lib/image-builders/aws-image-builder/builder.d.ts +4 -2
- package/lib/image-builders/aws-image-builder/builder.js +36 -34
- package/lib/image-builders/aws-image-builder/container.d.ts +2 -2
- package/lib/image-builders/aws-image-builder/container.js +7 -12
- package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/container.js +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
- package/lib/image-builders/aws-image-builder/index.d.ts +0 -1
- package/lib/image-builders/aws-image-builder/index.js +1 -2
- package/lib/image-builders/aws-image-builder/workflow.d.ts +4 -4
- package/lib/image-builders/aws-image-builder/workflow.js +7 -10
- package/lib/image-builders/codebuild-deprecated.js +1 -1
- package/lib/image-builders/components.js +1 -1
- package/lib/image-builders/static.js +1 -1
- package/lib/index.d.ts +1 -0
- package/lib/index.js +2 -1
- package/lib/providers/codebuild.js +16 -10
- package/lib/providers/common.d.ts +53 -0
- package/lib/providers/common.js +11 -4
- package/lib/providers/composite.d.ts +61 -0
- package/lib/providers/composite.js +229 -0
- package/lib/providers/ec2.js +9 -8
- package/lib/providers/ecs.js +11 -6
- package/lib/providers/fargate.js +14 -9
- package/lib/providers/index.d.ts +1 -0
- package/lib/providers/index.js +2 -1
- package/lib/providers/lambda.js +6 -5
- package/lib/runner.d.ts +29 -5
- package/lib/runner.js +57 -24
- package/lib/secrets.js +1 -1
- package/lib/webhook-handler.lambda.d.ts +11 -0
- package/lib/webhook-handler.lambda.js +81 -14
- package/lib/webhook.d.ts +52 -7
- package/lib/webhook.js +4 -2
- package/package.json +15 -19
- package/assets/image-builders/aws-image-builder/versioner.lambda/index.js +0 -2115
- package/lib/image-builders/aws-image-builder/common.d.ts +0 -10
- package/lib/image-builders/aws-image-builder/common.js +0 -48
- package/lib/image-builders/aws-image-builder/versioner-function.d.ts +0 -13
- package/lib/image-builders/aws-image-builder/versioner-function.js +0 -23
- package/lib/image-builders/aws-image-builder/versioner.lambda.d.ts +0 -7
- 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
|
-
|
|
62
|
-
|
|
63
|
-
|
|
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
|
|
66
|
-
const
|
|
67
|
-
if (
|
|
68
|
-
return
|
|
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.
|
|
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
|
-
//
|
|
145
|
-
const
|
|
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
|
-
|
|
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
|
-
*
|
|
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
|
|
10
|
-
|
|
11
|
-
|
|
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
|
-
*
|
|
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
|
|
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
|
-
|
|
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,
|
|
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.
|
|
76
|
-
"@aws-sdk/client-codebuild": "^3.
|
|
77
|
-
"@aws-sdk/client-ec2": "^3.
|
|
78
|
-
"@aws-sdk/client-ecr": "^3.
|
|
79
|
-
"@aws-sdk/client-imagebuilder": "^3.
|
|
80
|
-
"@aws-sdk/client-lambda": "^3.
|
|
81
|
-
"@aws-sdk/client-secrets-manager": "^3.
|
|
82
|
-
"@aws-sdk/client-sfn": "^3.
|
|
83
|
-
"@aws-sdk/client-sns": "^3.
|
|
84
|
-
"@aws-sdk/client-ssm": "^3.
|
|
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.
|
|
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.
|
|
108
|
+
"jsii-diff": "^1.121.0",
|
|
112
109
|
"jsii-docgen": "^10.5.0",
|
|
113
|
-
"jsii-pacmak": "^1.
|
|
110
|
+
"jsii-pacmak": "^1.121.0",
|
|
114
111
|
"jsii-rosetta": "5.8.x",
|
|
115
|
-
"projen": "^0.98.
|
|
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.
|
|
144
|
+
"version": "0.14.16",
|
|
149
145
|
"jest": {
|
|
150
146
|
"coverageProvider": "v8",
|
|
151
147
|
"testMatch": [
|