@cloudsnorkel/cdk-github-runners 0.14.24 → 0.15.0
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 +5400 -255
- package/API.md +1048 -24
- package/README.md +52 -0
- package/assets/delete-failed-runner.lambda/index.js +105 -9
- package/assets/idle-runner-repear.lambda/index.js +136 -14
- package/assets/image-builders/aws-image-builder/delete-resources.lambda/index.js +1 -1
- package/assets/image-builders/build-image.lambda/index.js +1 -1
- package/assets/providers/ami-root-device.lambda/index.js +1 -1
- package/assets/setup.lambda/index.html +7 -7
- package/assets/setup.lambda/index.js +101 -8
- package/assets/status.lambda/index.js +104 -8
- package/assets/token-retriever.lambda/index.js +104 -8
- package/assets/warm-runner-manager.lambda/index.js +5892 -0
- package/assets/webhook-handler.lambda/index.js +109 -11
- package/assets/webhook-redelivery.lambda/index.js +122 -24
- package/lib/access.js +1 -1
- package/lib/delete-failed-runner.lambda.js +2 -2
- package/lib/idle-runner-repear.lambda.js +33 -7
- package/lib/image-builders/api.js +1 -1
- package/lib/image-builders/aws-image-builder/base-image.d.ts +13 -0
- package/lib/image-builders/aws-image-builder/base-image.js +36 -3
- package/lib/image-builders/aws-image-builder/builder.js +4 -4
- package/lib/image-builders/aws-image-builder/delete-resources.lambda.js +2 -2
- 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/build-image.lambda.js +2 -2
- package/lib/image-builders/codebuild-deprecated.js +1 -1
- package/lib/image-builders/components.js +3 -3
- package/lib/image-builders/static.js +1 -1
- package/lib/index.d.ts +1 -0
- package/lib/index.js +2 -1
- package/lib/lambda-github.d.ts +1 -1
- package/lib/lambda-github.js +3 -2
- package/lib/lambda-helpers.js +4 -4
- package/lib/providers/ami-root-device.lambda.js +2 -2
- package/lib/providers/codebuild.d.ts +16 -0
- package/lib/providers/codebuild.js +15 -4
- package/lib/providers/common.js +3 -3
- package/lib/providers/composite.js +1 -1
- package/lib/providers/ec2.d.ts +5 -0
- package/lib/providers/ec2.js +31 -17
- package/lib/providers/ecs.d.ts +17 -0
- package/lib/providers/ecs.js +43 -38
- package/lib/providers/fargate.js +9 -31
- package/lib/providers/lambda.js +2 -2
- package/lib/runner.d.ts +25 -2
- package/lib/runner.js +119 -17
- package/lib/secrets.js +1 -1
- package/lib/setup.lambda.js +2 -2
- package/lib/utils.d.ts +10 -1
- package/lib/utils.js +15 -1
- package/lib/warm-runner-manager-function.d.ts +18 -0
- package/lib/warm-runner-manager-function.js +24 -0
- package/lib/warm-runner-manager.lambda.d.ts +41 -0
- package/lib/warm-runner-manager.lambda.js +487 -0
- package/lib/warm-runner.d.ts +147 -0
- package/lib/warm-runner.js +210 -0
- package/lib/webhook-handler.lambda.js +5 -3
- package/lib/webhook-redelivery.lambda.js +17 -16
- package/lib/webhook.d.ts +4 -0
- package/lib/webhook.js +2 -1
- package/node_modules/cron-parser/LICENSE +21 -0
- package/node_modules/cron-parser/README.md +408 -0
- package/node_modules/cron-parser/dist/CronDate.js +518 -0
- package/node_modules/cron-parser/dist/CronExpression.js +520 -0
- package/node_modules/cron-parser/dist/CronExpressionParser.js +382 -0
- package/node_modules/cron-parser/dist/CronFieldCollection.js +371 -0
- package/node_modules/cron-parser/dist/CronFileParser.js +109 -0
- package/node_modules/cron-parser/dist/fields/CronDayOfMonth.js +44 -0
- package/node_modules/cron-parser/dist/fields/CronDayOfWeek.js +51 -0
- package/node_modules/cron-parser/dist/fields/CronField.js +214 -0
- package/node_modules/cron-parser/dist/fields/CronHour.js +40 -0
- package/node_modules/cron-parser/dist/fields/CronMinute.js +40 -0
- package/node_modules/cron-parser/dist/fields/CronMonth.js +44 -0
- package/node_modules/cron-parser/dist/fields/CronSecond.js +40 -0
- package/node_modules/cron-parser/dist/fields/index.js +24 -0
- package/node_modules/cron-parser/dist/fields/types.js +2 -0
- package/node_modules/cron-parser/dist/index.js +31 -0
- package/node_modules/cron-parser/dist/types/CronDate.d.ts +288 -0
- package/node_modules/cron-parser/dist/types/CronExpression.d.ts +118 -0
- package/node_modules/cron-parser/dist/types/CronExpressionParser.d.ts +70 -0
- package/node_modules/cron-parser/dist/types/CronFieldCollection.d.ts +153 -0
- package/node_modules/cron-parser/dist/types/CronFileParser.d.ts +30 -0
- package/node_modules/cron-parser/dist/types/fields/CronDayOfMonth.d.ts +25 -0
- package/node_modules/cron-parser/dist/types/fields/CronDayOfWeek.d.ts +30 -0
- package/node_modules/cron-parser/dist/types/fields/CronField.d.ts +130 -0
- package/node_modules/cron-parser/dist/types/fields/CronHour.d.ts +23 -0
- package/node_modules/cron-parser/dist/types/fields/CronMinute.d.ts +23 -0
- package/node_modules/cron-parser/dist/types/fields/CronMonth.d.ts +24 -0
- package/node_modules/cron-parser/dist/types/fields/CronSecond.d.ts +23 -0
- package/node_modules/cron-parser/dist/types/fields/index.d.ts +8 -0
- package/node_modules/cron-parser/dist/types/fields/types.d.ts +18 -0
- package/node_modules/cron-parser/dist/types/index.d.ts +8 -0
- package/node_modules/cron-parser/dist/types/utils/random.d.ts +10 -0
- package/node_modules/cron-parser/dist/utils/random.js +38 -0
- package/node_modules/cron-parser/package.json +117 -0
- package/node_modules/luxon/LICENSE.md +7 -0
- package/node_modules/luxon/README.md +55 -0
- package/node_modules/luxon/build/amd/luxon.js +8741 -0
- package/node_modules/luxon/build/amd/luxon.js.map +1 -0
- package/node_modules/luxon/build/cjs-browser/luxon.js +8739 -0
- package/node_modules/luxon/build/cjs-browser/luxon.js.map +1 -0
- package/node_modules/luxon/build/es6/luxon.mjs +8133 -0
- package/node_modules/luxon/build/es6/luxon.mjs.map +1 -0
- package/node_modules/luxon/build/global/luxon.js +8744 -0
- package/node_modules/luxon/build/global/luxon.js.map +1 -0
- package/node_modules/luxon/build/global/luxon.min.js +1 -0
- package/node_modules/luxon/build/global/luxon.min.js.map +1 -0
- package/node_modules/luxon/build/node/luxon.js +7792 -0
- package/node_modules/luxon/build/node/luxon.js.map +1 -0
- package/node_modules/luxon/package.json +87 -0
- package/node_modules/luxon/src/datetime.js +2603 -0
- package/node_modules/luxon/src/duration.js +1009 -0
- package/node_modules/luxon/src/errors.js +61 -0
- package/node_modules/luxon/src/impl/conversions.js +206 -0
- package/node_modules/luxon/src/impl/diff.js +95 -0
- package/node_modules/luxon/src/impl/digits.js +94 -0
- package/node_modules/luxon/src/impl/english.js +233 -0
- package/node_modules/luxon/src/impl/formats.js +176 -0
- package/node_modules/luxon/src/impl/formatter.js +434 -0
- package/node_modules/luxon/src/impl/invalid.js +14 -0
- package/node_modules/luxon/src/impl/locale.js +569 -0
- package/node_modules/luxon/src/impl/regexParser.js +335 -0
- package/node_modules/luxon/src/impl/tokenParser.js +505 -0
- package/node_modules/luxon/src/impl/util.js +330 -0
- package/node_modules/luxon/src/impl/zoneUtil.js +34 -0
- package/node_modules/luxon/src/info.js +205 -0
- package/node_modules/luxon/src/interval.js +669 -0
- package/node_modules/luxon/src/luxon.js +26 -0
- package/node_modules/luxon/src/package.json +4 -0
- package/node_modules/luxon/src/settings.js +180 -0
- package/node_modules/luxon/src/zone.js +97 -0
- package/node_modules/luxon/src/zones/IANAZone.js +235 -0
- package/node_modules/luxon/src/zones/fixedOffsetZone.js +150 -0
- package/node_modules/luxon/src/zones/invalidZone.js +53 -0
- package/node_modules/luxon/src/zones/systemZone.js +61 -0
- package/package.json +33 -24
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import * as cdk from 'aws-cdk-lib';
|
|
2
|
+
import { aws_events as events } from 'aws-cdk-lib';
|
|
3
|
+
import { Construct } from 'constructs';
|
|
4
|
+
import { ICompositeProvider, IRunnerProvider } from './providers';
|
|
5
|
+
import { GitHubRunners } from './runner';
|
|
6
|
+
import { WarmRunnerFillPayload } from './warm-runner-manager.lambda';
|
|
7
|
+
/**
|
|
8
|
+
* Common properties for warm runner constructs.
|
|
9
|
+
*
|
|
10
|
+
* @internal
|
|
11
|
+
*/
|
|
12
|
+
export interface WarmRunnerBaseProps {
|
|
13
|
+
/**
|
|
14
|
+
* The GitHubRunners construct that owns the shared warm runner infrastructure.
|
|
15
|
+
*/
|
|
16
|
+
readonly runners: GitHubRunners;
|
|
17
|
+
/**
|
|
18
|
+
* Provider to use. Warm runners bypass the provider selector — they always use
|
|
19
|
+
* this provider, regardless of job characteristics. Labels cannot be modified.
|
|
20
|
+
*/
|
|
21
|
+
readonly provider: IRunnerProvider | ICompositeProvider;
|
|
22
|
+
/**
|
|
23
|
+
* Number of warm runners to maintain.
|
|
24
|
+
*/
|
|
25
|
+
readonly count: number;
|
|
26
|
+
/**
|
|
27
|
+
* GitHub owner where runners will be registered (org or user login).
|
|
28
|
+
*/
|
|
29
|
+
readonly owner: string;
|
|
30
|
+
/**
|
|
31
|
+
* Registration level — must match how your runners are set up in GitHub. Choose
|
|
32
|
+
* 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard or
|
|
33
|
+
* {@link SETUP_GITHUB.md} for choosing repo vs org.
|
|
34
|
+
*
|
|
35
|
+
* @default 'repo'
|
|
36
|
+
*/
|
|
37
|
+
readonly registrationLevel?: 'org' | 'repo';
|
|
38
|
+
/**
|
|
39
|
+
* Repository name (without owner) where runners will be registered. Required when `registrationLevel` is 'repo'.
|
|
40
|
+
*/
|
|
41
|
+
readonly repo?: string;
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Properties for always on warm runners.
|
|
45
|
+
*/
|
|
46
|
+
export interface AlwaysOnWarmRunnerProps extends WarmRunnerBaseProps {
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Properties for scheduled warm runners.
|
|
50
|
+
*/
|
|
51
|
+
export interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {
|
|
52
|
+
/**
|
|
53
|
+
* When to start filling the pool (e.g. start of business hours).
|
|
54
|
+
*/
|
|
55
|
+
readonly schedule: events.Schedule;
|
|
56
|
+
/**
|
|
57
|
+
* How long the warm runners should be maintained from the fill time (schedule). Defines the end of the
|
|
58
|
+
* window (schedule time + duration).
|
|
59
|
+
*/
|
|
60
|
+
readonly duration: cdk.Duration;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Warm runners that run 24/7. Fills at midnight UTC and each runner stays alive for 24 hours.
|
|
64
|
+
*
|
|
65
|
+
* Runners will be provisioned using the specified provider and registered in the specified repository or organization.
|
|
66
|
+
*
|
|
67
|
+
* Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
|
|
68
|
+
*
|
|
69
|
+
* ## Limitations
|
|
70
|
+
*
|
|
71
|
+
* - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.
|
|
72
|
+
* - You may briefly see more than `count` runners when changing config or at rotation.
|
|
73
|
+
* - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.
|
|
74
|
+
* If you don't follow this procedure, warm runners may linger until they expire.
|
|
75
|
+
* - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a
|
|
76
|
+
* gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
|
|
77
|
+
* can be tweaked using `retryOptions`. This will be improved in the future.
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {
|
|
81
|
+
* runners,
|
|
82
|
+
* provider: myProvider,
|
|
83
|
+
* count: 3,
|
|
84
|
+
* owner: 'my-org',
|
|
85
|
+
* repo: 'my-repo',
|
|
86
|
+
* });
|
|
87
|
+
*/
|
|
88
|
+
export declare class AlwaysOnWarmRunner extends Construct {
|
|
89
|
+
/**
|
|
90
|
+
* The fill payload for this warm runner configuration.
|
|
91
|
+
* @internal
|
|
92
|
+
*/
|
|
93
|
+
readonly _fillPayload: WarmRunnerFillPayload;
|
|
94
|
+
constructor(scope: Construct, id: string, props: AlwaysOnWarmRunnerProps);
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Warm runners active during a time window specified by start time (`schedule`) and duration (`duration`).
|
|
98
|
+
*
|
|
99
|
+
* Runners will be provisioned using the specified provider and registered in the specified repository or organization.
|
|
100
|
+
*
|
|
101
|
+
* Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
|
|
102
|
+
*
|
|
103
|
+
* ## Limitations
|
|
104
|
+
*
|
|
105
|
+
* - **No deployment-fill**: Unlike `AlwaysOnWarmRunner`, scheduled warm runners do not get an initial
|
|
106
|
+
* fill on deploy. The first fill happens at the next schedule occurrence. If you deploy at 1pm for
|
|
107
|
+
* a 2pm schedule, runners will not appear until 2pm.
|
|
108
|
+
* - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.
|
|
109
|
+
* - You may briefly see more than `count` runners when changing config or at rotation.
|
|
110
|
+
* - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.
|
|
111
|
+
* If you don't follow this procedure, warm runners may linger until they expire.
|
|
112
|
+
* - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a
|
|
113
|
+
* gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
|
|
114
|
+
* can be tweaked using `retryOptions`. This will be improved in the future.
|
|
115
|
+
*
|
|
116
|
+
* @example
|
|
117
|
+
* // Cron: fill at 1pm on weekdays
|
|
118
|
+
* new ScheduledWarmRunner(stack, 'Business Hours', {
|
|
119
|
+
* runners,
|
|
120
|
+
* provider: myProvider,
|
|
121
|
+
* count: 3,
|
|
122
|
+
* owner: 'my-org',
|
|
123
|
+
* repo: 'my-repo',
|
|
124
|
+
* schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),
|
|
125
|
+
* duration: cdk.Duration.hours(2),
|
|
126
|
+
* });
|
|
127
|
+
*
|
|
128
|
+
* @example
|
|
129
|
+
* // Rate: fill every 12 hours
|
|
130
|
+
* new ScheduledWarmRunner(stack, 'Every 12 Hours', {
|
|
131
|
+
* runners,
|
|
132
|
+
* provider: myProvider,
|
|
133
|
+
* count: 2,
|
|
134
|
+
* owner: 'my-org',
|
|
135
|
+
* repo: 'my-repo',
|
|
136
|
+
* schedule: events.Schedule.rate(cdk.Duration.hours(5)),
|
|
137
|
+
* duration: cdk.Duration.hours(12),
|
|
138
|
+
* });
|
|
139
|
+
*/
|
|
140
|
+
export declare class ScheduledWarmRunner extends Construct {
|
|
141
|
+
/**
|
|
142
|
+
* The fill payload for this warm runner configuration.
|
|
143
|
+
* @internal
|
|
144
|
+
*/
|
|
145
|
+
readonly _fillPayload: WarmRunnerFillPayload;
|
|
146
|
+
constructor(scope: Construct, id: string, props: ScheduledWarmRunnerProps);
|
|
147
|
+
}
|
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a, _b;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.ScheduledWarmRunner = exports.AlwaysOnWarmRunner = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const crypto = require("crypto");
|
|
7
|
+
const cdk = require("aws-cdk-lib");
|
|
8
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
9
|
+
const constructs_1 = require("constructs");
|
|
10
|
+
const cron_parser_1 = require("cron-parser");
|
|
11
|
+
function buildWarmRunner(scope, props, schedule, duration, createInitialFill) {
|
|
12
|
+
const registrationLevel = props.registrationLevel ?? 'repo';
|
|
13
|
+
if (registrationLevel === 'org' && props.repo) {
|
|
14
|
+
throw new Error('Do not specify repo when registrationLevel is \'org\'');
|
|
15
|
+
}
|
|
16
|
+
if (registrationLevel === 'repo' && !props.repo) {
|
|
17
|
+
throw new Error('repo is required when registrationLevel is \'repo\'');
|
|
18
|
+
}
|
|
19
|
+
const providerPath = props.provider.node.path;
|
|
20
|
+
if (!props.runners.providers.some(p => p.node.path === providerPath)) {
|
|
21
|
+
throw new Error(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);
|
|
22
|
+
}
|
|
23
|
+
const labels = props.provider.labels;
|
|
24
|
+
const repo = registrationLevel === 'repo' ? props.repo : '';
|
|
25
|
+
const configHash = crypto.createHash('sha256')
|
|
26
|
+
.update(JSON.stringify({ providerPath, providerLabels: labels, count: props.count, duration, owner: props.owner, repo }))
|
|
27
|
+
.digest('hex')
|
|
28
|
+
.slice(0, 16);
|
|
29
|
+
const fillPayload = {
|
|
30
|
+
action: 'fill',
|
|
31
|
+
providerPath,
|
|
32
|
+
providerLabels: labels,
|
|
33
|
+
count: props.count,
|
|
34
|
+
duration,
|
|
35
|
+
owner: props.owner,
|
|
36
|
+
repo,
|
|
37
|
+
configHash,
|
|
38
|
+
};
|
|
39
|
+
const { lambda: managerFn, queue } = props.runners._ensureWarmRunnerInfra();
|
|
40
|
+
props.runners._registerWarmConfigHash(configHash);
|
|
41
|
+
// Schedule to fill the warm pool (usually daily). Sends to SQS so we get stable messageId for idempotent fills.
|
|
42
|
+
new aws_cdk_lib_1.aws_events.Rule(scope, 'Schedule', {
|
|
43
|
+
schedule,
|
|
44
|
+
targets: [new aws_cdk_lib_1.aws_events_targets.SqsQueue(queue, {
|
|
45
|
+
message: aws_cdk_lib_1.aws_events.RuleTargetInput.fromObject(fillPayload),
|
|
46
|
+
})],
|
|
47
|
+
});
|
|
48
|
+
// Fill the warm pool immediately on deploy (AlwaysOnWarmRunner only).
|
|
49
|
+
// ScheduledWarmRunner does not get deployment-fill. First fill happens at the next schedule fire.
|
|
50
|
+
if (createInitialFill) {
|
|
51
|
+
new cdk.CustomResource(scope, 'Initial Fill', {
|
|
52
|
+
serviceToken: managerFn.functionArn,
|
|
53
|
+
resourceType: 'Custom::WarmRunnerFill',
|
|
54
|
+
properties: fillPayload,
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
return fillPayload;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Warm runners that run 24/7. Fills at midnight UTC and each runner stays alive for 24 hours.
|
|
61
|
+
*
|
|
62
|
+
* Runners will be provisioned using the specified provider and registered in the specified repository or organization.
|
|
63
|
+
*
|
|
64
|
+
* Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
|
|
65
|
+
*
|
|
66
|
+
* ## Limitations
|
|
67
|
+
*
|
|
68
|
+
* - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.
|
|
69
|
+
* - You may briefly see more than `count` runners when changing config or at rotation.
|
|
70
|
+
* - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.
|
|
71
|
+
* If you don't follow this procedure, warm runners may linger until they expire.
|
|
72
|
+
* - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a
|
|
73
|
+
* gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
|
|
74
|
+
* can be tweaked using `retryOptions`. This will be improved in the future.
|
|
75
|
+
*
|
|
76
|
+
* @example
|
|
77
|
+
* new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {
|
|
78
|
+
* runners,
|
|
79
|
+
* provider: myProvider,
|
|
80
|
+
* count: 3,
|
|
81
|
+
* owner: 'my-org',
|
|
82
|
+
* repo: 'my-repo',
|
|
83
|
+
* });
|
|
84
|
+
*/
|
|
85
|
+
class AlwaysOnWarmRunner extends constructs_1.Construct {
|
|
86
|
+
constructor(scope, id, props) {
|
|
87
|
+
super(scope, id);
|
|
88
|
+
this._fillPayload = buildWarmRunner(this, props, aws_cdk_lib_1.aws_events.Schedule.cron({ hour: '0', minute: '0' }), cdk.Duration.days(1).toSeconds(), true);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
exports.AlwaysOnWarmRunner = AlwaysOnWarmRunner;
|
|
92
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
93
|
+
AlwaysOnWarmRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.AlwaysOnWarmRunner", version: "0.15.0" };
|
|
94
|
+
/**
|
|
95
|
+
* Convert AWS EventBridge cron format to cron-parser format.
|
|
96
|
+
* AWS: cron(min hour dom month dow year), cron-parser: sec min hour dom month dow
|
|
97
|
+
*/
|
|
98
|
+
function awsCronToParserFormat(expressionString) {
|
|
99
|
+
const match = expressionString.match(/^cron\((.+)\)$/);
|
|
100
|
+
if (!match)
|
|
101
|
+
return expressionString;
|
|
102
|
+
const [, inner] = match;
|
|
103
|
+
const parts = inner.trim().split(/\s+/);
|
|
104
|
+
if (parts.length !== 6)
|
|
105
|
+
return expressionString;
|
|
106
|
+
const [minute, hour, dom, month, dow] = parts;
|
|
107
|
+
return `0 ${minute} ${hour} ${dom} ${month} ${dow}`;
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Parse AWS EventBridge rate expression and return interval in seconds.
|
|
111
|
+
* Format: rate(value unit) e.g. rate(2 hours), rate(5 minutes), rate(1 day)
|
|
112
|
+
*/
|
|
113
|
+
function parseRateInterval(expressionString) {
|
|
114
|
+
const match = expressionString.match(/^rate\((\d+)\s+(minute|minutes|hour|hours|day|days)\)$/i);
|
|
115
|
+
if (!match)
|
|
116
|
+
return undefined;
|
|
117
|
+
const value = parseInt(match[1], 10);
|
|
118
|
+
const unit = match[2].toLowerCase();
|
|
119
|
+
if (value <= 0)
|
|
120
|
+
return undefined;
|
|
121
|
+
const secondsPerUnit = {
|
|
122
|
+
minute: 60,
|
|
123
|
+
minutes: 60,
|
|
124
|
+
hour: 3600,
|
|
125
|
+
hours: 3600,
|
|
126
|
+
day: 86400,
|
|
127
|
+
days: 86400,
|
|
128
|
+
};
|
|
129
|
+
return value * secondsPerUnit[unit];
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Get the interval between schedule occurrences in seconds.
|
|
133
|
+
* Supports both cron and rate expressions.
|
|
134
|
+
*/
|
|
135
|
+
function getScheduleIntervalSeconds(expressionString) {
|
|
136
|
+
const rateInterval = parseRateInterval(expressionString);
|
|
137
|
+
if (rateInterval !== undefined)
|
|
138
|
+
return rateInterval;
|
|
139
|
+
try {
|
|
140
|
+
const cronExpression = cron_parser_1.CronExpressionParser.parse(awsCronToParserFormat(expressionString));
|
|
141
|
+
const next = cronExpression.take(2);
|
|
142
|
+
return (next[1].getTime() - next[0].getTime()) / 1000;
|
|
143
|
+
}
|
|
144
|
+
catch {
|
|
145
|
+
return undefined;
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Warm runners active during a time window specified by start time (`schedule`) and duration (`duration`).
|
|
150
|
+
*
|
|
151
|
+
* Runners will be provisioned using the specified provider and registered in the specified repository or organization.
|
|
152
|
+
*
|
|
153
|
+
* Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.
|
|
154
|
+
*
|
|
155
|
+
* ## Limitations
|
|
156
|
+
*
|
|
157
|
+
* - **No deployment-fill**: Unlike `AlwaysOnWarmRunner`, scheduled warm runners do not get an initial
|
|
158
|
+
* fill on deploy. The first fill happens at the next schedule occurrence. If you deploy at 1pm for
|
|
159
|
+
* a 2pm schedule, runners will not appear until 2pm.
|
|
160
|
+
* - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.
|
|
161
|
+
* - You may briefly see more than `count` runners when changing config or at rotation.
|
|
162
|
+
* - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.
|
|
163
|
+
* If you don't follow this procedure, warm runners may linger until they expire.
|
|
164
|
+
* - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a
|
|
165
|
+
* gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and
|
|
166
|
+
* can be tweaked using `retryOptions`. This will be improved in the future.
|
|
167
|
+
*
|
|
168
|
+
* @example
|
|
169
|
+
* // Cron: fill at 1pm on weekdays
|
|
170
|
+
* new ScheduledWarmRunner(stack, 'Business Hours', {
|
|
171
|
+
* runners,
|
|
172
|
+
* provider: myProvider,
|
|
173
|
+
* count: 3,
|
|
174
|
+
* owner: 'my-org',
|
|
175
|
+
* repo: 'my-repo',
|
|
176
|
+
* schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),
|
|
177
|
+
* duration: cdk.Duration.hours(2),
|
|
178
|
+
* });
|
|
179
|
+
*
|
|
180
|
+
* @example
|
|
181
|
+
* // Rate: fill every 12 hours
|
|
182
|
+
* new ScheduledWarmRunner(stack, 'Every 12 Hours', {
|
|
183
|
+
* runners,
|
|
184
|
+
* provider: myProvider,
|
|
185
|
+
* count: 2,
|
|
186
|
+
* owner: 'my-org',
|
|
187
|
+
* repo: 'my-repo',
|
|
188
|
+
* schedule: events.Schedule.rate(cdk.Duration.hours(5)),
|
|
189
|
+
* duration: cdk.Duration.hours(12),
|
|
190
|
+
* });
|
|
191
|
+
*/
|
|
192
|
+
class ScheduledWarmRunner extends constructs_1.Construct {
|
|
193
|
+
constructor(scope, id, props) {
|
|
194
|
+
super(scope, id);
|
|
195
|
+
// make sure the duration is not longer than the interval between next two schedule occurrences
|
|
196
|
+
const interval = getScheduleIntervalSeconds(props.schedule.expressionString);
|
|
197
|
+
if (interval !== undefined && interval < props.duration.toSeconds()) {
|
|
198
|
+
cdk.Annotations.of(this).addError(`ScheduledWarmRunner duration ${props.duration.toHumanString()} is longer than the interval ${cdk.Duration.seconds(interval).toHumanString()} between next two schedule occurrences. This will result in overlapping warm runners at the start of the next schedule occurrence.`);
|
|
199
|
+
}
|
|
200
|
+
// warn for short interval
|
|
201
|
+
if (interval !== undefined && interval < cdk.Duration.hours(1).toSeconds()) {
|
|
202
|
+
cdk.Annotations.of(this).addWarningV2('@cloudsnorkel/cdk-github-runners:ScheduledWarmRunner.intervalTooShort', `ScheduledWarmRunner interval ${cdk.Duration.seconds(interval).toHumanString()} is less than 1 hour, which may result in more warm runners than expected`);
|
|
203
|
+
}
|
|
204
|
+
this._fillPayload = buildWarmRunner(this, props, props.schedule, props.duration.toSeconds(), false);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
exports.ScheduledWarmRunner = ScheduledWarmRunner;
|
|
208
|
+
_b = JSII_RTTI_SYMBOL_1;
|
|
209
|
+
ScheduledWarmRunner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.ScheduledWarmRunner", version: "0.15.0" };
|
|
210
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"warm-runner.js","sourceRoot":"","sources":["../src/warm-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  aws_events as events,\n  aws_events_targets as events_targets,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { CronExpressionParser } from 'cron-parser';\nimport { ICompositeProvider, IRunnerProvider } from './providers';\nimport { GitHubRunners } from './runner';\nimport { WarmRunnerFillPayload } from './warm-runner-manager.lambda';\n\n/**\n * Common properties for warm runner constructs.\n *\n * @internal\n */\nexport interface WarmRunnerBaseProps {\n  /**\n   * The GitHubRunners construct that owns the shared warm runner infrastructure.\n   */\n  readonly runners: GitHubRunners;\n\n  /**\n   * Provider to use. Warm runners bypass the provider selector — they always use\n   * this provider, regardless of job characteristics. Labels cannot be modified.\n   */\n  readonly provider: IRunnerProvider | ICompositeProvider;\n\n  /**\n   * Number of warm runners to maintain.\n   */\n  readonly count: number;\n\n  /**\n   * GitHub owner where runners will be registered (org or user login).\n   */\n  readonly owner: string;\n\n  /**\n   * Registration level — must match how your runners are set up in GitHub. Choose\n   * 'org' for org-wide runners, 'repo' for repo-level. See the setup wizard or\n   * {@link SETUP_GITHUB.md} for choosing repo vs org.\n   *\n   * @default 'repo'\n   */\n  readonly registrationLevel?: 'org' | 'repo';\n\n  /**\n   * Repository name (without owner) where runners will be registered. Required when `registrationLevel` is 'repo'.\n   */\n  readonly repo?: string;\n}\n\n/**\n * Properties for always on warm runners.\n */\nexport interface AlwaysOnWarmRunnerProps extends WarmRunnerBaseProps { }\n\n/**\n * Properties for scheduled warm runners.\n */\nexport interface ScheduledWarmRunnerProps extends WarmRunnerBaseProps {\n  /**\n   * When to start filling the pool (e.g. start of business hours).\n   */\n  readonly schedule: events.Schedule;\n\n  /**\n   * How long the warm runners should be maintained from the fill time (schedule). Defines the end of the\n   * window (schedule time + duration).\n   */\n  readonly duration: cdk.Duration;\n}\n\nfunction buildWarmRunner(scope: Construct, props: WarmRunnerBaseProps, schedule: events.Schedule, duration: number, createInitialFill: boolean) {\n  const registrationLevel = props.registrationLevel ?? 'repo';\n  if (registrationLevel === 'org' && props.repo) {\n    throw new Error('Do not specify repo when registrationLevel is \\'org\\'');\n  }\n  if (registrationLevel === 'repo' && !props.repo) {\n    throw new Error('repo is required when registrationLevel is \\'repo\\'');\n  }\n\n  const providerPath = props.provider.node.path;\n  if (!props.runners.providers.some(p => p.node.path === providerPath)) {\n    throw new Error(`Provider ${providerPath} is not in the providers list of the GitHubRunners construct`);\n  }\n\n  const labels = props.provider.labels;\n\n  const repo = registrationLevel === 'repo' ? props.repo! : '';\n  const configHash = crypto.createHash('sha256')\n    .update(JSON.stringify({ providerPath, providerLabels: labels, count: props.count, duration, owner: props.owner, repo }))\n    .digest('hex')\n    .slice(0, 16);\n\n  const fillPayload: WarmRunnerFillPayload = {\n    action: 'fill' as const,\n    providerPath,\n    providerLabels: labels,\n    count: props.count,\n    duration,\n    owner: props.owner,\n    repo,\n    configHash,\n  };\n\n  const { lambda: managerFn, queue } = props.runners._ensureWarmRunnerInfra();\n  props.runners._registerWarmConfigHash(configHash);\n\n  // Schedule to fill the warm pool (usually daily). Sends to SQS so we get stable messageId for idempotent fills.\n  new events.Rule(scope, 'Schedule', {\n    schedule,\n    targets: [new events_targets.SqsQueue(queue, {\n      message: events.RuleTargetInput.fromObject(fillPayload),\n    })],\n  });\n\n  // Fill the warm pool immediately on deploy (AlwaysOnWarmRunner only).\n  // ScheduledWarmRunner does not get deployment-fill. First fill happens at the next schedule fire.\n  if (createInitialFill) {\n    new cdk.CustomResource(scope, 'Initial Fill', {\n      serviceToken: managerFn.functionArn,\n      resourceType: 'Custom::WarmRunnerFill',\n      properties: fillPayload,\n    });\n  }\n\n  return fillPayload;\n}\n\n/**\n * Warm runners that run 24/7. Fills at midnight UTC and each runner stays alive for 24 hours.\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.\n *\n * ## Limitations\n *\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * @example\n * new AlwaysOnWarmRunner(stack, 'AlwaysOnLinux', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n * });\n */\nexport class AlwaysOnWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: AlwaysOnWarmRunnerProps) {\n    super(scope, id);\n    this._fillPayload = buildWarmRunner(this, props, events.Schedule.cron({ hour: '0', minute: '0' }), cdk.Duration.days(1).toSeconds(), true);\n  }\n}\n\n/**\n * Convert AWS EventBridge cron format to cron-parser format.\n * AWS: cron(min hour dom month dow year), cron-parser: sec min hour dom month dow\n */\nfunction awsCronToParserFormat(expressionString: string): string {\n  const match = expressionString.match(/^cron\\((.+)\\)$/);\n  if (!match) return expressionString;\n  const [, inner] = match;\n  const parts = inner.trim().split(/\\s+/);\n  if (parts.length !== 6) return expressionString;\n  const [minute, hour, dom, month, dow] = parts;\n  return `0 ${minute} ${hour} ${dom} ${month} ${dow}`;\n}\n\n/**\n * Parse AWS EventBridge rate expression and return interval in seconds.\n * Format: rate(value unit) e.g. rate(2 hours), rate(5 minutes), rate(1 day)\n */\nfunction parseRateInterval(expressionString: string): number | undefined {\n  const match = expressionString.match(/^rate\\((\\d+)\\s+(minute|minutes|hour|hours|day|days)\\)$/i);\n  if (!match) return undefined;\n  const value = parseInt(match[1], 10);\n  const unit = match[2].toLowerCase();\n  if (value <= 0) return undefined;\n  const secondsPerUnit: Record<string, number> = {\n    minute: 60,\n    minutes: 60,\n    hour: 3600,\n    hours: 3600,\n    day: 86400,\n    days: 86400,\n  };\n  return value * secondsPerUnit[unit];\n}\n\n/**\n * Get the interval between schedule occurrences in seconds.\n * Supports both cron and rate expressions.\n */\nfunction getScheduleIntervalSeconds(expressionString: string): number | undefined {\n  const rateInterval = parseRateInterval(expressionString);\n  if (rateInterval !== undefined) return rateInterval;\n\n  try {\n    const cronExpression = CronExpressionParser.parse(awsCronToParserFormat(expressionString));\n    const next = cronExpression.take(2);\n    return (next[1].getTime() - next[0].getTime()) / 1000;\n  } catch {\n    return undefined;\n  }\n}\n\n/**\n * Warm runners active during a time window specified by start time (`schedule`) and duration (`duration`).\n *\n * Runners will be provisioned using the specified provider and registered in the specified repository or organization.\n *\n * Registration level must match the one selected during setup. See {@link SETUP_GITHUB.md} for more information on the selection.\n *\n * ## Limitations\n *\n * - **No deployment-fill**: Unlike `AlwaysOnWarmRunner`, scheduled warm runners do not get an initial\n *   fill on deploy. The first fill happens at the next schedule occurrence. If you deploy at 1pm for\n *   a 2pm schedule, runners will not appear until 2pm.\n * - Jobs will still trigger provisioning of on-demand runners, even if a warm runner ends up being used.\n * - You may briefly see more than `count` runners when changing config or at rotation.\n * - To remove: set `count` to 0, deploy, wait for warm runners to stop, then remove and deploy again.\n *   If you don't follow this procedure, warm runners may linger until they expire.\n * - Provider failures or timeouts (like Lambda provider timing out after 15 minutes) will result in a\n *   gap in coverage until the retry succeeds. Current retry mechanism has built-in back-off rate and\n *   can be tweaked using `retryOptions`. This will be improved in the future.\n *\n * @example\n * // Cron: fill at 1pm on weekdays\n * new ScheduledWarmRunner(stack, 'Business Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 3,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.cron({ hour: '13', minute: '0', weekDay: 'MON-FRI' }),\n *   duration: cdk.Duration.hours(2),\n * });\n *\n * @example\n * // Rate: fill every 12 hours\n * new ScheduledWarmRunner(stack, 'Every 12 Hours', {\n *   runners,\n *   provider: myProvider,\n *   count: 2,\n *   owner: 'my-org',\n *   repo: 'my-repo',\n *   schedule: events.Schedule.rate(cdk.Duration.hours(5)),\n *   duration: cdk.Duration.hours(12),\n * });\n */\nexport class ScheduledWarmRunner extends Construct {\n  /**\n   * The fill payload for this warm runner configuration.\n   * @internal\n   */\n  public readonly _fillPayload: WarmRunnerFillPayload;\n\n  constructor(scope: Construct, id: string, props: ScheduledWarmRunnerProps) {\n    super(scope, id);\n\n    // make sure the duration is not longer than the interval between next two schedule occurrences\n    const interval = getScheduleIntervalSeconds(props.schedule.expressionString);\n    if (interval !== undefined && interval < props.duration.toSeconds()) {\n      cdk.Annotations.of(this).addError(`ScheduledWarmRunner duration ${props.duration.toHumanString()} is longer than the interval ${cdk.Duration.seconds(interval).toHumanString()} between next two schedule occurrences. This will result in overlapping warm runners at the start of the next schedule occurrence.`);\n    }\n\n    // warn for short interval\n    if (interval !== undefined && interval < cdk.Duration.hours(1).toSeconds()) {\n      cdk.Annotations.of(this).addWarningV2(\n        '@cloudsnorkel/cdk-github-runners:ScheduledWarmRunner.intervalTooShort',\n        `ScheduledWarmRunner interval ${cdk.Duration.seconds(interval).toHumanString()} is less than 1 hour, which may result in more warm runners than expected`,\n      );\n    }\n\n    this._fillPayload = buildWarmRunner(this, props, props.schedule, props.duration.toSeconds(), false);\n  }\n}\n"]}
|
|
@@ -63,7 +63,7 @@ async function isDeploymentPending(payload) {
|
|
|
63
63
|
catch (e) {
|
|
64
64
|
console.error({
|
|
65
65
|
notice: 'Unable to check deployment. Try adding deployment read permission.',
|
|
66
|
-
error:
|
|
66
|
+
error: e,
|
|
67
67
|
});
|
|
68
68
|
return false;
|
|
69
69
|
}
|
|
@@ -178,7 +178,7 @@ async function handler(event) {
|
|
|
178
178
|
catch (e) {
|
|
179
179
|
console.error({
|
|
180
180
|
notice: 'Bad signature',
|
|
181
|
-
error:
|
|
181
|
+
error: e,
|
|
182
182
|
});
|
|
183
183
|
return {
|
|
184
184
|
statusCode: 403,
|
|
@@ -259,6 +259,7 @@ async function handler(event) {
|
|
|
259
259
|
}
|
|
260
260
|
// start execution
|
|
261
261
|
const executionName = generateExecutionName(event, payload);
|
|
262
|
+
const idleTimeoutSeconds = process.env.IDLE_TIMEOUT_SECONDS ? parseInt(process.env.IDLE_TIMEOUT_SECONDS, 10) : 300; // default 5 minutes
|
|
262
263
|
const input = {
|
|
263
264
|
owner: payload.repository.owner.login,
|
|
264
265
|
repo: payload.repository.name,
|
|
@@ -268,6 +269,7 @@ async function handler(event) {
|
|
|
268
269
|
jobLabels: payload.workflow_job.labels.join(','), // original labels requested by the job
|
|
269
270
|
provider: selection.provider,
|
|
270
271
|
labels: selection.labels.join(','), // labels to use when registering runner
|
|
272
|
+
maxIdleSeconds: idleTimeoutSeconds,
|
|
271
273
|
};
|
|
272
274
|
const execution = await sf.send(new client_sfn_1.StartExecutionCommand({
|
|
273
275
|
stateMachineArn: process.env.STEP_FUNCTION_ARN,
|
|
@@ -286,4 +288,4 @@ async function handler(event) {
|
|
|
286
288
|
body: executionName,
|
|
287
289
|
};
|
|
288
290
|
}
|
|
289
|
-
//# 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({\n    notice: 'Calculated signature',\n    signature: expectedSig.toString(),\n  });\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({\n      notice: 'Unable to check deployment. Try adding deployment read permission.',\n      error: `${e}`,\n    });\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    const selectorResponsePayload = result.Payload ? Buffer.from(result.Payload).toString() : undefined;\n    console.error({\n      notice: 'Provider selector failed',\n      functionError: result.FunctionError,\n      payload: selectorResponsePayload,\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({\n    notice: 'Before provider selector',\n    provider: defaultProvider,\n    labels: defaultLabels,\n    jobLabels: jobLabels,\n  });\n  console.log({\n    notice: 'After provider selector',\n    provider: selectorResult.provider,\n    labels: selectorResult.labels,\n    jobLabels: jobLabels,\n  });\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({\n      notice: 'Bad signature',\n      error: `${e}`,\n    });\n    return {\n      statusCode: 403,\n      body: 'Bad signature',\n    };\n  }\n\n  if (getHeader(event, 'content-type') !== 'application/json') {\n    console.error({\n      notice: 'This webhook only accepts JSON payloads',\n      contentType: getHeader(event, 'content-type'),\n    });\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({\n      notice: 'This webhook only accepts workflow_job',\n      githubEvent: getHeader(event, 'x-github-event'),\n    });\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"]}
|
|
291
|
+
//# 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({\n    notice: 'Calculated signature',\n    signature: expectedSig.toString(),\n  });\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({\n      notice: 'Unable to check deployment. Try adding deployment read permission.',\n      error: e,\n    });\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    const selectorResponsePayload = result.Payload ? Buffer.from(result.Payload).toString() : undefined;\n    console.error({\n      notice: 'Provider selector failed',\n      functionError: result.FunctionError,\n      payload: selectorResponsePayload,\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({\n    notice: 'Before provider selector',\n    provider: defaultProvider,\n    labels: defaultLabels,\n    jobLabels: jobLabels,\n  });\n  console.log({\n    notice: 'After provider selector',\n    provider: selectorResult.provider,\n    labels: selectorResult.labels,\n    jobLabels: jobLabels,\n  });\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({\n      notice: 'Bad signature',\n      error: e,\n    });\n    return {\n      statusCode: 403,\n      body: 'Bad signature',\n    };\n  }\n\n  if (getHeader(event, 'content-type') !== 'application/json') {\n    console.error({\n      notice: 'This webhook only accepts JSON payloads',\n      contentType: getHeader(event, 'content-type'),\n    });\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({\n      notice: 'This webhook only accepts workflow_job',\n      githubEvent: getHeader(event, 'x-github-event'),\n    });\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 idleTimeoutSeconds = process.env.IDLE_TIMEOUT_SECONDS ? parseInt(process.env.IDLE_TIMEOUT_SECONDS, 10) : 300; // default 5 minutes\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    maxIdleSeconds: idleTimeoutSeconds,\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"]}
|