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.

127 lines 18.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const client_ec2_1 = require("@aws-sdk/client-ec2"); const client_imagebuilder_1 = require("@aws-sdk/client-imagebuilder"); const client_ssm_1 = require("@aws-sdk/client-ssm"); const lambda_helpers_1 = require("../lambda-helpers"); const ssm = new client_ssm_1.SSMClient(); const ec2 = new client_ec2_1.EC2Client(); const ib = new client_imagebuilder_1.ImagebuilderClient(); async function handleAmi(event, ami) { const imageDescs = (await ec2.send(new client_ec2_1.DescribeImagesCommand({ ImageIds: [ami] }))); if (imageDescs.Images?.length !== 1) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} doesn't exist`, 'ERROR', {}); return; } const rootDevice = imageDescs.Images[0].RootDeviceName; if (!rootDevice) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} has no root device`, 'ERROR', {}); return; } console.log({ notice: 'Resolved AMI root device', ami, rootDevice, }); await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', rootDevice, {}); return; } async function handler(event, context) { try { console.log({ notice: 'CloudFormation custom resource request', ...event, ResponseURL: '...', }); const ami = event.ResourceProperties.Ami; switch (event.RequestType) { case 'Create': case 'Update': if (ami.startsWith('ami-')) { console.log({ notice: 'Checking AMI', ami, }); await handleAmi(event, ami); break; } if (ami.startsWith('resolve:ssm:')) { const ssmParam = ami.substring('resolve:ssm:'.length); console.log({ notice: 'Checking SSM', ssmParam, }); const ssmValue = (await ssm.send(new client_ssm_1.GetParameterCommand({ Name: ssmParam }))).Parameter?.Value; if (!ssmValue) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} has no value`, 'ERROR', {}); break; } await handleAmi(event, ssmValue); break; } if (ami.startsWith('ssm:')) { const ssmParam = ami.substring('ssm:'.length); console.log({ notice: 'Checking SSM', ssmParam, }); const ssmValue = (await ssm.send(new client_ssm_1.GetParameterCommand({ Name: ssmParam }))).Parameter?.Value; if (!ssmValue) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} has no value`, 'ERROR', {}); break; } await handleAmi(event, ssmValue); break; } if (ami.startsWith('lt-')) { console.log({ notice: 'Checking Launch Template', launchTemplateId: ami, }); const lts = await ec2.send(new client_ec2_1.DescribeLaunchTemplateVersionsCommand({ LaunchTemplateId: ami, Versions: ['$Latest'] })); if (lts.LaunchTemplateVersions?.length !== 1) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} doesn't exist`, 'ERROR', {}); break; } if (!lts.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} doesn't have an AMI`, 'ERROR', {}); break; } await handleAmi(event, lts.LaunchTemplateVersions[0].LaunchTemplateData.ImageId); break; } if (ami.match('^arn:aws[^:]*:imagebuilder:[^:]+:[^:]+:image/.*$')) { console.log({ notice: 'Checking Image Builder', imageBuildVersionArn: ami, }); const img = await ib.send(new client_imagebuilder_1.GetImageCommand({ imageBuildVersionArn: ami })); const actualAmi = img.image?.outputResources?.amis?.[0]?.image; if (!actualAmi) { await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `${ami} doesn't have an AMI`, 'ERROR', {}); break; } await handleAmi(event, actualAmi); break; } await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', `Unknown type of AMI ${ami}`, 'ERROR', {}); break; case 'Delete': console.log({ notice: 'Nothing to delete', ami, }); await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {}); break; } } catch (e) { console.error({ notice: 'Failed to resolve AMI root device', error: `${e}`, }); await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', e.message || 'Internal Error', context.logStreamName, {}); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ami-root-device.lambda.js","sourceRoot":"","sources":["../../src/providers/ami-root-device.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { DescribeImagesCommand, DescribeLaunchTemplateVersionsCommand, EC2Client } from '@aws-sdk/client-ec2';\nimport { GetImageCommand, ImagebuilderClient } from '@aws-sdk/client-imagebuilder';\nimport { GetParameterCommand, SSMClient } from '@aws-sdk/client-ssm';\nimport * as AWSLambda from 'aws-lambda';\nimport { customResourceRespond } from '../lambda-helpers';\n\nconst ssm = new SSMClient();\nconst ec2 = new EC2Client();\nconst ib = new ImagebuilderClient();\n\n\nasync function handleAmi(event: AWSLambda.CloudFormationCustomResourceEvent, ami: string) {\n  const imageDescs = (await ec2.send(new DescribeImagesCommand({ ImageIds: [ami] })));\n  if (imageDescs.Images?.length !== 1) {\n    await customResourceRespond(event, 'FAILED', `${ami} doesn't exist`, 'ERROR', {});\n    return;\n  }\n\n  const rootDevice = imageDescs.Images[0].RootDeviceName;\n  if (!rootDevice) {\n    await customResourceRespond(event, 'FAILED', `${ami} has no root device`, 'ERROR', {});\n    return;\n  }\n\n  console.log({\n    notice: 'Resolved AMI root device',\n    ami,\n    rootDevice,\n  });\n\n  await customResourceRespond(event, 'SUCCESS', 'OK', rootDevice, {});\n  return;\n}\n\n\nexport async function handler(event: AWSLambda.CloudFormationCustomResourceEvent, context: AWSLambda.Context) {\n  try {\n    console.log({\n      notice: 'CloudFormation custom resource request',\n      ...event,\n      ResponseURL: '...',\n    });\n\n    const ami = event.ResourceProperties.Ami as string;\n\n    switch (event.RequestType) {\n      case 'Create':\n      case 'Update':\n        if (ami.startsWith('ami-')) {\n          console.log({\n            notice: 'Checking AMI',\n            ami,\n          });\n\n          await handleAmi(event, ami);\n          break;\n        }\n\n        if (ami.startsWith('resolve:ssm:')) {\n          const ssmParam = ami.substring('resolve:ssm:'.length);\n          console.log({\n            notice: 'Checking SSM',\n            ssmParam,\n          });\n\n          const ssmValue = (await ssm.send(new GetParameterCommand({ Name: ssmParam }))).Parameter?.Value;\n          if (!ssmValue) {\n            await customResourceRespond(event, 'FAILED', `${ami} has no value`, 'ERROR', {});\n            break;\n          }\n\n          await handleAmi(event, ssmValue);\n          break;\n        }\n\n        if (ami.startsWith('ssm:')) {\n          const ssmParam = ami.substring('ssm:'.length);\n          console.log({\n            notice: 'Checking SSM',\n            ssmParam,\n          });\n\n          const ssmValue = (await ssm.send(new GetParameterCommand({ Name: ssmParam }))).Parameter?.Value;\n          if (!ssmValue) {\n            await customResourceRespond(event, 'FAILED', `${ami} has no value`, 'ERROR', {});\n            break;\n          }\n\n          await handleAmi(event, ssmValue);\n          break;\n        }\n\n        if (ami.startsWith('lt-')) {\n          console.log({\n            notice: 'Checking Launch Template',\n            launchTemplateId: ami,\n          });\n\n          const lts = await ec2.send(new DescribeLaunchTemplateVersionsCommand({ LaunchTemplateId: ami, Versions: ['$Latest'] }));\n          if (lts.LaunchTemplateVersions?.length !== 1) {\n            await customResourceRespond(event, 'FAILED', `${ami} doesn't exist`, 'ERROR', {});\n            break;\n          }\n\n          if (!lts.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId) {\n            await customResourceRespond(event, 'FAILED', `${ami} doesn't have an AMI`, 'ERROR', {});\n            break;\n          }\n\n          await handleAmi(event, lts.LaunchTemplateVersions[0].LaunchTemplateData.ImageId);\n          break;\n        }\n\n        if (ami.match('^arn:aws[^:]*:imagebuilder:[^:]+:[^:]+:image/.*$')) {\n          console.log({\n            notice: 'Checking Image Builder',\n            imageBuildVersionArn: ami,\n          });\n\n          const img = await ib.send(new GetImageCommand({ imageBuildVersionArn: ami }));\n          const actualAmi = img.image?.outputResources?.amis?.[0]?.image;\n          if (!actualAmi) {\n            await customResourceRespond(event, 'FAILED', `${ami} doesn't have an AMI`, 'ERROR', {});\n            break;\n          }\n\n          await handleAmi(event, actualAmi);\n          break;\n        }\n\n        await customResourceRespond(event, 'FAILED', `Unknown type of AMI ${ami}`, 'ERROR', {});\n        break;\n      case 'Delete':\n        console.log({\n          notice: 'Nothing to delete',\n          ami,\n        });\n        await customResourceRespond(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});\n        break;\n    }\n  } catch (e) {\n    console.error({\n      notice: 'Failed to resolve AMI root device',\n      error: `${e}`,\n    });\n    await customResourceRespond(event, 'FAILED', (e as Error).message || 'Internal Error', context.logStreamName, {});\n  }\n}\n"]}