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.

314 lines 44 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const client_cloudformation_1 = require("@aws-sdk/client-cloudformation"); const client_ec2_1 = require("@aws-sdk/client-ec2"); const client_ecr_1 = require("@aws-sdk/client-ecr"); const client_sfn_1 = require("@aws-sdk/client-sfn"); const auth_app_1 = require("@octokit/auth-app"); const core_1 = require("@octokit/core"); const lambda_github_1 = require("./lambda-github"); const lambda_helpers_1 = require("./lambda-helpers"); const cfn = new client_cloudformation_1.CloudFormationClient(); const ec2 = new client_ec2_1.EC2Client(); const ecr = new client_ecr_1.ECRClient(); const sf = new client_sfn_1.SFNClient(); function secretArnToUrl(arn) { const parts = arn.split(':'); // arn:aws:secretsmanager:us-east-1:12345678:secret:secret-name-REVISION const region = parts[3]; const fullName = parts[6]; const name = fullName.slice(0, fullName.lastIndexOf('-')); return `https://${region}.console.aws.amazon.com/secretsmanager/home?region=${region}#!/secret?name=${name}`; } function lambdaArnToUrl(arn) { const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ const region = parts[3]; const name = parts[6]; return `https://${region}.console.aws.amazon.com/lambda/home?region=${region}#/functions/${name}?tab=monitoring`; } function lambdaArnToLogGroup(arn) { const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ const name = parts[6]; return `/aws/lambda/${name}`; } function stepFunctionArnToUrl(arn) { const parts = arn.split(':'); // arn:aws:states:us-east-1:12345678:stateMachine:name-XYZ const region = parts[3]; return `https://${region}.console.aws.amazon.com/states/home?region=${region}#/statemachines/view/${arn}`; } async function generateProvidersStatus(stack, logicalId) { const resource = await cfn.send(new client_cloudformation_1.DescribeStackResourceCommand({ StackName: stack, LogicalResourceId: logicalId })); const providers = JSON.parse(resource.StackResourceDetail?.Metadata ?? '{}').providers; if (!providers) { return {}; } return Promise.all(providers.map(async (p) => { // add ECR data, if image is from ECR if (p.image?.imageRepository?.match(/[0-9]+\.dkr\.ecr\.[a-z0-9\-]+\.amazonaws\.com\/.+/)) { const tags = await ecr.send(new client_ecr_1.DescribeImagesCommand({ repositoryName: p.image.imageRepository.split('/')[1], filter: { tagStatus: 'TAGGED', }, maxResults: 1, })); if (tags.imageDetails && tags.imageDetails?.length >= 1) { p.image.latestImage = { tags: tags.imageDetails[0].imageTags, digest: tags.imageDetails[0].imageDigest, date: tags.imageDetails[0].imagePushedAt, }; } } // add AMI data, if image is AMI if (p.ami?.launchTemplate) { const versions = await ec2.send(new client_ec2_1.DescribeLaunchTemplateVersionsCommand({ LaunchTemplateId: p.ami.launchTemplate, Versions: ['$Default'], })); if (versions.LaunchTemplateVersions && versions.LaunchTemplateVersions.length >= 1) { p.ami.latestAmi = versions.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId; } } return p; })); } function safeReturnValue(event, status) { if (event.path) { return { statusCode: 200, headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(status), }; } return status; } async function handler(event) { // confirm required environment variables 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.STACK_NAME) { throw new Error('Missing environment variables'); } // base status const status = { github: { setup: { status: 'Unknown', url: '', secretArn: process.env.SETUP_SECRET_ARN, secretUrl: secretArnToUrl(process.env.SETUP_SECRET_ARN), }, domain: 'Unknown', runnerLevel: 'Unknown', webhook: { url: process.env.WEBHOOK_URL, status: 'Unable to check', secretArn: process.env.WEBHOOK_SECRET_ARN, secretUrl: secretArnToUrl(process.env.WEBHOOK_SECRET_ARN), }, auth: { type: 'Unknown', status: 'Unknown', secretArn: process.env.GITHUB_SECRET_ARN, secretUrl: secretArnToUrl(process.env.GITHUB_SECRET_ARN), privateKeySecretArn: process.env.GITHUB_PRIVATE_KEY_SECRET_ARN, privateKeySecretUrl: secretArnToUrl(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN), app: { id: -1, url: '', installations: [], }, personalAuthToken: '', }, }, providers: await generateProvidersStatus(process.env.STACK_NAME, process.env.LOGICAL_ID), troubleshooting: { webhookHandlerArn: process.env.WEBHOOK_HANDLER_ARN, webhookHandlerUrl: lambdaArnToUrl(process.env.WEBHOOK_HANDLER_ARN), webhookHandlerLogGroup: lambdaArnToLogGroup(process.env.WEBHOOK_HANDLER_ARN), stepFunctionArn: process.env.STEP_FUNCTION_ARN, stepFunctionUrl: stepFunctionArnToUrl(process.env.STEP_FUNCTION_ARN), stepFunctionLogGroup: process.env.STEP_FUNCTION_LOG_GROUP, recentRuns: [], }, }; // setup url if (process.env.SETUP_FUNCTION_URL) { const setupToken = (await (0, lambda_helpers_1.getSecretJsonValue)(process.env.SETUP_SECRET_ARN)).token; if (setupToken) { status.github.setup.status = 'Pending'; status.github.setup.url = `${process.env.SETUP_FUNCTION_URL}?token=${setupToken}`; } else { status.github.setup.status = 'Complete'; } } else { status.github.setup.status = 'Disabled'; } // list last 10 executions and their status try { const executions = await sf.send(new client_sfn_1.ListExecutionsCommand({ stateMachineArn: process.env.STEP_FUNCTION_ARN, maxResults: 10, })); for (const execution of executions.executions ?? []) { const executionDetails = await sf.send(new client_sfn_1.DescribeExecutionCommand({ executionArn: execution.executionArn, })); const input = JSON.parse(executionDetails.input || '{}'); status.troubleshooting.recentRuns.push({ executionArn: execution.executionArn, status: execution.status ?? '<unknown>', owner: input.owner, repo: input.repo, jobId: input.jobId, }); } } catch (e) { status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` }); } // get secrets let githubSecrets; try { githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN); } catch (e) { status.github.auth.status = `Unable to read secret: ${e}`; return safeReturnValue(event, status); } let privateKey; try { privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN); } catch (e) { status.github.auth.status = `Unable to read private key secret: ${e}`; return safeReturnValue(event, status); } // calculate base url let baseUrl = (0, lambda_github_1.baseUrlFromDomain)(githubSecrets.domain); status.github.domain = githubSecrets.domain; // copy runner level status.github.runnerLevel = githubSecrets.runnerLevel ?? 'repo'; if (githubSecrets.personalAuthToken) { // try authenticating with personal access token status.github.auth.type = 'Personal Access Token'; status.github.auth.personalAuthToken = '*redacted*'; let octokit; try { octokit = new core_1.Octokit({ baseUrl, auth: githubSecrets.personalAuthToken }); } catch (e) { status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`; return safeReturnValue(event, status); } try { const user = await octokit.request('GET /user'); status.github.auth.personalAuthToken = `username: ${user.data.login}`; } catch (e) { status.github.auth.status = `Unable to call /user with personal auth token: ${e}`; return safeReturnValue(event, status); } status.github.auth.status = 'OK'; status.github.webhook.status = 'Unable to verify automatically'; } else { // try authenticating with GitHub app status.github.auth.type = 'GitHub App'; status.github.auth.app.id = githubSecrets.appId; let appOctokit; try { appOctokit = new core_1.Octokit({ baseUrl, authStrategy: auth_app_1.createAppAuth, auth: { appId: githubSecrets.appId, privateKey: privateKey, }, }); } catch (e) { status.github.auth.status = `Unable to authenticate app: ${e}`; return safeReturnValue(event, status); } // get app url try { const app = (await appOctokit.request('GET /app')).data; status.github.auth.app.url = app.html_url; } catch (e) { status.github.auth.status = `Unable to get app details: ${e}`; return safeReturnValue(event, status); } // list all app installations try { const installations = (await appOctokit.request('GET /app/installations')).data; for (const installation of installations) { let installationDetails = { id: installation.id, url: installation.html_url, status: 'Unable to query', repositories: [], }; let token; try { token = (await appOctokit.auth({ type: 'installation', installationId: installation.id, })).token; } catch (e) { installationDetails.status = `Unable to authenticate app installation: ${e}`; continue; } let octokit; try { octokit = new core_1.Octokit({ baseUrl, auth: token }); } catch (e) { installationDetails.status = `Unable to authenticate using app: ${e}`; continue; } try { const repositories = (await octokit.request('GET /installation/repositories')).data.repositories; for (const repo of repositories) { installationDetails.repositories.push(repo.full_name); } } catch (e) { installationDetails.status = `Unable to authenticate using installation token: ${e}`; continue; } installationDetails.status = 'OK'; status.github.auth.app.installations.push(installationDetails); } } catch (e) { status.github.auth.status = 'Unable to list app installations'; return safeReturnValue(event, status); } status.github.auth.status = 'OK'; // check webhook config try { const response = await appOctokit.request('GET /app/hook/config', {}); if (response.data.url !== process.env.WEBHOOK_URL) { status.github.webhook.status = 'GitHub has wrong webhook URL configured'; } else { // TODO check secret by doing a dummy delivery? force apply secret? status.github.webhook.status = 'OK (note that secret cannot be checked automatically)'; } } catch (e) { status.github.webhook.status = `Unable to check app configuration: ${e}`; return safeReturnValue(event, status); } } return safeReturnValue(event, status); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.lambda.js","sourceRoot":"","sources":["../src/status.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { CloudFormationClient, DescribeStackResourceCommand } from '@aws-sdk/client-cloudformation';\nimport { DescribeLaunchTemplateVersionsCommand, EC2Client } from '@aws-sdk/client-ec2';\nimport { ECRClient, DescribeImagesCommand } from '@aws-sdk/client-ecr';\nimport { DescribeExecutionCommand, ListExecutionsCommand, SFNClient } from '@aws-sdk/client-sfn';\nimport { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/core';\nimport * as AWSLambda from 'aws-lambda';\nimport { baseUrlFromDomain, GitHubSecrets } from './lambda-github';\nimport { getSecretJsonValue, getSecretValue } from './lambda-helpers';\n\nconst cfn = new CloudFormationClient();\nconst ec2 = new EC2Client();\nconst ecr = new ECRClient();\nconst sf = new SFNClient();\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 lambdaArnToLogGroup(arn: string) {\n  const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ\n  const name = parts[6];\n\n  return `/aws/lambda/${name}`;\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.send(new DescribeStackResourceCommand({ StackName: stack, LogicalResourceId: logicalId }));\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.send(new DescribeImagesCommand({\n        repositoryName: p.image.imageRepository.split('/')[1],\n        filter: {\n          tagStatus: 'TAGGED',\n        },\n        maxResults: 1,\n      }));\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.send(new DescribeLaunchTemplateVersionsCommand({\n        LaunchTemplateId: p.ami.launchTemplate,\n        Versions: ['$Default'],\n      }));\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 jobId?: 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\nexport async function handler(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 ||\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      runnerLevel: '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: -1,\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      webhookHandlerLogGroup: lambdaArnToLogGroup(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  if (process.env.SETUP_FUNCTION_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  } else {\n    status.github.setup.status = 'Disabled';\n  }\n\n  // list last 10 executions and their status\n  try {\n    const executions = await sf.send(new ListExecutionsCommand({\n      stateMachineArn: process.env.STEP_FUNCTION_ARN,\n      maxResults: 10,\n    }));\n    for (const execution of executions.executions ?? []) {\n      const executionDetails = await sf.send(new DescribeExecutionCommand({\n        executionArn: execution.executionArn,\n      }));\n      const input = JSON.parse(executionDetails.input || '{}');\n\n      status.troubleshooting.recentRuns.push({\n        executionArn: execution.executionArn,\n        status: execution.status ?? '<unknown>',\n        owner: input.owner,\n        repo: input.repo,\n        jobId: input.jobId,\n      });\n    }\n  } catch (e) {\n    status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` });\n  }\n\n  // get secrets\n  let githubSecrets: 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  // copy runner level\n  status.github.runnerLevel = githubSecrets.runnerLevel ?? 'repo';\n\n  if (githubSecrets.personalAuthToken) {\n    // try authenticating with personal access token\n    status.github.auth.type = 'Personal Access 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"]}