@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.
87 lines • 14.8 kB
JavaScript
;
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(`Root device for ${ami} is ${rootDevice}`);
await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', rootDevice, {});
return;
}
async function handler(event, context) {
try {
console.log({ ...event, ResponseURL: '...' });
const ami = event.ResourceProperties.Ami;
switch (event.RequestType) {
case 'Create':
case 'Update':
if (ami.startsWith('ami-')) {
console.log(`Checking AMI ${ami}`);
await handleAmi(event, ami);
break;
}
if (ami.startsWith('resolve:ssm:')) {
const ssmParam = ami.substring('resolve:ssm:'.length);
console.log(`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(`Checking Launch Template ${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(`Checking Image Builder ${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('Nothing to delete');
await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});
break;
}
}
catch (e) {
console.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(`Root device for ${ami} is ${rootDevice}`);\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({ ...event, ResponseURL: '...' });\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(`Checking AMI ${ami}`);\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(`Checking SSM ${ssmParam}`);\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(`Checking Launch Template ${ami}`);\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(`Checking Image Builder ${ami}`);\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('Nothing to delete');\n        await customResourceRespond(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});\n        break;\n    }\n  } catch (e) {\n    console.error(e);\n    await customResourceRespond(event, 'FAILED', (e as Error).message || 'Internal Error', context.logStreamName, {});\n  }\n}\n"]}