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.

151 lines 17.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const client_ec2_1 = require("@aws-sdk/client-ec2"); const client_ecr_1 = require("@aws-sdk/client-ecr"); const client_imagebuilder_1 = require("@aws-sdk/client-imagebuilder"); const lambda_helpers_1 = require("../../lambda-helpers"); const ec2 = new client_ec2_1.EC2Client(); const ecr = new client_ecr_1.ECRClient(); const ib = new client_imagebuilder_1.ImagebuilderClient(); async function deleteResources(props) { const buildsToDelete = []; const amisToDelete = []; const dockerImagesToDelete = []; let result = {}; do { result = await ib.send(new client_imagebuilder_1.ListImageBuildVersionsCommand({ imageVersionArn: props.ImageVersionArn, nextToken: result.nextToken, })); if (result.imageSummaryList) { for (const image of result.imageSummaryList) { if (image.arn) { buildsToDelete.push(image.arn); } for (const output of image.outputResources?.amis ?? []) { if (output.image) { amisToDelete.push(output.image); } } for (const output of image.outputResources?.containers ?? []) { if (output.imageUris) { dockerImagesToDelete.push(...output.imageUris); } } } } } while (result.nextToken); // delete amis for (const imageId of amisToDelete) { try { console.log({ notice: 'Deleting AMI', image: imageId, }); const imageDesc = await ec2.send(new client_ec2_1.DescribeImagesCommand({ Owners: ['self'], ImageIds: [imageId], })); if (imageDesc.Images?.length !== 1) { console.warn({ notice: 'Unable to find AMI', image: imageId, }); continue; } await ec2.send(new client_ec2_1.DeregisterImageCommand({ ImageId: imageId, DeleteAssociatedSnapshots: true, })); } catch (e) { console.warn({ notice: 'Failed to delete AMI', image: imageId, error: e, }); } } // delete docker images for (const image of dockerImagesToDelete) { try { console.log({ notice: 'Deleting Docker Image', image, }); // image looks like 0123456789.dkr.ecr.us-east-1.amazonaws.com/github-runners-test-windowsimagebuilderrepositorya4cbb6d8-hehdl99r7s3d:1.0.10-1 const parts = image.split('/')[1].split(':'); const repo = parts[0]; const tag = parts[1]; // delete image await ecr.send(new client_ecr_1.BatchDeleteImageCommand({ repositoryName: repo, imageIds: [ { imageTag: tag, }, ], })); } catch (e) { console.warn({ notice: 'Failed to delete docker image', image, error: e, }); } } // delete builds (last so retries would still work) for (const build of buildsToDelete) { try { console.log({ notice: 'Deleting Image Build', build, }); await ib.send(new client_imagebuilder_1.DeleteImageCommand({ imageBuildVersionArn: build, })); } catch (e) { console.warn({ notice: 'Failed to delete image version build', build, error: e, }); } } } async function handler(event, _context) { try { console.log({ notice: 'CloudFormation custom resource request', ...event, ResponseURL: '...', }); const props = event.ResourceProperties; switch (event.RequestType) { case 'Create': case 'Update': // we just return the arn as the physical id // this way a change in the version will trigger delete of the old version on cleanup of stack // it will also trigger delete on stack deletion await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', props.ImageVersionArn, {}); break; case 'Delete': if (event.PhysicalResourceId != 'FAIL') { await deleteResources(props); } await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {}); break; } } catch (e) { console.error({ notice: 'Failed to delete Image Builder resources', error: `${e}`, }); await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', e.message || 'Internal Error', 'FAIL', {}); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delete-resources.lambda.js","sourceRoot":"","sources":["../../../src/image-builders/aws-image-builder/delete-resources.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { DeregisterImageCommand, DescribeImagesCommand, EC2Client } from '@aws-sdk/client-ec2';\nimport { BatchDeleteImageCommand, ECRClient } from '@aws-sdk/client-ecr';\nimport { DeleteImageCommand, ImagebuilderClient, ListImageBuildVersionsCommand, ListImageBuildVersionsResponse } from '@aws-sdk/client-imagebuilder';\nimport * as AWSLambda from 'aws-lambda';\nimport { customResourceRespond } from '../../lambda-helpers';\n\nconst ec2 = new EC2Client();\nconst ecr = new ECRClient();\nconst ib = new ImagebuilderClient();\n\n/**\n * @internal\n */\nexport interface DeleteResourcesProps {\n  ServiceToken: string;\n  ImageVersionArn: string;\n}\n\nasync function deleteResources(props: DeleteResourcesProps) {\n  const buildsToDelete: string[] = [];\n  const amisToDelete: string[] = [];\n  const dockerImagesToDelete: string[] = [];\n\n  let result: ListImageBuildVersionsResponse = {};\n  do {\n    result = await ib.send(new ListImageBuildVersionsCommand({\n      imageVersionArn: props.ImageVersionArn,\n      nextToken: result.nextToken,\n    }));\n    if (result.imageSummaryList) {\n      for (const image of result.imageSummaryList) {\n        if (image.arn) {\n          buildsToDelete.push(image.arn);\n        }\n        for (const output of image.outputResources?.amis ?? []) {\n          if (output.image) {\n            amisToDelete.push(output.image);\n          }\n        }\n        for (const output of image.outputResources?.containers ?? []) {\n          if (output.imageUris) {\n            dockerImagesToDelete.push(...output.imageUris);\n          }\n        }\n      }\n    }\n  } while (result.nextToken);\n\n  // delete amis\n  for (const imageId of amisToDelete) {\n    try {\n      console.log({\n        notice: 'Deleting AMI',\n        image: imageId,\n      });\n\n      const imageDesc = await ec2.send(new DescribeImagesCommand({\n        Owners: ['self'],\n        ImageIds: [imageId],\n      }));\n\n      if (imageDesc.Images?.length !== 1) {\n        console.warn({\n          notice: 'Unable to find AMI',\n          image: imageId,\n        });\n        continue;\n      }\n\n      await ec2.send(new DeregisterImageCommand({\n        ImageId: imageId,\n        DeleteAssociatedSnapshots: true,\n      }));\n    } catch (e) {\n      console.warn({\n        notice: 'Failed to delete AMI',\n        image: imageId,\n        error: e,\n      });\n    }\n  }\n\n  // delete docker images\n  for (const image of dockerImagesToDelete) {\n    try {\n      console.log({\n        notice: 'Deleting Docker Image',\n        image,\n      });\n\n      // image looks like 0123456789.dkr.ecr.us-east-1.amazonaws.com/github-runners-test-windowsimagebuilderrepositorya4cbb6d8-hehdl99r7s3d:1.0.10-1\n      const parts = image.split('/')[1].split(':');\n      const repo = parts[0];\n      const tag = parts[1];\n\n      // delete image\n      await ecr.send(new BatchDeleteImageCommand({\n        repositoryName: repo,\n        imageIds: [\n          {\n            imageTag: tag,\n          },\n        ],\n      }));\n    } catch (e) {\n      console.warn({\n        notice: 'Failed to delete docker image',\n        image,\n        error: e,\n      });\n    }\n  }\n\n  // delete builds (last so retries would still work)\n  for (const build of buildsToDelete) {\n    try {\n      console.log({\n        notice: 'Deleting Image Build',\n        build,\n      });\n\n      await ib.send(new DeleteImageCommand({\n        imageBuildVersionArn: build,\n      }));\n    } catch (e) {\n      console.warn({\n        notice: 'Failed to delete image version build',\n        build,\n        error: e,\n      });\n    }\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 props = event.ResourceProperties as DeleteResourcesProps;\n\n    switch (event.RequestType) {\n      case 'Create':\n      case 'Update':\n        // we just return the arn as the physical id\n        // this way a change in the version will trigger delete of the old version on cleanup of stack\n        // it will also trigger delete on stack deletion\n        await customResourceRespond(event, 'SUCCESS', 'OK', props.ImageVersionArn, {});\n        break;\n      case 'Delete':\n        if (event.PhysicalResourceId != 'FAIL') {\n          await deleteResources(props);\n        }\n        await customResourceRespond(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});\n        break;\n    }\n  } catch (e) {\n    console.error({\n      notice: 'Failed to delete Image Builder resources',\n      error: `${e}`,\n    });\n    await customResourceRespond(event, 'FAILED', (e as Error).message || 'Internal Error', 'FAIL', {});\n  }\n}\n"]}