UNPKG

@cloudsnorkel/cdk-github-runners

Version:

CDK construct to create GitHub Actions self-hosted runners. Creates ephemeral runners on demand. Easy to deploy and highly customizable.

161 lines 26.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const crypto = require("crypto"); const fs = require("fs"); const lambda_github_1 = require("./lambda-github"); const lambda_helpers_1 = require("./lambda-helpers"); const nonce = crypto.randomBytes(64).toString('hex'); function getHtml(baseUrl, token, domain) { return fs.readFileSync('index.html', 'utf-8') .replace(/INSERT_WEBHOOK_URL_HERE/g, process.env.WEBHOOK_URL) .replace(/INSERT_BASE_URL_HERE/g, baseUrl) .replace(/INSERT_TOKEN_HERE/g, token) .replace(/INSERT_SECRET_ARN_HERE/g, process.env.SETUP_SECRET_ARN) .replace(/INSERT_DOMAIN_HERE/g, domain) .replace(/<script/g, `<script nonce="${nonce}"`) .replace(/<style/g, `<style nonce="${nonce}"`); } function response(code, body) { return { statusCode: code, headers: { 'Content-Type': 'text/html', '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'`, }, body: body, }; } async function handleRoot(event, setupToken) { const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`; const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`; const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN); return response(200, getHtml(setupBaseUrl, setupToken, githubSecrets.domain)); } function decodeBody(event) { let body = event.body; if (!body) { throw new Error('No body found'); } if (event.isBase64Encoded) { body = Buffer.from(body, 'base64').toString('utf-8'); } return JSON.parse(body); } async function handleDomain(event) { const body = decodeBody(event); if (!body.domain) { return response(400, 'Invalid domain'); } if (body.runnerLevel !== 'repo' && body.runnerLevel !== 'org') { return response(400, 'Invalid runner registration level'); } const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN); githubSecrets.domain = body.domain; githubSecrets.runnerLevel = body.runnerLevel; await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets)); return response(200, 'Domain set'); } async function handlePat(event) { const body = decodeBody(event); if (!body.pat || !body.domain) { return response(400, 'Invalid personal access token'); } await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({ domain: body.domain, appId: -1, personalAuthToken: body.pat, runnerLevel: 'repo', })); await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' })); return response(200, 'Personal access token set'); } async function handleNewApp(event) { if (!event.queryStringParameters) { return response(400, 'Invalid code'); } const code = event.queryStringParameters.code; if (!code) { return response(400, 'Invalid code'); } const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN); const baseUrl = (0, lambda_github_1.baseUrlFromDomain)(githubSecrets.domain); const { Octokit } = await (0, lambda_github_1.loadOctokitRest)(); const newApp = await new Octokit({ baseUrl }).rest.apps.createFromManifest({ code }); githubSecrets.appId = newApp.data.id; githubSecrets.domain = new URL(newApp.data.html_url).host; // just in case it's different githubSecrets.personalAuthToken = ''; // don't update runnerLevel as it was set by handleDomain() above await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets)); await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, newApp.data.pem); await (0, lambda_helpers_1.updateSecretValue)(process.env.WEBHOOK_SECRET_ARN, JSON.stringify({ webhookSecret: newApp.data.webhook_secret, })); await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' })); return response(200, `New app set. <a href="${newApp.data.html_url}/installations/new">Install it</a> for your repositories.`); } async function handleExistingApp(event) { const body = decodeBody(event); if (!body.appid || !body.pk || !body.domain || (body.runnerLevel !== 'repo' && body.runnerLevel !== 'org')) { return response(400, 'Missing fields'); } await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_SECRET_ARN, JSON.stringify({ domain: body.domain, appId: body.appid, personalAuthToken: '', runnerLevel: body.runnerLevel, })); await (0, lambda_helpers_1.updateSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, body.pk); await (0, lambda_helpers_1.updateSecretValue)(process.env.SETUP_SECRET_ARN, JSON.stringify({ token: '' })); return response(200, 'Existing app set. Don\'t forget to set up the webhook.'); } async function handler(event) { // confirm required environment variables if (!process.env.WEBHOOK_URL) { throw new Error('Missing environment variables'); } const setupToken = (await (0, lambda_helpers_1.getSecretJsonValue)(process.env.SETUP_SECRET_ARN)).token; // bail out if setup was already completed if (!setupToken) { return response(200, 'Setup already complete. Put a new token in the setup secret if you want to redo it.'); } if (!event.queryStringParameters) { return response(403, 'Wrong setup token.'); } // safely confirm url token matches our secret const urlToken = event.queryStringParameters.token || event.queryStringParameters.state || ''; if (urlToken.length != setupToken.length || !crypto.timingSafeEqual(Buffer.from(urlToken, 'utf-8'), Buffer.from(setupToken, 'utf-8'))) { return response(403, 'Wrong setup token.'); } // handle requests try { const path = event.path ?? event.rawPath; const method = event.httpMethod ?? event.requestContext.http.method; if (path == '/') { return await handleRoot(event, setupToken); } else if (path == '/domain' && method == 'POST') { return await handleDomain(event); } else if (path == '/pat' && method == 'POST') { return await handlePat(event); } else if (path == '/complete-new-app' && method == 'GET') { return await handleNewApp(event); } else if (path == '/app' && method == 'POST') { return await handleExistingApp(event); } else { return response(404, 'Not found'); } } catch (e) { console.error({ notice: 'Setup handler failed', error: `${e}`, }); return response(500, `<b>Error:</b> ${e}`); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.lambda.js","sourceRoot":"","sources":["../src/setup.lambda.ts"],"names":[],"mappings":";;AAsIA,0BA+CC;AArLD,iCAAiC;AACjC,yBAAyB;AAEzB,mDAAoF;AACpF,qDAAyE;AAIzE,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;AAErD,SAAS,OAAO,CAAC,OAAe,EAAE,KAAa,EAAE,MAAc;IAC7D,OAAO,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC;SAC1C,OAAO,CAAC,0BAA0B,EAAE,OAAO,CAAC,GAAG,CAAC,WAAY,CAAC;SAC7D,OAAO,CAAC,uBAAuB,EAAE,OAAO,CAAC;SACzC,OAAO,CAAC,oBAAoB,EAAE,KAAK,CAAC;SACpC,OAAO,CAAC,yBAAyB,EAAE,OAAO,CAAC,GAAG,CAAC,gBAAiB,CAAC;SACjE,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC;SACtC,OAAO,CAAC,UAAU,EAAE,kBAAkB,KAAK,GAAG,CAAC;SAC/C,OAAO,CAAC,SAAS,EAAE,iBAAiB,KAAK,GAAG,CAAC,CAAC;AACnD,CAAC;AAED,SAAS,QAAQ,CAAC,IAAY,EAAE,IAAY;IAC1C,OAAO;QACL,UAAU,EAAE,IAAI;QAChB,OAAO,EAAE;YACP,cAAc,EAAE,WAAW;YAC3B,yBAAyB,EAAE,sCAAsC,KAAK,sHAAsH;SAC7L;QACD,IAAI,EAAE,IAAI;KACX,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,UAAU,CAAC,KAAsB,EAAE,UAAkB;IAClE,MAAM,KAAK,GAAG,KAAK,CAAC,cAAc,CAAC,KAAK,IAAI,UAAU,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;IAC/F,MAAM,YAAY,GAAG,WAAW,KAAK,CAAC,cAAc,CAAC,UAAU,GAAG,KAAK,EAAE,CAAC;IAC1E,MAAM,aAAa,GAAkB,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAE7F,OAAO,QAAQ,CAAC,GAAG,EAAE,OAAO,CAAC,YAAY,EAAE,UAAU,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC;AAChF,CAAC;AAED,SAAS,UAAU,CAAC,KAAsB;IACxC,IAAI,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;IACtB,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;IACnC,CAAC;IACD,IAAI,KAAK,CAAC,eAAe,EAAE,CAAC;QAC1B,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IACvD,CAAC;IACD,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,KAAK,UAAU,YAAY,CAAC,KAAsB;IAChD,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;QACjB,OAAO,QAAQ,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;IACzC,CAAC;IACD,IAAI,IAAI,CAAC,WAAW,KAAK,MAAM,IAAI,IAAI,CAAC,WAAW,KAAK,KAAK,EAAE,CAAC;QAC9D,OAAO,QAAQ,CAAC,GAAG,EAAE,mCAAmC,CAAC,CAAC;IAC5D,CAAC;IAED,MAAM,aAAa,GAAkB,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC7F,aAAa,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;IACnC,aAAa,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;IAC7C,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC;IACtF,OAAO,QAAQ,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;AACrC,CAAC;AAED,KAAK,UAAU,SAAS,CAAC,KAAsB;IAC7C,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;QAC9B,OAAO,QAAQ,CAAC,GAAG,EAAE,+BAA+B,CAAC,CAAC;IACxD,CAAC;IAED,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAgB;QACnF,MAAM,EAAE,IAAI,CAAC,MAAM;QACnB,KAAK,EAAE,CAAC,CAAC;QACT,iBAAiB,EAAE,IAAI,CAAC,GAAG;QAC3B,WAAW,EAAE,MAAM;KACpB,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAC,GAAG,EAAE,2BAA2B,CAAC,CAAC;AACpD,CAAC;AAED,KAAK,UAAU,YAAY,CAAC,KAAsB;IAChD,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE,CAAC;QACjC,OAAO,QAAQ,CAAC,GAAG,EAAE,cAAc,CAAC,CAAC;IACvC,CAAC;IAED,MAAM,IAAI,GAAG,KAAK,CAAC,qBAAqB,CAAC,IAAI,CAAC;IAE9C,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO,QAAQ,CAAC,GAAG,EAAE,cAAc,CAAC,CAAC;IACvC,CAAC;IAED,MAAM,aAAa,GAAkB,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;IAC7F,MAAM,OAAO,GAAG,IAAA,iCAAiB,EAAC,aAAa,CAAC,MAAM,CAAC,CAAC;IACxD,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,IAAA,+BAAe,GAAE,CAAC;IAC5C,MAAM,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;IAErF,aAAa,CAAC,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;IACrC,aAAa,CAAC,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,8BAA8B;IACzF,aAAa,CAAC,iBAAiB,GAAG,EAAE,CAAC;IACrC,iEAAiE;IAEjE,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC;IACtF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACpF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC;QACrE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,cAAc;KAC1C,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAC,GAAG,EAAE,yBAAyB,MAAM,CAAC,IAAI,CAAC,QAAQ,2DAA2D,CAAC,CAAC;AACjI,CAAC;AAED,KAAK,UAAU,iBAAiB,CAAC,KAAsB;IACrD,MAAM,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;IAE/B,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,WAAW,KAAK,MAAM,IAAI,IAAI,CAAC,WAAW,KAAK,KAAK,CAAC,EAAE,CAAC;QAC3G,OAAO,QAAQ,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;IACzC,CAAC;IAED,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAgB;QACnF,MAAM,EAAE,IAAI,CAAC,MAAM;QACnB,KAAK,EAAE,IAAI,CAAC,KAAK;QACjB,iBAAiB,EAAE,EAAE;QACrB,WAAW,EAAE,IAAI,CAAC,WAAW;KAC9B,CAAC,CAAC,CAAC;IACJ,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,IAAI,CAAC,EAAY,CAAC,CAAC;IACtF,MAAM,IAAA,kCAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,QAAQ,CAAC,GAAG,EAAE,wDAAwD,CAAC,CAAC;AACjF,CAAC;AAEM,KAAK,UAAU,OAAO,CAAC,KAAsB;IAClD,yCAAyC;IACzC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;QAC7B,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;IACnD,CAAC;IAED,MAAM,UAAU,GAAG,CAAC,MAAM,IAAA,mCAAkB,EAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC;IAElF,0CAA0C;IAC1C,IAAI,CAAC,UAAU,EAAE,CAAC;QAChB,OAAO,QAAQ,CAAC,GAAG,EAAE,qFAAqF,CAAC,CAAC;IAC9G,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE,CAAC;QACjC,OAAO,QAAQ,CAAC,GAAG,EAAE,oBAAoB,CAAC,CAAC;IAC7C,CAAC;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,CAAC;QACtI,OAAO,QAAQ,CAAC,GAAG,EAAE,oBAAoB,CAAC,CAAC;IAC7C,CAAC;IAED,kBAAkB;IAClB,IAAI,CAAC;QACH,MAAM,IAAI,GAAI,KAAwC,CAAC,IAAI,IAAK,KAA0C,CAAC,OAAO,CAAC;QACnH,MAAM,MAAM,GAAI,KAAwC,CAAC,UAAU,IAAK,KAA0C,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC;QAC9I,IAAI,IAAI,IAAI,GAAG,EAAE,CAAC;YAChB,OAAO,MAAM,UAAU,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAC7C,CAAC;aAAM,IAAI,IAAI,IAAI,SAAS,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;YACjD,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;QACnC,CAAC;aAAM,IAAI,IAAI,IAAI,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;YAC9C,OAAO,MAAM,SAAS,CAAC,KAAK,CAAC,CAAC;QAChC,CAAC;aAAM,IAAI,IAAI,IAAI,mBAAmB,IAAI,MAAM,IAAI,KAAK,EAAE,CAAC;YAC1D,OAAO,MAAM,YAAY,CAAC,KAAK,CAAC,CAAC;QACnC,CAAC;aAAM,IAAI,IAAI,IAAI,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;YAC9C,OAAO,MAAM,iBAAiB,CAAC,KAAK,CAAC,CAAC;QACxC,CAAC;aAAM,CAAC;YACN,OAAO,QAAQ,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CAAC;YACZ,MAAM,EAAE,sBAAsB;YAC9B,KAAK,EAAE,GAAG,CAAC,EAAE;SACd,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,GAAG,EAAE,iBAAiB,CAAC,EAAE,CAAC,CAAC;IAC7C,CAAC;AACH,CAAC","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport * as AWSLambda from 'aws-lambda';\nimport { baseUrlFromDomain, GitHubSecrets, loadOctokitRest } from './lambda-github';\nimport { getSecretJsonValue, updateSecretValue } from './lambda-helpers';\n\ntype ApiGatewayEvent = AWSLambda.APIGatewayProxyEvent | AWSLambda.APIGatewayProxyEventV2;\n\nconst nonce = crypto.randomBytes(64).toString('hex');\n\nfunction getHtml(baseUrl: string, token: string, domain: string): string {\n  return fs.readFileSync('index.html', 'utf-8')\n    .replace(/INSERT_WEBHOOK_URL_HERE/g, process.env.WEBHOOK_URL!)\n    .replace(/INSERT_BASE_URL_HERE/g, baseUrl)\n    .replace(/INSERT_TOKEN_HERE/g, token)\n    .replace(/INSERT_SECRET_ARN_HERE/g, process.env.SETUP_SECRET_ARN!)\n    .replace(/INSERT_DOMAIN_HERE/g, domain)\n    .replace(/<script/g, `<script nonce=\"${nonce}\"`)\n    .replace(/<style/g, `<style nonce=\"${nonce}\"`);\n}\n\nfunction response(code: number, body: string): AWSLambda.APIGatewayProxyResultV2 {\n  return {\n    statusCode: code,\n    headers: {\n      'Content-Type': 'text/html',\n      'Content-Security-Policy': `default-src 'unsafe-inline' 'nonce-${nonce}'; img-src data:; connect-src 'self'; form-action https:; frame-ancestors 'none'; object-src 'none'; base-uri 'self'`,\n    },\n    body: body,\n  };\n}\n\nasync function handleRoot(event: ApiGatewayEvent, setupToken: string): Promise<AWSLambda.APIGatewayProxyResultV2> {\n  const stage = event.requestContext.stage == '$default' ? '' : `/${event.requestContext.stage}`;\n  const setupBaseUrl = `https://${event.requestContext.domainName}${stage}`;\n  const githubSecrets: 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  if (body.runnerLevel !== 'repo' && body.runnerLevel !== 'org') {\n    return response(400, 'Invalid runner registration level');\n  }\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  githubSecrets.domain = body.domain;\n  githubSecrets.runnerLevel = body.runnerLevel;\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets));\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(<GitHubSecrets>{\n    domain: body.domain,\n    appId: -1,\n    personalAuthToken: body.pat,\n    runnerLevel: 'repo',\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: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n  const { Octokit } = await loadOctokitRest();\n  const newApp = await new Octokit({ baseUrl }).rest.apps.createFromManifest({ code });\n\n  githubSecrets.appId = newApp.data.id;\n  githubSecrets.domain = new URL(newApp.data.html_url).host; // just in case it's different\n  githubSecrets.personalAuthToken = '';\n  // don't update runnerLevel as it was set by handleDomain() above\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify(githubSecrets));\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 || (body.runnerLevel !== 'repo' && body.runnerLevel !== 'org')) {\n    return response(400, 'Missing fields');\n  }\n\n  await updateSecretValue(process.env.GITHUB_SECRET_ARN, JSON.stringify(<GitHubSecrets>{\n    domain: body.domain,\n    appId: body.appid,\n    personalAuthToken: '',\n    runnerLevel: body.runnerLevel,\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\nexport async function handler(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({\n      notice: 'Setup handler failed',\n      error: `${e}`,\n    });\n    return response(500, `<b>Error:</b> ${e}`);\n  }\n}\n"]}