@cloudsnorkel/cdk-github-runners 0.9.7 → 0.10.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/.gitattributes +2 -0
- package/.jsii +275 -303
- package/API.md +215 -39
- package/README.md +8 -7
- package/assets/delete-runner.lambda/index.js +2983 -5504
- package/assets/docker-images/lambda/linux-arm64/runner.sh +1 -1
- package/assets/docker-images/lambda/linux-x64/runner.sh +1 -1
- package/assets/idle-runner-repear.lambda/index.js +20062 -0
- package/assets/setup.lambda/index.html +14 -14
- package/assets/setup.lambda/index.js +102 -153
- package/assets/status.lambda/index.js +2925 -5418
- package/assets/token-retriever.lambda/index.js +3000 -5497
- package/assets/webhook-handler.lambda/index.js +1 -1
- package/lib/access.d.ts +11 -1
- package/lib/access.js +56 -6
- package/lib/delete-runner.lambda.js +7 -35
- package/lib/idle-runner-repear-function.d.ts +13 -0
- package/lib/idle-runner-repear-function.js +23 -0
- package/lib/idle-runner-repear.lambda.d.ts +1 -0
- package/lib/idle-runner-repear.lambda.js +67 -0
- package/lib/image-builders/api.js +1 -1
- package/lib/image-builders/aws-image-builder/builder.js +1 -1
- package/lib/image-builders/aws-image-builder/delete-ami.lambda.js +1 -3
- 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/filter-failed-builds.lambda.js +1 -2
- package/lib/image-builders/aws-image-builder/reaper.lambda.js +1 -3
- package/lib/image-builders/aws-image-builder/versioner.lambda.js +1 -3
- 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/{github.d.ts → lambda-github.d.ts} +14 -0
- package/lib/lambda-github.js +69 -0
- package/lib/lambda-helpers.d.ts +0 -1
- package/lib/lambda-helpers.js +1 -2
- package/lib/providers/build-image.lambda.js +1 -3
- package/lib/providers/codebuild.d.ts +1 -0
- package/lib/providers/codebuild.js +9 -6
- package/lib/providers/common.d.ts +8 -8
- package/lib/providers/common.js +5 -16
- package/lib/providers/ec2.d.ts +1 -0
- package/lib/providers/ec2.js +11 -12
- package/lib/providers/ecs.d.ts +1 -0
- package/lib/providers/ecs.js +7 -3
- package/lib/providers/fargate.d.ts +1 -0
- package/lib/providers/fargate.js +10 -6
- package/lib/providers/lambda.d.ts +1 -0
- package/lib/providers/lambda.js +10 -4
- package/lib/providers/update-lambda.lambda.js +1 -2
- package/lib/runner.d.ts +24 -3
- package/lib/runner.js +64 -21
- package/lib/secrets.js +1 -1
- package/lib/setup.lambda.js +3 -4
- package/lib/status.lambda.js +4 -6
- package/lib/token-retriever.lambda.js +25 -11
- package/lib/webhook-handler.lambda.js +2 -3
- package/package.json +11 -9
- package/lib/github.js +0 -50
package/lib/runner.js
CHANGED
|
@@ -8,6 +8,7 @@ const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
|
8
8
|
const constructs_1 = require("constructs");
|
|
9
9
|
const access_1 = require("./access");
|
|
10
10
|
const delete_runner_function_1 = require("./delete-runner-function");
|
|
11
|
+
const idle_runner_repear_function_1 = require("./idle-runner-repear-function");
|
|
11
12
|
const providers_1 = require("./providers");
|
|
12
13
|
const secrets_1 = require("./secrets");
|
|
13
14
|
const setup_function_1 = require("./setup-function");
|
|
@@ -108,7 +109,6 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
108
109
|
runnerName: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$$.Execution.Name'),
|
|
109
110
|
owner: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.owner'),
|
|
110
111
|
repo: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.repo'),
|
|
111
|
-
runId: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.runId'),
|
|
112
112
|
installationId: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.installationId'),
|
|
113
113
|
idleOnly: false,
|
|
114
114
|
}),
|
|
@@ -121,21 +121,18 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
121
121
|
backoffRate: 1,
|
|
122
122
|
maxAttempts: 60,
|
|
123
123
|
});
|
|
124
|
-
const
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
payloadResponseOnly: true,
|
|
130
|
-
resultPath: '$.delete',
|
|
131
|
-
payload: aws_cdk_lib_1.aws_stepfunctions.TaskInput.fromObject({
|
|
124
|
+
const idleReaper = this.idleReaper();
|
|
125
|
+
const queueIdleReaperTask = new aws_cdk_lib_1.aws_stepfunctions_tasks.SqsSendMessage(this, 'Queue Idle Reaper', {
|
|
126
|
+
queue: this.idleReaperQueue(idleReaper),
|
|
127
|
+
messageBody: aws_cdk_lib_1.aws_stepfunctions.TaskInput.fromObject({
|
|
128
|
+
executionArn: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$$.Execution.Id'),
|
|
132
129
|
runnerName: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$$.Execution.Name'),
|
|
133
130
|
owner: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.owner'),
|
|
134
131
|
repo: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.repo'),
|
|
135
|
-
runId: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.runId'),
|
|
136
132
|
installationId: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.installationId'),
|
|
137
|
-
|
|
133
|
+
maxIdleSeconds: (props?.idleTimeout ?? cdk.Duration.minutes(5)).toSeconds(),
|
|
138
134
|
}),
|
|
135
|
+
resultPath: aws_cdk_lib_1.aws_stepfunctions.JsonPath.DISCARD,
|
|
139
136
|
});
|
|
140
137
|
const providerChooser = new aws_cdk_lib_1.aws_stepfunctions.Choice(this, 'Choose provider');
|
|
141
138
|
for (const provider of this.providers) {
|
|
@@ -149,19 +146,39 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
149
146
|
providerChooser.when(aws_cdk_lib_1.aws_stepfunctions.Condition.and(...provider.labels.map(label => aws_cdk_lib_1.aws_stepfunctions.Condition.isPresent(`$.labels.${label.toLowerCase()}`))), providerTask);
|
|
150
147
|
}
|
|
151
148
|
providerChooser.otherwise(new aws_cdk_lib_1.aws_stepfunctions.Succeed(this, 'Unknown label'));
|
|
152
|
-
const
|
|
153
|
-
.branch(providerChooser)
|
|
154
|
-
.
|
|
155
|
-
|
|
156
|
-
|
|
149
|
+
const runProviders = new aws_cdk_lib_1.aws_stepfunctions.Parallel(this, 'Retry on Error', { resultPath: '$.result' })
|
|
150
|
+
.branch(tokenRetrieverTask.next(providerChooser))
|
|
151
|
+
.addCatch(
|
|
152
|
+
// delete runner on failure as it won't remove itself and there is a limit on the number of registered runners
|
|
153
|
+
deleteRunnerTask.next(new aws_cdk_lib_1.aws_stepfunctions.Fail(this, 'Runner Failed')), {
|
|
157
154
|
resultPath: '$.error',
|
|
158
|
-
})
|
|
155
|
+
});
|
|
156
|
+
if (props?.retryOptions?.retry ?? true) {
|
|
157
|
+
const interval = props?.retryOptions?.interval ?? cdk.Duration.minutes(1);
|
|
158
|
+
const maxAttempts = props?.retryOptions?.maxAttempts ?? 23;
|
|
159
|
+
const backoffRate = props?.retryOptions?.backoffRate ?? 1.3;
|
|
160
|
+
const totalSeconds = interval.toSeconds() * backoffRate ** maxAttempts / (backoffRate - 1);
|
|
161
|
+
if (totalSeconds >= cdk.Duration.days(1).toSeconds()) {
|
|
162
|
+
// https://docs.github.com/en/actions/hosting-your-own-runners/managing-self-hosted-runners/about-self-hosted-runners#usage-limits
|
|
163
|
+
// "Job queue time - Each job for self-hosted runners can be queued for a maximum of 24 hours. If a self-hosted runner does not start executing the job within this limit, the job is terminated and fails to complete."
|
|
164
|
+
aws_cdk_lib_1.Annotations.of(this).addWarning(`Total retry time is greater than 24 hours (${Math.floor(totalSeconds / 60 / 60)} hours). Jobs expire after 24 hours so it would be a waste of resources to retry further.`);
|
|
165
|
+
}
|
|
166
|
+
runProviders.addRetry({
|
|
167
|
+
interval,
|
|
168
|
+
maxAttempts,
|
|
169
|
+
backoffRate,
|
|
170
|
+
// we can't retry on all errors mainly because we don't want to retry on idle runner error
|
|
171
|
+
// when the idle reaper kills a runner, it will fail the state machine
|
|
172
|
+
// if we retry, we would just waste resources and end up killing it again
|
|
173
|
+
errors: ['RunnerTokenError'].concat(...this.providers.map(provider => provider.retryableErrors)),
|
|
174
|
+
});
|
|
175
|
+
}
|
|
159
176
|
let logOptions;
|
|
160
177
|
if (this.props?.logOptions) {
|
|
161
178
|
this.stateMachineLogGroup = new aws_cdk_lib_1.aws_logs.LogGroup(this, 'Logs', {
|
|
162
179
|
logGroupName: props?.logOptions?.logGroupName,
|
|
163
180
|
retention: props?.logOptions?.logRetention ?? aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
|
|
164
|
-
removalPolicy:
|
|
181
|
+
removalPolicy: cdk.RemovalPolicy.DESTROY,
|
|
165
182
|
});
|
|
166
183
|
logOptions = {
|
|
167
184
|
destination: this.stateMachineLogGroup,
|
|
@@ -170,9 +187,10 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
170
187
|
};
|
|
171
188
|
}
|
|
172
189
|
const stateMachine = new aws_cdk_lib_1.aws_stepfunctions.StateMachine(this, 'Runner Orchestrator', {
|
|
173
|
-
definition:
|
|
190
|
+
definition: queueIdleReaperTask.next(runProviders),
|
|
174
191
|
logs: logOptions,
|
|
175
192
|
});
|
|
193
|
+
stateMachine.grantRead(idleReaper);
|
|
176
194
|
for (const provider of this.providers) {
|
|
177
195
|
provider.grantStateMachine(stateMachine);
|
|
178
196
|
}
|
|
@@ -299,6 +317,31 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
299
317
|
}
|
|
300
318
|
}
|
|
301
319
|
}
|
|
320
|
+
idleReaper() {
|
|
321
|
+
return new idle_runner_repear_function_1.IdleRunnerRepearFunction(this, 'Idle Reaper', {
|
|
322
|
+
description: 'Stop idle GitHub runners to avoid paying for runners when the job was already canceled',
|
|
323
|
+
environment: {
|
|
324
|
+
GITHUB_SECRET_ARN: this.secrets.github.secretArn,
|
|
325
|
+
GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,
|
|
326
|
+
...this.extraLambdaEnv,
|
|
327
|
+
},
|
|
328
|
+
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
|
|
329
|
+
timeout: cdk.Duration.minutes(1),
|
|
330
|
+
...this.extraLambdaProps,
|
|
331
|
+
});
|
|
332
|
+
}
|
|
333
|
+
idleReaperQueue(reaper) {
|
|
334
|
+
const queue = new aws_cdk_lib_1.aws_sqs.Queue(this, 'Idle Reaper Queue', {
|
|
335
|
+
deliveryDelay: cdk.Duration.minutes(10),
|
|
336
|
+
visibilityTimeout: cdk.Duration.minutes(10),
|
|
337
|
+
});
|
|
338
|
+
reaper.addEventSource(new aws_cdk_lib_1.aws_lambda_event_sources.SqsEventSource(queue, {
|
|
339
|
+
reportBatchItemFailures: true,
|
|
340
|
+
}));
|
|
341
|
+
this.secrets.github.grantRead(reaper);
|
|
342
|
+
this.secrets.githubPrivateKey.grantRead(reaper);
|
|
343
|
+
return queue;
|
|
344
|
+
}
|
|
302
345
|
/**
|
|
303
346
|
* Metric for the number of GitHub Actions jobs completed. It has `ProviderLabels` and `Status` dimensions. The status can be one of "Succeeded", "SucceededWithIssues", "Failed", "Canceled", "Skipped", or "Abandoned".
|
|
304
347
|
*
|
|
@@ -378,6 +421,6 @@ class GitHubRunners extends constructs_1.Construct {
|
|
|
378
421
|
}
|
|
379
422
|
}
|
|
380
423
|
_a = JSII_RTTI_SYMBOL_1;
|
|
381
|
-
GitHubRunners[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.GitHubRunners", version: "0.
|
|
424
|
+
GitHubRunners[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.GitHubRunners", version: "0.10.0" };
|
|
382
425
|
exports.GitHubRunners = GitHubRunners;
|
|
383
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner.js","sourceRoot":"","sources":["../src/runner.ts"],"names":[],"mappings":";;;;;AAAA,mCAAmC;AACnC,6CAWqB;AACrB,2CAAuC;AACvC,qCAAwC;AACxC,qEAAgE;AAChE,2CAOqB;AACrB,uCAAoC;AACpC,qDAAiD;AACjD,uDAAmD;AACnD,yEAAoE;AACpE,uCAAiD;AAuIjD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCG;AACH,MAAa,aAAc,SAAQ,sBAAS;IAmB1C,YAAY,KAAgB,EAAE,EAAU,EAAW,KAA0B;QAC3E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QADgC,UAAK,GAAL,KAAK,CAAqB;QAL5D,mBAAc,GAA0B,EAAE,CAAC;QAQ1D,IAAI,CAAC,OAAO,GAAG,IAAI,iBAAO,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QAC5C,IAAI,CAAC,gBAAgB,GAAG;YACtB,GAAG,EAAE,IAAI,CAAC,KAAK,EAAE,GAAG;YACpB,UAAU,EAAE,IAAI,CAAC,KAAK,EAAE,UAAU;YAClC,iBAAiB,EAAE,IAAI,CAAC,KAAK,EAAE,iBAAiB;YAChD,cAAc,EAAE,IAAI,CAAC,KAAK,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,SAAS;YAClF,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,IAAI,wBAAM,CAAC,YAAY,CAAC,KAAK,EAAE,mBAAmB,EAAE;oBAC3F,WAAW,EAAE,8EAA8E;oBAC3F,IAAI,EAAE,wBAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC;iBAC1D,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS;SAChB,CAAC;QACF,IAAI,IAAI,CAAC,KAAK,EAAE,iBAAiB,EAAE;YACjC,IAAI,CAAC,cAAc,CAAC,mBAAmB,GAAG,gBAAgB,CAAC;SAC5D;QAED,IAAI,IAAI,CAAC,KAAK,EAAE,SAAS,EAAE;YACzB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC;SACvC;aAAM;YACL,IAAI,CAAC,SAAS,GAAG;gBACf,IAAI,mCAAuB,CAAC,IAAI,EAAE,WAAW,CAAC;gBAC9C,IAAI,gCAAoB,CAAC,IAAI,EAAE,QAAQ,CAAC;gBACxC,IAAI,iCAAqB,CAAC,IAAI,EAAE,SAAS,CAAC;aAC3C,CAAC;SACH;QAED,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAE/B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC7C,IAAI,CAAC,OAAO,GAAG,IAAI,8BAAoB,CAAC,IAAI,EAAE,iBAAiB,EAAE;YAC/D,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,aAAa,IAAI,qBAAY,CAAC,SAAS,EAAE;SAC9D,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACrC,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEO,YAAY,CAAC,KAA0B;QAC7C,MAAM,kBAAkB,GAAG,IAAI,qCAAmB,CAAC,YAAY,CAC7D,IAAI,EACJ,kBAAkB,EAClB;YACE,cAAc,EAAE,IAAI,CAAC,cAAc,EAAE;YACrC,mBAAmB,EAAE,IAAI;YACzB,UAAU,EAAE,UAAU;SACvB,CACF,CAAC;QAEF,IAAI,oBAAoB,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QAC/C,MAAM,gBAAgB,GAAG,IAAI,qCAAmB,CAAC,YAAY,CAC3D,IAAI,EACJ,eAAe,EACf;YACE,cAAc,EAAE,oBAAoB;YACpC,mBAAmB,EAAE,IAAI;YACzB,UAAU,EAAE,UAAU;YACtB,OAAO,EAAE,+BAAa,CAAC,SAAS,CAAC,UAAU,CAAC;gBAC1C,UAAU,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,CAAC;gBAChE,KAAK,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACjD,IAAI,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAC/C,KAAK,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACjD,cAAc,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,kBAAkB,CAAC;gBACnE,QAAQ,EAAE,KAAK;aAChB,CAAC;SACH,CACF,CAAC;QACF,gBAAgB,CAAC,QAAQ,CAAC;YACxB,MAAM,EAAE;gBACN,YAAY;aACb;YACD,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YACjC,WAAW,EAAE,CAAC;YACd,WAAW,EAAE,EAAE;SAChB,CAAC,CAAC;QAEH,MAAM,iBAAiB,GAAG,IAAI,+BAAa,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE;YAC7D,IAAI,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,EAAE,WAAW,IAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;SACtF,CAAC,CAAC;QACH,MAAM,oBAAoB,GAAG,IAAI,qCAAmB,CAAC,YAAY,CAC/D,IAAI,EACJ,oBAAoB,EACpB;YACE,cAAc,EAAE,oBAAoB;YACpC,mBAAmB,EAAE,IAAI;YACzB,UAAU,EAAE,UAAU;YACtB,OAAO,EAAE,+BAAa,CAAC,SAAS,CAAC,UAAU,CAAC;gBAC1C,UAAU,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,CAAC;gBAChE,KAAK,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACjD,IAAI,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAC/C,KAAK,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACjD,cAAc,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,kBAAkB,CAAC;gBACnE,QAAQ,EAAE,IAAI;aACf,CAAC;SACH,CACF,CAAC;QAEF,MAAM,eAAe,GAAG,IAAI,+BAAa,CAAC,MAAM,CAAC,IAAI,EAAE,iBAAiB,CAAC,CAAC;QAC1E,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,SAAS,EAAE;YACrC,MAAM,YAAY,GAAG,QAAQ,CAAC,mBAAmB,CAC/C;gBACE,eAAe,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,gBAAgB,CAAC;gBAClE,cAAc,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,CAAC;gBACpE,gBAAgB,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,iBAAiB,CAAC;gBACpE,SAAS,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACrD,QAAQ,EAAE,+BAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC;aACpD,CACF,CAAC;YACF,eAAe,CAAC,IAAI,CAClB,+BAAa,CAAC,SAAS,CAAC,GAAG,CACzB,GAAG,QAAQ,CAAC,MAAM,CAAC,GAAG,CACpB,KAAK,CAAC,EAAE,CAAC,+BAAa,CAAC,SAAS,CAAC,SAAS,CAAC,YAAY,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC,CAC9E,CACF,EACD,YAAY,CACb,CAAC;SACH;QAED,eAAe,CAAC,SAAS,CAAC,IAAI,+BAAa,CAAC,OAAO,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC,CAAC;QAE5E,MAAM,IAAI,GAAG,kBAAkB,CAAC,IAAI,CAClC,IAAI,+BAAa,CAAC,QAAQ,CAAC,IAAI,EAAE,eAAe,EAAE,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC;aAC1E,MAAM,CAAC,eAAe,CAAC;aACvB,MAAM,CAAC,iBAAiB,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;aACpD,QAAQ,CACP,gBAAgB;aACb,IAAI,CAAC,IAAI,+BAAa,CAAC,IAAI,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC,EACtD;YACE,UAAU,EAAE,SAAS;SACtB,CACF,CACJ,CAAC;QAEF,IAAI,UAAwD,CAAC;QAC7D,IAAI,IAAI,CAAC,KAAK,EAAE,UAAU,EAAE;YAC1B,IAAI,CAAC,oBAAoB,GAAG,IAAI,sBAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE;gBAC1D,YAAY,EAAE,KAAK,EAAE,UAAU,EAAE,YAAY;gBAC7C,SAAS,EAAE,KAAK,EAAE,UAAU,EAAE,YAAY,IAAI,sBAAI,CAAC,aAAa,CAAC,SAAS;gBAC1E,aAAa,EAAE,2BAAa,CAAC,OAAO;aACrC,CAAC,CAAC;YAEH,UAAU,GAAG;gBACX,WAAW,EAAE,IAAI,CAAC,oBAAoB;gBACtC,oBAAoB,EAAE,KAAK,EAAE,UAAU,EAAE,oBAAoB,IAAI,IAAI;gBACrE,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,IAAI,+BAAa,CAAC,QAAQ,CAAC,GAAG;aAC9D,CAAC;SACH;QAED,MAAM,YAAY,GAAG,IAAI,+BAAa,CAAC,YAAY,CACjD,IAAI,EACJ,qBAAqB,EACrB;YACE,UAAU,EAAE,IAAI;YAChB,IAAI,EAAE,UAAU;SACjB,CACF,CAAC;QAEF,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,SAAS,EAAE;YACrC,QAAQ,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;SAC1C;QAED,OAAO,YAAY,CAAC;IACtB,CAAC;IAEO,cAAc;QACpB,MAAM,IAAI,GAAG,IAAI,iDAAsB,CACrC,IAAI,EACJ,iBAAiB,EACjB;YACE,WAAW,EAAE,oEAAoE;YACjF,WAAW,EAAE;gBACX,iBAAiB,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS;gBAChD,6BAA6B,EAAE,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS;gBACtE,GAAG,IAAI,CAAC,cAAc;aACvB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,YAAY,EAAE,sBAAI,CAAC,aAAa,CAAC,SAAS;YAC1C,GAAG,IAAI,CAAC,gBAAgB;SACzB,CACF,CAAC;QAEF,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACpC,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAE9C,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,YAAY;QAClB,MAAM,IAAI,GAAG,IAAI,6CAAoB,CACnC,IAAI,EACJ,eAAe,EACf;YACE,WAAW,EAAE,uCAAuC;YACpD,WAAW,EAAE;gBACX,iBAAiB,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS;gBAChD,6BAA6B,EAAE,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS;gBACtE,GAAG,IAAI,CAAC,cAAc;aACvB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,YAAY,EAAE,sBAAI,CAAC,aAAa,CAAC,SAAS;YAC1C,GAAG,IAAI,CAAC,gBAAgB;SACzB,CACF,CAAC;QAEF,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACpC,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAE9C,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,cAAc;QACpB,MAAM,cAAc,GAAG,IAAI,gCAAc,CACvC,IAAI,EACJ,QAAQ,EACR;YACE,WAAW,EAAE,mEAAmE;YAChF,WAAW,EAAE;gBACX,kBAAkB,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS;gBAClD,iBAAiB,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS;gBAChD,6BAA6B,EAAE,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS;gBACtE,gBAAgB,EAAE,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS;gBAC9C,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG;gBAC7B,mBAAmB,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,aAAa,CAAC,WAAW;gBACnE,iBAAiB,EAAE,IAAI,CAAC,YAAY,CAAC,eAAe;gBACpD,uBAAuB,EAAE,IAAI,CAAC,oBAAoB,EAAE,YAAY,IAAI,EAAE;gBACtE,kBAAkB,EAAE,IAAI,CAAC,QAAQ;gBACjC,GAAG,IAAI,CAAC,cAAc;aACvB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAChC,YAAY,EAAE,sBAAI,CAAC,aAAa,CAAC,SAAS;YAC1C,GAAG,IAAI,CAAC,gBAAgB;SACzB,CACF,CAAC;QAEF,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC;QAElF,sFAAsF;QACtF,uFAAuF;QACvF,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QACjC,MAAM,CAAC,GAAI,cAAc,CAAC,IAAI,CAAC,YAAmC,CAAC;QACnE,CAAC,CAAC,mBAAmB,CAAC,kCAAkC,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC;QACvE,CAAC,CAAC,mBAAmB,CAAC,kCAAkC,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;QAC3E,CAAC,CAAC,WAAW,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;QACtC,cAAc,CAAC,eAAe,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;YACrD,OAAO,EAAE,CAAC,sCAAsC,CAAC;YACjD,SAAS,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC;SAC3B,CAAC,CAAC,CAAC;QAEJ,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QAC/C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QAC9C,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QACxD,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QAC7C,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;QAE5C,IAAI,GAAG,CAAC,SAAS,CACf,IAAI,EACJ,gBAAgB,EAChB;YACE,KAAK,EAAE,gBAAgB,KAAK,CAAC,MAAM,kCAAkC,cAAc,CAAC,YAAY,cAAc;SAC/G,CACF,CAAC;QAEF,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,EAAE,YAAY,IAAI,qBAAY,CAAC,QAAQ,EAAE,CAAC;QACnE,MAAM,GAAG,GAAG,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,eAAe,EAAE,cAAc,CAAC,CAAC;QAEhE,IAAI,GAAG,KAAK,EAAE,EAAE;YACd,IAAI,GAAG,CAAC,SAAS,CACf,IAAI,EACJ,YAAY,EACZ;gBACE,KAAK,EAAE,GAAG;aACX,CACF,CAAC;SACH;IACH,CAAC;IAEO,aAAa;QACnB,MAAM,aAAa,GAAG,IAAI,8BAAa,CACrC,IAAI,EACJ,OAAO,EACP;YACE,WAAW,EAAE,2DAA2D;YACxE,WAAW,EAAE;gBACX,gBAAgB,EAAE,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS;gBAC9C,kBAAkB,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS;gBAClD,iBAAiB,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS;gBAChD,6BAA6B,EAAE,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,SAAS;gBACtE,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG;gBAC7B,GAAG,IAAI,CAAC,cAAc;aACvB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAChC,YAAY,EAAE,sBAAI,CAAC,aAAa,CAAC,SAAS;YAC1C,GAAG,IAAI,CAAC,gBAAgB;SACzB,CACF,CAAC;QAEF,iDAAiD;QACjD,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAC/C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;QAC7C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAC9C,0DAA0D;QAC1D,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QACxD,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAE7C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,EAAE,WAAW,IAAI,qBAAY,CAAC,SAAS,EAAE,CAAC;QACnE,OAAO,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,cAAc,EAAE,aAAa,CAAC,CAAC;IAC3D,CAAC;IAEO,uBAAuB;QAC7B,wFAAwF;QACxF,KAAK,MAAM,EAAE,IAAI,IAAI,CAAC,SAAS,EAAE;YAC/B,KAAK,MAAM,EAAE,IAAI,IAAI,CAAC,SAAS,EAAE;gBAC/B,IAAI,EAAE,IAAI,EAAE,EAAE;oBACZ,SAAS;iBACV;gBACD,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE;oBAC/C,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE;wBAC/C,MAAM,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC,IAAI,CAAC,IAAI,QAAQ,EAAE,CAAC,IAAI,CAAC,IAAI,yBAAyB,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;qBAC3G;oBACD,yBAAW,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,sCAAsC,EAAE,CAAC,IAAI,CAAC,IAAI,QAAQ,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,2CAA2C,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,2GAA2G,CAAC,CAAC;iBACxT;aACF;SACF;IACH,CAAC;IAED;;;;OAIG;IACI,kBAAkB,CAAC,KAA8B;QACtD,IAAI,CAAC,IAAI,CAAC,0BAA0B,EAAE;YACpC,oFAAoF;YACpF,+IAA+I;YAC/I,sEAAsE;YACtE,MAAM,OAAO,GAAG,sBAAI,CAAC,aAAa,CAAC,OAAO,CAAC,gNAAgN,CAAC,CAAC;YAE7P,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CACvD,CAAC,CAAC,QAAQ,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE;gBACzD,eAAe,EAAE,eAAe;gBAChC,UAAU,EAAE,cAAc;gBAC1B,aAAa,EAAE,OAAO;gBACtB,WAAW,EAAE,GAAG;gBAChB,4CAA4C;gBAC5C,UAAU,EAAE;oBACV,cAAc,EAAE,SAAS;oBACzB,MAAM,EAAE,SAAS;iBAClB;aACF,CAAC,CACH,CAAC;YAEF,KAAK,MAAM,YAAY,IAAI,IAAI,CAAC,0BAA0B,EAAE;gBAC1D,IAAI,YAAY,CAAC,IAAI,CAAC,YAAY,YAAY,sBAAI,CAAC,eAAe,EAAE;oBAClE,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,mBAAmB,CAAC,8BAA8B,EAAE,OAAO,CAAC,CAAC;iBAC7F;qBAAM;oBACL,yBAAW,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC,UAAU,CAAC,2CAA2C,CAAC,CAAC;iBACtF;aACF;SACF;QAED,OAAO,IAAI,4BAAU,CAAC,MAAM,CAAC;YAC3B,SAAS,EAAE,eAAe;YAC1B,UAAU,EAAE,eAAe;YAC3B,IAAI,EAAE,4BAAU,CAAC,IAAI,CAAC,KAAK;YAC3B,SAAS,EAAE,4BAAU,CAAC,SAAS,CAAC,GAAG;YACnC,GAAG,KAAK;SACT,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IACpB,CAAC;IAED;;;;;;OAMG;IACI,eAAe,CAAC,KAA8B;QACnD,OAAO,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;IAClD,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,KAA8B;QAChD,OAAO,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;IAC/C,CAAC;IAED;;OAEG;IACI,UAAU,CAAC,KAA8B;QAC9C,OAAO,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;IAC7C,CAAC;IAED;;;;;;OAMG;IACI,sBAAsB;QAC3B,MAAM,KAAK,GAAG,IAAI,qBAAG,CAAC,KAAK,CAAC,IAAI,EAAE,4BAA4B,CAAC,CAAC;QAChE,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QACjC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,oDAAwC,CAAC,KAAK,CAAC,CAAC,CAAC;QAC/E,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CACvB,IAAI,8CAAkC,CACpC,8CAAkC,CAAC,oBAAoB,CAAC,IAAI,EAAE,KAAK,CAAC,CACrE,CACF,CAAC;QACF,OAAO,KAAK,CAAC;IACf,CAAC;;;;AApbU,sCAAa","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_cloudwatch as cloudwatch,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_sns as sns,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { LambdaAccess } from './access';\nimport { DeleteRunnerFunction } from './delete-runner-function';\nimport {\n  AwsImageBuilderFailedBuildNotifier,\n  CodeBuildImageBuilderFailedBuildNotifier,\n  CodeBuildRunnerProvider,\n  FargateRunnerProvider,\n  IRunnerProvider,\n  LambdaRunnerProvider,\n} from './providers';\nimport { Secrets } from './secrets';\nimport { SetupFunction } from './setup-function';\nimport { StatusFunction } from './status-function';\nimport { TokenRetrieverFunction } from './token-retriever-function';\nimport { GithubWebhookHandler } from './webhook';\n\n\n/**\n * Properties for GitHubRunners\n */\nexport interface GitHubRunnersProps {\n  /**\n   * List of runner providers to use. At least one provider is required. Provider will be selected when its label matches the labels requested by the workflow job.\n   *\n   * @default CodeBuild, Lambda and Fargate runners with all the defaults (no VPC or default account VPC)\n   */\n  readonly providers?: IRunnerProvider[];\n\n  /**\n   * VPC used for all management functions. Use this with GitHub Enterprise Server hosted that's inaccessible from outside the VPC.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * VPC subnets used for all management functions. Use this with GitHub Enterprise Server hosted that's inaccessible from outside the VPC.\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * Allow management functions to run in public subnets. Lambda Functions in a public subnet can NOT access the internet.\n   *\n   * @default false\n   */\n  readonly allowPublicSubnet?: boolean;\n\n  /**\n   * Security group attached to all management functions. Use this with to provide access to GitHub Enterprise Server hosted inside a VPC.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Path to a directory containing a file named certs.pem containing any additional certificates required to trust GitHub Enterprise Server. Use this when GitHub Enterprise Server certificates are self-signed.\n   *\n   * You may also want to use custom images for your runner providers that contain the same certificates. See {@link CodeBuildImageBuilder.addCertificates}.\n   *\n   * ```typescript\n   * const imageBuilder = CodeBuildRunnerProvider.imageBuilder(this, 'Image Builder with Certs');\n   * imageBuilder.addComponent(RunnerImageComponent.extraCertificates('path-to-my-extra-certs-folder/certs.pem', 'private-ca');\n   *\n   * const provider = new CodeBuildRunnerProvider(this, 'CodeBuild', {\n   *     imageBuilder: imageBuilder,\n   * });\n   *\n   * new GitHubRunners(\n   *   this,\n   *   'runners',\n   *   {\n   *     providers: [provider],\n   *     extraCertificates: 'path-to-my-extra-certs-folder',\n   *   }\n   * );\n   * ```\n   */\n  readonly extraCertificates?: string;\n\n  /**\n   * Time to wait before stopping a runner that remains idle. If the user cancelled the job, or if another runner stole it, this stops the runner to avoid wasting resources.\n   *\n   * @default 10 minutes\n   */\n  readonly idleTimeout?: cdk.Duration;\n\n  /**\n   * Logging options for the state machine that manages the runners.\n   *\n   * @default no logs\n   */\n  readonly logOptions?: LogOptions;\n\n  /**\n   * Access configuration for the setup function. Once you finish the setup process, you can set this to `LambdaAccess.noAccess()` to remove access to the setup function. You can also use `LambdaAccess.apiGateway({ allowedIps: ['my-ip/0']})` to limit access to your IP only.\n   *\n   * @default LambdaAccess.lambdaUrl()\n   */\n  readonly setupAccess?: LambdaAccess;\n\n\n  /**\n   * Access configuration for the webhook function. This function is called by GitHub when a new workflow job is scheduled. For an extra layer of security, you can set this to `LambdaAccess.apiGateway({ allowedIps: LambdaAccess.githubWebhookIps() })`.\n   *\n   * You can also set this to `LambdaAccess.privateApiGateway()` if your GitHub Enterprise Server is hosted in a VPC. This will create an API Gateway endpoint that's only accessible from within the VPC.\n   *\n   * *WARNING*: changing access type may change the URL. When the URL changes, you must update GitHub as well.\n   *\n   * @default LambdaAccess.lambdaUrl()\n   */\n  readonly webhookAccess?: LambdaAccess;\n\n  /**\n   * Access configuration for the status function. This function returns a lot of sensitive information about the runner, so you should only allow access to it from trusted IPs, if at all.\n   *\n   * @default LambdaAccess.noAccess()\n   */\n  readonly statusAccess?: LambdaAccess;\n}\n\n/**\n * Defines what execution history events are logged and where they are logged.\n */\nexport interface LogOptions {\n  /**\n   * The log group where the execution history events will be logged.\n   */\n  readonly logGroupName?: string;\n\n  /**\n   * Determines whether execution data is included in your log.\n   *\n   * @default false\n   */\n  readonly includeExecutionData?: boolean;\n\n  /**\n   * Defines which category of execution history events are logged.\n   *\n   * @default ERROR\n   */\n  readonly level?: stepfunctions.LogLevel;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n}\n\n/**\n * Create all the required infrastructure to provide self-hosted GitHub runners. It creates a webhook, secrets, and a step function to orchestrate all runs. Secrets are not automatically filled. See README.md for instructions on how to setup GitHub integration.\n *\n * By default, this will create a runner provider of each available type with the defaults. This is good enough for the initial setup stage when you just want to get GitHub integration working.\n *\n * ```typescript\n * new GitHubRunners(this, 'runners');\n * ```\n *\n * Usually you'd want to configure the runner providers so the runners can run in a certain VPC or have certain permissions.\n *\n * ```typescript\n * const vpc = ec2.Vpc.fromLookup(this, 'vpc', { vpcId: 'vpc-1234567' });\n * const runnerSg = new ec2.SecurityGroup(this, 'runner security group', { vpc: vpc });\n * const dbSg = ec2.SecurityGroup.fromSecurityGroupId(this, 'database security group', 'sg-1234567');\n * const bucket = new s3.Bucket(this, 'runner bucket');\n *\n * // create a custom CodeBuild provider\n * const myProvider = new CodeBuildRunnerProvider(\n *   this, 'codebuild runner',\n *   {\n *      labels: ['my-codebuild'],\n *      vpc: vpc,\n *      securityGroups: [runnerSg],\n *   },\n * );\n * // grant some permissions to the provider\n * bucket.grantReadWrite(myProvider);\n * dbSg.connections.allowFrom(runnerSg, ec2.Port.tcp(3306), 'allow runners to connect to MySQL database');\n *\n * // create the runner infrastructure\n * new GitHubRunners(\n *   this,\n *   'runners',\n *   {\n *     providers: [myProvider],\n *   }\n * );\n * ```\n */\nexport class GitHubRunners extends Construct {\n  /**\n   * Configured runner providers.\n   */\n  readonly providers: IRunnerProvider[];\n\n  /**\n   * Secrets for GitHub communication including webhook secret and runner authentication.\n   */\n  readonly secrets: Secrets;\n\n  private readonly webhook: GithubWebhookHandler;\n  private readonly orchestrator: stepfunctions.StateMachine;\n  private readonly setupUrl: string;\n  private readonly extraLambdaEnv: {[p: string]: string} = {};\n  private readonly extraLambdaProps: lambda.FunctionOptions;\n  private stateMachineLogGroup?: logs.LogGroup;\n  private jobsCompletedMetricFilters?: logs.MetricFilter[];\n\n  constructor(scope: Construct, id: string, readonly props?: GitHubRunnersProps) {\n    super(scope, id);\n\n    this.secrets = new Secrets(this, 'Secrets');\n    this.extraLambdaProps = {\n      vpc: this.props?.vpc,\n      vpcSubnets: this.props?.vpcSubnets,\n      allowPublicSubnet: this.props?.allowPublicSubnet,\n      securityGroups: this.props?.securityGroup ? [this.props.securityGroup] : undefined,\n      layers: this.props?.extraCertificates ? [new lambda.LayerVersion(scope, 'Certificate Layer', {\n        description: 'Layer containing GitHub Enterprise Server certificate for cdk-github-runners',\n        code: lambda.Code.fromAsset(this.props.extraCertificates),\n      })] : undefined,\n    };\n    if (this.props?.extraCertificates) {\n      this.extraLambdaEnv.NODE_EXTRA_CA_CERTS = '/opt/certs.pem';\n    }\n\n    if (this.props?.providers) {\n      this.providers = this.props.providers;\n    } else {\n      this.providers = [\n        new CodeBuildRunnerProvider(this, 'CodeBuild'),\n        new LambdaRunnerProvider(this, 'Lambda'),\n        new FargateRunnerProvider(this, 'Fargate'),\n      ];\n    }\n\n    this.checkIntersectingLabels();\n\n    this.orchestrator = this.stateMachine(props);\n    this.webhook = new GithubWebhookHandler(this, 'Webhook Handler', {\n      orchestrator: this.orchestrator,\n      secrets: this.secrets,\n      access: this.props?.webhookAccess ?? LambdaAccess.lambdaUrl(),\n    });\n\n    this.setupUrl = this.setupFunction();\n    this.statusFunction();\n  }\n\n  private stateMachine(props?: GitHubRunnersProps) {\n    const tokenRetrieverTask = new stepfunctions_tasks.LambdaInvoke(\n      this,\n      'Get Runner Token',\n      {\n        lambdaFunction: this.tokenRetriever(),\n        payloadResponseOnly: true,\n        resultPath: '$.runner',\n      },\n    );\n\n    let deleteRunnerFunction = this.deleteRunner();\n    const deleteRunnerTask = new stepfunctions_tasks.LambdaInvoke(\n      this,\n      'Delete Runner',\n      {\n        lambdaFunction: deleteRunnerFunction,\n        payloadResponseOnly: true,\n        resultPath: '$.delete',\n        payload: stepfunctions.TaskInput.fromObject({\n          runnerName: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n          owner: stepfunctions.JsonPath.stringAt('$.owner'),\n          repo: stepfunctions.JsonPath.stringAt('$.repo'),\n          runId: stepfunctions.JsonPath.stringAt('$.runId'),\n          installationId: stepfunctions.JsonPath.stringAt('$.installationId'),\n          idleOnly: false,\n        }),\n      },\n    );\n    deleteRunnerTask.addRetry({\n      errors: [\n        'RunnerBusy',\n      ],\n      interval: cdk.Duration.minutes(1),\n      backoffRate: 1,\n      maxAttempts: 60,\n    });\n\n    const waitForIdleRunner = new stepfunctions.Wait(this, 'Wait', {\n      time: stepfunctions.WaitTime.duration(props?.idleTimeout ?? cdk.Duration.minutes(10)),\n    });\n    const deleteIdleRunnerTask = new stepfunctions_tasks.LambdaInvoke(\n      this,\n      'Delete Idle Runner',\n      {\n        lambdaFunction: deleteRunnerFunction,\n        payloadResponseOnly: true,\n        resultPath: '$.delete',\n        payload: stepfunctions.TaskInput.fromObject({\n          runnerName: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n          owner: stepfunctions.JsonPath.stringAt('$.owner'),\n          repo: stepfunctions.JsonPath.stringAt('$.repo'),\n          runId: stepfunctions.JsonPath.stringAt('$.runId'),\n          installationId: stepfunctions.JsonPath.stringAt('$.installationId'),\n          idleOnly: true,\n        }),\n      },\n    );\n\n    const providerChooser = new stepfunctions.Choice(this, 'Choose provider');\n    for (const provider of this.providers) {\n      const providerTask = provider.getStepFunctionTask(\n        {\n          runnerTokenPath: stepfunctions.JsonPath.stringAt('$.runner.token'),\n          runnerNamePath: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n          githubDomainPath: stepfunctions.JsonPath.stringAt('$.runner.domain'),\n          ownerPath: stepfunctions.JsonPath.stringAt('$.owner'),\n          repoPath: stepfunctions.JsonPath.stringAt('$.repo'),\n        },\n      );\n      providerChooser.when(\n        stepfunctions.Condition.and(\n          ...provider.labels.map(\n            label => stepfunctions.Condition.isPresent(`$.labels.${label.toLowerCase()}`),\n          ),\n        ),\n        providerTask,\n      );\n    }\n\n    providerChooser.otherwise(new stepfunctions.Succeed(this, 'Unknown label'));\n\n    const work = tokenRetrieverTask.next(\n      new stepfunctions.Parallel(this, 'Error Catcher', { resultPath: '$.result' })\n        .branch(providerChooser)\n        .branch(waitForIdleRunner.next(deleteIdleRunnerTask))\n        .addCatch(\n          deleteRunnerTask\n            .next(new stepfunctions.Fail(this, 'Runner Failed')),\n          {\n            resultPath: '$.error',\n          },\n        ),\n    );\n\n    let logOptions: cdk.aws_stepfunctions.LogOptions | undefined;\n    if (this.props?.logOptions) {\n      this.stateMachineLogGroup = new logs.LogGroup(this, 'Logs', {\n        logGroupName: props?.logOptions?.logGroupName,\n        retention: props?.logOptions?.logRetention ?? logs.RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      });\n\n      logOptions = {\n        destination: this.stateMachineLogGroup,\n        includeExecutionData: props?.logOptions?.includeExecutionData ?? true,\n        level: props?.logOptions?.level ?? stepfunctions.LogLevel.ALL,\n      };\n    }\n\n    const stateMachine = new stepfunctions.StateMachine(\n      this,\n      'Runner Orchestrator',\n      {\n        definition: work,\n        logs: logOptions,\n      },\n    );\n\n    for (const provider of this.providers) {\n      provider.grantStateMachine(stateMachine);\n    }\n\n    return stateMachine;\n  }\n\n  private tokenRetriever() {\n    const func = new TokenRetrieverFunction(\n      this,\n      'token-retriever',\n      {\n        description: 'Get token from GitHub Actions used to start new self-hosted runner',\n        environment: {\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.seconds(30),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    this.secrets.github.grantRead(func);\n    this.secrets.githubPrivateKey.grantRead(func);\n\n    return func;\n  }\n\n  private deleteRunner() {\n    const func = new DeleteRunnerFunction(\n      this,\n      'delete-runner',\n      {\n        description: 'Delete GitHub Actions runner on error',\n        environment: {\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.seconds(30),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    this.secrets.github.grantRead(func);\n    this.secrets.githubPrivateKey.grantRead(func);\n\n    return func;\n  }\n\n  private statusFunction() {\n    const statusFunction = new StatusFunction(\n      this,\n      'status',\n      {\n        description: 'Provide user with status about self-hosted GitHub Actions runners',\n        environment: {\n          WEBHOOK_SECRET_ARN: this.secrets.webhook.secretArn,\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          SETUP_SECRET_ARN: this.secrets.setup.secretArn,\n          WEBHOOK_URL: this.webhook.url,\n          WEBHOOK_HANDLER_ARN: this.webhook.handler.latestVersion.functionArn,\n          STEP_FUNCTION_ARN: this.orchestrator.stateMachineArn,\n          STEP_FUNCTION_LOG_GROUP: this.stateMachineLogGroup?.logGroupName ?? '',\n          SETUP_FUNCTION_URL: this.setupUrl,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.minutes(3),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    const providers = this.providers.map(provider => provider.status(statusFunction));\n\n    // expose providers as stack metadata as it's too big for Lambda environment variables\n    // specifically integration testing got an error because lambda update request was >5kb\n    const stack = cdk.Stack.of(this);\n    const f = (statusFunction.node.defaultChild as lambda.CfnFunction);\n    f.addPropertyOverride('Environment.Variables.LOGICAL_ID', f.logicalId);\n    f.addPropertyOverride('Environment.Variables.STACK_NAME', stack.stackName);\n    f.addMetadata('providers', providers);\n    statusFunction.addToRolePolicy(new iam.PolicyStatement({\n      actions: ['cloudformation:DescribeStackResource'],\n      resources: [stack.stackId],\n    }));\n\n    this.secrets.webhook.grantRead(statusFunction);\n    this.secrets.github.grantRead(statusFunction);\n    this.secrets.githubPrivateKey.grantRead(statusFunction);\n    this.secrets.setup.grantRead(statusFunction);\n    this.orchestrator.grantRead(statusFunction);\n\n    new cdk.CfnOutput(\n      this,\n      'status command',\n      {\n        value: `aws --region ${stack.region} lambda invoke --function-name ${statusFunction.functionName} status.json`,\n      },\n    );\n\n    const access = this.props?.statusAccess ?? LambdaAccess.noAccess();\n    const url = access._bind(this, 'status access', statusFunction);\n\n    if (url !== '') {\n      new cdk.CfnOutput(\n        this,\n        'status url',\n        {\n          value: url,\n        },\n      );\n    }\n  }\n\n  private setupFunction(): string {\n    const setupFunction = new SetupFunction(\n      this,\n      'setup',\n      {\n        description: 'Setup GitHub Actions integration with self-hosted runners',\n        environment: {\n          SETUP_SECRET_ARN: this.secrets.setup.secretArn,\n          WEBHOOK_SECRET_ARN: this.secrets.webhook.secretArn,\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          WEBHOOK_URL: this.webhook.url,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.minutes(3),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    // this.secrets.webhook.grantRead(setupFunction);\n    this.secrets.webhook.grantWrite(setupFunction);\n    this.secrets.github.grantRead(setupFunction);\n    this.secrets.github.grantWrite(setupFunction);\n    // this.secrets.githubPrivateKey.grantRead(setupFunction);\n    this.secrets.githubPrivateKey.grantWrite(setupFunction);\n    this.secrets.setup.grantRead(setupFunction);\n    this.secrets.setup.grantWrite(setupFunction);\n\n    const access = this.props?.setupAccess ?? LambdaAccess.lambdaUrl();\n    return access._bind(this, 'setup access', setupFunction);\n  }\n\n  private checkIntersectingLabels() {\n    // this \"algorithm\" is very inefficient, but good enough for the tiny datasets we expect\n    for (const p1 of this.providers) {\n      for (const p2 of this.providers) {\n        if (p1 == p2) {\n          continue;\n        }\n        if (p1.labels.every(l => p2.labels.includes(l))) {\n          if (p2.labels.every(l => p1.labels.includes(l))) {\n            throw new Error(`Both ${p1.node.path} and ${p2.node.path} use the same labels [${p1.labels.join(', ')}]`);\n          }\n          Annotations.of(p1).addWarning(`Labels [${p1.labels.join(', ')}] intersect with another provider (${p2.node.path} -- [${p2.labels.join(', ')}]). If a workflow specifies the labels [${p1.labels.join(', ')}], it is not guaranteed which provider will be used. It is recommended you do not use intersecting labels`);\n        }\n      }\n    }\n  }\n\n  /**\n   * Metric for the number of GitHub Actions jobs completed. It has `ProviderLabels` and `Status` dimensions. The status can be one of \"Succeeded\", \"SucceededWithIssues\", \"Failed\", \"Canceled\", \"Skipped\", or \"Abandoned\".\n   *\n   * **WARNING:** this method creates a metric filter for each provider. Each metric has a status dimension with six possible values. These resources may incur cost.\n   */\n  public metricJobCompleted(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    if (!this.jobsCompletedMetricFilters) {\n      // we can't use logs.FilterPattern.spaceDelimited() because it has no support for ||\n      // status list taken from https://github.com/actions/runner/blob/be9632302ceef50bfb36ea998cea9c94c75e5d4d/src/Sdk/DTWebApi/WebApi/TaskResult.cs\n      // we need \"...\" for Lambda that prefixes some extra data to log lines\n      const pattern = logs.FilterPattern.literal('[..., marker = \"CDKGHA\", job = \"JOB\", done = \"DONE\", labels, status = \"Succeeded\" || status = \"SucceededWithIssues\" || status = \"Failed\" || status = \"Canceled\" || status = \"Skipped\" || status = \"Abandoned\"]');\n\n      this.jobsCompletedMetricFilters = this.providers.map(p =>\n        p.logGroup.addMetricFilter(`${p.logGroup.node.id} filter`, {\n          metricNamespace: 'GitHubRunners',\n          metricName: 'JobCompleted',\n          filterPattern: pattern,\n          metricValue: '1',\n          // can't with dimensions -- defaultValue: 0,\n          dimensions: {\n            ProviderLabels: '$labels',\n            Status: '$status',\n          },\n        }),\n      );\n\n      for (const metricFilter of this.jobsCompletedMetricFilters) {\n        if (metricFilter.node.defaultChild instanceof logs.CfnMetricFilter) {\n          metricFilter.node.defaultChild.addPropertyOverride('MetricTransformations.0.Unit', 'Count');\n        } else {\n          Annotations.of(metricFilter).addWarning('Unable to set metric filter Unit to Count');\n        }\n      }\n    }\n\n    return new cloudwatch.Metric({\n      namespace: 'GitHubRunners',\n      metricName: 'JobsCompleted',\n      unit: cloudwatch.Unit.COUNT,\n      statistic: cloudwatch.Statistic.SUM,\n      ...props,\n    }).attachTo(this);\n  }\n\n  /**\n   * Metric for successful executions.\n   *\n   * A successful execution doesn't always mean a runner was started. It can be successful even without any label matches.\n   *\n   * A successful runner doesn't mean the job it executed was successful. For that, see {@link metricJobCompleted}.\n   */\n  public metricSucceeded(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricSucceeded(props);\n  }\n\n  /**\n   * Metric for failed runner executions.\n   *\n   * A failed runner usually means the runner failed to start and so a job was never executed. It doesn't necessarily mean the job was executed and failed. For that, see {@link metricJobCompleted}.\n   */\n  public metricFailed(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricFailed(props);\n  }\n\n  /**\n   * Metric for the interval, in milliseconds, between the time the execution starts and the time it closes. This time may be longer than the time the runner took.\n   */\n  public metricTime(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricTime(props);\n  }\n\n  /**\n   * Creates a topic for notifications when a runner image build fails.\n   *\n   * Runner images are rebuilt every week by default. This provides the latest GitHub Runner version and software updates.\n   *\n   * If you want to be sure you are using the latest runner version, you can use this topic to be notified when a build fails.\n   */\n  public failedImageBuildsTopic() {\n    const topic = new sns.Topic(this, 'Failed Runner Image Builds');\n    const stack = cdk.Stack.of(this);\n    cdk.Aspects.of(stack).add(new CodeBuildImageBuilderFailedBuildNotifier(topic));\n    cdk.Aspects.of(stack).add(\n      new AwsImageBuilderFailedBuildNotifier(\n        AwsImageBuilderFailedBuildNotifier.createFilteringTopic(this, topic),\n      ),\n    );\n    return topic;\n  }\n}\n"]}
|
|
426
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner.js","sourceRoot":"","sources":["../src/runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_cloudwatch as cloudwatch,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_lambda_event_sources as lambda_event_sources,\n  aws_logs as logs,\n  aws_sns as sns,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  aws_sqs as sqs,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { LambdaAccess } from './access';\nimport { DeleteRunnerFunction } from './delete-runner-function';\nimport { IdleRunnerRepearFunction } from './idle-runner-repear-function';\nimport {\n  AwsImageBuilderFailedBuildNotifier,\n  CodeBuildImageBuilderFailedBuildNotifier,\n  CodeBuildRunnerProvider,\n  FargateRunnerProvider,\n  IRunnerProvider,\n  LambdaRunnerProvider,\n  ProviderRetryOptions,\n} from './providers';\nimport { Secrets } from './secrets';\nimport { SetupFunction } from './setup-function';\nimport { StatusFunction } from './status-function';\nimport { TokenRetrieverFunction } from './token-retriever-function';\nimport { GithubWebhookHandler } from './webhook';\n\n\n/**\n * Properties for GitHubRunners\n */\nexport interface GitHubRunnersProps {\n  /**\n   * List of runner providers to use. At least one provider is required. Provider will be selected when its label matches the labels requested by the workflow job.\n   *\n   * @default CodeBuild, Lambda and Fargate runners with all the defaults (no VPC or default account VPC)\n   */\n  readonly providers?: IRunnerProvider[];\n\n  /**\n   * VPC used for all management functions. Use this with GitHub Enterprise Server hosted that's inaccessible from outside the VPC.\n   *\n   * Make sure the selected VPC and subnets have access to the following with either NAT Gateway or VPC Endpoints:\n   * * GitHub Enterprise Server\n   * * Secrets Manager\n   * * SQS\n   * * Step Functions\n   * * CloudFormation (status function only)\n   * * EC2 (status function only)\n   * * ECR (status function only)\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * VPC subnets used for all management functions. Use this with GitHub Enterprise Server hosted that's inaccessible from outside the VPC.\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * Allow management functions to run in public subnets. Lambda Functions in a public subnet can NOT access the internet.\n   *\n   * @default false\n   */\n  readonly allowPublicSubnet?: boolean;\n\n  /**\n   * Security group attached to all management functions. Use this with to provide access to GitHub Enterprise Server hosted inside a VPC.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Path to a directory containing a file named certs.pem containing any additional certificates required to trust GitHub Enterprise Server. Use this when GitHub Enterprise Server certificates are self-signed.\n   *\n   * You may also want to use custom images for your runner providers that contain the same certificates. See {@link CodeBuildImageBuilder.addCertificates}.\n   *\n   * ```typescript\n   * const imageBuilder = CodeBuildRunnerProvider.imageBuilder(this, 'Image Builder with Certs');\n   * imageBuilder.addComponent(RunnerImageComponent.extraCertificates('path-to-my-extra-certs-folder/certs.pem', 'private-ca');\n   *\n   * const provider = new CodeBuildRunnerProvider(this, 'CodeBuild', {\n   *     imageBuilder: imageBuilder,\n   * });\n   *\n   * new GitHubRunners(\n   *   this,\n   *   'runners',\n   *   {\n   *     providers: [provider],\n   *     extraCertificates: 'path-to-my-extra-certs-folder',\n   *   }\n   * );\n   * ```\n   */\n  readonly extraCertificates?: string;\n\n  /**\n   * Time to wait before stopping a runner that remains idle. If the user cancelled the job, or if another runner stole it, this stops the runner to avoid wasting resources.\n   *\n   * @default 5 minutes\n   */\n  readonly idleTimeout?: cdk.Duration;\n\n  /**\n   * Logging options for the state machine that manages the runners.\n   *\n   * @default no logs\n   */\n  readonly logOptions?: LogOptions;\n\n  /**\n   * Access configuration for the setup function. Once you finish the setup process, you can set this to `LambdaAccess.noAccess()` to remove access to the setup function. You can also use `LambdaAccess.apiGateway({ allowedIps: ['my-ip/0']})` to limit access to your IP only.\n   *\n   * @default LambdaAccess.lambdaUrl()\n   */\n  readonly setupAccess?: LambdaAccess;\n\n\n  /**\n   * Access configuration for the webhook function. This function is called by GitHub when a new workflow job is scheduled. For an extra layer of security, you can set this to `LambdaAccess.apiGateway({ allowedIps: LambdaAccess.githubWebhookIps() })`.\n   *\n   * You can also set this to `LambdaAccess.apiGateway({allowedVpc: vpc, allowedIps: ['GHES.IP.ADDRESS/32']})` if your GitHub Enterprise Server is hosted in a VPC. This will create an API Gateway endpoint that's only accessible from within the VPC.\n   *\n   * *WARNING*: changing access type may change the URL. When the URL changes, you must update GitHub as well.\n   *\n   * @default LambdaAccess.lambdaUrl()\n   */\n  readonly webhookAccess?: LambdaAccess;\n\n  /**\n   * Access configuration for the status function. This function returns a lot of sensitive information about the runner, so you should only allow access to it from trusted IPs, if at all.\n   *\n   * @default LambdaAccess.noAccess()\n   */\n  readonly statusAccess?: LambdaAccess;\n\n  /**\n   * Options to retry operation in case of failure like missing capacity, or API quota issues.\n   *\n   * GitHub jobs time out after not being able to get a runner for 24 hours. You should not retry for more than 24 hours.\n   *\n   * Total time spent waiting can be calculated with interval * (backoffRate ^ maxAttempts) / (backoffRate - 1).\n   *\n   * @default retry 23 times up to about 24 hours\n   */\n  readonly retryOptions?: ProviderRetryOptions;\n}\n\n/**\n * Defines what execution history events are logged and where they are logged.\n */\nexport interface LogOptions {\n  /**\n   * The log group where the execution history events will be logged.\n   */\n  readonly logGroupName?: string;\n\n  /**\n   * Determines whether execution data is included in your log.\n   *\n   * @default false\n   */\n  readonly includeExecutionData?: boolean;\n\n  /**\n   * Defines which category of execution history events are logged.\n   *\n   * @default ERROR\n   */\n  readonly level?: stepfunctions.LogLevel;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n}\n\n/**\n * Create all the required infrastructure to provide self-hosted GitHub runners. It creates a webhook, secrets, and a step function to orchestrate all runs. Secrets are not automatically filled. See README.md for instructions on how to setup GitHub integration.\n *\n * By default, this will create a runner provider of each available type with the defaults. This is good enough for the initial setup stage when you just want to get GitHub integration working.\n *\n * ```typescript\n * new GitHubRunners(this, 'runners');\n * ```\n *\n * Usually you'd want to configure the runner providers so the runners can run in a certain VPC or have certain permissions.\n *\n * ```typescript\n * const vpc = ec2.Vpc.fromLookup(this, 'vpc', { vpcId: 'vpc-1234567' });\n * const runnerSg = new ec2.SecurityGroup(this, 'runner security group', { vpc: vpc });\n * const dbSg = ec2.SecurityGroup.fromSecurityGroupId(this, 'database security group', 'sg-1234567');\n * const bucket = new s3.Bucket(this, 'runner bucket');\n *\n * // create a custom CodeBuild provider\n * const myProvider = new CodeBuildRunnerProvider(\n *   this, 'codebuild runner',\n *   {\n *      labels: ['my-codebuild'],\n *      vpc: vpc,\n *      securityGroups: [runnerSg],\n *   },\n * );\n * // grant some permissions to the provider\n * bucket.grantReadWrite(myProvider);\n * dbSg.connections.allowFrom(runnerSg, ec2.Port.tcp(3306), 'allow runners to connect to MySQL database');\n *\n * // create the runner infrastructure\n * new GitHubRunners(\n *   this,\n *   'runners',\n *   {\n *     providers: [myProvider],\n *   }\n * );\n * ```\n */\nexport class GitHubRunners extends Construct {\n  /**\n   * Configured runner providers.\n   */\n  readonly providers: IRunnerProvider[];\n\n  /**\n   * Secrets for GitHub communication including webhook secret and runner authentication.\n   */\n  readonly secrets: Secrets;\n\n  private readonly webhook: GithubWebhookHandler;\n  private readonly orchestrator: stepfunctions.StateMachine;\n  private readonly setupUrl: string;\n  private readonly extraLambdaEnv: {[p: string]: string} = {};\n  private readonly extraLambdaProps: lambda.FunctionOptions;\n  private stateMachineLogGroup?: logs.LogGroup;\n  private jobsCompletedMetricFilters?: logs.MetricFilter[];\n\n  constructor(scope: Construct, id: string, readonly props?: GitHubRunnersProps) {\n    super(scope, id);\n\n    this.secrets = new Secrets(this, 'Secrets');\n    this.extraLambdaProps = {\n      vpc: this.props?.vpc,\n      vpcSubnets: this.props?.vpcSubnets,\n      allowPublicSubnet: this.props?.allowPublicSubnet,\n      securityGroups: this.props?.securityGroup ? [this.props.securityGroup] : undefined,\n      layers: this.props?.extraCertificates ? [new lambda.LayerVersion(scope, 'Certificate Layer', {\n        description: 'Layer containing GitHub Enterprise Server certificate for cdk-github-runners',\n        code: lambda.Code.fromAsset(this.props.extraCertificates),\n      })] : undefined,\n    };\n    if (this.props?.extraCertificates) {\n      this.extraLambdaEnv.NODE_EXTRA_CA_CERTS = '/opt/certs.pem';\n    }\n\n    if (this.props?.providers) {\n      this.providers = this.props.providers;\n    } else {\n      this.providers = [\n        new CodeBuildRunnerProvider(this, 'CodeBuild'),\n        new LambdaRunnerProvider(this, 'Lambda'),\n        new FargateRunnerProvider(this, 'Fargate'),\n      ];\n    }\n\n    this.checkIntersectingLabels();\n\n    this.orchestrator = this.stateMachine(props);\n    this.webhook = new GithubWebhookHandler(this, 'Webhook Handler', {\n      orchestrator: this.orchestrator,\n      secrets: this.secrets,\n      access: this.props?.webhookAccess ?? LambdaAccess.lambdaUrl(),\n    });\n\n    this.setupUrl = this.setupFunction();\n    this.statusFunction();\n  }\n\n  private stateMachine(props?: GitHubRunnersProps) {\n    const tokenRetrieverTask = new stepfunctions_tasks.LambdaInvoke(\n      this,\n      'Get Runner Token',\n      {\n        lambdaFunction: this.tokenRetriever(),\n        payloadResponseOnly: true,\n        resultPath: '$.runner',\n      },\n    );\n\n    let deleteRunnerFunction = this.deleteRunner();\n    const deleteRunnerTask = new stepfunctions_tasks.LambdaInvoke(\n      this,\n      'Delete Runner',\n      {\n        lambdaFunction: deleteRunnerFunction,\n        payloadResponseOnly: true,\n        resultPath: '$.delete',\n        payload: stepfunctions.TaskInput.fromObject({\n          runnerName: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n          owner: stepfunctions.JsonPath.stringAt('$.owner'),\n          repo: stepfunctions.JsonPath.stringAt('$.repo'),\n          installationId: stepfunctions.JsonPath.stringAt('$.installationId'),\n          idleOnly: false,\n        }),\n      },\n    );\n    deleteRunnerTask.addRetry({\n      errors: [\n        'RunnerBusy',\n      ],\n      interval: cdk.Duration.minutes(1),\n      backoffRate: 1,\n      maxAttempts: 60,\n    });\n\n    const idleReaper = this.idleReaper();\n    const queueIdleReaperTask = new stepfunctions_tasks.SqsSendMessage(this, 'Queue Idle Reaper', {\n      queue: this.idleReaperQueue(idleReaper),\n      messageBody: stepfunctions.TaskInput.fromObject({\n        executionArn: stepfunctions.JsonPath.stringAt('$$.Execution.Id'),\n        runnerName: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n        owner: stepfunctions.JsonPath.stringAt('$.owner'),\n        repo: stepfunctions.JsonPath.stringAt('$.repo'),\n        installationId: stepfunctions.JsonPath.stringAt('$.installationId'),\n        maxIdleSeconds: (props?.idleTimeout ?? cdk.Duration.minutes(5)).toSeconds(),\n      }),\n      resultPath: stepfunctions.JsonPath.DISCARD,\n    });\n\n    const providerChooser = new stepfunctions.Choice(this, 'Choose provider');\n    for (const provider of this.providers) {\n      const providerTask = provider.getStepFunctionTask(\n        {\n          runnerTokenPath: stepfunctions.JsonPath.stringAt('$.runner.token'),\n          runnerNamePath: stepfunctions.JsonPath.stringAt('$$.Execution.Name'),\n          githubDomainPath: stepfunctions.JsonPath.stringAt('$.runner.domain'),\n          ownerPath: stepfunctions.JsonPath.stringAt('$.owner'),\n          repoPath: stepfunctions.JsonPath.stringAt('$.repo'),\n        },\n      );\n      providerChooser.when(\n        stepfunctions.Condition.and(\n          ...provider.labels.map(\n            label => stepfunctions.Condition.isPresent(`$.labels.${label.toLowerCase()}`),\n          ),\n        ),\n        providerTask,\n      );\n    }\n\n    providerChooser.otherwise(new stepfunctions.Succeed(this, 'Unknown label'));\n\n    const runProviders = new stepfunctions.Parallel(this, 'Retry on Error', { resultPath: '$.result' })\n      .branch(tokenRetrieverTask.next(providerChooser))\n      .addCatch(\n        // delete runner on failure as it won't remove itself and there is a limit on the number of registered runners\n        deleteRunnerTask.next(new stepfunctions.Fail(this, 'Runner Failed')),\n        {\n          resultPath: '$.error',\n        },\n      );\n\n    if (props?.retryOptions?.retry ?? true) {\n      const interval = props?.retryOptions?.interval ?? cdk.Duration.minutes(1);\n      const maxAttempts = props?.retryOptions?.maxAttempts ?? 23;\n      const backoffRate = props?.retryOptions?.backoffRate ?? 1.3;\n\n      const totalSeconds = interval.toSeconds() * backoffRate ** maxAttempts / (backoffRate - 1);\n      if (totalSeconds >= cdk.Duration.days(1).toSeconds()) {\n        // https://docs.github.com/en/actions/hosting-your-own-runners/managing-self-hosted-runners/about-self-hosted-runners#usage-limits\n        // \"Job queue time - Each job for self-hosted runners can be queued for a maximum of 24 hours. If a self-hosted runner does not start executing the job within this limit, the job is terminated and fails to complete.\"\n        Annotations.of(this).addWarning(`Total retry time is greater than 24 hours (${Math.floor(totalSeconds / 60 / 60)} hours). Jobs expire after 24 hours so it would be a waste of resources to retry further.`);\n      }\n\n      runProviders.addRetry({\n        interval,\n        maxAttempts,\n        backoffRate,\n        // we can't retry on all errors mainly because we don't want to retry on idle runner error\n        // when the idle reaper kills a runner, it will fail the state machine\n        // if we retry, we would just waste resources and end up killing it again\n        errors: ['RunnerTokenError'].concat(...this.providers.map(provider => provider.retryableErrors)),\n      });\n    }\n\n    let logOptions: cdk.aws_stepfunctions.LogOptions | undefined;\n    if (this.props?.logOptions) {\n      this.stateMachineLogGroup = new logs.LogGroup(this, 'Logs', {\n        logGroupName: props?.logOptions?.logGroupName,\n        retention: props?.logOptions?.logRetention ?? logs.RetentionDays.ONE_MONTH,\n        removalPolicy: cdk.RemovalPolicy.DESTROY,\n      });\n\n      logOptions = {\n        destination: this.stateMachineLogGroup,\n        includeExecutionData: props?.logOptions?.includeExecutionData ?? true,\n        level: props?.logOptions?.level ?? stepfunctions.LogLevel.ALL,\n      };\n    }\n\n    const stateMachine = new stepfunctions.StateMachine(\n      this,\n      'Runner Orchestrator',\n      {\n        definition: queueIdleReaperTask.next(runProviders),\n        logs: logOptions,\n      },\n    );\n\n    stateMachine.grantRead(idleReaper);\n    for (const provider of this.providers) {\n      provider.grantStateMachine(stateMachine);\n    }\n\n    return stateMachine;\n  }\n\n  private tokenRetriever() {\n    const func = new TokenRetrieverFunction(\n      this,\n      'token-retriever',\n      {\n        description: 'Get token from GitHub Actions used to start new self-hosted runner',\n        environment: {\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.seconds(30),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    this.secrets.github.grantRead(func);\n    this.secrets.githubPrivateKey.grantRead(func);\n\n    return func;\n  }\n\n  private deleteRunner() {\n    const func = new DeleteRunnerFunction(\n      this,\n      'delete-runner',\n      {\n        description: 'Delete GitHub Actions runner on error',\n        environment: {\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.seconds(30),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    this.secrets.github.grantRead(func);\n    this.secrets.githubPrivateKey.grantRead(func);\n\n    return func;\n  }\n\n  private statusFunction() {\n    const statusFunction = new StatusFunction(\n      this,\n      'status',\n      {\n        description: 'Provide user with status about self-hosted GitHub Actions runners',\n        environment: {\n          WEBHOOK_SECRET_ARN: this.secrets.webhook.secretArn,\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          SETUP_SECRET_ARN: this.secrets.setup.secretArn,\n          WEBHOOK_URL: this.webhook.url,\n          WEBHOOK_HANDLER_ARN: this.webhook.handler.latestVersion.functionArn,\n          STEP_FUNCTION_ARN: this.orchestrator.stateMachineArn,\n          STEP_FUNCTION_LOG_GROUP: this.stateMachineLogGroup?.logGroupName ?? '',\n          SETUP_FUNCTION_URL: this.setupUrl,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.minutes(3),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    const providers = this.providers.map(provider => provider.status(statusFunction));\n\n    // expose providers as stack metadata as it's too big for Lambda environment variables\n    // specifically integration testing got an error because lambda update request was >5kb\n    const stack = cdk.Stack.of(this);\n    const f = (statusFunction.node.defaultChild as lambda.CfnFunction);\n    f.addPropertyOverride('Environment.Variables.LOGICAL_ID', f.logicalId);\n    f.addPropertyOverride('Environment.Variables.STACK_NAME', stack.stackName);\n    f.addMetadata('providers', providers);\n    statusFunction.addToRolePolicy(new iam.PolicyStatement({\n      actions: ['cloudformation:DescribeStackResource'],\n      resources: [stack.stackId],\n    }));\n\n    this.secrets.webhook.grantRead(statusFunction);\n    this.secrets.github.grantRead(statusFunction);\n    this.secrets.githubPrivateKey.grantRead(statusFunction);\n    this.secrets.setup.grantRead(statusFunction);\n    this.orchestrator.grantRead(statusFunction);\n\n    new cdk.CfnOutput(\n      this,\n      'status command',\n      {\n        value: `aws --region ${stack.region} lambda invoke --function-name ${statusFunction.functionName} status.json`,\n      },\n    );\n\n    const access = this.props?.statusAccess ?? LambdaAccess.noAccess();\n    const url = access._bind(this, 'status access', statusFunction);\n\n    if (url !== '') {\n      new cdk.CfnOutput(\n        this,\n        'status url',\n        {\n          value: url,\n        },\n      );\n    }\n  }\n\n  private setupFunction(): string {\n    const setupFunction = new SetupFunction(\n      this,\n      'setup',\n      {\n        description: 'Setup GitHub Actions integration with self-hosted runners',\n        environment: {\n          SETUP_SECRET_ARN: this.secrets.setup.secretArn,\n          WEBHOOK_SECRET_ARN: this.secrets.webhook.secretArn,\n          GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n          GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n          WEBHOOK_URL: this.webhook.url,\n          ...this.extraLambdaEnv,\n        },\n        timeout: cdk.Duration.minutes(3),\n        logRetention: logs.RetentionDays.ONE_MONTH,\n        ...this.extraLambdaProps,\n      },\n    );\n\n    // this.secrets.webhook.grantRead(setupFunction);\n    this.secrets.webhook.grantWrite(setupFunction);\n    this.secrets.github.grantRead(setupFunction);\n    this.secrets.github.grantWrite(setupFunction);\n    // this.secrets.githubPrivateKey.grantRead(setupFunction);\n    this.secrets.githubPrivateKey.grantWrite(setupFunction);\n    this.secrets.setup.grantRead(setupFunction);\n    this.secrets.setup.grantWrite(setupFunction);\n\n    const access = this.props?.setupAccess ?? LambdaAccess.lambdaUrl();\n    return access._bind(this, 'setup access', setupFunction);\n  }\n\n  private checkIntersectingLabels() {\n    // this \"algorithm\" is very inefficient, but good enough for the tiny datasets we expect\n    for (const p1 of this.providers) {\n      for (const p2 of this.providers) {\n        if (p1 == p2) {\n          continue;\n        }\n        if (p1.labels.every(l => p2.labels.includes(l))) {\n          if (p2.labels.every(l => p1.labels.includes(l))) {\n            throw new Error(`Both ${p1.node.path} and ${p2.node.path} use the same labels [${p1.labels.join(', ')}]`);\n          }\n          Annotations.of(p1).addWarning(`Labels [${p1.labels.join(', ')}] intersect with another provider (${p2.node.path} -- [${p2.labels.join(', ')}]). If a workflow specifies the labels [${p1.labels.join(', ')}], it is not guaranteed which provider will be used. It is recommended you do not use intersecting labels`);\n        }\n      }\n    }\n  }\n\n  private idleReaper() {\n    return new IdleRunnerRepearFunction(this, 'Idle Reaper', {\n      description: 'Stop idle GitHub runners to avoid paying for runners when the job was already canceled',\n      environment: {\n        GITHUB_SECRET_ARN: this.secrets.github.secretArn,\n        GITHUB_PRIVATE_KEY_SECRET_ARN: this.secrets.githubPrivateKey.secretArn,\n        ...this.extraLambdaEnv,\n      },\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      timeout: cdk.Duration.minutes(1),\n      ...this.extraLambdaProps,\n    });\n  }\n\n  private idleReaperQueue(reaper: lambda.Function) {\n    const queue = new sqs.Queue(this, 'Idle Reaper Queue', {\n      deliveryDelay: cdk.Duration.minutes(10),\n      visibilityTimeout: cdk.Duration.minutes(10),\n    });\n\n    reaper.addEventSource(new lambda_event_sources.SqsEventSource(queue, {\n      reportBatchItemFailures: true,\n    }));\n\n    this.secrets.github.grantRead(reaper);\n    this.secrets.githubPrivateKey.grantRead(reaper);\n\n    return queue;\n  }\n\n  /**\n   * Metric for the number of GitHub Actions jobs completed. It has `ProviderLabels` and `Status` dimensions. The status can be one of \"Succeeded\", \"SucceededWithIssues\", \"Failed\", \"Canceled\", \"Skipped\", or \"Abandoned\".\n   *\n   * **WARNING:** this method creates a metric filter for each provider. Each metric has a status dimension with six possible values. These resources may incur cost.\n   */\n  public metricJobCompleted(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    if (!this.jobsCompletedMetricFilters) {\n      // we can't use logs.FilterPattern.spaceDelimited() because it has no support for ||\n      // status list taken from https://github.com/actions/runner/blob/be9632302ceef50bfb36ea998cea9c94c75e5d4d/src/Sdk/DTWebApi/WebApi/TaskResult.cs\n      // we need \"...\" for Lambda that prefixes some extra data to log lines\n      const pattern = logs.FilterPattern.literal('[..., marker = \"CDKGHA\", job = \"JOB\", done = \"DONE\", labels, status = \"Succeeded\" || status = \"SucceededWithIssues\" || status = \"Failed\" || status = \"Canceled\" || status = \"Skipped\" || status = \"Abandoned\"]');\n\n      this.jobsCompletedMetricFilters = this.providers.map(p =>\n        p.logGroup.addMetricFilter(`${p.logGroup.node.id} filter`, {\n          metricNamespace: 'GitHubRunners',\n          metricName: 'JobCompleted',\n          filterPattern: pattern,\n          metricValue: '1',\n          // can't with dimensions -- defaultValue: 0,\n          dimensions: {\n            ProviderLabels: '$labels',\n            Status: '$status',\n          },\n        }),\n      );\n\n      for (const metricFilter of this.jobsCompletedMetricFilters) {\n        if (metricFilter.node.defaultChild instanceof logs.CfnMetricFilter) {\n          metricFilter.node.defaultChild.addPropertyOverride('MetricTransformations.0.Unit', 'Count');\n        } else {\n          Annotations.of(metricFilter).addWarning('Unable to set metric filter Unit to Count');\n        }\n      }\n    }\n\n    return new cloudwatch.Metric({\n      namespace: 'GitHubRunners',\n      metricName: 'JobsCompleted',\n      unit: cloudwatch.Unit.COUNT,\n      statistic: cloudwatch.Statistic.SUM,\n      ...props,\n    }).attachTo(this);\n  }\n\n  /**\n   * Metric for successful executions.\n   *\n   * A successful execution doesn't always mean a runner was started. It can be successful even without any label matches.\n   *\n   * A successful runner doesn't mean the job it executed was successful. For that, see {@link metricJobCompleted}.\n   */\n  public metricSucceeded(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricSucceeded(props);\n  }\n\n  /**\n   * Metric for failed runner executions.\n   *\n   * A failed runner usually means the runner failed to start and so a job was never executed. It doesn't necessarily mean the job was executed and failed. For that, see {@link metricJobCompleted}.\n   */\n  public metricFailed(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricFailed(props);\n  }\n\n  /**\n   * Metric for the interval, in milliseconds, between the time the execution starts and the time it closes. This time may be longer than the time the runner took.\n   */\n  public metricTime(props?: cloudwatch.MetricProps): cloudwatch.Metric {\n    return this.orchestrator.metricTime(props);\n  }\n\n  /**\n   * Creates a topic for notifications when a runner image build fails.\n   *\n   * Runner images are rebuilt every week by default. This provides the latest GitHub Runner version and software updates.\n   *\n   * If you want to be sure you are using the latest runner version, you can use this topic to be notified when a build fails.\n   */\n  public failedImageBuildsTopic() {\n    const topic = new sns.Topic(this, 'Failed Runner Image Builds');\n    const stack = cdk.Stack.of(this);\n    cdk.Aspects.of(stack).add(new CodeBuildImageBuilderFailedBuildNotifier(topic));\n    cdk.Aspects.of(stack).add(\n      new AwsImageBuilderFailedBuildNotifier(\n        AwsImageBuilderFailedBuildNotifier.createFilteringTopic(this, topic),\n      ),\n    );\n    return topic;\n  }\n}\n"]}
|
package/lib/secrets.js
CHANGED
|
@@ -49,6 +49,6 @@ class Secrets extends constructs_1.Construct {
|
|
|
49
49
|
}
|
|
50
50
|
}
|
|
51
51
|
_a = JSII_RTTI_SYMBOL_1;
|
|
52
|
-
Secrets[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Secrets", version: "0.
|
|
52
|
+
Secrets[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Secrets", version: "0.10.0" };
|
|
53
53
|
exports.Secrets = Secrets;
|
|
54
54
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VjcmV0cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9zZWNyZXRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkNBQW1FO0FBQ25FLG1DQUFtQztBQUNuQywyQ0FBdUM7QUFFdkM7O0dBRUc7QUFDSCxNQUFhLE9BQVEsU0FBUSxzQkFBUztJQTBCcEMsWUFBWSxLQUFnQixFQUFFLEVBQVU7UUFDdEMsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLENBQUMsT0FBTyxHQUFHLElBQUksZ0NBQWMsQ0FBQyxNQUFNLENBQ3RDLElBQUksRUFDSixTQUFTLEVBQ1Q7WUFDRSxvQkFBb0IsRUFBRTtnQkFDcEIsb0JBQW9CLEVBQUUsSUFBSTtnQkFDMUIsaUJBQWlCLEVBQUUsZUFBZTtnQkFDbEMsWUFBWSxFQUFFLEtBQUs7Z0JBQ25CLGtCQUFrQixFQUFFLElBQUk7YUFDekI7U0FDRixDQUNGLENBQUM7UUFFRixJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksZ0NBQWMsQ0FBQyxNQUFNLENBQ3JDLElBQUksRUFDSixRQUFRLEVBQ1I7WUFDRSxvQkFBb0IsRUFBRTtnQkFDcEIsb0JBQW9CLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQztvQkFDbkMsTUFBTSxFQUFFLFlBQVk7b0JBQ3BCLEtBQUssRUFBRSxFQUFFO29CQUNULGlCQUFpQixFQUFFLEVBQUU7aUJBQ3RCLENBQUM7Z0JBQ0YsaUJBQWlCLEVBQUUsT0FBTztnQkFDMUIsWUFBWSxFQUFFLEtBQUs7Z0JBQ25CLGtCQUFrQixFQUFFLElBQUk7YUFDekI7U0FDRixDQUNGLENBQUM7UUFFRix5R0FBeUc7UUFDekcsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksZ0NBQWMsQ0FBQyxNQUFNLENBQy9DLElBQUksRUFDSixvQkFBb0IsRUFDcEI7WUFDRSxpQkFBaUIsRUFBRSxHQUFHLENBQUMsV0FBVyxDQUFDLGVBQWUsQ0FBQyxxRUFBcUUsQ0FBQztTQUMxSCxDQUNGLENBQUM7UUFFRixJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksZ0NBQWMsQ0FBQyxNQUFNLENBQ3BDLElBQUksRUFDSixPQUFPLEVBQ1A7WUFDRSxvQkFBb0IsRUFBRTtnQkFDcEIsb0JBQW9CLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQztvQkFDbkMsS0FBSyxFQUFFLEVBQUU7aUJBQ1YsQ0FBQztnQkFDRixpQkFBaUIsRUFBRSxPQUFPO2dCQUMxQixZQUFZLEVBQUUsS0FBSztnQkFDbkIsa0JBQWtCLEVBQUUsSUFBSTthQUN6QjtTQUNGLENBQ0YsQ0FBQztJQUNKLENBQUM7Ozs7QUFsRlUsMEJBQU8iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBhd3Nfc2VjcmV0c21hbmFnZXIgYXMgc2VjcmV0c21hbmFnZXIgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbi8qKlxuICogU2VjcmV0cyByZXF1aXJlZCBmb3IgR2l0SHViIHJ1bm5lcnMgb3BlcmF0aW9uLlxuICovXG5leHBvcnQgY2xhc3MgU2VjcmV0cyBleHRlbmRzIENvbnN0cnVjdCB7XG4gIC8qKlxuICAgKiBXZWJob29rIHNlY3JldCB1c2VkIHRvIGNvbmZpcm0gZXZlbnRzIGFyZSBjb21pbmcgZnJvbSBHaXRIdWIgYW5kIG5vd2hlcmUgZWxzZS5cbiAgICovXG4gIHJlYWRvbmx5IHdlYmhvb2s6IHNlY3JldHNtYW5hZ2VyLlNlY3JldDtcblxuICAvKipcbiAgICogQXV0aGVudGljYXRpb24gc2VjcmV0IGZvciBHaXRIdWIgY29udGFpbmluZyBlaXRoZXIgYXBwIGRldGFpbHMgb3IgcGVyc29uYWwgYXV0aGVudGljYXRpb24gdG9rZW4uIFRoaXMgc2VjcmV0IGlzIHVzZWQgdG8gcmVnaXN0ZXIgcnVubmVycyBhbmRcbiAgICogY2FuY2VsIGpvYnMgd2hlbiB0aGUgcnVubmVyIGZhaWxzIHRvIHN0YXJ0LlxuICAgKlxuICAgKiBUaGlzIHNlY3JldCBpcyBtZWFudCB0byBiZSBlZGl0ZWQgYnkgdGhlIHVzZXIgYWZ0ZXIgYmVpbmcgY3JlYXRlZC5cbiAgICovXG4gIHJlYWRvbmx5IGdpdGh1Yjogc2VjcmV0c21hbmFnZXIuU2VjcmV0O1xuXG4gIC8qKlxuICAgKiBHaXRIdWIgYXBwIHByaXZhdGUga2V5LiBOb3QgbmVlZGVkIHdoZW4gdXNpbmcgcGVyc29uYWwgYXV0aGVudGljYXRpb24gdG9rZW5zLlxuICAgKlxuICAgKiBUaGlzIHNlY3JldCBpcyBtZWFudCB0byBiZSBlZGl0ZWQgYnkgdGhlIHVzZXIgYWZ0ZXIgYmVpbmcgY3JlYXRlZC4gSXQgaXMgc2VwYXJhdGUgdGhhbiB0aGUgbWFpbiBHaXRIdWIgc2VjcmV0IGJlY2F1c2UgaW5zZXJ0aW5nIHByaXZhdGUga2V5cyBpbnRvIEpTT04gaXMgaGFyZC5cbiAgICovXG4gIHJlYWRvbmx5IGdpdGh1YlByaXZhdGVLZXk6IHNlY3JldHNtYW5hZ2VyLlNlY3JldDtcblxuICAvKipcbiAgICogU2V0dXAgc2VjcmV0IHVzZWQgdG8gYXV0aGVudGljYXRlIHVzZXIgZm9yIG91ciBzZXR1cCB3aXphcmQuIFNob3VsZCBiZSBlbXB0eSBhZnRlciBzZXR1cCBoYXMgYmVlbiBjb21wbGV0ZWQuXG4gICAqL1xuICByZWFkb25seSBzZXR1cDogc2VjcmV0c21hbmFnZXIuU2VjcmV0O1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgdGhpcy53ZWJob29rID0gbmV3IHNlY3JldHNtYW5hZ2VyLlNlY3JldChcbiAgICAgIHRoaXMsXG4gICAgICAnV2ViaG9vaycsXG4gICAgICB7XG4gICAgICAgIGdlbmVyYXRlU2VjcmV0U3RyaW5nOiB7XG4gICAgICAgICAgc2VjcmV0U3RyaW5nVGVtcGxhdGU6ICd7fScsXG4gICAgICAgICAgZ2VuZXJhdGVTdHJpbmdLZXk6ICd3ZWJob29rU2VjcmV0JyxcbiAgICAgICAgICBpbmNsdWRlU3BhY2U6IGZhbHNlLFxuICAgICAgICAgIGV4Y2x1ZGVQdW5jdHVhdGlvbjogdHJ1ZSxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgKTtcblxuICAgIHRoaXMuZ2l0aHViID0gbmV3IHNlY3JldHNtYW5hZ2VyLlNlY3JldChcbiAgICAgIHRoaXMsXG4gICAgICAnR2l0SHViJyxcbiAgICAgIHtcbiAgICAgICAgZ2VuZXJhdGVTZWNyZXRTdHJpbmc6IHtcbiAgICAgICAgICBzZWNyZXRTdHJpbmdUZW1wbGF0ZTogSlNPTi5zdHJpbmdpZnkoe1xuICAgICAgICAgICAgZG9tYWluOiAnZ2l0aHViLmNvbScsXG4gICAgICAgICAgICBhcHBJZDogJycsXG4gICAgICAgICAgICBwZXJzb25hbEF1dGhUb2tlbjogJycsXG4gICAgICAgICAgfSksXG4gICAgICAgICAgZ2VuZXJhdGVTdHJpbmdLZXk6ICdkdW1teScsXG4gICAgICAgICAgaW5jbHVkZVNwYWNlOiBmYWxzZSxcbiAgICAgICAgICBleGNsdWRlUHVuY3R1YXRpb246IHRydWUsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgICk7XG5cbiAgICAvLyB3ZSBjcmVhdGUgYSBzZXBhcmF0ZSBzZWNyZXQgZm9yIHRoZSBwcml2YXRlIGtleSBiZWNhdXNlIHB1dHRpbmcgaXQgaW4gSlNPTiBzZWNyZXQgaXMgaGFyZCBmb3IgdGhlIHVzZXJcbiAgICB0aGlzLmdpdGh1YlByaXZhdGVLZXkgPSBuZXcgc2VjcmV0c21hbmFnZXIuU2VjcmV0KFxuICAgICAgdGhpcyxcbiAgICAgICdHaXRIdWIgUHJpdmF0ZSBLZXknLFxuICAgICAge1xuICAgICAgICBzZWNyZXRTdHJpbmdWYWx1ZTogY2RrLlNlY3JldFZhbHVlLnVuc2FmZVBsYWluVGV4dCgnLS0tLS1CRUdJTiBSU0EgUFJJVkFURSBLRVktLS0tLVxcbi4uLlxcbi0tLS0tRU5EIFJTQSBQUklWQVRFIEtFWS0tLS0tJyksXG4gICAgICB9LFxuICAgICk7XG5cbiAgICB0aGlzLnNldHVwID0gbmV3IHNlY3JldHNtYW5hZ2VyLlNlY3JldChcbiAgICAgIHRoaXMsXG4gICAgICAnU2V0dXAnLFxuICAgICAge1xuICAgICAgICBnZW5lcmF0ZVNlY3JldFN0cmluZzoge1xuICAgICAgICAgIHNlY3JldFN0cmluZ1RlbXBsYXRlOiBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgICAgICB0b2tlbjogJycsXG4gICAgICAgICAgfSksXG4gICAgICAgICAgZ2VuZXJhdGVTdHJpbmdLZXk6ICd0b2tlbicsXG4gICAgICAgICAgaW5jbHVkZVNwYWNlOiBmYWxzZSxcbiAgICAgICAgICBleGNsdWRlUHVuY3R1YXRpb246IHRydWUsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgICk7XG4gIH1cbn0iXX0=
|
package/lib/setup.lambda.js
CHANGED
|
@@ -1,10 +1,9 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
/* eslint-disable import/no-extraneous-dependencies */
|
|
4
3
|
const crypto = require("crypto");
|
|
5
4
|
const fs = require("fs");
|
|
6
5
|
const rest_1 = require("@octokit/rest");
|
|
7
|
-
const
|
|
6
|
+
const lambda_github_1 = require("./lambda-github");
|
|
8
7
|
const lambda_helpers_1 = require("./lambda-helpers");
|
|
9
8
|
const nonce = crypto.randomBytes(64).toString('hex');
|
|
10
9
|
function getHtml(baseUrl, token, domain) {
|
|
@@ -75,7 +74,7 @@ async function handleNewApp(event) {
|
|
|
75
74
|
return response(400, 'Invalid code');
|
|
76
75
|
}
|
|
77
76
|
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
78
|
-
const baseUrl = (0,
|
|
77
|
+
const baseUrl = (0, lambda_github_1.baseUrlFromDomain)(githubSecrets.domain);
|
|
79
78
|
const newApp = await new rest_1.Octokit({ baseUrl }).rest.apps.createFromManifest({ code });
|
|
80
79
|
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({
|
|
81
80
|
domain: new URL(newApp.data.html_url).host,
|
|
@@ -149,4 +148,4 @@ exports.handler = async function (event) {
|
|
|
149
148
|
return response(500, `<b>Error:</b> ${e}`);
|
|
150
149
|
}
|
|
151
150
|
};
|
|
152
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.lambda.js","sourceRoot":"","sources":["../src/setup.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/no-extraneous-dependencies */\nimport * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport { Octokit } from '@octokit/rest';\n/* eslint-disable-next-line import/no-extraneous-dependencies,import/no-unresolved */\nimport * as AWSLambda from 'aws-lambda';\nimport { baseUrlFromDomain } from './github';\nimport { getSecretJsonValue, updateSecretValue } from './lambda-helpers';\n\ntype ApiGatewayEvent = AWSLambda.APIGatewayProxyEvent | AWSLambda.APIGatewayProxyEventV2;\n\nconst nonce = crypto.randomBytes(64).toString('hex');\n\nfunction getHtml(baseUrl: string, token: string, domain: string): string {\n  return fs.readFileSync('index.html', 'utf-8')\n    .replace(/INSERT_WEBHOOK_URL_HERE/g, process.env.WEBHOOK_URL!)\n    .replace(/INSERT_BASE_URL_HERE/g, baseUrl)\n    .replace(/INSERT_TOKEN_HERE/g, token)\n    .replace(/INSERT_SECRET_ARN_HERE/g, process.env.SETUP_SECRET_ARN!)\n    .replace(/INSERT_DOMAIN_HERE/g, domain)\n    .replace(/<script/g, `<script nonce=\"${nonce}\"`)\n    .replace(/<style/g, `<style nonce=\"${nonce}\"`);\n}\n\nfunction response(code: number, body: string): AWSLambda.APIGatewayProxyResultV2 {\n  return {\n    statusCode: code,\n    headers: {\n      'Content-Type': 'text/html',\n      'Content-Security-Policy': `default-src 'unsafe-inline' 'nonce-${nonce}'; img-src data:; connect-src 'self'; form-action https:; frame-ancestors 'none'; object-src 'none'; base-uri 'self'`,\n    },\n    body: body,\n  };\n}\n\nasync function handleRoot(event: ApiGatewayEvent, setupToken: string): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`;\n  const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n\n  return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain));\n}\n\nfunction decodeBody(event: ApiGatewayEvent) {\n  let body = event.body;\n  if (!body) {\n    throw new Error('No body found');\n  }\n  if (event.isBase64Encoded) {\n    body = Buffer.from(body, 'base64').toString('utf-8');\n  }\n  return JSON.parse(body);\n}\n\nasync function handleDomain(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n  if (!body.domain) {\n    return response(400, 'Invalid domain');\n  }\n\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  githubSecrets.domain = body.domain;\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets));\n\n  return response(200, 'Domain set');\n}\n\nasync function handlePat(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n  if (!body.pat || !body.domain) {\n    return response(400, 'Invalid personal access token');\n  }\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: body.domain,\n    appId: '',\n    personalAuthToken: body.pat,\n  }));\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, 'Personal access token set');\n}\n\nasync function handleNewApp(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  if (!event.queryStringParameters) {\n    return response( 400, 'Invalid code');\n  }\n\n  const code = event.queryStringParameters.code;\n\n  if (!code) {\n    return response( 400, 'Invalid code');\n  }\n\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n  const newApp = await new Octokit({ baseUrl }).rest.apps.createFromManifest({ code });\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: new URL(newApp.data.html_url).host,\n    appId: newApp.data.id,\n    personalAuthToken: '',\n  }));\n  await updateSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, newApp.data.pem);\n  await updateSecretValue(process.env.WEBHOOK_SECRET_ARN, JSON.stringify({\n    webhookSecret: newApp.data.webhook_secret,\n  }));\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, `New app set. <a href=\"${newApp.data.html_url}/installations/new\">Install it</a> for your repositories.`);\n}\n\nasync function handleExistingApp(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n\n  if (!body.appid || !body.pk || !body.domain) {\n    return response( 400, 'Missing fields');\n  }\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: body.domain,\n    appId: body.appid,\n    personalAuthToken: '',\n  }));\n  await updateSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, body.pk as string);\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, 'Existing app set. Don\\'t forget to set up the webhook.');\n}\n\nexports.handler = async function (event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  // confirm required environment variables\n  if (!process.env.WEBHOOK_URL) {\n    throw new Error('Missing environment variables');\n  }\n\n  const setupToken = (await getSecretJsonValue(process.env.SETUP_SECRET_ARN)).token;\n\n  // bail out if setup was already completed\n  if (!setupToken) {\n    return response(200, 'Setup already complete. Put a new token in the setup secret if you want to redo it.');\n  }\n\n  if (!event.queryStringParameters) {\n    return response(403, 'Wrong setup token.');\n  }\n\n  // safely confirm url token matches our secret\n  const urlToken = event.queryStringParameters.token || event.queryStringParameters.state || '';\n  if (urlToken.length != setupToken.length || !crypto.timingSafeEqual(Buffer.from(urlToken, 'utf-8'), Buffer.from(setupToken, 'utf-8'))) {\n    return response(403, 'Wrong setup token.');\n  }\n\n  // handle requests\n  try {\n    const path = (event as AWSLambda.APIGatewayProxyEvent).path ?? (event as AWSLambda.APIGatewayProxyEventV2).rawPath;\n    const method = (event as AWSLambda.APIGatewayProxyEvent).httpMethod ?? (event as AWSLambda.APIGatewayProxyEventV2).requestContext.http.method;\n    if (path == '/') {\n      return await handleRoot(event, setupToken);\n    } else if (path == '/domain' && method == 'POST') {\n      return await handleDomain(event);\n    } else if (path == '/pat' && method == 'POST') {\n      return await handlePat(event);\n    } else if (path == '/complete-new-app' && method == 'GET') {\n      return await handleNewApp(event);\n    } else if (path == '/app' && method == 'POST') {\n      return await handleExistingApp(event);\n    } else {\n      return response(404, 'Not found');\n    }\n  } catch (e) {\n    console.error(e);\n    return response(500, `<b>Error:</b> ${e}`);\n  }\n};\n"]}
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.lambda.js","sourceRoot":"","sources":["../src/setup.lambda.ts"],"names":[],"mappings":";;AAAA,iCAAiC;AACjC,yBAAyB;AACzB,wCAAwC;AAExC,mDAAoD;AACpD,qDAAyE;AAIzE,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;AAErD,SAAS,OAAO,CAAC,OAAe,EAAE,KAAa,EAAE,MAAc;IAC7D,OAAO,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC;SAC1C,OAAO,CAAC,0BAA0B,EAAE,OAAO,CAAC,GAAG,CAAC,WAAY,CAAC;SAC7D,OAAO,CAAC,uBAAuB,EAAE,OAAO,CAAC;SACzC,OAAO,CAAC,oBAAoB,EAAE,KAAK,CAAC;SACpC,OAAO,CAAC,yBAAyB,EAAE,OAAO,CAAC,GAAG,CAAC,gBAAiB,CAAC;SACjE,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC;SACtC,OAAO,CAAC,UAAU,EAAE,kBAAkB,KAAK,GAAG,CAAC;SAC/C,OAAO,CAAC,SAAS,EAAE,iBAAiB,KAAK,GAAG,CAAC,CAAC;AACnD,CAAC;AAED,SAAS,QAAQ,CAAC,IAAY,EAAE,IAAY;IAC1C,OAAO;QACL,UAAU,EAAE,IAAI;QAChB,OAAO,EAAE;YACP,cAAc,EAAE,WAAW;YAC3B,yBAAyB,EAAE,sCAAsC,KAAK,sHAAsH;SAC7L;QACD,IAAI,EAAE,IAAI;KACX,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,UAAU,CAAC,KAAsB,EAAE,UAAkB;IAClE,MAAM,KAAK,GAAG,KAAK,CAAC,cAAc,CAAC,KAAK,IAAI,UAAU,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;IAC/F,MAAM,YAAY,GAAG,WAAW,KAAK,CAAC,cAAc,CAAC,UAAU,GAAG,KAAK,EAAE,CAAC;IAC1E,MAAM,aAAa,GAAG,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAE9E,OAAO,QAAQ,CAAC,GAAG,EAAE,OAAO,CAAC,YAAY,EAAE,UAAU,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC;AAChF,CAAC;AAED,SAAS,UAAU,CAAC,KAAsB;IACxC,IAAI,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;IACtB,IAAI,CAAC,IAAI,EAAE;QACT,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;KAClC;IACD,IAAI,KAAK,CAAC,eAAe,EAAE;QACzB,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;KACtD;IACD,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,KAAK,UAAU,YAAY,CAAC,KAAsB;IAChD,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;QAChB,OAAO,QAAQ,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;KACxC;IAED,MAAM,aAAa,GAAG,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC9E,aAAa,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;IACnC,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC;IAEtF,OAAO,QAAQ,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;AACrC,CAAC;AAED,KAAK,UAAU,SAAS,CAAC,KAAsB;IAC7C,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;QAC7B,OAAO,QAAQ,CAAC,GAAG,EAAE,+BAA+B,CAAC,CAAC;KACvD;IAED,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC;QACpE,MAAM,EAAE,IAAI,CAAC,MAAM;QACnB,KAAK,EAAE,EAAE;QACT,iBAAiB,EAAE,IAAI,CAAC,GAAG;KAC5B,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAE,GAAG,EAAE,2BAA2B,CAAC,CAAC;AACrD,CAAC;AAED,KAAK,UAAU,YAAY,CAAC,KAAsB;IAChD,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE;QAChC,OAAO,QAAQ,CAAE,GAAG,EAAE,cAAc,CAAC,CAAC;KACvC;IAED,MAAM,IAAI,GAAG,KAAK,CAAC,qBAAqB,CAAC,IAAI,CAAC;IAE9C,IAAI,CAAC,IAAI,EAAE;QACT,OAAO,QAAQ,CAAE,GAAG,EAAE,cAAc,CAAC,CAAC;KACvC;IAED,MAAM,aAAa,GAAG,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC9E,MAAM,OAAO,GAAG,IAAA,iCAAiB,EAAC,aAAa,CAAC,MAAM,CAAC,CAAC;IACxD,MAAM,MAAM,GAAG,MAAM,IAAI,cAAO,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;IAErF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC;QACpE,MAAM,EAAE,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI;QAC1C,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE;QACrB,iBAAiB,EAAE,EAAE;KACtB,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACpF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC;QACrE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,cAAc;KAC1C,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAE,GAAG,EAAE,yBAAyB,MAAM,CAAC,IAAI,CAAC,QAAQ,2DAA2D,CAAC,CAAC;AAClI,CAAC;AAED,KAAK,UAAU,iBAAiB,CAAC,KAAsB;IACrD,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAE/B,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;QAC3C,OAAO,QAAQ,CAAE,GAAG,EAAE,gBAAgB,CAAC,CAAC;KACzC;IAED,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC;QACpE,MAAM,EAAE,IAAI,CAAC,MAAM;QACnB,KAAK,EAAE,IAAI,CAAC,KAAK;QACjB,iBAAiB,EAAE,EAAE;KACtB,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,IAAI,CAAC,EAAY,CAAC,CAAC;IACtF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAE,GAAG,EAAE,wDAAwD,CAAC,CAAC;AAClF,CAAC;AAED,OAAO,CAAC,OAAO,GAAG,KAAK,WAAW,KAAsB;IACtD,yCAAyC;IACzC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,EAAE;QAC5B,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;KAClD;IAED,MAAM,UAAU,GAAG,CAAC,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC;IAElF,0CAA0C;IAC1C,IAAI,CAAC,UAAU,EAAE;QACf,OAAO,QAAQ,CAAC,GAAG,EAAE,qFAAqF,CAAC,CAAC;KAC7G;IAED,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE;QAChC,OAAO,QAAQ,CAAC,GAAG,EAAE,oBAAoB,CAAC,CAAC;KAC5C;IAED,8CAA8C;IAC9C,MAAM,QAAQ,GAAG,KAAK,CAAC,qBAAqB,CAAC,KAAK,IAAI,KAAK,CAAC,qBAAqB,CAAC,KAAK,IAAI,EAAE,CAAC;IAC9F,IAAI,QAAQ,CAAC,MAAM,IAAI,UAAU,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC,EAAE;QACrI,OAAO,QAAQ,CAAC,GAAG,EAAE,oBAAoB,CAAC,CAAC;KAC5C;IAED,kBAAkB;IAClB,IAAI;QACF,MAAM,IAAI,GAAI,KAAwC,CAAC,IAAI,IAAK,KAA0C,CAAC,OAAO,CAAC;QACnH,MAAM,MAAM,GAAI,KAAwC,CAAC,UAAU,IAAK,KAA0C,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC;QAC9I,IAAI,IAAI,IAAI,GAAG,EAAE;YACf,OAAO,MAAM,UAAU,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;SAC5C;aAAM,IAAI,IAAI,IAAI,SAAS,IAAI,MAAM,IAAI,MAAM,EAAE;YAChD,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;SAClC;aAAM,IAAI,IAAI,IAAI,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE;YAC7C,OAAO,MAAM,SAAS,CAAC,KAAK,CAAC,CAAC;SAC/B;aAAM,IAAI,IAAI,IAAI,mBAAmB,IAAI,MAAM,IAAI,KAAK,EAAE;YACzD,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;SAClC;aAAM,IAAI,IAAI,IAAI,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE;YAC7C,OAAO,MAAM,iBAAiB,CAAC,KAAK,CAAC,CAAC;SACvC;aAAM;YACL,OAAO,QAAQ,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;SACnC;KACF;IAAC,OAAO,CAAC,EAAE;QACV,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACjB,OAAO,QAAQ,CAAC,GAAG,EAAE,iBAAiB,CAAC,EAAE,CAAC,CAAC;KAC5C;AACH,CAAC,CAAC","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport { Octokit } from '@octokit/rest';\nimport * as AWSLambda from 'aws-lambda';\nimport { baseUrlFromDomain } from './lambda-github';\nimport { getSecretJsonValue, updateSecretValue } from './lambda-helpers';\n\ntype ApiGatewayEvent = AWSLambda.APIGatewayProxyEvent | AWSLambda.APIGatewayProxyEventV2;\n\nconst nonce = crypto.randomBytes(64).toString('hex');\n\nfunction getHtml(baseUrl: string, token: string, domain: string): string {\n  return fs.readFileSync('index.html', 'utf-8')\n    .replace(/INSERT_WEBHOOK_URL_HERE/g, process.env.WEBHOOK_URL!)\n    .replace(/INSERT_BASE_URL_HERE/g, baseUrl)\n    .replace(/INSERT_TOKEN_HERE/g, token)\n    .replace(/INSERT_SECRET_ARN_HERE/g, process.env.SETUP_SECRET_ARN!)\n    .replace(/INSERT_DOMAIN_HERE/g, domain)\n    .replace(/<script/g, `<script nonce=\"${nonce}\"`)\n    .replace(/<style/g, `<style nonce=\"${nonce}\"`);\n}\n\nfunction response(code: number, body: string): AWSLambda.APIGatewayProxyResultV2 {\n  return {\n    statusCode: code,\n    headers: {\n      'Content-Type': 'text/html',\n      'Content-Security-Policy': `default-src 'unsafe-inline' 'nonce-${nonce}'; img-src data:; connect-src 'self'; form-action https:; frame-ancestors 'none'; object-src 'none'; base-uri 'self'`,\n    },\n    body: body,\n  };\n}\n\nasync function handleRoot(event: ApiGatewayEvent, setupToken: string): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`;\n  const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n\n  return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain));\n}\n\nfunction decodeBody(event: ApiGatewayEvent) {\n  let body = event.body;\n  if (!body) {\n    throw new Error('No body found');\n  }\n  if (event.isBase64Encoded) {\n    body = Buffer.from(body, 'base64').toString('utf-8');\n  }\n  return JSON.parse(body);\n}\n\nasync function handleDomain(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n  if (!body.domain) {\n    return response(400, 'Invalid domain');\n  }\n\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  githubSecrets.domain = body.domain;\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets));\n\n  return response(200, 'Domain set');\n}\n\nasync function handlePat(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n  if (!body.pat || !body.domain) {\n    return response(400, 'Invalid personal access token');\n  }\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: body.domain,\n    appId: '',\n    personalAuthToken: body.pat,\n  }));\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, 'Personal access token set');\n}\n\nasync function handleNewApp(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  if (!event.queryStringParameters) {\n    return response( 400, 'Invalid code');\n  }\n\n  const code = event.queryStringParameters.code;\n\n  if (!code) {\n    return response( 400, 'Invalid code');\n  }\n\n  const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n  const newApp = await new Octokit({ baseUrl }).rest.apps.createFromManifest({ code });\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: new URL(newApp.data.html_url).host,\n    appId: newApp.data.id,\n    personalAuthToken: '',\n  }));\n  await updateSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, newApp.data.pem);\n  await updateSecretValue(process.env.WEBHOOK_SECRET_ARN, JSON.stringify({\n    webhookSecret: newApp.data.webhook_secret,\n  }));\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, `New app set. <a href=\"${newApp.data.html_url}/installations/new\">Install it</a> for your repositories.`);\n}\n\nasync function handleExistingApp(event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const body = decodeBody(event);\n\n  if (!body.appid || !body.pk || !body.domain) {\n    return response( 400, 'Missing fields');\n  }\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify({\n    domain: body.domain,\n    appId: body.appid,\n    personalAuthToken: '',\n  }));\n  await updateSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, body.pk as string);\n  await updateSecretValue(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));\n\n  return response( 200, 'Existing app set. Don\\'t forget to set up the webhook.');\n}\n\nexports.handler = async function (event: ApiGatewayEvent): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  // confirm required environment variables\n  if (!process.env.WEBHOOK_URL) {\n    throw new Error('Missing environment variables');\n  }\n\n  const setupToken = (await getSecretJsonValue(process.env.SETUP_SECRET_ARN)).token;\n\n  // bail out if setup was already completed\n  if (!setupToken) {\n    return response(200, 'Setup already complete. Put a new token in the setup secret if you want to redo it.');\n  }\n\n  if (!event.queryStringParameters) {\n    return response(403, 'Wrong setup token.');\n  }\n\n  // safely confirm url token matches our secret\n  const urlToken = event.queryStringParameters.token || event.queryStringParameters.state || '';\n  if (urlToken.length != setupToken.length || !crypto.timingSafeEqual(Buffer.from(urlToken, 'utf-8'), Buffer.from(setupToken, 'utf-8'))) {\n    return response(403, 'Wrong setup token.');\n  }\n\n  // handle requests\n  try {\n    const path = (event as AWSLambda.APIGatewayProxyEvent).path ?? (event as AWSLambda.APIGatewayProxyEventV2).rawPath;\n    const method = (event as AWSLambda.APIGatewayProxyEvent).httpMethod ?? (event as AWSLambda.APIGatewayProxyEventV2).requestContext.http.method;\n    if (path == '/') {\n      return await handleRoot(event, setupToken);\n    } else if (path == '/domain' && method == 'POST') {\n      return await handleDomain(event);\n    } else if (path == '/pat' && method == 'POST') {\n      return await handlePat(event);\n    } else if (path == '/complete-new-app' && method == 'GET') {\n      return await handleNewApp(event);\n    } else if (path == '/app' && method == 'POST') {\n      return await handleExistingApp(event);\n    } else {\n      return response(404, 'Not found');\n    }\n  } catch (e) {\n    console.error(e);\n    return response(500, `<b>Error:</b> ${e}`);\n  }\n};\n"]}
|