@cloudsnorkel/cdk-github-runners 0.9.3 → 0.9.5
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 +10 -9
- package/.jsii +397 -332
- package/API.md +56 -9
- package/README.md +15 -2
- package/assets/{lambdas/delete-runner.lambda → delete-runner.lambda}/index.js +96 -56
- package/assets/{lambdas → image-builders/aws-image-builder}/delete-ami.lambda/index.js +3 -3
- package/assets/image-builders/aws-image-builder/filter-failed-builds.lambda/index.js +39 -0
- package/assets/{lambdas/aws-image-builder-versioner.lambda → image-builders/aws-image-builder/versioner.lambda}/index.js +98 -58
- package/assets/{lambdas → providers}/build-image.lambda/index.js +3 -3
- package/assets/{lambdas → providers}/update-lambda.lambda/index.js +1 -1
- package/assets/{lambdas/setup.lambda → setup.lambda}/index.js +4 -4
- package/assets/{lambdas/status.lambda → status.lambda}/index.js +96 -56
- package/assets/{lambdas/token-retriever.lambda → token-retriever.lambda}/index.js +96 -56
- package/assets/{lambdas/webhook-handler.lambda → webhook-handler.lambda}/index.js +3 -3
- package/lib/access.js +1 -1
- package/lib/{lambdas/delete-runner-function.d.ts → delete-runner-function.d.ts} +1 -1
- package/lib/delete-runner-function.js +23 -0
- package/lib/delete-runner.lambda.js +69 -0
- package/lib/github.js +50 -0
- package/lib/image-builders/api.js +47 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/ami.d.ts +2 -3
- package/lib/image-builders/aws-image-builder/ami.js +93 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/builder.d.ts +21 -5
- package/lib/image-builders/aws-image-builder/builder.js +529 -0
- package/lib/image-builders/aws-image-builder/common.js +46 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/container.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/container.js +63 -0
- package/lib/{lambdas → image-builders/aws-image-builder}/delete-ami-function.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/delete-ami-function.js +23 -0
- package/lib/image-builders/aws-image-builder/delete-ami.lambda.js +87 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/ami.d.ts +2 -3
- package/lib/image-builders/aws-image-builder/deprecated/ami.js +240 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/common.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/common.js +144 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/container.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/container.js +222 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/index.js +1 -1
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/linux-components.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +172 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/windows-components.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +126 -0
- package/lib/image-builders/aws-image-builder/filter-failed-builds-function.d.ts +13 -0
- package/lib/image-builders/aws-image-builder/filter-failed-builds-function.js +23 -0
- package/lib/image-builders/aws-image-builder/filter-failed-builds.lambda.js +18 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/index.js +1 -1
- package/lib/image-builders/aws-image-builder/versioner-function.d.ts +13 -0
- package/lib/image-builders/aws-image-builder/versioner-function.js +23 -0
- package/lib/image-builders/aws-image-builder/versioner.lambda.js +96 -0
- package/lib/{providers/image-builders → image-builders}/codebuild-deprecated.d.ts +2 -2
- package/lib/image-builders/codebuild-deprecated.js +373 -0
- package/lib/{providers/image-builders → image-builders}/codebuild.d.ts +13 -4
- package/lib/image-builders/codebuild.js +287 -0
- package/lib/{providers/image-builders → image-builders}/common.d.ts +4 -2
- package/lib/image-builders/common.js +61 -0
- package/lib/{providers/image-builders → image-builders}/components.d.ts +8 -2
- package/lib/image-builders/components.js +568 -0
- package/lib/{providers/image-builders → image-builders}/index.js +1 -1
- package/lib/{providers/image-builders → image-builders}/static.d.ts +1 -1
- package/lib/image-builders/static.js +58 -0
- package/lib/lambda-helpers.js +66 -0
- package/lib/{lambdas → providers}/build-image-function.d.ts +1 -1
- package/lib/providers/build-image-function.js +23 -0
- package/lib/providers/build-image.lambda.js +92 -0
- package/lib/providers/codebuild.d.ts +1 -1
- package/lib/providers/codebuild.js +4 -4
- package/lib/providers/common.js +3 -3
- package/lib/providers/ec2.d.ts +1 -1
- package/lib/providers/ec2.js +4 -4
- package/lib/providers/ecs.d.ts +1 -1
- package/lib/providers/ecs.js +8 -4
- package/lib/providers/fargate.d.ts +1 -1
- package/lib/providers/fargate.js +4 -4
- package/lib/providers/index.d.ts +1 -1
- package/lib/providers/index.js +2 -2
- package/lib/providers/lambda.d.ts +1 -1
- package/lib/providers/lambda.js +5 -5
- package/lib/{lambdas → providers}/update-lambda-function.d.ts +1 -1
- package/lib/providers/update-lambda-function.js +23 -0
- package/lib/providers/update-lambda.lambda.js +34 -0
- package/lib/runner.d.ts +9 -1
- package/lib/runner.js +24 -12
- package/lib/secrets.js +1 -1
- package/lib/{lambdas/setup-function.d.ts → setup-function.d.ts} +1 -1
- package/lib/setup-function.js +23 -0
- package/lib/setup.lambda.js +152 -0
- package/lib/{lambdas/status-function.d.ts → status-function.d.ts} +1 -1
- package/lib/status-function.js +23 -0
- package/lib/status.lambda.js +298 -0
- package/lib/{lambdas/token-retriever-function.d.ts → token-retriever-function.d.ts} +1 -1
- package/lib/token-retriever-function.js +23 -0
- package/lib/token-retriever.lambda.js +15 -0
- package/lib/{lambdas/webhook-handler-function.d.ts → webhook-handler-function.d.ts} +1 -1
- package/lib/webhook-handler-function.js +23 -0
- package/lib/webhook-handler.lambda.d.ts +1 -0
- package/lib/webhook-handler.lambda.js +116 -0
- package/lib/webhook.d.ts +1 -1
- package/lib/webhook.js +2 -2
- package/package.json +28 -26
- package/lib/lambdas/aws-image-builder-versioner-function.d.ts +0 -13
- package/lib/lambdas/aws-image-builder-versioner-function.js +0 -23
- package/lib/lambdas/aws-image-builder-versioner.lambda.js +0 -96
- package/lib/lambdas/build-image-function.js +0 -23
- package/lib/lambdas/build-image.lambda.js +0 -92
- package/lib/lambdas/delete-ami-function.js +0 -23
- package/lib/lambdas/delete-ami.lambda.js +0 -87
- package/lib/lambdas/delete-runner-function.js +0 -23
- package/lib/lambdas/delete-runner.lambda.js +0 -69
- package/lib/lambdas/github.js +0 -50
- package/lib/lambdas/helpers.js +0 -66
- package/lib/lambdas/setup-function.js +0 -23
- package/lib/lambdas/setup.lambda.js +0 -152
- package/lib/lambdas/status-function.js +0 -23
- package/lib/lambdas/status.lambda.js +0 -298
- package/lib/lambdas/token-retriever-function.js +0 -23
- package/lib/lambdas/token-retriever.lambda.js +0 -15
- package/lib/lambdas/update-lambda-function.js +0 -23
- package/lib/lambdas/update-lambda.lambda.js +0 -34
- package/lib/lambdas/webhook-handler-function.js +0 -23
- package/lib/lambdas/webhook-handler.lambda.js +0 -116
- package/lib/providers/image-builders/api.js +0 -47
- package/lib/providers/image-builders/aws-image-builder/ami.js +0 -81
- package/lib/providers/image-builders/aws-image-builder/builder.js +0 -488
- package/lib/providers/image-builders/aws-image-builder/common.js +0 -46
- package/lib/providers/image-builders/aws-image-builder/container.js +0 -63
- package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +0 -239
- package/lib/providers/image-builders/aws-image-builder/deprecated/common.js +0 -139
- package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +0 -222
- package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +0 -180
- package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +0 -142
- package/lib/providers/image-builders/codebuild-deprecated.js +0 -373
- package/lib/providers/image-builders/codebuild.js +0 -271
- package/lib/providers/image-builders/common.js +0 -61
- package/lib/providers/image-builders/components.js +0 -535
- package/lib/providers/image-builders/static.js +0 -58
- /package/assets/{lambdas/setup.lambda → setup.lambda}/index.html +0 -0
- /package/lib/{lambdas/delete-runner.lambda.d.ts → delete-runner.lambda.d.ts} +0 -0
- /package/lib/{lambdas/github.d.ts → github.d.ts} +0 -0
- /package/lib/{providers/image-builders → image-builders}/api.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/common.d.ts +0 -0
- /package/lib/{lambdas → image-builders/aws-image-builder}/delete-ami.lambda.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/index.d.ts +0 -0
- /package/lib/{lambdas/setup.lambda.d.ts → image-builders/aws-image-builder/filter-failed-builds.lambda.d.ts} +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/index.d.ts +0 -0
- /package/lib/{lambdas/aws-image-builder-versioner.lambda.d.ts → image-builders/aws-image-builder/versioner.lambda.d.ts} +0 -0
- /package/lib/{providers/image-builders → image-builders}/index.d.ts +0 -0
- /package/lib/{lambdas/helpers.d.ts → lambda-helpers.d.ts} +0 -0
- /package/lib/{lambdas → providers}/build-image.lambda.d.ts +0 -0
- /package/lib/{lambdas → providers}/update-lambda.lambda.d.ts +0 -0
- /package/lib/{lambdas/status.lambda.d.ts → setup.lambda.d.ts} +0 -0
- /package/lib/{lambdas/token-retriever.lambda.d.ts → status.lambda.d.ts} +0 -0
- /package/lib/{lambdas/webhook-handler.lambda.d.ts → token-retriever.lambda.d.ts} +0 -0
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.SetupFunction = void 0;
|
|
4
|
+
// ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
7
|
+
/**
|
|
8
|
+
* An AWS Lambda function which executes src/setup.
|
|
9
|
+
*/
|
|
10
|
+
class SetupFunction extends lambda.Function {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id, {
|
|
13
|
+
description: 'src/setup.lambda.ts',
|
|
14
|
+
...props,
|
|
15
|
+
runtime: new lambda.Runtime('nodejs16.x', lambda.RuntimeFamily.NODEJS),
|
|
16
|
+
handler: 'index.handler',
|
|
17
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../assets/setup.lambda')),
|
|
18
|
+
});
|
|
19
|
+
this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.SetupFunction = SetupFunction;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2V0dXAtZnVuY3Rpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvc2V0dXAtZnVuY3Rpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkVBQTZFO0FBQzdFLDZCQUE2QjtBQUM3QixpREFBaUQ7QUFTakQ7O0dBRUc7QUFDSCxNQUFhLGFBQWMsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUNoRCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQTBCO1FBQ2xFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLHFCQUFxQjtZQUNsQyxHQUFHLEtBQUs7WUFDUixPQUFPLEVBQUUsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQztZQUN0RSxPQUFPLEVBQUUsZUFBZTtZQUN4QixJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsd0JBQXdCLENBQUMsQ0FBQztTQUM1RSxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxFQUFFLEdBQUcsRUFBRSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQzFGLENBQUM7Q0FDRjtBQVhELHNDQVdDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gfn4gR2VuZXJhdGVkIGJ5IHByb2plbi4gVG8gbW9kaWZ5LCBlZGl0IC5wcm9qZW5yYy5qcyBhbmQgcnVuIFwibnB4IHByb2plblwiLlxuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIFByb3BzIGZvciBTZXR1cEZ1bmN0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2V0dXBGdW5jdGlvblByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbi8qKlxuICogQW4gQVdTIExhbWJkYSBmdW5jdGlvbiB3aGljaCBleGVjdXRlcyBzcmMvc2V0dXAuXG4gKi9cbmV4cG9ydCBjbGFzcyBTZXR1cEZ1bmN0aW9uIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBTZXR1cEZ1bmN0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnc3JjL3NldHVwLmxhbWJkYS50cycsXG4gICAgICAuLi5wcm9wcyxcbiAgICAgIHJ1bnRpbWU6IG5ldyBsYW1iZGEuUnVudGltZSgnbm9kZWpzMTYueCcsIGxhbWJkYS5SdW50aW1lRmFtaWx5Lk5PREVKUyksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy4uL2Fzc2V0cy9zZXR1cC5sYW1iZGEnKSksXG4gICAgfSk7XG4gICAgdGhpcy5hZGRFbnZpcm9ubWVudCgnQVdTX05PREVKU19DT05ORUNUSU9OX1JFVVNFX0VOQUJMRUQnLCAnMScsIHsgcmVtb3ZlSW5FZGdlOiB0cnVlIH0pO1xuICB9XG59Il19
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
/* eslint-disable import/no-extraneous-dependencies */
|
|
4
|
+
const crypto = require("crypto");
|
|
5
|
+
const fs = require("fs");
|
|
6
|
+
const rest_1 = require("@octokit/rest");
|
|
7
|
+
const github_1 = require("./github");
|
|
8
|
+
const lambda_helpers_1 = require("./lambda-helpers");
|
|
9
|
+
const nonce = crypto.randomBytes(64).toString('hex');
|
|
10
|
+
function getHtml(baseUrl, token, domain) {
|
|
11
|
+
return fs.readFileSync('index.html', 'utf-8')
|
|
12
|
+
.replace(/INSERT_WEBHOOK_URL_HERE/g, process.env.WEBHOOK_URL)
|
|
13
|
+
.replace(/INSERT_BASE_URL_HERE/g, baseUrl)
|
|
14
|
+
.replace(/INSERT_TOKEN_HERE/g, token)
|
|
15
|
+
.replace(/INSERT_SECRET_ARN_HERE/g, process.env.SETUP_SECRET_ARN)
|
|
16
|
+
.replace(/INSERT_DOMAIN_HERE/g, domain)
|
|
17
|
+
.replace(/<script/g, `<script nonce="${nonce}"`)
|
|
18
|
+
.replace(/<style/g, `<style nonce="${nonce}"`);
|
|
19
|
+
}
|
|
20
|
+
function response(code, body) {
|
|
21
|
+
return {
|
|
22
|
+
statusCode: code,
|
|
23
|
+
headers: {
|
|
24
|
+
'Content-Type': 'text/html',
|
|
25
|
+
'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'`,
|
|
26
|
+
},
|
|
27
|
+
body: body,
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
async function handleRoot(event, setupToken) {
|
|
31
|
+
const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`;
|
|
32
|
+
const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;
|
|
33
|
+
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
34
|
+
return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain));
|
|
35
|
+
}
|
|
36
|
+
function decodeBody(event) {
|
|
37
|
+
let body = event.body;
|
|
38
|
+
if (!body) {
|
|
39
|
+
throw new Error('No body found');
|
|
40
|
+
}
|
|
41
|
+
if (event.isBase64Encoded) {
|
|
42
|
+
body = Buffer.from(body, 'base64').toString('utf-8');
|
|
43
|
+
}
|
|
44
|
+
return JSON.parse(body);
|
|
45
|
+
}
|
|
46
|
+
async function handleDomain(event) {
|
|
47
|
+
const body = decodeBody(event);
|
|
48
|
+
if (!body.domain) {
|
|
49
|
+
return response(400, 'Invalid domain');
|
|
50
|
+
}
|
|
51
|
+
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
52
|
+
githubSecrets.domain = body.domain;
|
|
53
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets));
|
|
54
|
+
return response(200, 'Domain set');
|
|
55
|
+
}
|
|
56
|
+
async function handlePat(event) {
|
|
57
|
+
const body = decodeBody(event);
|
|
58
|
+
if (!body.pat || !body.domain) {
|
|
59
|
+
return response(400, 'Invalid personal access token');
|
|
60
|
+
}
|
|
61
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({
|
|
62
|
+
domain: body.domain,
|
|
63
|
+
appId: '',
|
|
64
|
+
personalAuthToken: body.pat,
|
|
65
|
+
}));
|
|
66
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));
|
|
67
|
+
return response(200, 'Personal access token set');
|
|
68
|
+
}
|
|
69
|
+
async function handleNewApp(event) {
|
|
70
|
+
if (!event.queryStringParameters) {
|
|
71
|
+
return response(400, 'Invalid code');
|
|
72
|
+
}
|
|
73
|
+
const code = event.queryStringParameters.code;
|
|
74
|
+
if (!code) {
|
|
75
|
+
return response(400, 'Invalid code');
|
|
76
|
+
}
|
|
77
|
+
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
78
|
+
const baseUrl = (0, github_1.baseUrlFromDomain)(githubSecrets.domain);
|
|
79
|
+
const newApp = await new rest_1.Octokit({ baseUrl }).rest.apps.createFromManifest({ code });
|
|
80
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({
|
|
81
|
+
domain: new URL(newApp.data.html_url).host,
|
|
82
|
+
appId: newApp.data.id,
|
|
83
|
+
personalAuthToken: '',
|
|
84
|
+
}));
|
|
85
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, newApp.data.pem);
|
|
86
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.WEBHOOK_SECRET_ARN, JSON.stringify({
|
|
87
|
+
webhookSecret: newApp.data.webhook_secret,
|
|
88
|
+
}));
|
|
89
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));
|
|
90
|
+
return response(200, `New app set. <a href="${newApp.data.html_url}/installations/new">Install it</a> for your repositories.`);
|
|
91
|
+
}
|
|
92
|
+
async function handleExistingApp(event) {
|
|
93
|
+
const body = decodeBody(event);
|
|
94
|
+
if (!body.appid || !body.pk || !body.domain) {
|
|
95
|
+
return response(400, 'Missing fields');
|
|
96
|
+
}
|
|
97
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({
|
|
98
|
+
domain: body.domain,
|
|
99
|
+
appId: body.appid,
|
|
100
|
+
personalAuthToken: '',
|
|
101
|
+
}));
|
|
102
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, body.pk);
|
|
103
|
+
await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' }));
|
|
104
|
+
return response(200, 'Existing app set. Don\'t forget to set up the webhook.');
|
|
105
|
+
}
|
|
106
|
+
exports.handler = async function (event) {
|
|
107
|
+
// confirm required environment variables
|
|
108
|
+
if (!process.env.WEBHOOK_URL) {
|
|
109
|
+
throw new Error('Missing environment variables');
|
|
110
|
+
}
|
|
111
|
+
const setupToken = (await (0, lambda_helpers_1.getSecretJsonValue)(process.env.SETUP_SECRET_ARN)).token;
|
|
112
|
+
// bail out if setup was already completed
|
|
113
|
+
if (!setupToken) {
|
|
114
|
+
return response(200, 'Setup already complete. Put a new token in the setup secret if you want to redo it.');
|
|
115
|
+
}
|
|
116
|
+
if (!event.queryStringParameters) {
|
|
117
|
+
return response(403, 'Wrong setup token.');
|
|
118
|
+
}
|
|
119
|
+
// safely confirm url token matches our secret
|
|
120
|
+
const urlToken = event.queryStringParameters.token || event.queryStringParameters.state || '';
|
|
121
|
+
if (urlToken.length != setupToken.length || !crypto.timingSafeEqual(Buffer.from(urlToken, 'utf-8'), Buffer.from(setupToken, 'utf-8'))) {
|
|
122
|
+
return response(403, 'Wrong setup token.');
|
|
123
|
+
}
|
|
124
|
+
// handle requests
|
|
125
|
+
try {
|
|
126
|
+
const path = event.path ?? event.rawPath;
|
|
127
|
+
const method = event.httpMethod ?? event.requestContext.http.method;
|
|
128
|
+
if (path == '/') {
|
|
129
|
+
return await handleRoot(event, setupToken);
|
|
130
|
+
}
|
|
131
|
+
else if (path == '/domain' && method == 'POST') {
|
|
132
|
+
return await handleDomain(event);
|
|
133
|
+
}
|
|
134
|
+
else if (path == '/pat' && method == 'POST') {
|
|
135
|
+
return await handlePat(event);
|
|
136
|
+
}
|
|
137
|
+
else if (path == '/complete-new-app' && method == 'GET') {
|
|
138
|
+
return await handleNewApp(event);
|
|
139
|
+
}
|
|
140
|
+
else if (path == '/app' && method == 'POST') {
|
|
141
|
+
return await handleExistingApp(event);
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
return response(404, 'Not found');
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
catch (e) {
|
|
148
|
+
console.error(e);
|
|
149
|
+
return response(500, `<b>Error:</b> ${e}`);
|
|
150
|
+
}
|
|
151
|
+
};
|
|
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"]}
|
|
@@ -6,7 +6,7 @@ import { Construct } from 'constructs';
|
|
|
6
6
|
export interface StatusFunctionProps extends lambda.FunctionOptions {
|
|
7
7
|
}
|
|
8
8
|
/**
|
|
9
|
-
* An AWS Lambda function which executes src/
|
|
9
|
+
* An AWS Lambda function which executes src/status.
|
|
10
10
|
*/
|
|
11
11
|
export declare class StatusFunction extends lambda.Function {
|
|
12
12
|
constructor(scope: Construct, id: string, props?: StatusFunctionProps);
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.StatusFunction = void 0;
|
|
4
|
+
// ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
7
|
+
/**
|
|
8
|
+
* An AWS Lambda function which executes src/status.
|
|
9
|
+
*/
|
|
10
|
+
class StatusFunction extends lambda.Function {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id, {
|
|
13
|
+
description: 'src/status.lambda.ts',
|
|
14
|
+
...props,
|
|
15
|
+
runtime: new lambda.Runtime('nodejs16.x', lambda.RuntimeFamily.NODEJS),
|
|
16
|
+
handler: 'index.handler',
|
|
17
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../assets/status.lambda')),
|
|
18
|
+
});
|
|
19
|
+
this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.StatusFunction = StatusFunction;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdHVzLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3N0YXR1cy1mdW5jdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2RUFBNkU7QUFDN0UsNkJBQTZCO0FBQzdCLGlEQUFpRDtBQVNqRDs7R0FFRztBQUNILE1BQWEsY0FBZSxTQUFRLE1BQU0sQ0FBQyxRQUFRO0lBQ2pELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMkI7UUFDbkUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDZixXQUFXLEVBQUUsc0JBQXNCO1lBQ25DLEdBQUcsS0FBSztZQUNSLE9BQU8sRUFBRSxJQUFJLE1BQU0sQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLE1BQU0sQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDO1lBQ3RFLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSx5QkFBeUIsQ0FBQyxDQUFDO1NBQzdFLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxjQUFjLENBQUMscUNBQXFDLEVBQUUsR0FBRyxFQUFFLEVBQUUsWUFBWSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7SUFDMUYsQ0FBQztDQUNGO0FBWEQsd0NBV0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyB+fiBHZW5lcmF0ZWQgYnkgcHJvamVuLiBUbyBtb2RpZnksIGVkaXQgLnByb2plbnJjLmpzIGFuZCBydW4gXCJucHggcHJvamVuXCIuXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbi8qKlxuICogUHJvcHMgZm9yIFN0YXR1c0Z1bmN0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhdHVzRnVuY3Rpb25Qcm9wcyBleHRlbmRzIGxhbWJkYS5GdW5jdGlvbk9wdGlvbnMge1xufVxuXG4vKipcbiAqIEFuIEFXUyBMYW1iZGEgZnVuY3Rpb24gd2hpY2ggZXhlY3V0ZXMgc3JjL3N0YXR1cy5cbiAqL1xuZXhwb3J0IGNsYXNzIFN0YXR1c0Z1bmN0aW9uIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBTdGF0dXNGdW5jdGlvblByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkLCB7XG4gICAgICBkZXNjcmlwdGlvbjogJ3NyYy9zdGF0dXMubGFtYmRhLnRzJyxcbiAgICAgIC4uLnByb3BzLFxuICAgICAgcnVudGltZTogbmV3IGxhbWJkYS5SdW50aW1lKCdub2RlanMxNi54JywgbGFtYmRhLlJ1bnRpbWVGYW1pbHkuTk9ERUpTKSxcbiAgICAgIGhhbmRsZXI6ICdpbmRleC5oYW5kbGVyJyxcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vYXNzZXRzL3N0YXR1cy5sYW1iZGEnKSksXG4gICAgfSk7XG4gICAgdGhpcy5hZGRFbnZpcm9ubWVudCgnQVdTX05PREVKU19DT05ORUNUSU9OX1JFVVNFX0VOQUJMRUQnLCAnMScsIHsgcmVtb3ZlSW5FZGdlOiB0cnVlIH0pO1xuICB9XG59Il19
|
|
@@ -0,0 +1,298 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
/* eslint-disable import/no-extraneous-dependencies */
|
|
4
|
+
const auth_app_1 = require("@octokit/auth-app");
|
|
5
|
+
const core_1 = require("@octokit/core");
|
|
6
|
+
/* eslint-disable-next-line import/no-extraneous-dependencies,import/no-unresolved */
|
|
7
|
+
const AWS = require("aws-sdk");
|
|
8
|
+
const github_1 = require("./github");
|
|
9
|
+
const lambda_helpers_1 = require("./lambda-helpers");
|
|
10
|
+
const cfn = new AWS.CloudFormation();
|
|
11
|
+
const ec2 = new AWS.EC2();
|
|
12
|
+
const ecr = new AWS.ECR();
|
|
13
|
+
const sf = new AWS.StepFunctions();
|
|
14
|
+
function secretArnToUrl(arn) {
|
|
15
|
+
const parts = arn.split(':'); // arn:aws:secretsmanager:us-east-1:12345678:secret:secret-name-REVISION
|
|
16
|
+
const region = parts[3];
|
|
17
|
+
const fullName = parts[6];
|
|
18
|
+
const name = fullName.slice(0, fullName.lastIndexOf('-'));
|
|
19
|
+
return `https://${region}.console.aws.amazon.com/secretsmanager/home?region=${region}#!/secret?name=${name}`;
|
|
20
|
+
}
|
|
21
|
+
function lambdaArnToUrl(arn) {
|
|
22
|
+
const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ
|
|
23
|
+
const region = parts[3];
|
|
24
|
+
const name = parts[6];
|
|
25
|
+
return `https://${region}.console.aws.amazon.com/lambda/home?region=${region}#/functions/${name}?tab=monitoring`;
|
|
26
|
+
}
|
|
27
|
+
function stepFunctionArnToUrl(arn) {
|
|
28
|
+
const parts = arn.split(':'); // arn:aws:states:us-east-1:12345678:stateMachine:name-XYZ
|
|
29
|
+
const region = parts[3];
|
|
30
|
+
return `https://${region}.console.aws.amazon.com/states/home?region=${region}#/statemachines/view/${arn}`;
|
|
31
|
+
}
|
|
32
|
+
async function generateProvidersStatus(stack, logicalId) {
|
|
33
|
+
const resource = await cfn.describeStackResource({ StackName: stack, LogicalResourceId: logicalId }).promise();
|
|
34
|
+
const providers = JSON.parse(resource.StackResourceDetail?.Metadata ?? '{}').providers;
|
|
35
|
+
if (!providers) {
|
|
36
|
+
return {};
|
|
37
|
+
}
|
|
38
|
+
return Promise.all(providers.map(async (p) => {
|
|
39
|
+
// add ECR data, if image is from ECR
|
|
40
|
+
if (p.image?.imageRepository?.match(/[0-9]+\.dkr\.ecr\.[a-z0-9\-]+\.amazonaws\.com\/.+/)) {
|
|
41
|
+
const tags = await ecr.describeImages({
|
|
42
|
+
repositoryName: p.image.imageRepository.split('/')[1],
|
|
43
|
+
filter: {
|
|
44
|
+
tagStatus: 'TAGGED',
|
|
45
|
+
},
|
|
46
|
+
maxResults: 1,
|
|
47
|
+
}).promise();
|
|
48
|
+
if (tags.imageDetails && tags.imageDetails?.length >= 1) {
|
|
49
|
+
p.image.latestImage = {
|
|
50
|
+
tags: tags.imageDetails[0].imageTags,
|
|
51
|
+
digest: tags.imageDetails[0].imageDigest,
|
|
52
|
+
date: tags.imageDetails[0].imagePushedAt,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// add AMI data, if image is AMI
|
|
57
|
+
if (p.ami?.launchTemplate) {
|
|
58
|
+
const versions = await ec2.describeLaunchTemplateVersions({
|
|
59
|
+
LaunchTemplateId: p.ami.launchTemplate,
|
|
60
|
+
Versions: ['$Default'],
|
|
61
|
+
}).promise();
|
|
62
|
+
if (versions.LaunchTemplateVersions && versions.LaunchTemplateVersions.length >= 1) {
|
|
63
|
+
p.ami.latestAmi = versions.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
return p;
|
|
67
|
+
}));
|
|
68
|
+
}
|
|
69
|
+
function safeReturnValue(event, status) {
|
|
70
|
+
if (event.path) {
|
|
71
|
+
return {
|
|
72
|
+
statusCode: 200,
|
|
73
|
+
headers: {
|
|
74
|
+
'Content-Type': 'application/json',
|
|
75
|
+
},
|
|
76
|
+
body: JSON.stringify(status),
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
return status;
|
|
80
|
+
}
|
|
81
|
+
exports.handler = async function (event) {
|
|
82
|
+
// confirm required environment variables
|
|
83
|
+
if (!process.env.WEBHOOK_SECRET_ARN || !process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN || !process.env.LOGICAL_ID ||
|
|
84
|
+
!process.env.WEBHOOK_HANDLER_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SETUP_SECRET_ARN || !process.env.SETUP_FUNCTION_URL ||
|
|
85
|
+
!process.env.STACK_NAME) {
|
|
86
|
+
throw new Error('Missing environment variables');
|
|
87
|
+
}
|
|
88
|
+
// base status
|
|
89
|
+
const status = {
|
|
90
|
+
github: {
|
|
91
|
+
setup: {
|
|
92
|
+
status: 'Unknown',
|
|
93
|
+
url: '',
|
|
94
|
+
secretArn: process.env.SETUP_SECRET_ARN,
|
|
95
|
+
secretUrl: secretArnToUrl(process.env.SETUP_SECRET_ARN),
|
|
96
|
+
},
|
|
97
|
+
domain: 'Unknown',
|
|
98
|
+
webhook: {
|
|
99
|
+
url: process.env.WEBHOOK_URL,
|
|
100
|
+
status: 'Unable to check',
|
|
101
|
+
secretArn: process.env.WEBHOOK_SECRET_ARN,
|
|
102
|
+
secretUrl: secretArnToUrl(process.env.WEBHOOK_SECRET_ARN),
|
|
103
|
+
},
|
|
104
|
+
auth: {
|
|
105
|
+
type: 'Unknown',
|
|
106
|
+
status: 'Unknown',
|
|
107
|
+
secretArn: process.env.GITHUB_SECRET_ARN,
|
|
108
|
+
secretUrl: secretArnToUrl(process.env.GITHUB_SECRET_ARN),
|
|
109
|
+
privateKeySecretArn: process.env.GITHUB_PRIVATE_KEY_SECRET_ARN,
|
|
110
|
+
privateKeySecretUrl: secretArnToUrl(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN),
|
|
111
|
+
app: {
|
|
112
|
+
id: '',
|
|
113
|
+
url: '',
|
|
114
|
+
installations: [],
|
|
115
|
+
},
|
|
116
|
+
personalAuthToken: '',
|
|
117
|
+
},
|
|
118
|
+
},
|
|
119
|
+
providers: await generateProvidersStatus(process.env.STACK_NAME, process.env.LOGICAL_ID),
|
|
120
|
+
troubleshooting: {
|
|
121
|
+
webhookHandlerArn: process.env.WEBHOOK_HANDLER_ARN,
|
|
122
|
+
webhookHandlerUrl: lambdaArnToUrl(process.env.WEBHOOK_HANDLER_ARN),
|
|
123
|
+
stepFunctionArn: process.env.STEP_FUNCTION_ARN,
|
|
124
|
+
stepFunctionUrl: stepFunctionArnToUrl(process.env.STEP_FUNCTION_ARN),
|
|
125
|
+
stepFunctionLogGroup: process.env.STEP_FUNCTION_LOG_GROUP,
|
|
126
|
+
recentRuns: [],
|
|
127
|
+
},
|
|
128
|
+
};
|
|
129
|
+
// setup url
|
|
130
|
+
const setupToken = (await (0, lambda_helpers_1.getSecretJsonValue)(process.env.SETUP_SECRET_ARN)).token;
|
|
131
|
+
if (setupToken) {
|
|
132
|
+
status.github.setup.status = 'Pending';
|
|
133
|
+
status.github.setup.url = `${process.env.SETUP_FUNCTION_URL}?token=${setupToken}`;
|
|
134
|
+
}
|
|
135
|
+
else {
|
|
136
|
+
status.github.setup.status = 'Complete';
|
|
137
|
+
}
|
|
138
|
+
// list last 10 executions and their status
|
|
139
|
+
try {
|
|
140
|
+
const executions = await sf.listExecutions({
|
|
141
|
+
stateMachineArn: process.env.STEP_FUNCTION_ARN,
|
|
142
|
+
maxResults: 10,
|
|
143
|
+
}).promise();
|
|
144
|
+
for (const execution of executions.executions) {
|
|
145
|
+
const executionDetails = await sf.describeExecution({
|
|
146
|
+
executionArn: execution.executionArn,
|
|
147
|
+
}).promise();
|
|
148
|
+
const input = JSON.parse(executionDetails.input || '{}');
|
|
149
|
+
status.troubleshooting.recentRuns.push({
|
|
150
|
+
executionArn: execution.executionArn,
|
|
151
|
+
status: execution.status,
|
|
152
|
+
owner: input.owner,
|
|
153
|
+
repo: input.repo,
|
|
154
|
+
runId: input.runId,
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
catch (e) {
|
|
159
|
+
status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` });
|
|
160
|
+
}
|
|
161
|
+
// get secrets
|
|
162
|
+
let githubSecrets;
|
|
163
|
+
try {
|
|
164
|
+
githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
165
|
+
}
|
|
166
|
+
catch (e) {
|
|
167
|
+
status.github.auth.status = `Unable to read secret: ${e}`;
|
|
168
|
+
return safeReturnValue(event, status);
|
|
169
|
+
}
|
|
170
|
+
let privateKey;
|
|
171
|
+
try {
|
|
172
|
+
privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
|
|
173
|
+
}
|
|
174
|
+
catch (e) {
|
|
175
|
+
status.github.auth.status = `Unable to read private key secret: ${e}`;
|
|
176
|
+
return safeReturnValue(event, status);
|
|
177
|
+
}
|
|
178
|
+
// calculate base url
|
|
179
|
+
let baseUrl = (0, github_1.baseUrlFromDomain)(githubSecrets.domain);
|
|
180
|
+
status.github.domain = githubSecrets.domain;
|
|
181
|
+
if (githubSecrets.personalAuthToken) {
|
|
182
|
+
// try authenticating with personal authentication token
|
|
183
|
+
status.github.auth.type = 'Personal Auth Token';
|
|
184
|
+
status.github.auth.personalAuthToken = '*redacted*';
|
|
185
|
+
let octokit;
|
|
186
|
+
try {
|
|
187
|
+
octokit = new core_1.Octokit({ baseUrl, auth: githubSecrets.personalAuthToken });
|
|
188
|
+
}
|
|
189
|
+
catch (e) {
|
|
190
|
+
status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`;
|
|
191
|
+
return safeReturnValue(event, status);
|
|
192
|
+
}
|
|
193
|
+
try {
|
|
194
|
+
const user = await octokit.request('GET /user');
|
|
195
|
+
status.github.auth.personalAuthToken = `username: ${user.data.login}`;
|
|
196
|
+
}
|
|
197
|
+
catch (e) {
|
|
198
|
+
status.github.auth.status = `Unable to call /user with personal auth token: ${e}`;
|
|
199
|
+
return safeReturnValue(event, status);
|
|
200
|
+
}
|
|
201
|
+
status.github.auth.status = 'OK';
|
|
202
|
+
status.github.webhook.status = 'Unable to verify automatically';
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
// try authenticating with GitHub app
|
|
206
|
+
status.github.auth.type = 'GitHub App';
|
|
207
|
+
status.github.auth.app.id = githubSecrets.appId;
|
|
208
|
+
let appOctokit;
|
|
209
|
+
try {
|
|
210
|
+
appOctokit = new core_1.Octokit({
|
|
211
|
+
baseUrl,
|
|
212
|
+
authStrategy: auth_app_1.createAppAuth,
|
|
213
|
+
auth: {
|
|
214
|
+
appId: githubSecrets.appId,
|
|
215
|
+
privateKey: privateKey,
|
|
216
|
+
},
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
catch (e) {
|
|
220
|
+
status.github.auth.status = `Unable to authenticate app: ${e}`;
|
|
221
|
+
return safeReturnValue(event, status);
|
|
222
|
+
}
|
|
223
|
+
// get app url
|
|
224
|
+
try {
|
|
225
|
+
const app = (await appOctokit.request('GET /app')).data;
|
|
226
|
+
status.github.auth.app.url = app.html_url;
|
|
227
|
+
}
|
|
228
|
+
catch (e) {
|
|
229
|
+
status.github.auth.status = `Unable to get app details: ${e}`;
|
|
230
|
+
return safeReturnValue(event, status);
|
|
231
|
+
}
|
|
232
|
+
// list all app installations
|
|
233
|
+
try {
|
|
234
|
+
const installations = (await appOctokit.request('GET /app/installations')).data;
|
|
235
|
+
for (const installation of installations) {
|
|
236
|
+
let installationDetails = {
|
|
237
|
+
id: installation.id,
|
|
238
|
+
url: installation.html_url,
|
|
239
|
+
status: 'Unable to query',
|
|
240
|
+
repositories: [],
|
|
241
|
+
};
|
|
242
|
+
let token;
|
|
243
|
+
try {
|
|
244
|
+
token = (await appOctokit.auth({
|
|
245
|
+
type: 'installation',
|
|
246
|
+
installationId: installation.id,
|
|
247
|
+
})).token;
|
|
248
|
+
}
|
|
249
|
+
catch (e) {
|
|
250
|
+
installationDetails.status = `Unable to authenticate app installation: ${e}`;
|
|
251
|
+
continue;
|
|
252
|
+
}
|
|
253
|
+
let octokit;
|
|
254
|
+
try {
|
|
255
|
+
octokit = new core_1.Octokit({ baseUrl, auth: token });
|
|
256
|
+
}
|
|
257
|
+
catch (e) {
|
|
258
|
+
installationDetails.status = `Unable to authenticate using app: ${e}`;
|
|
259
|
+
continue;
|
|
260
|
+
}
|
|
261
|
+
try {
|
|
262
|
+
const repositories = (await octokit.request('GET /installation/repositories')).data.repositories;
|
|
263
|
+
for (const repo of repositories) {
|
|
264
|
+
installationDetails.repositories.push(repo.full_name);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
catch (e) {
|
|
268
|
+
installationDetails.status = `Unable to authenticate using installation token: ${e}`;
|
|
269
|
+
continue;
|
|
270
|
+
}
|
|
271
|
+
installationDetails.status = 'OK';
|
|
272
|
+
status.github.auth.app.installations.push(installationDetails);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
catch (e) {
|
|
276
|
+
status.github.auth.status = 'Unable to list app installations';
|
|
277
|
+
return safeReturnValue(event, status);
|
|
278
|
+
}
|
|
279
|
+
status.github.auth.status = 'OK';
|
|
280
|
+
// check webhook config
|
|
281
|
+
try {
|
|
282
|
+
const response = await appOctokit.request('GET /app/hook/config', {});
|
|
283
|
+
if (response.data.url !== process.env.WEBHOOK_URL) {
|
|
284
|
+
status.github.webhook.status = 'GitHub has wrong webhook URL configured';
|
|
285
|
+
}
|
|
286
|
+
else {
|
|
287
|
+
// TODO check secret by doing a dummy delivery? force apply secret?
|
|
288
|
+
status.github.webhook.status = 'OK (note that secret cannot be checked automatically)';
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
catch (e) {
|
|
292
|
+
status.github.webhook.status = `Unable to check app configuration: ${e}`;
|
|
293
|
+
return safeReturnValue(event, status);
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
return safeReturnValue(event, status);
|
|
297
|
+
};
|
|
298
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.lambda.js","sourceRoot":"","sources":["../src/status.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/no-extraneous-dependencies */\nimport { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/core';\nimport * as AWSLambda from 'aws-lambda';\n/* eslint-disable-next-line import/no-extraneous-dependencies,import/no-unresolved */\nimport * as AWS from 'aws-sdk';\nimport { baseUrlFromDomain } from './github';\nimport { getSecretJsonValue, getSecretValue } from './lambda-helpers';\n\nconst cfn = new AWS.CloudFormation();\nconst ec2 = new AWS.EC2();\nconst ecr = new AWS.ECR();\nconst sf = new AWS.StepFunctions();\n\nfunction secretArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:secretsmanager:us-east-1:12345678:secret:secret-name-REVISION\n  const region = parts[3];\n  const fullName = parts[6];\n  const name = fullName.slice(0, fullName.lastIndexOf('-'));\n\n  return `https://${region}.console.aws.amazon.com/secretsmanager/home?region=${region}#!/secret?name=${name}`;\n}\n\nfunction lambdaArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ\n  const region = parts[3];\n  const name = parts[6];\n\n  return `https://${region}.console.aws.amazon.com/lambda/home?region=${region}#/functions/${name}?tab=monitoring`;\n}\n\nfunction stepFunctionArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:states:us-east-1:12345678:stateMachine:name-XYZ\n  const region = parts[3];\n\n  return `https://${region}.console.aws.amazon.com/states/home?region=${region}#/statemachines/view/${arn}`;\n}\n\nasync function generateProvidersStatus(stack: string, logicalId: string) {\n  const resource = await cfn.describeStackResource({ StackName: stack, LogicalResourceId: logicalId }).promise();\n  const providers = JSON.parse(resource.StackResourceDetail?.Metadata ?? '{}').providers as any[] | undefined;\n\n  if (!providers) {\n    return {};\n  }\n\n  return Promise.all(providers.map(async (p) => {\n    // add ECR data, if image is from ECR\n    if (p.image?.imageRepository?.match(/[0-9]+\\.dkr\\.ecr\\.[a-z0-9\\-]+\\.amazonaws\\.com\\/.+/)) {\n      const tags = await ecr.describeImages({\n        repositoryName: p.image.imageRepository.split('/')[1],\n        filter: {\n          tagStatus: 'TAGGED',\n        },\n        maxResults: 1,\n      }).promise();\n      if (tags.imageDetails && tags.imageDetails?.length >= 1) {\n        p.image.latestImage = {\n          tags: tags.imageDetails[0].imageTags,\n          digest: tags.imageDetails[0].imageDigest,\n          date: tags.imageDetails[0].imagePushedAt,\n        };\n      }\n    }\n    // add AMI data, if image is AMI\n    if (p.ami?.launchTemplate) {\n      const versions = await ec2.describeLaunchTemplateVersions({\n        LaunchTemplateId: p.ami.launchTemplate,\n        Versions: ['$Default'],\n      }).promise();\n      if (versions.LaunchTemplateVersions && versions.LaunchTemplateVersions.length >= 1) {\n        p.ami.latestAmi = versions.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId;\n      }\n    }\n    return p;\n  }));\n}\n\ninterface AppInstallation {\n  readonly id: number;\n  readonly url: string;\n  readonly status: string;\n  readonly repositories: string[];\n}\n\ninterface RecentRun {\n  readonly owner?: string;\n  readonly repo?: string;\n  readonly runId?: string;\n  readonly executionArn?: string;\n  readonly status: string;\n}\n\nfunction safeReturnValue(event: Partial<AWSLambda.APIGatewayProxyEvent>, status: any) {\n  if (event.path) {\n    return {\n      statusCode: 200,\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify(status),\n    };\n  }\n\n  return status;\n}\n\nexports.handler = async function (event: Partial<AWSLambda.APIGatewayProxyEvent>) {\n  // confirm required environment variables\n  if (!process.env.WEBHOOK_SECRET_ARN || !process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN || !process.env.LOGICAL_ID ||\n      !process.env.WEBHOOK_HANDLER_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SETUP_SECRET_ARN || !process.env.SETUP_FUNCTION_URL ||\n      !process.env.STACK_NAME) {\n    throw new Error('Missing environment variables');\n  }\n\n  // base status\n  const status = {\n    github: {\n      setup: {\n        status: 'Unknown',\n        url: '',\n        secretArn: process.env.SETUP_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.SETUP_SECRET_ARN),\n      },\n      domain: 'Unknown',\n      webhook: {\n        url: process.env.WEBHOOK_URL,\n        status: 'Unable to check',\n        secretArn: process.env.WEBHOOK_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.WEBHOOK_SECRET_ARN),\n      },\n      auth: {\n        type: 'Unknown',\n        status: 'Unknown',\n        secretArn: process.env.GITHUB_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.GITHUB_SECRET_ARN),\n        privateKeySecretArn: process.env.GITHUB_PRIVATE_KEY_SECRET_ARN,\n        privateKeySecretUrl: secretArnToUrl(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN),\n        app: {\n          id: '',\n          url: '',\n          installations: [] as AppInstallation[],\n        },\n        personalAuthToken: '',\n      },\n    },\n    providers: await generateProvidersStatus(process.env.STACK_NAME, process.env.LOGICAL_ID),\n    troubleshooting: {\n      webhookHandlerArn: process.env.WEBHOOK_HANDLER_ARN,\n      webhookHandlerUrl: lambdaArnToUrl(process.env.WEBHOOK_HANDLER_ARN),\n      stepFunctionArn: process.env.STEP_FUNCTION_ARN,\n      stepFunctionUrl: stepFunctionArnToUrl(process.env.STEP_FUNCTION_ARN),\n      stepFunctionLogGroup: process.env.STEP_FUNCTION_LOG_GROUP,\n      recentRuns: [] as RecentRun[],\n    },\n  };\n\n  // setup url\n  const setupToken = (await getSecretJsonValue(process.env.SETUP_SECRET_ARN)).token;\n  if (setupToken) {\n    status.github.setup.status = 'Pending';\n    status.github.setup.url = `${process.env.SETUP_FUNCTION_URL}?token=${setupToken}`;\n  } else {\n    status.github.setup.status = 'Complete';\n  }\n\n  // list last 10 executions and their status\n  try {\n    const executions = await sf.listExecutions({\n      stateMachineArn: process.env.STEP_FUNCTION_ARN,\n      maxResults: 10,\n    }).promise();\n    for (const execution of executions.executions) {\n      const executionDetails = await sf.describeExecution({\n        executionArn: execution.executionArn,\n      }).promise();\n      const input = JSON.parse(executionDetails.input || '{}');\n\n      status.troubleshooting.recentRuns.push({\n        executionArn: execution.executionArn,\n        status: execution.status,\n        owner: input.owner,\n        repo: input.repo,\n        runId: input.runId,\n      });\n    }\n  } catch (e) {\n    status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` });\n  }\n\n  // get secrets\n  let githubSecrets;\n  try {\n    githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  } catch (e) {\n    status.github.auth.status = `Unable to read secret: ${e}`;\n    return safeReturnValue(event, status);\n  }\n\n  let privateKey;\n  try {\n    privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n  } catch (e) {\n    status.github.auth.status = `Unable to read private key secret: ${e}`;\n    return safeReturnValue(event, status);\n  }\n\n  // calculate base url\n  let baseUrl = baseUrlFromDomain(githubSecrets.domain);\n  status.github.domain = githubSecrets.domain;\n\n  if (githubSecrets.personalAuthToken) {\n    // try authenticating with personal authentication token\n    status.github.auth.type = 'Personal Auth Token';\n    status.github.auth.personalAuthToken = '*redacted*';\n\n    let octokit;\n    try {\n      octokit = new Octokit({ baseUrl, auth: githubSecrets.personalAuthToken });\n    } catch (e) {\n      status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`;\n      return safeReturnValue(event, status);\n    }\n\n    try {\n      const user = await octokit.request('GET /user');\n      status.github.auth.personalAuthToken = `username: ${user.data.login}`;\n    } catch (e) {\n      status.github.auth.status = `Unable to call /user with personal auth token: ${e}`;\n      return safeReturnValue(event, status);\n    }\n\n    status.github.auth.status = 'OK';\n    status.github.webhook.status = 'Unable to verify automatically';\n  } else {\n    // try authenticating with GitHub app\n    status.github.auth.type = 'GitHub App';\n    status.github.auth.app.id = githubSecrets.appId;\n\n    let appOctokit;\n    try {\n      appOctokit = new Octokit({\n        baseUrl,\n        authStrategy: createAppAuth,\n        auth: {\n          appId: githubSecrets.appId,\n          privateKey: privateKey,\n        },\n      });\n    } catch (e) {\n      status.github.auth.status = `Unable to authenticate app: ${e}`;\n      return safeReturnValue(event, status);\n    }\n\n    // get app url\n    try {\n      const app = (await appOctokit.request('GET /app')).data;\n      status.github.auth.app.url = app.html_url;\n    } catch (e) {\n      status.github.auth.status = `Unable to get app details: ${e}`;\n      return safeReturnValue(event, status);\n    }\n\n    // list all app installations\n    try {\n      const installations = (await appOctokit.request('GET /app/installations')).data;\n      for (const installation of installations) {\n        let installationDetails = {\n          id: installation.id,\n          url: installation.html_url,\n          status: 'Unable to query',\n          repositories: [] as string[],\n        };\n\n        let token;\n        try {\n          token = (await appOctokit.auth({\n            type: 'installation',\n            installationId: installation.id,\n          }) as any).token;\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate app installation: ${e}`;\n          continue;\n        }\n\n        let octokit;\n        try {\n          octokit = new Octokit({ baseUrl, auth: token });\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate using app: ${e}`;\n          continue;\n        }\n\n        try {\n          const repositories = (await octokit.request('GET /installation/repositories')).data.repositories;\n          for (const repo of repositories) {\n            installationDetails.repositories.push(repo.full_name as string);\n          }\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate using installation token: ${e}`;\n          continue;\n        }\n\n        installationDetails.status = 'OK';\n        status.github.auth.app.installations.push(installationDetails);\n      }\n    } catch (e) {\n      status.github.auth.status = 'Unable to list app installations';\n      return safeReturnValue(event, status);\n    }\n\n    status.github.auth.status = 'OK';\n\n    // check webhook config\n    try {\n      const response = await appOctokit.request('GET /app/hook/config', {});\n\n      if (response.data.url !== process.env.WEBHOOK_URL) {\n        status.github.webhook.status = 'GitHub has wrong webhook URL configured';\n      } else {\n        // TODO check secret by doing a dummy delivery? force apply secret?\n        status.github.webhook.status = 'OK (note that secret cannot be checked automatically)';\n      }\n    } catch (e) {\n      status.github.webhook.status = `Unable to check app configuration: ${e}`;\n      return safeReturnValue(event, status);\n    }\n  }\n\n  return safeReturnValue(event, status);\n};\n"]}
|
|
@@ -6,7 +6,7 @@ import { Construct } from 'constructs';
|
|
|
6
6
|
export interface TokenRetrieverFunctionProps extends lambda.FunctionOptions {
|
|
7
7
|
}
|
|
8
8
|
/**
|
|
9
|
-
* An AWS Lambda function which executes src/
|
|
9
|
+
* An AWS Lambda function which executes src/token-retriever.
|
|
10
10
|
*/
|
|
11
11
|
export declare class TokenRetrieverFunction extends lambda.Function {
|
|
12
12
|
constructor(scope: Construct, id: string, props?: TokenRetrieverFunctionProps);
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TokenRetrieverFunction = void 0;
|
|
4
|
+
// ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
7
|
+
/**
|
|
8
|
+
* An AWS Lambda function which executes src/token-retriever.
|
|
9
|
+
*/
|
|
10
|
+
class TokenRetrieverFunction extends lambda.Function {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id, {
|
|
13
|
+
description: 'src/token-retriever.lambda.ts',
|
|
14
|
+
...props,
|
|
15
|
+
runtime: new lambda.Runtime('nodejs16.x', lambda.RuntimeFamily.NODEJS),
|
|
16
|
+
handler: 'index.handler',
|
|
17
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../assets/token-retriever.lambda')),
|
|
18
|
+
});
|
|
19
|
+
this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.TokenRetrieverFunction = TokenRetrieverFunction;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9rZW4tcmV0cmlldmVyLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3Rva2VuLXJldHJpZXZlci1mdW5jdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2RUFBNkU7QUFDN0UsNkJBQTZCO0FBQzdCLGlEQUFpRDtBQVNqRDs7R0FFRztBQUNILE1BQWEsc0JBQXVCLFNBQVEsTUFBTSxDQUFDLFFBQVE7SUFDekQsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUFtQztRQUMzRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRTtZQUNmLFdBQVcsRUFBRSwrQkFBK0I7WUFDNUMsR0FBRyxLQUFLO1lBQ1IsT0FBTyxFQUFFLElBQUksTUFBTSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsTUFBTSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUM7WUFDdEUsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLGtDQUFrQyxDQUFDLENBQUM7U0FDdEYsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQ0FBcUMsRUFBRSxHQUFHLEVBQUUsRUFBRSxZQUFZLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUMxRixDQUFDO0NBQ0Y7QUFYRCx3REFXQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIH5+IEdlbmVyYXRlZCBieSBwcm9qZW4uIFRvIG1vZGlmeSwgZWRpdCAucHJvamVucmMuanMgYW5kIHJ1biBcIm5weCBwcm9qZW5cIi5cbmltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuLyoqXG4gKiBQcm9wcyBmb3IgVG9rZW5SZXRyaWV2ZXJGdW5jdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRva2VuUmV0cmlldmVyRnVuY3Rpb25Qcm9wcyBleHRlbmRzIGxhbWJkYS5GdW5jdGlvbk9wdGlvbnMge1xufVxuXG4vKipcbiAqIEFuIEFXUyBMYW1iZGEgZnVuY3Rpb24gd2hpY2ggZXhlY3V0ZXMgc3JjL3Rva2VuLXJldHJpZXZlci5cbiAqL1xuZXhwb3J0IGNsYXNzIFRva2VuUmV0cmlldmVyRnVuY3Rpb24gZXh0ZW5kcyBsYW1iZGEuRnVuY3Rpb24ge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wcz86IFRva2VuUmV0cmlldmVyRnVuY3Rpb25Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwge1xuICAgICAgZGVzY3JpcHRpb246ICdzcmMvdG9rZW4tcmV0cmlldmVyLmxhbWJkYS50cycsXG4gICAgICAuLi5wcm9wcyxcbiAgICAgIHJ1bnRpbWU6IG5ldyBsYW1iZGEuUnVudGltZSgnbm9kZWpzMTYueCcsIGxhbWJkYS5SdW50aW1lRmFtaWx5Lk5PREVKUyksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy4uL2Fzc2V0cy90b2tlbi1yZXRyaWV2ZXIubGFtYmRhJykpLFxuICAgIH0pO1xuICAgIHRoaXMuYWRkRW52aXJvbm1lbnQoJ0FXU19OT0RFSlNfQ09OTkVDVElPTl9SRVVTRV9FTkFCTEVEJywgJzEnLCB7IHJlbW92ZUluRWRnZTogdHJ1ZSB9KTtcbiAgfVxufSJdfQ==
|