@cloudsnorkel/cdk-github-runners 0.9.1 → 0.9.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +999 -98
- package/API.md +776 -8
- package/assets/lambdas/setup.lambda/index.js +10 -6
- package/assets/lambdas/status.lambda/index.js +22 -10
- package/assets/lambdas/webhook-handler.lambda/index.js +15 -7
- package/lib/access.d.ts +65 -0
- package/lib/access.js +160 -0
- package/lib/index.d.ts +1 -0
- package/lib/index.js +2 -1
- package/lib/lambdas/setup.lambda.js +11 -7
- package/lib/lambdas/status.lambda.js +24 -11
- package/lib/lambdas/webhook-handler.lambda.js +19 -10
- package/lib/providers/codebuild.js +2 -2
- package/lib/providers/common.js +3 -3
- package/lib/providers/ec2.js +2 -2
- package/lib/providers/ecs.js +5 -1
- package/lib/providers/fargate.js +2 -2
- package/lib/providers/image-builders/api.js +1 -1
- package/lib/providers/image-builders/aws-image-builder/builder.js +1 -1
- package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +1 -1
- package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +1 -1
- package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
- package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
- package/lib/providers/image-builders/codebuild-deprecated.js +1 -1
- package/lib/providers/image-builders/components.js +1 -1
- package/lib/providers/image-builders/static.js +1 -1
- package/lib/providers/index.d.ts +1 -0
- package/lib/providers/index.js +2 -1
- package/lib/providers/lambda.js +2 -2
- package/lib/runner.d.ts +23 -0
- package/lib/runner.js +13 -4
- package/lib/secrets.js +1 -1
- package/lib/webhook.d.ts +5 -0
- package/lib/webhook.js +5 -4
- package/package.json +1 -1
|
@@ -9164,7 +9164,8 @@ function response(code, body) {
|
|
|
9164
9164
|
};
|
|
9165
9165
|
}
|
|
9166
9166
|
async function handleRoot(event, setupToken) {
|
|
9167
|
-
const
|
|
9167
|
+
const stage = event.requestContext.stage == "$default" ? "" : `/${event.requestContext.stage}`;
|
|
9168
|
+
const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;
|
|
9168
9169
|
const githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);
|
|
9169
9170
|
return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain));
|
|
9170
9171
|
}
|
|
@@ -9254,20 +9255,23 @@ exports.handler = async function(event) {
|
|
|
9254
9255
|
return response(403, "Wrong setup token.");
|
|
9255
9256
|
}
|
|
9256
9257
|
try {
|
|
9257
|
-
|
|
9258
|
+
const path = event.path ?? event.rawPath;
|
|
9259
|
+
const method = event.httpMethod ?? event.requestContext.http.method;
|
|
9260
|
+
if (path == "/") {
|
|
9258
9261
|
return await handleRoot(event, setupToken);
|
|
9259
|
-
} else if (
|
|
9262
|
+
} else if (path == "/domain" && method == "POST") {
|
|
9260
9263
|
return await handleDomain(event);
|
|
9261
|
-
} else if (
|
|
9264
|
+
} else if (path == "/pat" && method == "POST") {
|
|
9262
9265
|
return await handlePat(event);
|
|
9263
|
-
} else if (
|
|
9266
|
+
} else if (path == "/complete-new-app" && method == "GET") {
|
|
9264
9267
|
return await handleNewApp(event);
|
|
9265
|
-
} else if (
|
|
9268
|
+
} else if (path == "/app" && method == "POST") {
|
|
9266
9269
|
return await handleExistingApp(event);
|
|
9267
9270
|
} else {
|
|
9268
9271
|
return response(404, "Not found");
|
|
9269
9272
|
}
|
|
9270
9273
|
} catch (e) {
|
|
9274
|
+
console.error(e);
|
|
9271
9275
|
return response(500, `<b>Error:</b> ${e}`);
|
|
9272
9276
|
}
|
|
9273
9277
|
};
|
|
@@ -21184,7 +21184,19 @@ async function generateProvidersStatus(stack, logicalId) {
|
|
|
21184
21184
|
return p;
|
|
21185
21185
|
}));
|
|
21186
21186
|
}
|
|
21187
|
-
|
|
21187
|
+
function safeReturnValue(event, status) {
|
|
21188
|
+
if (event.path) {
|
|
21189
|
+
return {
|
|
21190
|
+
statusCode: 200,
|
|
21191
|
+
headers: {
|
|
21192
|
+
"Content-Type": "application/json"
|
|
21193
|
+
},
|
|
21194
|
+
body: JSON.stringify(status)
|
|
21195
|
+
};
|
|
21196
|
+
}
|
|
21197
|
+
return status;
|
|
21198
|
+
}
|
|
21199
|
+
exports.handler = async function(event) {
|
|
21188
21200
|
if (!process.env.WEBHOOK_SECRET_ARN || !process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN || !process.env.LOGICAL_ID || !process.env.WEBHOOK_HANDLER_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SETUP_SECRET_ARN || !process.env.SETUP_FUNCTION_URL || !process.env.STACK_NAME) {
|
|
21189
21201
|
throw new Error("Missing environment variables");
|
|
21190
21202
|
}
|
|
@@ -21261,14 +21273,14 @@ exports.handler = async function() {
|
|
|
21261
21273
|
githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);
|
|
21262
21274
|
} catch (e) {
|
|
21263
21275
|
status.github.auth.status = `Unable to read secret: ${e}`;
|
|
21264
|
-
return status;
|
|
21276
|
+
return safeReturnValue(event, status);
|
|
21265
21277
|
}
|
|
21266
21278
|
let privateKey;
|
|
21267
21279
|
try {
|
|
21268
21280
|
privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
|
|
21269
21281
|
} catch (e) {
|
|
21270
21282
|
status.github.auth.status = `Unable to read private key secret: ${e}`;
|
|
21271
|
-
return status;
|
|
21283
|
+
return safeReturnValue(event, status);
|
|
21272
21284
|
}
|
|
21273
21285
|
let baseUrl = baseUrlFromDomain(githubSecrets.domain);
|
|
21274
21286
|
status.github.domain = githubSecrets.domain;
|
|
@@ -21280,14 +21292,14 @@ exports.handler = async function() {
|
|
|
21280
21292
|
octokit = new import_core.Octokit({ baseUrl, auth: githubSecrets.personalAuthToken });
|
|
21281
21293
|
} catch (e) {
|
|
21282
21294
|
status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`;
|
|
21283
|
-
return status;
|
|
21295
|
+
return safeReturnValue(event, status);
|
|
21284
21296
|
}
|
|
21285
21297
|
try {
|
|
21286
21298
|
const user = await octokit.request("GET /user");
|
|
21287
21299
|
status.github.auth.personalAuthToken = `username: ${user.data.login}`;
|
|
21288
21300
|
} catch (e) {
|
|
21289
21301
|
status.github.auth.status = `Unable to call /user with personal auth token: ${e}`;
|
|
21290
|
-
return status;
|
|
21302
|
+
return safeReturnValue(event, status);
|
|
21291
21303
|
}
|
|
21292
21304
|
status.github.auth.status = "OK";
|
|
21293
21305
|
status.github.webhook.status = "Unable to verify automatically";
|
|
@@ -21306,14 +21318,14 @@ exports.handler = async function() {
|
|
|
21306
21318
|
});
|
|
21307
21319
|
} catch (e) {
|
|
21308
21320
|
status.github.auth.status = `Unable to authenticate app: ${e}`;
|
|
21309
|
-
return status;
|
|
21321
|
+
return safeReturnValue(event, status);
|
|
21310
21322
|
}
|
|
21311
21323
|
try {
|
|
21312
21324
|
const app = (await appOctokit.request("GET /app")).data;
|
|
21313
21325
|
status.github.auth.app.url = app.html_url;
|
|
21314
21326
|
} catch (e) {
|
|
21315
21327
|
status.github.auth.status = `Unable to get app details: ${e}`;
|
|
21316
|
-
return status;
|
|
21328
|
+
return safeReturnValue(event, status);
|
|
21317
21329
|
}
|
|
21318
21330
|
try {
|
|
21319
21331
|
const installations = (await appOctokit.request("GET /app/installations")).data;
|
|
@@ -21355,7 +21367,7 @@ exports.handler = async function() {
|
|
|
21355
21367
|
}
|
|
21356
21368
|
} catch (e) {
|
|
21357
21369
|
status.github.auth.status = "Unable to list app installations";
|
|
21358
|
-
return status;
|
|
21370
|
+
return safeReturnValue(event, status);
|
|
21359
21371
|
}
|
|
21360
21372
|
status.github.auth.status = "OK";
|
|
21361
21373
|
try {
|
|
@@ -21367,10 +21379,10 @@ exports.handler = async function() {
|
|
|
21367
21379
|
}
|
|
21368
21380
|
} catch (e) {
|
|
21369
21381
|
status.github.webhook.status = `Unable to check app configuration: ${e}`;
|
|
21370
|
-
return status;
|
|
21382
|
+
return safeReturnValue(event, status);
|
|
21371
21383
|
}
|
|
21372
21384
|
}
|
|
21373
|
-
return status;
|
|
21385
|
+
return safeReturnValue(event, status);
|
|
21374
21386
|
};
|
|
21375
21387
|
/*! Bundled license information:
|
|
21376
21388
|
|
|
@@ -45,8 +45,16 @@ async function getSecretJsonValue(arn) {
|
|
|
45
45
|
|
|
46
46
|
// src/lambdas/webhook-handler.lambda.ts
|
|
47
47
|
var sf = new AWS2.StepFunctions();
|
|
48
|
+
function getHeader(event, header) {
|
|
49
|
+
for (const headerName of Object.keys(event.headers)) {
|
|
50
|
+
if (headerName.toLowerCase() === header.toLowerCase()) {
|
|
51
|
+
return event.headers[headerName];
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
return void 0;
|
|
55
|
+
}
|
|
48
56
|
function verifyBody(event, secret) {
|
|
49
|
-
const sig = Buffer.from(event
|
|
57
|
+
const sig = Buffer.from(getHeader(event, "x-hub-signature-256") || "", "utf8");
|
|
50
58
|
if (!event.body) {
|
|
51
59
|
throw new Error("No body");
|
|
52
60
|
}
|
|
@@ -81,21 +89,21 @@ exports.handler = async function(event) {
|
|
|
81
89
|
body: "Bad signature"
|
|
82
90
|
};
|
|
83
91
|
}
|
|
84
|
-
if (event
|
|
85
|
-
console.error(`This webhook only accepts JSON payloads, got ${event
|
|
92
|
+
if (getHeader(event, "content-type") !== "application/json") {
|
|
93
|
+
console.error(`This webhook only accepts JSON payloads, got ${getHeader(event, "content-type")}`);
|
|
86
94
|
return {
|
|
87
95
|
statusCode: 400,
|
|
88
96
|
body: "Expecting JSON payload"
|
|
89
97
|
};
|
|
90
98
|
}
|
|
91
|
-
if (event
|
|
99
|
+
if (getHeader(event, "x-github-event") === "ping") {
|
|
92
100
|
return {
|
|
93
101
|
statusCode: 200,
|
|
94
102
|
body: "Pong"
|
|
95
103
|
};
|
|
96
104
|
}
|
|
97
|
-
if (event
|
|
98
|
-
console.error(`This webhook only accepts workflow_job, got ${event
|
|
105
|
+
if (getHeader(event, "x-github-event") !== "workflow_job") {
|
|
106
|
+
console.error(`This webhook only accepts workflow_job, got ${getHeader(event, "x-github-event")}`);
|
|
99
107
|
return {
|
|
100
108
|
statusCode: 400,
|
|
101
109
|
body: "Expecting workflow_job"
|
|
@@ -118,7 +126,7 @@ exports.handler = async function(event) {
|
|
|
118
126
|
}
|
|
119
127
|
let labels = {};
|
|
120
128
|
payload.workflow_job.labels.forEach((l) => labels[l.toLowerCase()] = true);
|
|
121
|
-
let executionName = `${payload.repository.full_name.replace("/", "-")}-${event
|
|
129
|
+
let executionName = `${payload.repository.full_name.replace("/", "-")}-${getHeader(event, "x-github-delivery")}`.slice(0, 64);
|
|
122
130
|
const execution = await sf.startExecution({
|
|
123
131
|
stateMachineArn: process.env.STEP_FUNCTION_ARN,
|
|
124
132
|
input: JSON.stringify({
|
package/lib/access.d.ts
ADDED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { aws_ec2 as ec2, aws_lambda as lambda } from 'aws-cdk-lib';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
export interface ApiGatewayAccessProps {
|
|
4
|
+
/**
|
|
5
|
+
* List of IP addresses in CIDR notation that are allowed to access the API Gateway.
|
|
6
|
+
*
|
|
7
|
+
* If not specified on public API Gateway, all IP addresses are allowed.
|
|
8
|
+
*
|
|
9
|
+
* If not specified on private API Gateway, no IP addresses are allowed (but specified security groups are).
|
|
10
|
+
*/
|
|
11
|
+
readonly allowedIps?: string[];
|
|
12
|
+
/**
|
|
13
|
+
* Creates a private API Gateway and allows access from the specified VPC.
|
|
14
|
+
*/
|
|
15
|
+
readonly allowedVpc?: ec2.IVpc;
|
|
16
|
+
/**
|
|
17
|
+
* List of security groups that are allowed to access the API Gateway.
|
|
18
|
+
*
|
|
19
|
+
* Only works for private API Gateways with {@link allowedVpc}.
|
|
20
|
+
*/
|
|
21
|
+
readonly allowedSecurityGroups?: ec2.ISecurityGroup[];
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Access configuration options for Lambda functions like setup and webhook function. Use this to limit access to these functions.
|
|
25
|
+
*/
|
|
26
|
+
export declare abstract class LambdaAccess {
|
|
27
|
+
/**
|
|
28
|
+
* Disables access to the configured Lambda function. This is useful for the setup function after setup is done.
|
|
29
|
+
*/
|
|
30
|
+
static noAccess(): LambdaAccess;
|
|
31
|
+
/**
|
|
32
|
+
* Provide access using Lambda URL. This is the default and simplest option. It puts no limits on the requester, but the Lambda functions themselves authenticate every request.
|
|
33
|
+
*/
|
|
34
|
+
static lambdaUrl(): LambdaAccess;
|
|
35
|
+
/**
|
|
36
|
+
* Provide access using API Gateway. This is the most secure option, but requires additional configuration. It allows you to limit access to specific IP addresses and even to a specific VPC.
|
|
37
|
+
*
|
|
38
|
+
* To limit access to GitHub.com use:
|
|
39
|
+
*
|
|
40
|
+
* ```
|
|
41
|
+
* LambdaAccess.apiGateway({
|
|
42
|
+
* allowedIps: LambdaAccess.githubWebhookIps(),
|
|
43
|
+
* });
|
|
44
|
+
* ```
|
|
45
|
+
*
|
|
46
|
+
* Alternatively, get and manually update the list manually with:
|
|
47
|
+
*
|
|
48
|
+
* ```
|
|
49
|
+
* curl https://api.github.com/meta | jq .hooks
|
|
50
|
+
* ```
|
|
51
|
+
*/
|
|
52
|
+
static apiGateway(props?: ApiGatewayAccessProps): LambdaAccess;
|
|
53
|
+
/**
|
|
54
|
+
* Downloads the list of IP addresses used by GitHub.com for webhooks.
|
|
55
|
+
*
|
|
56
|
+
* Note that downloading dynamic data during deployment is not recommended in CDK. This is a workaround for the lack of a better solution.
|
|
57
|
+
*/
|
|
58
|
+
static githubWebhookIps(): string[];
|
|
59
|
+
/**
|
|
60
|
+
* Creates all required resources and returns access URL or empty string if disabled.
|
|
61
|
+
*
|
|
62
|
+
* @internal
|
|
63
|
+
*/
|
|
64
|
+
abstract _bind(construct: Construct, id: string, lambdaFunction: lambda.Function): string;
|
|
65
|
+
}
|
package/lib/access.js
ADDED
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.LambdaAccess = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const child_process_1 = require("child_process");
|
|
7
|
+
const cdk = require("aws-cdk-lib");
|
|
8
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
9
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
10
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
11
|
+
/**
|
|
12
|
+
* Access configuration options for Lambda functions like setup and webhook function. Use this to limit access to these functions.
|
|
13
|
+
*/
|
|
14
|
+
class LambdaAccess {
|
|
15
|
+
/**
|
|
16
|
+
* Disables access to the configured Lambda function. This is useful for the setup function after setup is done.
|
|
17
|
+
*/
|
|
18
|
+
static noAccess() {
|
|
19
|
+
return new NoAccess();
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Provide access using Lambda URL. This is the default and simplest option. It puts no limits on the requester, but the Lambda functions themselves authenticate every request.
|
|
23
|
+
*/
|
|
24
|
+
static lambdaUrl() {
|
|
25
|
+
return new LambdaUrl();
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Provide access using API Gateway. This is the most secure option, but requires additional configuration. It allows you to limit access to specific IP addresses and even to a specific VPC.
|
|
29
|
+
*
|
|
30
|
+
* To limit access to GitHub.com use:
|
|
31
|
+
*
|
|
32
|
+
* ```
|
|
33
|
+
* LambdaAccess.apiGateway({
|
|
34
|
+
* allowedIps: LambdaAccess.githubWebhookIps(),
|
|
35
|
+
* });
|
|
36
|
+
* ```
|
|
37
|
+
*
|
|
38
|
+
* Alternatively, get and manually update the list manually with:
|
|
39
|
+
*
|
|
40
|
+
* ```
|
|
41
|
+
* curl https://api.github.com/meta | jq .hooks
|
|
42
|
+
* ```
|
|
43
|
+
*/
|
|
44
|
+
static apiGateway(props) {
|
|
45
|
+
return new ApiGateway(props);
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Downloads the list of IP addresses used by GitHub.com for webhooks.
|
|
49
|
+
*
|
|
50
|
+
* Note that downloading dynamic data during deployment is not recommended in CDK. This is a workaround for the lack of a better solution.
|
|
51
|
+
*/
|
|
52
|
+
static githubWebhookIps() {
|
|
53
|
+
const githubMeta = (0, child_process_1.execFileSync)('curl', ['-fsSL', 'https://api.github.com/meta']).toString();
|
|
54
|
+
const githubMetaJson = JSON.parse(githubMeta);
|
|
55
|
+
return githubMetaJson.hooks;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
59
|
+
LambdaAccess[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.LambdaAccess", version: "0.9.3" };
|
|
60
|
+
exports.LambdaAccess = LambdaAccess;
|
|
61
|
+
/**
|
|
62
|
+
* @internal
|
|
63
|
+
*/
|
|
64
|
+
class NoAccess extends LambdaAccess {
|
|
65
|
+
_bind(_construct, _id, _lambdaFunction) {
|
|
66
|
+
return '';
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* @internal
|
|
71
|
+
*/
|
|
72
|
+
class LambdaUrl extends LambdaAccess {
|
|
73
|
+
_bind(_construct, _id, lambdaFunction) {
|
|
74
|
+
return lambdaFunction.addFunctionUrl({
|
|
75
|
+
authType: aws_lambda_1.FunctionUrlAuthType.NONE,
|
|
76
|
+
}).url;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* @internal
|
|
81
|
+
*/
|
|
82
|
+
class ApiGateway {
|
|
83
|
+
constructor(props) {
|
|
84
|
+
this.props = props;
|
|
85
|
+
}
|
|
86
|
+
_bind(scope, id, lambdaFunction) {
|
|
87
|
+
let policy;
|
|
88
|
+
let endpointConfig = undefined;
|
|
89
|
+
if (this.props?.allowedVpc) {
|
|
90
|
+
// private api gateway
|
|
91
|
+
const sg = new aws_cdk_lib_1.aws_ec2.SecurityGroup(scope, `${id}/SG`, {
|
|
92
|
+
vpc: this.props.allowedVpc,
|
|
93
|
+
allowAllOutbound: true,
|
|
94
|
+
});
|
|
95
|
+
for (const otherSg of this.props?.allowedSecurityGroups ?? []) {
|
|
96
|
+
sg.connections.allowFrom(otherSg, aws_cdk_lib_1.aws_ec2.Port.tcp(443));
|
|
97
|
+
}
|
|
98
|
+
const vpcEndpoint = new aws_cdk_lib_1.aws_ec2.InterfaceVpcEndpoint(scope, `${id}/VpcEndpoint`, {
|
|
99
|
+
vpc: this.props.allowedVpc,
|
|
100
|
+
service: aws_cdk_lib_1.aws_ec2.InterfaceVpcEndpointAwsService.APIGATEWAY,
|
|
101
|
+
privateDnsEnabled: true,
|
|
102
|
+
securityGroups: [sg],
|
|
103
|
+
open: false,
|
|
104
|
+
});
|
|
105
|
+
endpointConfig = {
|
|
106
|
+
types: [aws_cdk_lib_1.aws_apigateway.EndpointType.PRIVATE],
|
|
107
|
+
vpcEndpoints: [vpcEndpoint],
|
|
108
|
+
};
|
|
109
|
+
policy = aws_iam_1.PolicyDocument.fromJson({
|
|
110
|
+
Version: '2012-10-17',
|
|
111
|
+
Statement: [
|
|
112
|
+
{
|
|
113
|
+
Effect: 'Allow',
|
|
114
|
+
Principal: '*',
|
|
115
|
+
Action: 'execute-api:Invoke',
|
|
116
|
+
Resource: 'execute-api:/*/*/*',
|
|
117
|
+
Condition: {
|
|
118
|
+
StringEquals: {
|
|
119
|
+
'aws:SourceVpce': vpcEndpoint.vpcEndpointId,
|
|
120
|
+
},
|
|
121
|
+
},
|
|
122
|
+
},
|
|
123
|
+
],
|
|
124
|
+
});
|
|
125
|
+
}
|
|
126
|
+
else {
|
|
127
|
+
// public api gateway
|
|
128
|
+
if (this.props?.allowedSecurityGroups) {
|
|
129
|
+
cdk.Annotations.of(scope).addWarning('allowedSecurityGroups is ignored when allowedVpc is not specified.');
|
|
130
|
+
}
|
|
131
|
+
policy = aws_iam_1.PolicyDocument.fromJson({
|
|
132
|
+
Version: '2012-10-17',
|
|
133
|
+
Statement: [
|
|
134
|
+
{
|
|
135
|
+
Effect: 'Allow',
|
|
136
|
+
Principal: '*',
|
|
137
|
+
Action: 'execute-api:Invoke',
|
|
138
|
+
Resource: 'execute-api:/*/*/*',
|
|
139
|
+
Condition: {
|
|
140
|
+
IpAddress: {
|
|
141
|
+
'aws:SourceIp': this.props?.allowedIps ?? ['0.0.0.0/0'],
|
|
142
|
+
},
|
|
143
|
+
},
|
|
144
|
+
},
|
|
145
|
+
],
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
const api = new aws_cdk_lib_1.aws_apigateway.LambdaRestApi(scope, id, {
|
|
149
|
+
handler: lambdaFunction,
|
|
150
|
+
proxy: true,
|
|
151
|
+
cloudWatchRole: false,
|
|
152
|
+
endpointConfiguration: endpointConfig,
|
|
153
|
+
policy,
|
|
154
|
+
});
|
|
155
|
+
// remove CfnOutput
|
|
156
|
+
api.node.tryRemoveChild('Endpoint');
|
|
157
|
+
return api.url;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"access.js","sourceRoot":"","sources":["../src/access.ts"],"names":[],"mappings":"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","sourcesContent":["import { execFileSync } from 'child_process';\nimport * as cdk from 'aws-cdk-lib';\nimport { aws_apigateway as apigateway, aws_ec2 as ec2, aws_iam as iam, aws_lambda as lambda } from 'aws-cdk-lib';\nimport { PolicyDocument } from 'aws-cdk-lib/aws-iam';\nimport { FunctionUrlAuthType } from 'aws-cdk-lib/aws-lambda';\nimport { Construct } from 'constructs';\n\n\nexport interface ApiGatewayAccessProps {\n  /**\n   * List of IP addresses in CIDR notation that are allowed to access the API Gateway.\n   *\n   * If not specified on public API Gateway, all IP addresses are allowed.\n   *\n   * If not specified on private API Gateway, no IP addresses are allowed (but specified security groups are).\n   */\n  readonly allowedIps?: string[];\n\n  /**\n   * Creates a private API Gateway and allows access from the specified VPC.\n   */\n  readonly allowedVpc?: ec2.IVpc;\n\n  /**\n   * List of security groups that are allowed to access the API Gateway.\n   *\n   * Only works for private API Gateways with {@link allowedVpc}.\n   */\n  readonly allowedSecurityGroups?: ec2.ISecurityGroup[];\n}\n\n/**\n * Access configuration options for Lambda functions like setup and webhook function. Use this to limit access to these functions.\n */\nexport abstract class LambdaAccess {\n  /**\n   * Disables access to the configured Lambda function. This is useful for the setup function after setup is done.\n   */\n  static noAccess(): LambdaAccess {\n    return new NoAccess();\n  }\n\n  /**\n   * Provide access using Lambda URL. This is the default and simplest option. It puts no limits on the requester, but the Lambda functions themselves authenticate every request.\n   */\n  static lambdaUrl(): LambdaAccess {\n    return new LambdaUrl();\n  }\n\n  /**\n   * Provide access using API Gateway. This is the most secure option, but requires additional configuration. It allows you to limit access to specific IP addresses and even to a specific VPC.\n   *\n   * To limit access to GitHub.com use:\n   *\n   * ```\n   * LambdaAccess.apiGateway({\n   *   allowedIps: LambdaAccess.githubWebhookIps(),\n   * });\n   * ```\n   *\n   * Alternatively, get and manually update the list manually with:\n   *\n   * ```\n   * curl https://api.github.com/meta | jq .hooks\n   * ```\n   */\n  static apiGateway(props?: ApiGatewayAccessProps): LambdaAccess {\n    return new ApiGateway(props);\n  }\n\n  /**\n   * Downloads the list of IP addresses used by GitHub.com for webhooks.\n   *\n   * Note that downloading dynamic data during deployment is not recommended in CDK. This is a workaround for the lack of a better solution.\n   */\n  static githubWebhookIps(): string[] {\n    const githubMeta = execFileSync('curl', ['-fsSL', 'https://api.github.com/meta']).toString();\n    const githubMetaJson = JSON.parse(githubMeta);\n    return githubMetaJson.hooks;\n  }\n\n  /**\n   * Creates all required resources and returns access URL or empty string if disabled.\n   *\n   * @internal\n   */\n  public abstract _bind(construct: Construct, id: string, lambdaFunction: lambda.Function): string;\n}\n\n/**\n * @internal\n */\nclass NoAccess extends LambdaAccess {\n  public _bind(_construct: Construct, _id: string, _lambdaFunction: lambda.Function): string {\n    return '';\n  }\n}\n\n/**\n * @internal\n */\nclass LambdaUrl extends LambdaAccess {\n  public _bind(_construct: Construct, _id: string, lambdaFunction: lambda.Function): string {\n    return lambdaFunction.addFunctionUrl({\n      authType: FunctionUrlAuthType.NONE,\n    }).url;\n  }\n}\n\n/**\n * @internal\n */\nclass ApiGateway {\n  constructor(private readonly props?: ApiGatewayAccessProps) {}\n\n  public _bind(scope: Construct, id: string, lambdaFunction: lambda.Function): string {\n    let policy: iam.PolicyDocument;\n    let endpointConfig: apigateway.EndpointConfiguration | undefined = undefined;\n\n    if (this.props?.allowedVpc) {\n      // private api gateway\n      const sg = new ec2.SecurityGroup(scope, `${id}/SG`, {\n        vpc: this.props.allowedVpc,\n        allowAllOutbound: true,\n      });\n\n      for (const otherSg of this.props?.allowedSecurityGroups ?? []) {\n        sg.connections.allowFrom(otherSg, ec2.Port.tcp(443));\n      }\n\n      const vpcEndpoint = new ec2.InterfaceVpcEndpoint(scope, `${id}/VpcEndpoint`, {\n        vpc: this.props.allowedVpc,\n        service: ec2.InterfaceVpcEndpointAwsService.APIGATEWAY,\n        privateDnsEnabled: true,\n        securityGroups: [sg],\n        open: false,\n      });\n\n      endpointConfig = {\n        types: [apigateway.EndpointType.PRIVATE],\n        vpcEndpoints: [vpcEndpoint],\n      };\n\n      policy = PolicyDocument.fromJson({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: '*',\n            Action: 'execute-api:Invoke',\n            Resource: 'execute-api:/*/*/*',\n            Condition: {\n              StringEquals: {\n                'aws:SourceVpce': vpcEndpoint.vpcEndpointId,\n              },\n            },\n          },\n        ],\n      });\n    } else {\n      // public api gateway\n      if (this.props?.allowedSecurityGroups) {\n        cdk.Annotations.of(scope).addWarning('allowedSecurityGroups is ignored when allowedVpc is not specified.');\n      }\n\n      policy = PolicyDocument.fromJson({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: '*',\n            Action: 'execute-api:Invoke',\n            Resource: 'execute-api:/*/*/*',\n            Condition: {\n              IpAddress: {\n                'aws:SourceIp': this.props?.allowedIps ?? ['0.0.0.0/0'],\n              },\n            },\n          },\n        ],\n      });\n    }\n\n    const api = new apigateway.LambdaRestApi(scope, id, {\n      handler: lambdaFunction,\n      proxy: true,\n      cloudWatchRole: false,\n      endpointConfiguration: endpointConfig,\n      policy,\n    });\n\n    // remove CfnOutput\n    api.node.tryRemoveChild('Endpoint');\n\n    return api.url;\n  }\n}\n"]}
|
package/lib/index.d.ts
CHANGED
package/lib/index.js
CHANGED
|
@@ -14,7 +14,8 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./access"), exports);
|
|
17
18
|
__exportStar(require("./secrets"), exports);
|
|
18
19
|
__exportStar(require("./runner"), exports);
|
|
19
20
|
__exportStar(require("./providers"), exports);
|
|
20
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDJDQUF5QjtBQUN6Qiw0Q0FBMEI7QUFDMUIsMkNBQXlCO0FBQ3pCLDhDQUE0QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vYWNjZXNzJztcbmV4cG9ydCAqIGZyb20gJy4vc2VjcmV0cyc7XG5leHBvcnQgKiBmcm9tICcuL3J1bm5lcic7XG5leHBvcnQgKiBmcm9tICcuL3Byb3ZpZGVycyc7XG4iXX0=
|
|
@@ -28,7 +28,8 @@ function response(code, body) {
|
|
|
28
28
|
};
|
|
29
29
|
}
|
|
30
30
|
async function handleRoot(event, setupToken) {
|
|
31
|
-
const
|
|
31
|
+
const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`;
|
|
32
|
+
const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;
|
|
32
33
|
const githubSecrets = await (0, helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
|
|
33
34
|
return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain));
|
|
34
35
|
}
|
|
@@ -122,19 +123,21 @@ exports.handler = async function (event) {
|
|
|
122
123
|
}
|
|
123
124
|
// handle requests
|
|
124
125
|
try {
|
|
125
|
-
|
|
126
|
+
const path = event.path ?? event.rawPath;
|
|
127
|
+
const method = event.httpMethod ?? event.requestContext.http.method;
|
|
128
|
+
if (path == '/') {
|
|
126
129
|
return await handleRoot(event, setupToken);
|
|
127
130
|
}
|
|
128
|
-
else if (
|
|
131
|
+
else if (path == '/domain' && method == 'POST') {
|
|
129
132
|
return await handleDomain(event);
|
|
130
133
|
}
|
|
131
|
-
else if (
|
|
134
|
+
else if (path == '/pat' && method == 'POST') {
|
|
132
135
|
return await handlePat(event);
|
|
133
136
|
}
|
|
134
|
-
else if (
|
|
137
|
+
else if (path == '/complete-new-app' && method == 'GET') {
|
|
135
138
|
return await handleNewApp(event);
|
|
136
139
|
}
|
|
137
|
-
else if (
|
|
140
|
+
else if (path == '/app' && method == 'POST') {
|
|
138
141
|
return await handleExistingApp(event);
|
|
139
142
|
}
|
|
140
143
|
else {
|
|
@@ -142,7 +145,8 @@ exports.handler = async function (event) {
|
|
|
142
145
|
}
|
|
143
146
|
}
|
|
144
147
|
catch (e) {
|
|
148
|
+
console.error(e);
|
|
145
149
|
return response(500, `<b>Error:</b> ${e}`);
|
|
146
150
|
}
|
|
147
151
|
};
|
|
148
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.lambda.js","sourceRoot":"","sources":["../../src/lambdas/setup.lambda.ts"],"names":[],"mappings":";;AAAA,sDAAsD;AACtD,iCAAiC;AACjC,yBAAyB;AACzB,wCAAwC;AAGxC,qCAA6C;AAC7C,uCAAkE;AAElE,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,KAAuC,EAAE,UAAkB;IACnF,MAAM,YAAY,GAAG,WAAW,KAAK,CAAC,cAAc,CAAC,UAAU,EAAE,CAAC;IAClE,MAAM,aAAa,GAAG,MAAM,IAAA,4BAAkB,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,KAAuC;IACzD,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,KAAuC;IACjE,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,4BAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC9E,aAAa,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;IACnC,MAAM,IAAA,2BAAiB,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,KAAuC;IAC9D,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,2BAAiB,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,2BAAiB,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,KAAuC;IACjE,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,4BAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC9E,MAAM,OAAO,GAAG,IAAA,0BAAiB,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,2BAAiB,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,2BAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACpF,MAAM,IAAA,2BAAiB,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,2BAAiB,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,KAAuC;IACtE,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,2BAAiB,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,2BAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,IAAI,CAAC,EAAY,CAAC,CAAC;IACtF,MAAM,IAAA,2BAAiB,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,KAAuC;IACvE,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,4BAAkB,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,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,IAAI,GAAG,EAAE;YACzC,OAAO,MAAM,UAAU,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;SAC5C;aAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,IAAI,SAAS,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,IAAI,MAAM,EAAE;YACpG,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;SAClC;aAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,IAAI,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,IAAI,MAAM,EAAE;YACjG,OAAO,MAAM,SAAS,CAAC,KAAK,CAAC,CAAC;SAC/B;aAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,IAAI,mBAAmB,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,IAAI,KAAK,EAAE;YAC7G,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;SAClC;aAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,IAAI,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,IAAI,MAAM,EAAE;YACjG,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,QAAQ,CAAC,GAAG,EAAE,iBAAiB,CAAC,EAAE,CAAC,CAAC;KAC5C;AACH,CAAC,CAAC","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 './helpers';\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: AWSLambda.APIGatewayProxyEventV2, setupToken: string): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const setupBaseUrl = `https://${event.requestContext.domainName}`;\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: AWSLambda.APIGatewayProxyEventV2) {\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: AWSLambda.APIGatewayProxyEventV2): 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: AWSLambda.APIGatewayProxyEventV2): 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: AWSLambda.APIGatewayProxyEventV2): 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: AWSLambda.APIGatewayProxyEventV2): 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: AWSLambda.APIGatewayProxyEventV2): 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    if (event.requestContext.http.path == '/') {\n      return await handleRoot(event, setupToken);\n    } else if (event.requestContext.http.path == '/domain' && event.requestContext.http.method == 'POST') {\n      return await handleDomain(event);\n    } else if (event.requestContext.http.path == '/pat' && event.requestContext.http.method == 'POST') {\n      return await handlePat(event);\n    } else if (event.requestContext.http.path == '/complete-new-app' && event.requestContext.http.method == 'GET') {\n      return await handleNewApp(event);\n    } else if (event.requestContext.http.path == '/app' && event.requestContext.http.method == 'POST') {\n      return await handleExistingApp(event);\n    } else {\n      return response(404, 'Not found');\n    }\n  } catch (e) {\n    return response(500, `<b>Error:</b> ${e}`);\n  }\n};\n"]}
|
|
152
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.lambda.js","sourceRoot":"","sources":["../../src/lambdas/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 './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"]}
|