@cloudsnorkel/cdk-github-runners 0.12.4 → 0.13.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (53) hide show
  1. package/.jsii +340 -202
  2. package/API.md +21 -1
  3. package/assets/idle-runner-repear.lambda/index.js +44 -11
  4. package/assets/image-builders/aws-image-builder/delete-ami.lambda/index.js +9 -4
  5. package/assets/image-builders/aws-image-builder/filter-failed-builds.lambda/index.js +1 -1
  6. package/assets/image-builders/aws-image-builder/versioner.lambda/index.js +1 -1
  7. package/assets/image-builders/build-image.lambda/index.js +11 -27
  8. package/assets/providers/ami-root-device.lambda/index.js +1 -1
  9. package/assets/providers/update-lambda.lambda/index.js +1 -1
  10. package/assets/setup.lambda/index.html +1 -1
  11. package/assets/webhook-handler.lambda/index.js +26 -10
  12. package/lib/access.js +1 -1
  13. package/lib/idle-runner-repear.lambda.js +45 -12
  14. package/lib/image-builders/api.js +3 -3
  15. package/lib/image-builders/aws-image-builder/ami.js +9 -1
  16. package/lib/image-builders/aws-image-builder/builder.d.ts +2 -1
  17. package/lib/image-builders/aws-image-builder/builder.js +24 -11
  18. package/lib/image-builders/aws-image-builder/common.js +3 -2
  19. package/lib/image-builders/aws-image-builder/container.d.ts +2 -4
  20. package/lib/image-builders/aws-image-builder/container.js +6 -2
  21. package/lib/image-builders/aws-image-builder/delete-ami.lambda.js +10 -5
  22. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  23. package/lib/image-builders/aws-image-builder/deprecated/common.js +2 -2
  24. package/lib/image-builders/aws-image-builder/deprecated/container.js +3 -3
  25. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  26. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  27. package/lib/image-builders/aws-image-builder/filter-failed-builds.lambda.js +2 -2
  28. package/lib/image-builders/aws-image-builder/versioner.lambda.js +2 -2
  29. package/lib/image-builders/build-image.lambda.d.ts +11 -0
  30. package/lib/image-builders/build-image.lambda.js +12 -30
  31. package/lib/image-builders/codebuild-deprecated.js +1 -1
  32. package/lib/image-builders/codebuild.d.ts +1 -0
  33. package/lib/image-builders/codebuild.js +52 -27
  34. package/lib/image-builders/common.d.ts +10 -0
  35. package/lib/image-builders/common.js +1 -1
  36. package/lib/image-builders/components.js +12 -4
  37. package/lib/image-builders/static.js +1 -1
  38. package/lib/providers/ami-root-device.lambda.js +2 -2
  39. package/lib/providers/codebuild.js +4 -4
  40. package/lib/providers/common.d.ts +5 -1
  41. package/lib/providers/common.js +12 -7
  42. package/lib/providers/ec2.js +2 -2
  43. package/lib/providers/ecs.js +6 -6
  44. package/lib/providers/fargate.js +6 -6
  45. package/lib/providers/lambda.js +14 -11
  46. package/lib/providers/update-lambda.lambda.js +2 -2
  47. package/lib/runner.js +44 -19
  48. package/lib/secrets.js +1 -1
  49. package/lib/utils.d.ts +19 -1
  50. package/lib/utils.js +32 -2
  51. package/lib/webhook-handler.lambda.js +27 -11
  52. package/lib/webhook.js +5 -3
  53. package/package.json +18 -18
@@ -8,49 +8,31 @@ const codebuild = new client_codebuild_1.CodeBuildClient();
8
8
  const ib = new client_imagebuilder_1.ImagebuilderClient();
9
9
  async function handler(event, context) {
10
10
  try {
11
- console.log(JSON.stringify({ ...event, ResponseURL: '...' }));
12
- const deleteOnly = event.ResourceProperties.DeleteOnly;
13
- const projectName = event.ResourceProperties.ProjectName;
14
- const ibName = event.ResourceProperties.ImageBuilderName;
15
- // let physicalResourceId: string;
16
- // let data: { [key: string]: string } = {};
11
+ console.log({ ...event, ResponseURL: '...' });
12
+ const props = event.ResourceProperties;
17
13
  switch (event.RequestType) {
18
14
  case 'Create':
19
15
  case 'Update':
20
- if (deleteOnly) {
16
+ if (props.DeleteOnly) {
21
17
  await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', 'Deleter', {});
22
18
  break;
23
19
  }
24
- console.log(`Starting CodeBuild project ${projectName}`);
25
- await codebuild.send(new client_codebuild_1.StartBuildCommand({
26
- projectName,
20
+ console.log(`Starting CodeBuild project ${props.ProjectName}`);
21
+ const cbRes = await codebuild.send(new client_codebuild_1.StartBuildCommand({
22
+ projectName: props.ProjectName,
27
23
  environmentVariablesOverride: [
28
24
  {
29
25
  type: 'PLAINTEXT',
30
- name: 'STACK_ID',
31
- value: event.StackId,
32
- },
33
- {
34
- type: 'PLAINTEXT',
35
- name: 'REQUEST_ID',
36
- value: event.RequestId,
37
- },
38
- {
39
- type: 'PLAINTEXT',
40
- name: 'LOGICAL_RESOURCE_ID',
41
- value: event.LogicalResourceId,
42
- },
43
- {
44
- type: 'PLAINTEXT',
45
- name: 'RESPONSE_URL',
46
- value: event.ResponseURL,
26
+ name: 'WAIT_HANDLE',
27
+ value: props.WaitHandle,
47
28
  },
48
29
  ],
49
30
  }));
31
+ await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', cbRes.build?.id ?? 'build', {});
50
32
  break;
51
33
  case 'Delete':
52
- if (ibName) {
53
- const ibImages = await ib.send(new client_imagebuilder_1.ListImagesCommand({ filters: [{ name: 'name', values: [ibName] }] }));
34
+ if (props.ImageBuilderName) {
35
+ const ibImages = await ib.send(new client_imagebuilder_1.ListImagesCommand({ filters: [{ name: 'name', values: [props.ImageBuilderName] }] }));
54
36
  if (ibImages.imageVersionList) {
55
37
  for (const v of ibImages.imageVersionList) {
56
38
  if (v.arn) {
@@ -77,4 +59,4 @@ async function handler(event, context) {
77
59
  }
78
60
  }
79
61
  exports.handler = handler;
80
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQtaW1hZ2UubGFtYmRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2ltYWdlLWJ1aWxkZXJzL2J1aWxkLWltYWdlLmxhbWJkYS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnRUFBK0U7QUFDL0Usc0VBS3NDO0FBRXRDLHNEQUEwRDtBQUUxRCxNQUFNLFNBQVMsR0FBRyxJQUFJLGtDQUFlLEVBQUUsQ0FBQztBQUN4QyxNQUFNLEVBQUUsR0FBRyxJQUFJLHdDQUFrQixFQUFFLENBQUM7QUFFN0IsS0FBSyxVQUFVLE9BQU8sQ0FBQyxLQUFrRCxFQUFFLE9BQTBCO0lBQzFHLElBQUksQ0FBQztRQUNILE9BQU8sQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxFQUFFLEdBQUcsS0FBSyxFQUFFLFdBQVcsRUFBRSxLQUFLLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFFOUQsTUFBTSxVQUFVLEdBQUcsS0FBSyxDQUFDLGtCQUFrQixDQUFDLFVBQWlDLENBQUM7UUFDOUUsTUFBTSxXQUFXLEdBQUcsS0FBSyxDQUFDLGtCQUFrQixDQUFDLFdBQVcsQ0FBQztRQUN6RCxNQUFNLE1BQU0sR0FBRyxLQUFLLENBQUMsa0JBQWtCLENBQUMsZ0JBQXNDLENBQUM7UUFFL0Usa0NBQWtDO1FBQ2xDLDRDQUE0QztRQUU1QyxRQUFRLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUMxQixLQUFLLFFBQVEsQ0FBQztZQUNkLEtBQUssUUFBUTtnQkFDWCxJQUFJLFVBQVUsRUFBRSxDQUFDO29CQUNmLE1BQU0sSUFBQSxzQ0FBcUIsRUFBQyxLQUFLLEVBQUUsU0FBUyxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsRUFBRSxDQUFDLENBQUM7b0JBQ25FLE1BQU07Z0JBQ1IsQ0FBQztnQkFFRCxPQUFPLENBQUMsR0FBRyxDQUFDLDhCQUE4QixXQUFXLEVBQUUsQ0FBQyxDQUFDO2dCQUN6RCxNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxvQ0FBaUIsQ0FBQztvQkFDekMsV0FBVztvQkFDWCw0QkFBNEIsRUFBRTt3QkFDNUI7NEJBQ0UsSUFBSSxFQUFFLFdBQVc7NEJBQ2pCLElBQUksRUFBRSxVQUFVOzRCQUNoQixLQUFLLEVBQUUsS0FBSyxDQUFDLE9BQU87eUJBQ3JCO3dCQUNEOzRCQUNFLElBQUksRUFBRSxXQUFXOzRCQUNqQixJQUFJLEVBQUUsWUFBWTs0QkFDbEIsS0FBSyxFQUFFLEtBQUssQ0FBQyxTQUFTO3lCQUN2Qjt3QkFDRDs0QkFDRSxJQUFJLEVBQUUsV0FBVzs0QkFDakIsSUFBSSxFQUFFLHFCQUFxQjs0QkFDM0IsS0FBSyxFQUFFLEtBQUssQ0FBQyxpQkFBaUI7eUJBQy9CO3dCQUNEOzRCQUNFLElBQUksRUFBRSxXQUFXOzRCQUNqQixJQUFJLEVBQUUsY0FBYzs0QkFDcEIsS0FBSyxFQUFFLEtBQUssQ0FBQyxXQUFXO3lCQUN6QjtxQkFDRjtpQkFDRixDQUFDLENBQUMsQ0FBQztnQkFDSixNQUFNO1lBQ1IsS0FBSyxRQUFRO2dCQUNYLElBQUksTUFBTSxFQUFFLENBQUM7b0JBQ1gsTUFBTSxRQUFRLEdBQUcsTUFBTSxFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksdUNBQW1CLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO29CQUMzRyxJQUFJLFFBQVEsQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO3dCQUM5QixLQUFLLE1BQU0sQ0FBQyxJQUFJLFFBQVEsQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDOzRCQUMxQyxJQUFJLENBQUMsQ0FBQyxHQUFHLEVBQUUsQ0FBQztnQ0FDVixNQUFNLGVBQWUsR0FBRyxNQUFNLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSxtREFBNkIsQ0FBQyxFQUFFLGVBQWUsRUFBRSxDQUFDLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDO2dDQUNyRyxJQUFJLGVBQWUsQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO29DQUNyQyxLQUFLLE1BQU0sRUFBRSxJQUFJLGVBQWUsQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO3dDQUNsRCxJQUFJLEVBQUUsQ0FBQyxHQUFHLEVBQUUsQ0FBQzs0Q0FDWCxPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksRUFBRSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUM7NENBQ2xDLE1BQU0sRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLHdDQUFrQixDQUFDLEVBQUUsb0JBQW9CLEVBQUUsRUFBRSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsQ0FBQzt3Q0FDMUUsQ0FBQztvQ0FDSCxDQUFDO2dDQUNILENBQUM7NEJBQ0gsQ0FBQzt3QkFDSCxDQUFDO29CQUNILENBQUM7Z0JBQ0gsQ0FBQztnQkFDRCxNQUFNLElBQUEsc0NBQXFCLEVBQUMsS0FBSyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsS0FBSyxDQUFDLGtCQUFrQixFQUFFLEVBQUUsQ0FBQyxDQUFDO2dCQUNsRixNQUFNO1FBQ1YsQ0FBQztJQUNILENBQUM7SUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ1gsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNqQixNQUFNLElBQUEsc0NBQXFCLEVBQUMsS0FBSyxFQUFFLFFBQVEsRUFBRyxDQUFXLENBQUMsT0FBTyxJQUFJLGdCQUFnQixFQUFFLE9BQU8sQ0FBQyxhQUFhLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDcEgsQ0FBQztBQUNILENBQUM7QUF4RUQsMEJBd0VDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29kZUJ1aWxkQ2xpZW50LCBTdGFydEJ1aWxkQ29tbWFuZCB9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1jb2RlYnVpbGQnO1xuaW1wb3J0IHtcbiAgRGVsZXRlSW1hZ2VDb21tYW5kLFxuICBJbWFnZWJ1aWxkZXJDbGllbnQsXG4gIExpc3RJbWFnZUJ1aWxkVmVyc2lvbnNDb21tYW5kLFxuICBMaXN0SW1hZ2VzQ29tbWFuZCBhcyBMaXN0SWJJbWFnZXNDb21tYW5kLFxufSBmcm9tICdAYXdzLXNkay9jbGllbnQtaW1hZ2VidWlsZGVyJztcbmltcG9ydCAqIGFzIEFXU0xhbWJkYSBmcm9tICdhd3MtbGFtYmRhJztcbmltcG9ydCB7IGN1c3RvbVJlc291cmNlUmVzcG9uZCB9IGZyb20gJy4uL2xhbWJkYS1oZWxwZXJzJztcblxuY29uc3QgY29kZWJ1aWxkID0gbmV3IENvZGVCdWlsZENsaWVudCgpO1xuY29uc3QgaWIgPSBuZXcgSW1hZ2VidWlsZGVyQ2xpZW50KCk7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBoYW5kbGVyKGV2ZW50OiBBV1NMYW1iZGEuQ2xvdWRGb3JtYXRpb25DdXN0b21SZXNvdXJjZUV2ZW50LCBjb250ZXh0OiBBV1NMYW1iZGEuQ29udGV4dCkge1xuICB0cnkge1xuICAgIGNvbnNvbGUubG9nKEpTT04uc3RyaW5naWZ5KHsgLi4uZXZlbnQsIFJlc3BvbnNlVVJMOiAnLi4uJyB9KSk7XG5cbiAgICBjb25zdCBkZWxldGVPbmx5ID0gZXZlbnQuUmVzb3VyY2VQcm9wZXJ0aWVzLkRlbGV0ZU9ubHkgYXMgYm9vbGVhbiB8IHVuZGVmaW5lZDtcbiAgICBjb25zdCBwcm9qZWN0TmFtZSA9IGV2ZW50LlJlc291cmNlUHJvcGVydGllcy5Qcm9qZWN0TmFtZTtcbiAgICBjb25zdCBpYk5hbWUgPSBldmVudC5SZXNvdXJjZVByb3BlcnRpZXMuSW1hZ2VCdWlsZGVyTmFtZSBhcyBzdHJpbmcgfCB1bmRlZmluZWQ7XG5cbiAgICAvLyBsZXQgcGh5c2ljYWxSZXNvdXJjZUlkOiBzdHJpbmc7XG4gICAgLy8gbGV0IGRhdGE6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIH0gPSB7fTtcblxuICAgIHN3aXRjaCAoZXZlbnQuUmVxdWVzdFR5cGUpIHtcbiAgICAgIGNhc2UgJ0NyZWF0ZSc6XG4gICAgICBjYXNlICdVcGRhdGUnOlxuICAgICAgICBpZiAoZGVsZXRlT25seSkge1xuICAgICAgICAgIGF3YWl0IGN1c3RvbVJlc291cmNlUmVzcG9uZChldmVudCwgJ1NVQ0NFU1MnLCAnT0snLCAnRGVsZXRlcicsIHt9KTtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnNvbGUubG9nKGBTdGFydGluZyBDb2RlQnVpbGQgcHJvamVjdCAke3Byb2plY3ROYW1lfWApO1xuICAgICAgICBhd2FpdCBjb2RlYnVpbGQuc2VuZChuZXcgU3RhcnRCdWlsZENvbW1hbmQoe1xuICAgICAgICAgIHByb2plY3ROYW1lLFxuICAgICAgICAgIGVudmlyb25tZW50VmFyaWFibGVzT3ZlcnJpZGU6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgdHlwZTogJ1BMQUlOVEVYVCcsXG4gICAgICAgICAgICAgIG5hbWU6ICdTVEFDS19JRCcsXG4gICAgICAgICAgICAgIHZhbHVlOiBldmVudC5TdGFja0lkLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgdHlwZTogJ1BMQUlOVEVYVCcsXG4gICAgICAgICAgICAgIG5hbWU6ICdSRVFVRVNUX0lEJyxcbiAgICAgICAgICAgICAgdmFsdWU6IGV2ZW50LlJlcXVlc3RJZCxcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHR5cGU6ICdQTEFJTlRFWFQnLFxuICAgICAgICAgICAgICBuYW1lOiAnTE9HSUNBTF9SRVNPVVJDRV9JRCcsXG4gICAgICAgICAgICAgIHZhbHVlOiBldmVudC5Mb2dpY2FsUmVzb3VyY2VJZCxcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHR5cGU6ICdQTEFJTlRFWFQnLFxuICAgICAgICAgICAgICBuYW1lOiAnUkVTUE9OU0VfVVJMJyxcbiAgICAgICAgICAgICAgdmFsdWU6IGV2ZW50LlJlc3BvbnNlVVJMLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICBdLFxuICAgICAgICB9KSk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnRGVsZXRlJzpcbiAgICAgICAgaWYgKGliTmFtZSkge1xuICAgICAgICAgIGNvbnN0IGliSW1hZ2VzID0gYXdhaXQgaWIuc2VuZChuZXcgTGlzdEliSW1hZ2VzQ29tbWFuZCh7IGZpbHRlcnM6IFt7IG5hbWU6ICduYW1lJywgdmFsdWVzOiBbaWJOYW1lXSB9XSB9KSk7XG4gICAgICAgICAgaWYgKGliSW1hZ2VzLmltYWdlVmVyc2lvbkxpc3QpIHtcbiAgICAgICAgICAgIGZvciAoY29uc3QgdiBvZiBpYkltYWdlcy5pbWFnZVZlcnNpb25MaXN0KSB7XG4gICAgICAgICAgICAgIGlmICh2LmFybikge1xuICAgICAgICAgICAgICAgIGNvbnN0IGliSW1hZ2VWZXJzaW9ucyA9IGF3YWl0IGliLnNlbmQobmV3IExpc3RJbWFnZUJ1aWxkVmVyc2lvbnNDb21tYW5kKHsgaW1hZ2VWZXJzaW9uQXJuOiB2LmFybiB9KSk7XG4gICAgICAgICAgICAgICAgaWYgKGliSW1hZ2VWZXJzaW9ucy5pbWFnZVN1bW1hcnlMaXN0KSB7XG4gICAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IHZzIG9mIGliSW1hZ2VWZXJzaW9ucy5pbWFnZVN1bW1hcnlMaXN0KSB7XG4gICAgICAgICAgICAgICAgICAgIGlmICh2cy5hcm4pIHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhgRGVsZXRpbmcgJHt2cy5hcm59YCk7XG4gICAgICAgICAgICAgICAgICAgICAgYXdhaXQgaWIuc2VuZChuZXcgRGVsZXRlSW1hZ2VDb21tYW5kKHsgaW1hZ2VCdWlsZFZlcnNpb25Bcm46IHZzLmFybiB9KSk7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgYXdhaXQgY3VzdG9tUmVzb3VyY2VSZXNwb25kKGV2ZW50LCAnU1VDQ0VTUycsICdPSycsIGV2ZW50LlBoeXNpY2FsUmVzb3VyY2VJZCwge30pO1xuICAgICAgICBicmVhaztcbiAgICB9XG4gIH0gY2F0Y2ggKGUpIHtcbiAgICBjb25zb2xlLmVycm9yKGUpO1xuICAgIGF3YWl0IGN1c3RvbVJlc291cmNlUmVzcG9uZChldmVudCwgJ0ZBSUxFRCcsIChlIGFzIEVycm9yKS5tZXNzYWdlIHx8ICdJbnRlcm5hbCBFcnJvcicsIGNvbnRleHQubG9nU3RyZWFtTmFtZSwge30pO1xuICB9XG59XG4iXX0=
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQtaW1hZ2UubGFtYmRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2ltYWdlLWJ1aWxkZXJzL2J1aWxkLWltYWdlLmxhbWJkYS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnRUFBK0U7QUFDL0Usc0VBS3NDO0FBRXRDLHNEQUEwRDtBQUUxRCxNQUFNLFNBQVMsR0FBRyxJQUFJLGtDQUFlLEVBQUUsQ0FBQztBQUN4QyxNQUFNLEVBQUUsR0FBRyxJQUFJLHdDQUFrQixFQUFFLENBQUM7QUFjN0IsS0FBSyxVQUFVLE9BQU8sQ0FBQyxLQUFrRCxFQUFFLE9BQTBCO0lBQzFHLElBQUksQ0FBQztRQUNILE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRSxHQUFHLEtBQUssRUFBRSxXQUFXLEVBQUUsS0FBSyxFQUFFLENBQUMsQ0FBQztRQUU5QyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsa0JBQWtELENBQUM7UUFFdkUsUUFBUSxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDMUIsS0FBSyxRQUFRLENBQUM7WUFDZCxLQUFLLFFBQVE7Z0JBQ1gsSUFBSSxLQUFLLENBQUMsVUFBVSxFQUFFLENBQUM7b0JBQ3JCLE1BQU0sSUFBQSxzQ0FBcUIsRUFBQyxLQUFLLEVBQUUsU0FBUyxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsRUFBRSxDQUFDLENBQUM7b0JBQ25FLE1BQU07Z0JBQ1IsQ0FBQztnQkFFRCxPQUFPLENBQUMsR0FBRyxDQUFDLDhCQUE4QixLQUFLLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQztnQkFDL0QsTUFBTSxLQUFLLEdBQUcsTUFBTSxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksb0NBQWlCLENBQUM7b0JBQ3ZELFdBQVcsRUFBRSxLQUFLLENBQUMsV0FBVztvQkFDOUIsNEJBQTRCLEVBQUU7d0JBQzVCOzRCQUNFLElBQUksRUFBRSxXQUFXOzRCQUNqQixJQUFJLEVBQUUsYUFBYTs0QkFDbkIsS0FBSyxFQUFFLEtBQUssQ0FBQyxVQUFXO3lCQUN6QjtxQkFDRjtpQkFDRixDQUFDLENBQUMsQ0FBQztnQkFDSixNQUFNLElBQUEsc0NBQXFCLEVBQUMsS0FBSyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLElBQUksT0FBTyxFQUFFLEVBQUUsQ0FBQyxDQUFDO2dCQUNwRixNQUFNO1lBQ1IsS0FBSyxRQUFRO2dCQUNYLElBQUksS0FBSyxDQUFDLGdCQUFnQixFQUFFLENBQUM7b0JBQzNCLE1BQU0sUUFBUSxHQUFHLE1BQU0sRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLHVDQUFtQixDQUFDLEVBQUUsT0FBTyxFQUFFLENBQUMsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztvQkFDM0gsSUFBSSxRQUFRLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQzt3QkFDOUIsS0FBSyxNQUFNLENBQUMsSUFBSSxRQUFRLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQzs0QkFDMUMsSUFBSSxDQUFDLENBQUMsR0FBRyxFQUFFLENBQUM7Z0NBQ1YsTUFBTSxlQUFlLEdBQUcsTUFBTSxFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksbURBQTZCLENBQUMsRUFBRSxlQUFlLEVBQUUsQ0FBQyxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsQ0FBQztnQ0FDckcsSUFBSSxlQUFlLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztvQ0FDckMsS0FBSyxNQUFNLEVBQUUsSUFBSSxlQUFlLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQzt3Q0FDbEQsSUFBSSxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUM7NENBQ1gsT0FBTyxDQUFDLEdBQUcsQ0FBQyxZQUFZLEVBQUUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDOzRDQUNsQyxNQUFNLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSx3Q0FBa0IsQ0FBQyxFQUFFLG9CQUFvQixFQUFFLEVBQUUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDLENBQUM7d0NBQzFFLENBQUM7b0NBQ0gsQ0FBQztnQ0FDSCxDQUFDOzRCQUNILENBQUM7d0JBQ0gsQ0FBQztvQkFDSCxDQUFDO2dCQUNILENBQUM7Z0JBQ0QsTUFBTSxJQUFBLHNDQUFxQixFQUFDLEtBQUssRUFBRSxTQUFTLEVBQUUsSUFBSSxFQUFFLEtBQUssQ0FBQyxrQkFBa0IsRUFBRSxFQUFFLENBQUMsQ0FBQztnQkFDbEYsTUFBTTtRQUNWLENBQUM7SUFDSCxDQUFDO0lBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztRQUNYLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDakIsTUFBTSxJQUFBLHNDQUFxQixFQUFDLEtBQUssRUFBRSxRQUFRLEVBQUcsQ0FBVyxDQUFDLE9BQU8sSUFBSSxnQkFBZ0IsRUFBRSxPQUFPLENBQUMsYUFBYSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ3BILENBQUM7QUFDSCxDQUFDO0FBckRELDBCQXFEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvZGVCdWlsZENsaWVudCwgU3RhcnRCdWlsZENvbW1hbmQgfSBmcm9tICdAYXdzLXNkay9jbGllbnQtY29kZWJ1aWxkJztcbmltcG9ydCB7XG4gIERlbGV0ZUltYWdlQ29tbWFuZCxcbiAgSW1hZ2VidWlsZGVyQ2xpZW50LFxuICBMaXN0SW1hZ2VCdWlsZFZlcnNpb25zQ29tbWFuZCxcbiAgTGlzdEltYWdlc0NvbW1hbmQgYXMgTGlzdEliSW1hZ2VzQ29tbWFuZCxcbn0gZnJvbSAnQGF3cy1zZGsvY2xpZW50LWltYWdlYnVpbGRlcic7XG5pbXBvcnQgKiBhcyBBV1NMYW1iZGEgZnJvbSAnYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBjdXN0b21SZXNvdXJjZVJlc3BvbmQgfSBmcm9tICcuLi9sYW1iZGEtaGVscGVycyc7XG5cbmNvbnN0IGNvZGVidWlsZCA9IG5ldyBDb2RlQnVpbGRDbGllbnQoKTtcbmNvbnN0IGliID0gbmV3IEltYWdlYnVpbGRlckNsaWVudCgpO1xuXG4vKipcbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEJ1aWxkSW1hZ2VGdW5jdGlvblByb3BlcnRpZXMge1xuICBTZXJ2aWNlVG9rZW46IHN0cmluZztcbiAgRGVsZXRlT25seT86IGJvb2xlYW47XG4gIFJlcG9OYW1lOiBzdHJpbmc7XG4gIFByb2plY3ROYW1lOiBzdHJpbmc7XG4gIEltYWdlQnVpbGRlck5hbWU/OiBzdHJpbmc7XG4gIFdhaXRIYW5kbGU/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBoYW5kbGVyKGV2ZW50OiBBV1NMYW1iZGEuQ2xvdWRGb3JtYXRpb25DdXN0b21SZXNvdXJjZUV2ZW50LCBjb250ZXh0OiBBV1NMYW1iZGEuQ29udGV4dCkge1xuICB0cnkge1xuICAgIGNvbnNvbGUubG9nKHsgLi4uZXZlbnQsIFJlc3BvbnNlVVJMOiAnLi4uJyB9KTtcblxuICAgIGNvbnN0IHByb3BzID0gZXZlbnQuUmVzb3VyY2VQcm9wZXJ0aWVzIGFzIEJ1aWxkSW1hZ2VGdW5jdGlvblByb3BlcnRpZXM7XG5cbiAgICBzd2l0Y2ggKGV2ZW50LlJlcXVlc3RUeXBlKSB7XG4gICAgICBjYXNlICdDcmVhdGUnOlxuICAgICAgY2FzZSAnVXBkYXRlJzpcbiAgICAgICAgaWYgKHByb3BzLkRlbGV0ZU9ubHkpIHtcbiAgICAgICAgICBhd2FpdCBjdXN0b21SZXNvdXJjZVJlc3BvbmQoZXZlbnQsICdTVUNDRVNTJywgJ09LJywgJ0RlbGV0ZXInLCB7fSk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zb2xlLmxvZyhgU3RhcnRpbmcgQ29kZUJ1aWxkIHByb2plY3QgJHtwcm9wcy5Qcm9qZWN0TmFtZX1gKTtcbiAgICAgICAgY29uc3QgY2JSZXMgPSBhd2FpdCBjb2RlYnVpbGQuc2VuZChuZXcgU3RhcnRCdWlsZENvbW1hbmQoe1xuICAgICAgICAgIHByb2plY3ROYW1lOiBwcm9wcy5Qcm9qZWN0TmFtZSxcbiAgICAgICAgICBlbnZpcm9ubWVudFZhcmlhYmxlc092ZXJyaWRlOiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHR5cGU6ICdQTEFJTlRFWFQnLFxuICAgICAgICAgICAgICBuYW1lOiAnV0FJVF9IQU5ETEUnLFxuICAgICAgICAgICAgICB2YWx1ZTogcHJvcHMuV2FpdEhhbmRsZSEsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIF0sXG4gICAgICAgIH0pKTtcbiAgICAgICAgYXdhaXQgY3VzdG9tUmVzb3VyY2VSZXNwb25kKGV2ZW50LCAnU1VDQ0VTUycsICdPSycsIGNiUmVzLmJ1aWxkPy5pZCA/PyAnYnVpbGQnLCB7fSk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnRGVsZXRlJzpcbiAgICAgICAgaWYgKHByb3BzLkltYWdlQnVpbGRlck5hbWUpIHtcbiAgICAgICAgICBjb25zdCBpYkltYWdlcyA9IGF3YWl0IGliLnNlbmQobmV3IExpc3RJYkltYWdlc0NvbW1hbmQoeyBmaWx0ZXJzOiBbeyBuYW1lOiAnbmFtZScsIHZhbHVlczogW3Byb3BzLkltYWdlQnVpbGRlck5hbWVdIH1dIH0pKTtcbiAgICAgICAgICBpZiAoaWJJbWFnZXMuaW1hZ2VWZXJzaW9uTGlzdCkge1xuICAgICAgICAgICAgZm9yIChjb25zdCB2IG9mIGliSW1hZ2VzLmltYWdlVmVyc2lvbkxpc3QpIHtcbiAgICAgICAgICAgICAgaWYgKHYuYXJuKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgaWJJbWFnZVZlcnNpb25zID0gYXdhaXQgaWIuc2VuZChuZXcgTGlzdEltYWdlQnVpbGRWZXJzaW9uc0NvbW1hbmQoeyBpbWFnZVZlcnNpb25Bcm46IHYuYXJuIH0pKTtcbiAgICAgICAgICAgICAgICBpZiAoaWJJbWFnZVZlcnNpb25zLmltYWdlU3VtbWFyeUxpc3QpIHtcbiAgICAgICAgICAgICAgICAgIGZvciAoY29uc3QgdnMgb2YgaWJJbWFnZVZlcnNpb25zLmltYWdlU3VtbWFyeUxpc3QpIHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKHZzLmFybikge1xuICAgICAgICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKGBEZWxldGluZyAke3ZzLmFybn1gKTtcbiAgICAgICAgICAgICAgICAgICAgICBhd2FpdCBpYi5zZW5kKG5ldyBEZWxldGVJbWFnZUNvbW1hbmQoeyBpbWFnZUJ1aWxkVmVyc2lvbkFybjogdnMuYXJuIH0pKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBhd2FpdCBjdXN0b21SZXNvdXJjZVJlc3BvbmQoZXZlbnQsICdTVUNDRVNTJywgJ09LJywgZXZlbnQuUGh5c2ljYWxSZXNvdXJjZUlkLCB7fSk7XG4gICAgICAgIGJyZWFrO1xuICAgIH1cbiAgfSBjYXRjaCAoZSkge1xuICAgIGNvbnNvbGUuZXJyb3IoZSk7XG4gICAgYXdhaXQgY3VzdG9tUmVzb3VyY2VSZXNwb25kKGV2ZW50LCAnRkFJTEVEJywgKGUgYXMgRXJyb3IpLm1lc3NhZ2UgfHwgJ0ludGVybmFsIEVycm9yJywgY29udGV4dC5sb2dTdHJlYW1OYW1lLCB7fSk7XG4gIH1cbn1cbiJdfQ==
@@ -361,7 +361,7 @@ class CodeBuildImageBuilder extends constructs_1.Construct {
361
361
  }
362
362
  exports.CodeBuildImageBuilder = CodeBuildImageBuilder;
363
363
  _a = JSII_RTTI_SYMBOL_1;
364
- CodeBuildImageBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildImageBuilder", version: "0.12.4" };
364
+ CodeBuildImageBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildImageBuilder", version: "0.13.0" };
365
365
  /**
366
366
  * Bump this number every time the buildspec or any important setting of the project changes. It will force a rebuild of the image.
367
367
  * @private
@@ -47,6 +47,7 @@ export declare class CodeBuildRunnerImageBuilder extends RunnerImageBuilderBase
47
47
  private readonly computeType;
48
48
  private readonly rebuildInterval;
49
49
  private readonly role;
50
+ private readonly waitOnDeploy;
50
51
  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps);
51
52
  bindAmi(): RunnerAmi;
52
53
  bindDockerImage(): RunnerImage;
@@ -1,6 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.CodeBuildImageBuilderFailedBuildNotifier = exports.CodeBuildRunnerImageBuilder = void 0;
4
+ const crypto = require("node:crypto");
4
5
  const cdk = require("aws-cdk-lib");
5
6
  const aws_cdk_lib_1 = require("aws-cdk-lib");
6
7
  const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild");
@@ -32,11 +33,16 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
32
33
  this.computeType = props?.codeBuildOptions?.computeType ?? aws_codebuild_1.ComputeType.SMALL;
33
34
  this.baseImage = props?.baseDockerImage ?? (0, aws_image_builder_1.defaultBaseDockerImage)(this.os);
34
35
  this.buildImage = props?.codeBuildOptions?.buildImage ?? this.getDefaultBuildImage();
36
+ this.waitOnDeploy = props?.waitOnDeploy ?? true;
35
37
  // warn against isolated networks
36
38
  if (props?.subnetSelection?.subnetType == aws_cdk_lib_1.aws_ec2.SubnetType.PRIVATE_ISOLATED) {
37
39
  aws_cdk_lib_1.Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +
38
40
  'See https://github.com/aws/containers-roadmap/issues/1160');
39
41
  }
42
+ // check timeout
43
+ if (this.timeout.toSeconds() > aws_cdk_lib_1.Duration.hours(8).toSeconds()) {
44
+ aws_cdk_lib_1.Annotations.of(this).addError('CodeBuild runner image builder timeout must 8 hours or less.');
45
+ }
40
46
  // create service role for CodeBuild
41
47
  this.role = new aws_cdk_lib_1.aws_iam.Role(this, 'Role', {
42
48
  assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('codebuild.amazonaws.com'),
@@ -46,7 +52,7 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
46
52
  imageScanOnPush: true,
47
53
  imageTagMutability: aws_ecr_1.TagMutability.MUTABLE,
48
54
  removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
49
- autoDeleteImages: true,
55
+ emptyOnDelete: true,
50
56
  lifecycleRules: [
51
57
  {
52
58
  description: 'Remove soci indexes for replaced images',
@@ -75,7 +81,7 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
75
81
  removalPolicy: this.logRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.DESTROY,
76
82
  });
77
83
  // generate buildSpec
78
- const buildSpec = this.getBuildSpec(this.repository);
84
+ const [buildSpec, buildSpecHash] = this.getBuildSpec(this.repository);
79
85
  // create CodeBuild project that builds Dockerfile and pushes to repository
80
86
  const project = new aws_cdk_lib_1.aws_codebuild.Project(this, 'CodeBuild', {
81
87
  description: `Build docker image for self-hosted GitHub runner ${this.node.path} (${this.os.name}/${this.architecture.name})`,
@@ -99,7 +105,7 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
99
105
  // permissions
100
106
  this.repository.grantPullPush(project);
101
107
  // call CodeBuild during deployment
102
- const cr = this.customResource(project, buildSpec.toBuildSpec());
108
+ const completedImage = this.customResource(project, buildSpecHash);
103
109
  // rebuild image on a schedule
104
110
  this.rebuildImageOnSchedule(project, this.rebuildInterval);
105
111
  // return the image
@@ -110,12 +116,12 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
110
116
  os: this.os,
111
117
  logGroup,
112
118
  runnerVersion: providers_1.RunnerVersion.specific('unknown'),
113
- _dependable: cr.getAttString('Random'),
119
+ _dependable: completedImage,
114
120
  };
115
121
  return this.boundDockerImage;
116
122
  }
117
123
  getDefaultBuildImage() {
118
- if (this.os.is(providers_1.Os.LINUX_UBUNTU) || this.os.is(providers_1.Os.LINUX_AMAZON_2) || this.os.is(providers_1.Os.LINUX_AMAZON_2023) || this.os.is(providers_1.Os.LINUX)) {
124
+ if (this.os.isIn(providers_1.Os._ALL_LINUX_VERSIONS)) {
119
125
  // CodeBuild just runs `docker build` so its OS doesn't really matter
120
126
  if (this.architecture.is(providers_1.Architecture.X86_64)) {
121
127
  return aws_cdk_lib_1.aws_codebuild.LinuxBuildImage.AMAZON_LINUX_2_5;
@@ -130,6 +136,7 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
130
136
  throw new Error(`Unable to find CodeBuild image for ${this.os.name}/${this.architecture.name}`);
131
137
  }
132
138
  getDockerfileGenerationCommands() {
139
+ let hashedComponents = [];
133
140
  let commands = [];
134
141
  let dockerfile = `FROM ${this.baseImage}\nVOLUME /var/lib/docker\n`;
135
142
  for (let i = 0; i < this.components.length; i++) {
@@ -153,18 +160,22 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
153
160
  throw new Error(`Unknown asset type: ${asset}`);
154
161
  }
155
162
  dockerfile += `COPY asset${i}-${componentName}-${j} ${assetDescriptors[j].target}\n`;
163
+ hashedComponents.push(`__ ASSET FILE ${asset.assetHash} ${i}-${componentName}-${j} ${assetDescriptors[j].target}`);
156
164
  asset.grantRead(this);
157
165
  }
158
166
  const componentCommands = this.components[i].getCommands(this.os, this.architecture);
159
167
  const script = '#!/bin/bash\nset -exuo pipefail\n' + componentCommands.join('\n');
160
168
  commands.push(`cat > component${i}-${componentName}.sh <<'EOFGITHUBRUNNERSDOCKERFILE'\n${script}\nEOFGITHUBRUNNERSDOCKERFILE`);
161
169
  commands.push(`chmod +x component${i}-${componentName}.sh`);
170
+ hashedComponents.push(`__ COMMAND ${i} ${componentName} ${script}`);
162
171
  dockerfile += `COPY component${i}-${componentName}.sh /tmp\n`;
163
172
  dockerfile += `RUN /tmp/component${i}-${componentName}.sh\n`;
164
- dockerfile += this.components[i].getDockerCommands(this.os, this.architecture).join('\n') + '\n';
173
+ const dockerCommands = this.components[i].getDockerCommands(this.os, this.architecture);
174
+ dockerfile += dockerCommands.join('\n') + '\n';
175
+ hashedComponents.push(`__ DOCKER COMMAND ${i} ${dockerCommands.join('\n')}`);
165
176
  }
166
177
  commands.push(`cat > Dockerfile <<'EOFGITHUBRUNNERSDOCKERFILE'\n${dockerfile}\nEOFGITHUBRUNNERSDOCKERFILE`);
167
- return commands;
178
+ return [commands, hashedComponents];
168
179
  }
169
180
  getBuildSpec(repository) {
170
181
  const thisStack = cdk.Stack.of(this);
@@ -178,16 +189,17 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
178
189
  else {
179
190
  throw new Error(`Unsupported architecture for required CodeBuild: ${this.architecture.name}`);
180
191
  }
181
- return aws_cdk_lib_1.aws_codebuild.BuildSpec.fromObject({
192
+ const [commands, commandsHashedComponents] = this.getDockerfileGenerationCommands();
193
+ const buildSpecVersion = 'v1'; // change this every time the build spec changes
194
+ const hashedComponents = commandsHashedComponents.concat(buildSpecVersion, this.architecture.name, this.baseImage, this.os.name);
195
+ const hash = crypto.createHash('md5').update(hashedComponents.join('\n')).digest('hex').slice(0, 10);
196
+ const buildSpec = aws_cdk_lib_1.aws_codebuild.BuildSpec.fromObject({
182
197
  version: '0.2',
183
198
  env: {
184
199
  variables: {
185
200
  REPO_ARN: repository.repositoryArn,
186
201
  REPO_URI: repository.repositoryUri,
187
- STACK_ID: 'unspecified',
188
- REQUEST_ID: 'unspecified',
189
- LOGICAL_RESOURCE_ID: 'unspecified',
190
- RESPONSE_URL: 'unspecified',
202
+ WAIT_HANDLE: 'unspecified',
191
203
  BASH_ENV: 'codebuild-log.sh',
192
204
  },
193
205
  shell: 'bash',
@@ -200,27 +212,24 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
200
212
  ],
201
213
  },
202
214
  build: {
203
- commands: this.getDockerfileGenerationCommands().concat('docker build --progress plain . -t "$REPO_URI"', 'docker push "$REPO_URI"'),
215
+ commands: commands.concat('docker build --progress plain . -t "$REPO_URI"', 'docker push "$REPO_URI"'),
204
216
  },
205
217
  post_build: {
206
218
  commands: [
207
219
  'rm -f codebuild-log.sh && STATUS="SUCCESS"',
208
- 'if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ]; then STATUS="FAILED"; fi',
220
+ 'if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ]; then STATUS="FAILURE"; fi',
209
221
  'cat <<EOF > /tmp/payload.json\n' +
210
222
  '{\n' +
211
- ' "StackId": "$STACK_ID",\n' +
212
- ' "RequestId": "$REQUEST_ID",\n' +
213
- ' "LogicalResourceId": "$LOGICAL_RESOURCE_ID",\n' +
214
- ' "PhysicalResourceId": "$REPO_ARN",\n' +
215
223
  ' "Status": "$STATUS",\n' +
224
+ ' "UniqueId": "build",\n' +
216
225
  // we remove non-printable characters from the log because CloudFormation doesn't like them
217
226
  // https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1601
218
227
  ' "Reason": `sed \'s/[^[:print:]]//g\' /tmp/codebuild.log | tail -c 400 | jq -Rsa .`,\n' +
219
228
  // for lambda always get a new value because there is always a new image hash
220
- ' "Data": {"Random": "$RANDOM"}\n' +
229
+ ' "Data": "$RANDOM"\n' +
221
230
  '}\n' +
222
231
  'EOF',
223
- 'if [ "$RESPONSE_URL" != "unspecified" ]; then jq . /tmp/payload.json; curl -fsSL -X PUT -H "Content-Type:" -d "@/tmp/payload.json" "$RESPONSE_URL"; fi',
232
+ 'if [ "$WAIT_HANDLE" != "unspecified" ]; then jq . /tmp/payload.json; curl -fsSL -X PUT -H "Content-Type:" -d "@/tmp/payload.json" "$WAIT_HANDLE"; fi',
224
233
  // generate and push soci index
225
234
  // we do this after finishing the build, so we don't have to wait. it's also not required, so it's ok if it fails
226
235
  'docker rmi "$REPO_URI"', // it downloads the image again to /tmp, so save on space
@@ -231,12 +240,14 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
231
240
  },
232
241
  },
233
242
  });
243
+ return [buildSpec, hash];
234
244
  }
235
- customResource(project, buildSpec) {
245
+ customResource(project, buildSpecHash) {
236
246
  const crHandler = (0, utils_1.singletonLambda)(build_image_function_1.BuildImageFunction, this, 'build-image', {
237
247
  description: 'Custom resource handler that triggers CodeBuild to build runner images, and cleans-up images on deletion',
238
248
  timeout: cdk.Duration.minutes(3),
239
- logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_MONTH,
249
+ logGroup: (0, utils_1.singletonLogGroup)(this, utils_1.SingletonLogType.RUNNER_IMAGE_BUILD),
250
+ logFormat: aws_cdk_lib_1.aws_lambda.LogFormat.JSON,
240
251
  });
241
252
  const policy = new aws_cdk_lib_1.aws_iam.Policy(this, 'CR Policy', {
242
253
  statements: [
@@ -247,15 +258,29 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
247
258
  ],
248
259
  });
249
260
  crHandler.role.attachInlinePolicy(policy);
261
+ let waitHandleRef = 'unspecified';
262
+ let waitDependable = '';
263
+ if (this.waitOnDeploy) {
264
+ // Wait handle lets us wait for longer than an hour for the image build to complete.
265
+ // We generate a new wait handle for build spec changes to guarantee a new image is built.
266
+ // This also helps make sure the changes are good. If they have a bug, the deployment will fail instead of just the scheduled build.
267
+ // Finally, it's recommended by CloudFormation docs to not reuse wait handles or old responses may interfere in some cases.
268
+ const handle = new aws_cdk_lib_1.aws_cloudformation.CfnWaitConditionHandle(this, `Build Wait Handle ${buildSpecHash}`);
269
+ const wait = new aws_cdk_lib_1.aws_cloudformation.CfnWaitCondition(this, `Build Wait ${buildSpecHash}`, {
270
+ handle: handle.ref,
271
+ timeout: this.timeout.toSeconds().toString(), // don't wait longer than the build timeout
272
+ count: 1,
273
+ });
274
+ waitHandleRef = handle.ref;
275
+ waitDependable = wait.ref;
276
+ }
250
277
  const cr = new aws_cdk_lib_1.CustomResource(this, 'Builder', {
251
278
  serviceToken: crHandler.functionArn,
252
279
  resourceType: 'Custom::ImageBuilder',
253
280
  properties: {
254
281
  RepoName: this.repository.repositoryName,
255
282
  ProjectName: project.projectName,
256
- // We include the full buildSpec so the image is built immediately on changes, and we don't have to wait for its scheduled build.
257
- // This also helps make sure the changes are good. If they have a bug, the deployment will fail instead of just the scheduled build.
258
- BuildSpec: buildSpec,
283
+ WaitHandle: waitHandleRef,
259
284
  },
260
285
  });
261
286
  // add dependencies to make sure resources are there when we need them
@@ -264,7 +289,7 @@ class CodeBuildRunnerImageBuilder extends common_1.RunnerImageBuilderBase {
264
289
  cr.node.addDependency(policy);
265
290
  cr.node.addDependency(crHandler.role);
266
291
  cr.node.addDependency(crHandler);
267
- return cr;
292
+ return waitDependable; // user needs to wait on wait handle which is triggered when the image is built
268
293
  }
269
294
  rebuildImageOnSchedule(project, rebuildInterval) {
270
295
  rebuildInterval = rebuildInterval ?? aws_cdk_lib_1.Duration.days(7);
@@ -308,4 +333,4 @@ class CodeBuildImageBuilderFailedBuildNotifier {
308
333
  }
309
334
  }
310
335
  exports.CodeBuildImageBuilderFailedBuildNotifier = CodeBuildImageBuilderFailedBuildNotifier;
311
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/image-builders/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_codebuild as codebuild,\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  aws_sns as sns,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct, IConstruct } from 'constructs';\nimport { defaultBaseDockerImage } from './aws-image-builder';\nimport { BuildImageFunction } from './build-image-function';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps } from './common';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers';\nimport { singletonLambda } from '../utils';\n\n\nexport interface CodeBuildRunnerImageBuilderProps {\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * Build image to use in CodeBuild. This is the image that's going to run the code that builds the runner image.\n   *\n   * The only action taken in CodeBuild is running `docker build`. You would therefore not need to change this setting often.\n   *\n   * @default Amazon Linux 2023\n   */\n  readonly buildImage?: codebuild.IBuildImage;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * @internal\n */\nexport class CodeBuildRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: string;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc | undefined;\n  private readonly securityGroups: ec2.ISecurityGroup[] | undefined;\n  private readonly buildImage: codebuild.IBuildImage;\n  private readonly repository: ecr.Repository;\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly timeout: cdk.Duration;\n  private readonly computeType: codebuild.ComputeType;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly role: iam.Role;\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.awsImageBuilderOptions) {\n      Annotations.of(this).addWarning('awsImageBuilderOptions are ignored when using CodeBuild runner image builder.');\n    }\n\n    this.os = props?.os ?? Os.LINUX_UBUNTU;\n    this.architecture = props?.architecture ?? Architecture.X86_64;\n    this.rebuildInterval = props?.rebuildInterval ?? Duration.days(7);\n    this.logRetention = props?.logRetention ?? RetentionDays.ONE_MONTH;\n    this.logRemovalPolicy = props?.logRemovalPolicy ?? RemovalPolicy.DESTROY;\n    this.vpc = props?.vpc;\n    this.securityGroups = props?.securityGroups;\n    this.subnetSelection = props?.subnetSelection;\n    this.timeout = props?.codeBuildOptions?.timeout ?? Duration.hours(1);\n    this.computeType = props?.codeBuildOptions?.computeType ?? ComputeType.SMALL;\n    this.baseImage = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.buildImage = props?.codeBuildOptions?.buildImage ?? this.getDefaultBuildImage();\n\n    // warn against isolated networks\n    if (props?.subnetSelection?.subnetType == ec2.SubnetType.PRIVATE_ISOLATED) {\n      Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +\n        'See https://github.com/aws/containers-roadmap/issues/1160');\n    }\n\n    // create service role for CodeBuild\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com'),\n    });\n\n    // create repository that only keeps one tag\n    this.repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteImages: true,\n      lifecycleRules: [\n        {\n          description: 'Remove soci indexes for replaced images',\n          tagStatus: TagStatus.TAGGED,\n          tagPrefixList: ['sha256-'],\n          maxImageCount: 1,\n        },\n        {\n          description: 'Remove untagged images that have been replaced by CodeBuild',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n    });\n  }\n\n  bindAmi(): RunnerAmi {\n    throw new Error('CodeBuild image builder cannot be used to build AMI');\n  }\n\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    // log group for the image builds\n    const logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: this.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: this.logRemovalPolicy ?? RemovalPolicy.DESTROY,\n      },\n    );\n\n    // generate buildSpec\n    const buildSpec = this.getBuildSpec(this.repository);\n\n    // create CodeBuild project that builds Dockerfile and pushes to repository\n    const project = new codebuild.Project(this, 'CodeBuild', {\n      description: `Build docker image for self-hosted GitHub runner ${this.node.path} (${this.os.name}/${this.architecture.name})`,\n      buildSpec,\n      vpc: this.vpc,\n      securityGroups: this.securityGroups,\n      subnetSelection: this.subnetSelection,\n      role: this.role,\n      timeout: this.timeout,\n      environment: {\n        buildImage: this.buildImage,\n        computeType: this.computeType,\n        privileged: true,\n      },\n      logging: {\n        cloudWatch: {\n          logGroup,\n        },\n      },\n    });\n\n    // permissions\n    this.repository.grantPullPush(project);\n\n    // call CodeBuild during deployment\n    const cr = this.customResource(project, buildSpec.toBuildSpec());\n\n    // rebuild image on a schedule\n    this.rebuildImageOnSchedule(project, this.rebuildInterval);\n\n    // return the image\n    this.boundDockerImage = {\n      imageRepository: this.repository,\n      imageTag: 'latest',\n      architecture: this.architecture,\n      os: this.os,\n      logGroup,\n      runnerVersion: RunnerVersion.specific('unknown'),\n      _dependable: cr.getAttString('Random'),\n    };\n    return this.boundDockerImage;\n  }\n\n  private getDefaultBuildImage(): codebuild.IBuildImage {\n    if (this.os.is(Os.LINUX_UBUNTU) || this.os.is(Os.LINUX_AMAZON_2) || this.os.is(Os.LINUX_AMAZON_2023) || this.os.is(Os.LINUX)) {\n      // CodeBuild just runs `docker build` so its OS doesn't really matter\n      if (this.architecture.is(Architecture.X86_64)) {\n        return codebuild.LinuxBuildImage.AMAZON_LINUX_2_5;\n      } else if (this.architecture.is(Architecture.ARM64)) {\n        return codebuild.LinuxArmBuildImage.AMAZON_LINUX_2_STANDARD_3_0;\n      }\n    }\n    if (this.os.is(Os.WINDOWS)) {\n      throw new Error('CodeBuild cannot be used to build Windows Docker images https://github.com/docker-library/docker/issues/49');\n    }\n\n    throw new Error(`Unable to find CodeBuild image for ${this.os.name}/${this.architecture.name}`);\n  }\n\n  private getDockerfileGenerationCommands() {\n    let commands = [];\n    let dockerfile = `FROM ${this.baseImage}\\nVOLUME /var/lib/docker\\n`;\n\n    for (let i = 0; i < this.components.length; i++) {\n      const componentName = this.components[i].name;\n      const assetDescriptors = this.components[i].getAssets(this.os, this.architecture);\n\n      for (let j = 0; j < assetDescriptors.length; j++) {\n        if (this.os.is(Os.WINDOWS)) {\n          throw new Error(\"Can't add asset as we can't build Windows Docker images on CodeBuild\");\n        }\n\n        const asset = new s3_assets.Asset(this, `Component ${i} ${componentName} Asset ${j}`, {\n          path: assetDescriptors[j].source,\n        });\n\n        if (asset.isFile) {\n          commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${componentName}-${j}`);\n        } else if (asset.isZipArchive) {\n          commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${componentName}-${j}.zip`);\n          commands.push(`unzip asset${i}-${componentName}-${j}.zip -d \"asset${i}-${componentName}-${j}\"`);\n        } else {\n          throw new Error(`Unknown asset type: ${asset}`);\n        }\n\n        dockerfile += `COPY asset${i}-${componentName}-${j} ${assetDescriptors[j].target}\\n`;\n\n        asset.grantRead(this);\n      }\n\n      const componentCommands = this.components[i].getCommands(this.os, this.architecture);\n      const script = '#!/bin/bash\\nset -exuo pipefail\\n' + componentCommands.join('\\n');\n      commands.push(`cat > component${i}-${componentName}.sh <<'EOFGITHUBRUNNERSDOCKERFILE'\\n${script}\\nEOFGITHUBRUNNERSDOCKERFILE`);\n      commands.push(`chmod +x component${i}-${componentName}.sh`);\n      dockerfile += `COPY component${i}-${componentName}.sh /tmp\\n`;\n      dockerfile += `RUN /tmp/component${i}-${componentName}.sh\\n`;\n\n      dockerfile += this.components[i].getDockerCommands(this.os, this.architecture).join('\\n') + '\\n';\n    }\n\n    commands.push(`cat > Dockerfile <<'EOFGITHUBRUNNERSDOCKERFILE'\\n${dockerfile}\\nEOFGITHUBRUNNERSDOCKERFILE`);\n\n    return commands;\n  }\n\n  private getBuildSpec(repository: ecr.Repository): codebuild.BuildSpec {\n    const thisStack = cdk.Stack.of(this);\n\n    let archUrl;\n    if (this.architecture.is(Architecture.X86_64)) {\n      archUrl = 'x86_64';\n    } else if (this.architecture.is(Architecture.ARM64)) {\n      archUrl = 'arm64';\n    } else {\n      throw new Error(`Unsupported architecture for required CodeBuild: ${this.architecture.name}`);\n    }\n\n    return codebuild.BuildSpec.fromObject({\n      version: '0.2',\n      env: {\n        variables: {\n          REPO_ARN: repository.repositoryArn,\n          REPO_URI: repository.repositoryUri,\n          STACK_ID: 'unspecified',\n          REQUEST_ID: 'unspecified',\n          LOGICAL_RESOURCE_ID: 'unspecified',\n          RESPONSE_URL: 'unspecified',\n          BASH_ENV: 'codebuild-log.sh',\n        },\n        shell: 'bash',\n      },\n      phases: {\n        pre_build: {\n          commands: [\n            'echo \"exec > >(tee -a /tmp/codebuild.log) 2>&1\" > codebuild-log.sh',\n            `aws ecr get-login-password --region \"$AWS_DEFAULT_REGION\" | docker login --username AWS --password-stdin ${thisStack.account}.dkr.ecr.${thisStack.region}.amazonaws.com`,\n          ],\n        },\n        build: {\n          commands: this.getDockerfileGenerationCommands().concat(\n            'docker build --progress plain . -t \"$REPO_URI\"',\n            'docker push \"$REPO_URI\"',\n          ),\n        },\n        post_build: {\n          commands: [\n            'rm -f codebuild-log.sh && STATUS=\"SUCCESS\"',\n            'if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ]; then STATUS=\"FAILED\"; fi',\n            'cat <<EOF > /tmp/payload.json\\n' +\n              '{\\n' +\n              '  \"StackId\": \"$STACK_ID\",\\n' +\n              '  \"RequestId\": \"$REQUEST_ID\",\\n' +\n              '  \"LogicalResourceId\": \"$LOGICAL_RESOURCE_ID\",\\n' +\n              '  \"PhysicalResourceId\": \"$REPO_ARN\",\\n' +\n              '  \"Status\": \"$STATUS\",\\n' +\n              // we remove non-printable characters from the log because CloudFormation doesn't like them\n              // https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1601\n              '  \"Reason\": `sed \\'s/[^[:print:]]//g\\' /tmp/codebuild.log | tail -c 400 | jq -Rsa .`,\\n' +\n              // for lambda always get a new value because there is always a new image hash\n              '  \"Data\": {\"Random\": \"$RANDOM\"}\\n' +\n              '}\\n' +\n              'EOF',\n            'if [ \"$RESPONSE_URL\" != \"unspecified\" ]; then jq . /tmp/payload.json; curl -fsSL -X PUT -H \"Content-Type:\" -d \"@/tmp/payload.json\" \"$RESPONSE_URL\"; fi',\n            // generate and push soci index\n            // we do this after finishing the build, so we don't have to wait. it's also not required, so it's ok if it fails\n            'docker rmi \"$REPO_URI\"', // it downloads the image again to /tmp, so save on space\n            'LATEST_SOCI_VERSION=`curl -w \"%{redirect_url}\" -fsS https://github.com/CloudSnorkel/standalone-soci-indexer/releases/latest | grep -oE \"[^/]+$\"`',\n            `curl -fsSL https://github.com/CloudSnorkel/standalone-soci-indexer/releases/download/$\\{LATEST_SOCI_VERSION}/standalone-soci-indexer_Linux_${archUrl}.tar.gz | tar xz`,\n            './standalone-soci-indexer \"$REPO_URI\"',\n          ],\n        },\n      },\n    });\n  }\n\n  private customResource(project: codebuild.Project, buildSpec: string) {\n    const crHandler = singletonLambda(BuildImageFunction, this, 'build-image', {\n      description: 'Custom resource handler that triggers CodeBuild to build runner images, and cleans-up images on deletion',\n      timeout: cdk.Duration.minutes(3),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    const policy = new iam.Policy(this, 'CR Policy', {\n      statements: [\n        new iam.PolicyStatement({\n          actions: ['codebuild:StartBuild'],\n          resources: [project.projectArn],\n        }),\n      ],\n    });\n    crHandler.role!.attachInlinePolicy(policy);\n\n    const cr = new CustomResource(this, 'Builder', {\n      serviceToken: crHandler.functionArn,\n      resourceType: 'Custom::ImageBuilder',\n      properties: {\n        RepoName: this.repository.repositoryName,\n        ProjectName: project.projectName,\n        // We include the full buildSpec so the image is built immediately on changes, and we don't have to wait for its scheduled build.\n        // This also helps make sure the changes are good. If they have a bug, the deployment will fail instead of just the scheduled build.\n        BuildSpec: buildSpec,\n      },\n    });\n\n    // add dependencies to make sure resources are there when we need them\n    cr.node.addDependency(project);\n    cr.node.addDependency(this.role);\n    cr.node.addDependency(policy);\n    cr.node.addDependency(crHandler.role!);\n    cr.node.addDependency(crHandler);\n\n    return cr;\n  }\n\n  private rebuildImageOnSchedule(project: codebuild.Project, rebuildInterval?: Duration) {\n    rebuildInterval = rebuildInterval ?? Duration.days(7);\n    if (rebuildInterval.toMilliseconds() != 0) {\n      const scheduleRule = new events.Rule(this, 'Build Schedule', {\n        description: `Rebuild runner image for ${this.repository.repositoryName}`,\n        schedule: events.Schedule.rate(rebuildInterval),\n      });\n      scheduleRule.addTarget(new events_targets.CodeBuildProject(project));\n    }\n  }\n\n  get connections(): ec2.Connections {\n    return new ec2.Connections({\n      securityGroups: this.securityGroups,\n    });\n  }\n\n  get grantPrincipal(): iam.IPrincipal {\n    return this.role;\n  }\n}\n\n/**\n * @internal\n */\nexport class CodeBuildImageBuilderFailedBuildNotifier implements cdk.IAspect {\n  constructor(private topic: sns.ITopic) {\n  }\n\n  public visit(node: IConstruct): void {\n    if (node instanceof CodeBuildRunnerImageBuilder) {\n      const builder = node as CodeBuildRunnerImageBuilder;\n      const projectNode = builder.node.tryFindChild('CodeBuild');\n      if (projectNode) {\n        const project = projectNode as codebuild.Project;\n        project.notifyOnBuildFailed('BuildFailed', this.topic);\n      } else {\n        cdk.Annotations.of(builder).addWarning('Unused builder cannot get notifications of failed builds');\n      }\n    }\n  }\n}\n"]}
336
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/image-builders/codebuild.ts"],"names":[],"mappings":";;;AAAA,sCAAsC;AACtC,mCAAmC;AACnC,6CAgBqB;AACrB,6DAAwD;AACxD,iDAA+D;AAC/D,mDAAqD;AAErD,2DAA6D;AAC7D,iEAA4D;AAE5D,qCAA2E;AAC3E,4CAAuF;AACvF,oCAAgF;AA+BhF;;GAEG;AACH,MAAa,2BAA4B,SAAQ,+BAAsB;IAkBrE,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA+B;QACvE,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QAExB,IAAI,KAAK,EAAE,sBAAsB,EAAE,CAAC;YAClC,yBAAW,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,UAAU,CAAC,+EAA+E,CAAC,CAAC;QACnH,CAAC;QAED,IAAI,CAAC,EAAE,GAAG,KAAK,EAAE,EAAE,IAAI,cAAE,CAAC,YAAY,CAAC;QACvC,IAAI,CAAC,YAAY,GAAG,KAAK,EAAE,YAAY,IAAI,wBAAY,CAAC,MAAM,CAAC;QAC/D,IAAI,CAAC,eAAe,GAAG,KAAK,EAAE,eAAe,IAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClE,IAAI,CAAC,YAAY,GAAG,KAAK,EAAE,YAAY,IAAI,wBAAa,CAAC,SAAS,CAAC;QACnE,IAAI,CAAC,gBAAgB,GAAG,KAAK,EAAE,gBAAgB,IAAI,2BAAa,CAAC,OAAO,CAAC;QACzE,IAAI,CAAC,GAAG,GAAG,KAAK,EAAE,GAAG,CAAC;QACtB,IAAI,CAAC,cAAc,GAAG,KAAK,EAAE,cAAc,CAAC;QAC5C,IAAI,CAAC,eAAe,GAAG,KAAK,EAAE,eAAe,CAAC;QAC9C,IAAI,CAAC,OAAO,GAAG,KAAK,EAAE,gBAAgB,EAAE,OAAO,IAAI,sBAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACrE,IAAI,CAAC,WAAW,GAAG,KAAK,EAAE,gBAAgB,EAAE,WAAW,IAAI,2BAAW,CAAC,KAAK,CAAC;QAC7E,IAAI,CAAC,SAAS,GAAG,KAAK,EAAE,eAAe,IAAI,IAAA,0CAAsB,EAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC3E,IAAI,CAAC,UAAU,GAAG,KAAK,EAAE,gBAAgB,EAAE,UAAU,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;QACrF,IAAI,CAAC,YAAY,GAAG,KAAK,EAAE,YAAY,IAAI,IAAI,CAAC;QAEhD,iCAAiC;QACjC,IAAI,KAAK,EAAE,eAAe,EAAE,UAAU,IAAI,qBAAG,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;YAC1E,yBAAW,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,UAAU,CAAC,8FAA8F;gBAC5H,2DAA2D,CAAC,CAAC;QACjE,CAAC;QAED,gBAAgB;QAChB,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,GAAG,sBAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC;YAC7D,yBAAW,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,8DAA8D,CAAC,CAAC;QAChG,CAAC;QAED,oCAAoC;QACpC,IAAI,CAAC,IAAI,GAAG,IAAI,qBAAG,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE;YACrC,SAAS,EAAE,IAAI,qBAAG,CAAC,gBAAgB,CAAC,yBAAyB,CAAC;SAC/D,CAAC,CAAC;QAEH,4CAA4C;QAC5C,IAAI,CAAC,UAAU,GAAG,IAAI,qBAAG,CAAC,UAAU,CAAC,IAAI,EAAE,YAAY,EAAE;YACvD,eAAe,EAAE,IAAI;YACrB,kBAAkB,EAAE,uBAAa,CAAC,OAAO;YACzC,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,aAAa,EAAE,IAAI;YACnB,cAAc,EAAE;gBACd;oBACE,WAAW,EAAE,yCAAyC;oBACtD,SAAS,EAAE,mBAAS,CAAC,MAAM;oBAC3B,aAAa,EAAE,CAAC,SAAS,CAAC;oBAC1B,aAAa,EAAE,CAAC;iBACjB;gBACD;oBACE,WAAW,EAAE,6DAA6D;oBAC1E,SAAS,EAAE,mBAAS,CAAC,QAAQ;oBAC7B,WAAW,EAAE,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;iBAC9B;aACF;SACF,CAAC,CAAC;IACL,CAAC;IAED,OAAO;QACL,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;IACzE,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,gBAAgB,CAAC;QAC/B,CAAC;QAED,iCAAiC;QACjC,MAAM,QAAQ,GAAG,IAAI,sBAAI,CAAC,QAAQ,CAChC,IAAI,EACJ,MAAM,EACN;YACE,SAAS,EAAE,IAAI,CAAC,YAAY,IAAI,wBAAa,CAAC,SAAS;YACvD,aAAa,EAAE,IAAI,CAAC,gBAAgB,IAAI,2BAAa,CAAC,OAAO;SAC9D,CACF,CAAC;QAEF,qBAAqB;QACrB,MAAM,CAAC,SAAS,EAAE,aAAa,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAEtE,2EAA2E;QAC3E,MAAM,OAAO,GAAG,IAAI,2BAAS,CAAC,OAAO,CAAC,IAAI,EAAE,WAAW,EAAE;YACvD,WAAW,EAAE,oDAAoD,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC,IAAI,IAAI,IAAI,CAAC,YAAY,CAAC,IAAI,GAAG;YAC7H,SAAS;YACT,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,cAAc,EAAE,IAAI,CAAC,cAAc;YACnC,eAAe,EAAE,IAAI,CAAC,eAAe;YACrC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,WAAW,EAAE;gBACX,UAAU,EAAE,IAAI,CAAC,UAAU;gBAC3B,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,UAAU,EAAE,IAAI;aACjB;YACD,OAAO,EAAE;gBACP,UAAU,EAAE;oBACV,QAAQ;iBACT;aACF;SACF,CAAC,CAAC;QAEH,cAAc;QACd,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAEvC,mCAAmC;QACnC,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QAEnE,8BAA8B;QAC9B,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC;QAE3D,mBAAmB;QACnB,IAAI,CAAC,gBAAgB,GAAG;YACtB,eAAe,EAAE,IAAI,CAAC,UAAU;YAChC,QAAQ,EAAE,QAAQ;YAClB,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,EAAE,EAAE,IAAI,CAAC,EAAE;YACX,QAAQ;YACR,aAAa,EAAE,yBAAa,CAAC,QAAQ,CAAC,SAAS,CAAC;YAChD,WAAW,EAAE,cAAc;SAC5B,CAAC;QACF,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;IAEO,oBAAoB;QAC1B,IAAI,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,cAAE,CAAC,mBAAmB,CAAC,EAAE,CAAC;YACzC,qEAAqE;YACrE,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,wBAAY,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC9C,OAAO,2BAAS,CAAC,eAAe,CAAC,gBAAgB,CAAC;YACpD,CAAC;iBAAM,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,wBAAY,CAAC,KAAK,CAAC,EAAE,CAAC;gBACpD,OAAO,2BAAS,CAAC,kBAAkB,CAAC,2BAA2B,CAAC;YAClE,CAAC;QACH,CAAC;QACD,IAAI,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,cAAE,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,4GAA4G,CAAC,CAAC;QAChI,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,sCAAsC,IAAI,CAAC,EAAE,CAAC,IAAI,IAAI,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;IAClG,CAAC;IAEO,+BAA+B;QACrC,IAAI,gBAAgB,GAAa,EAAE,CAAC;QACpC,IAAI,QAAQ,GAAG,EAAE,CAAC;QAClB,IAAI,UAAU,GAAG,QAAQ,IAAI,CAAC,SAAS,4BAA4B,CAAC;QAEpE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAChD,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YAC9C,MAAM,gBAAgB,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YAElF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,gBAAgB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACjD,IAAI,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,cAAE,CAAC,OAAO,CAAC,EAAE,CAAC;oBAC3B,MAAM,IAAI,KAAK,CAAC,sEAAsE,CAAC,CAAC;gBAC1F,CAAC;gBAED,MAAM,KAAK,GAAG,IAAI,2BAAS,CAAC,KAAK,CAAC,IAAI,EAAE,aAAa,CAAC,IAAI,aAAa,UAAU,CAAC,EAAE,EAAE;oBACpF,IAAI,EAAE,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM;iBACjC,CAAC,CAAC;gBAEH,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;oBACjB,QAAQ,CAAC,IAAI,CAAC,aAAa,KAAK,CAAC,WAAW,SAAS,CAAC,IAAI,aAAa,IAAI,CAAC,EAAE,CAAC,CAAC;gBAClF,CAAC;qBAAM,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;oBAC9B,QAAQ,CAAC,IAAI,CAAC,aAAa,KAAK,CAAC,WAAW,SAAS,CAAC,IAAI,aAAa,IAAI,CAAC,MAAM,CAAC,CAAC;oBACpF,QAAQ,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,IAAI,CAAC,iBAAiB,CAAC,IAAI,aAAa,IAAI,CAAC,GAAG,CAAC,CAAC;gBAClG,CAAC;qBAAM,CAAC;oBACN,MAAM,IAAI,KAAK,CAAC,uBAAuB,KAAK,EAAE,CAAC,CAAC;gBAClD,CAAC;gBAED,UAAU,IAAI,aAAa,CAAC,IAAI,aAAa,IAAI,CAAC,IAAI,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC;gBACrF,gBAAgB,CAAC,IAAI,CAAC,iBAAiB,KAAK,CAAC,SAAS,IAAI,CAAC,IAAI,aAAa,IAAI,CAAC,IAAI,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;gBAEnH,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YACxB,CAAC;YAED,MAAM,iBAAiB,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YACrF,MAAM,MAAM,GAAG,mCAAmC,GAAG,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClF,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,aAAa,uCAAuC,MAAM,8BAA8B,CAAC,CAAC;YAC/H,QAAQ,CAAC,IAAI,CAAC,qBAAqB,CAAC,IAAI,aAAa,KAAK,CAAC,CAAC;YAC5D,gBAAgB,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,aAAa,IAAI,MAAM,EAAE,CAAC,CAAC;YACpE,UAAU,IAAI,iBAAiB,CAAC,IAAI,aAAa,YAAY,CAAC;YAC9D,UAAU,IAAI,qBAAqB,CAAC,IAAI,aAAa,OAAO,CAAC;YAE7D,MAAM,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YACxF,UAAU,IAAI,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC;YAC/C,gBAAgB,CAAC,IAAI,CAAC,qBAAqB,CAAC,IAAI,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC/E,CAAC;QAED,QAAQ,CAAC,IAAI,CAAC,oDAAoD,UAAU,8BAA8B,CAAC,CAAC;QAE5G,OAAO,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;IACtC,CAAC;IAEO,YAAY,CAAC,UAA0B;QAC7C,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAErC,IAAI,OAAO,CAAC;QACZ,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,wBAAY,CAAC,MAAM,CAAC,EAAE,CAAC;YAC9C,OAAO,GAAG,QAAQ,CAAC;QACrB,CAAC;aAAM,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,wBAAY,CAAC,KAAK,CAAC,EAAE,CAAC;YACpD,OAAO,GAAG,OAAO,CAAC;QACpB,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,oDAAoD,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;QAChG,CAAC;QAED,MAAM,CAAC,QAAQ,EAAE,wBAAwB,CAAC,GAAG,IAAI,CAAC,+BAA+B,EAAE,CAAC;QAEpF,MAAM,gBAAgB,GAAG,IAAI,CAAC,CAAC,gDAAgD;QAC/E,MAAM,gBAAgB,GAAG,wBAAwB,CAAC,MAAM,CAAC,gBAAgB,EAAE,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QACjI,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAErG,MAAM,SAAS,GAAG,2BAAS,CAAC,SAAS,CAAC,UAAU,CAAC;YAC/C,OAAO,EAAE,KAAK;YACd,GAAG,EAAE;gBACH,SAAS,EAAE;oBACT,QAAQ,EAAE,UAAU,CAAC,aAAa;oBAClC,QAAQ,EAAE,UAAU,CAAC,aAAa;oBAClC,WAAW,EAAE,aAAa;oBAC1B,QAAQ,EAAE,kBAAkB;iBAC7B;gBACD,KAAK,EAAE,MAAM;aACd;YACD,MAAM,EAAE;gBACN,SAAS,EAAE;oBACT,QAAQ,EAAE;wBACR,oEAAoE;wBACpE,4GAA4G,SAAS,CAAC,OAAO,YAAY,SAAS,CAAC,MAAM,gBAAgB;qBAC1K;iBACF;gBACD,KAAK,EAAE;oBACL,QAAQ,EAAE,QAAQ,CAAC,MAAM,CACvB,gDAAgD,EAChD,yBAAyB,CAC1B;iBACF;gBACD,UAAU,EAAE;oBACV,QAAQ,EAAE;wBACR,4CAA4C;wBAC5C,qEAAqE;wBACrE,iCAAiC;4BAC/B,KAAK;4BACL,0BAA0B;4BAC1B,0BAA0B;4BAC1B,2FAA2F;4BAC3F,oFAAoF;4BACpF,yFAAyF;4BACzF,6EAA6E;4BAC7E,uBAAuB;4BACvB,KAAK;4BACL,KAAK;wBACP,sJAAsJ;wBACtJ,+BAA+B;wBAC/B,iHAAiH;wBACjH,wBAAwB,EAAE,yDAAyD;wBACnF,kJAAkJ;wBAClJ,8IAA8I,OAAO,kBAAkB;wBACvK,uCAAuC;qBACxC;iBACF;aACF;SACF,CAAC,CAAC;QAEH,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;IAC3B,CAAC;IAEO,cAAc,CAAC,OAA0B,EAAE,aAAqB;QACtE,MAAM,SAAS,GAAG,IAAA,uBAAe,EAAC,yCAAkB,EAAE,IAAI,EAAE,aAAa,EAAE;YACzE,WAAW,EAAE,0GAA0G;YACvH,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAChC,QAAQ,EAAE,IAAA,yBAAiB,EAAC,IAAI,EAAE,wBAAgB,CAAC,kBAAkB,CAAC;YACtE,SAAS,EAAE,wBAAM,CAAC,SAAS,CAAC,IAAI;SACjC,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,IAAI,qBAAG,CAAC,MAAM,CAAC,IAAI,EAAE,WAAW,EAAE;YAC/C,UAAU,EAAE;gBACV,IAAI,qBAAG,CAAC,eAAe,CAAC;oBACtB,OAAO,EAAE,CAAC,sBAAsB,CAAC;oBACjC,SAAS,EAAE,CAAC,OAAO,CAAC,UAAU,CAAC;iBAChC,CAAC;aACH;SACF,CAAC,CAAC;QACH,SAAS,CAAC,IAAK,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;QAE3C,IAAI,aAAa,GAAE,aAAa,CAAC;QACjC,IAAI,cAAc,GAAG,EAAE,CAAC;QAExB,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,oFAAoF;YACpF,0FAA0F;YAC1F,oIAAoI;YACpI,2HAA2H;YAC3H,MAAM,MAAM,GAAG,IAAI,gCAAc,CAAC,sBAAsB,CAAC,IAAI,EAAE,qBAAqB,aAAa,EAAE,CAAC,CAAC;YACrG,MAAM,IAAI,GAAG,IAAI,gCAAc,CAAC,gBAAgB,CAAC,IAAI,EAAE,cAAc,aAAa,EAAE,EAAE;gBACpF,MAAM,EAAE,MAAM,CAAC,GAAG;gBAClB,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,EAAE,2CAA2C;gBACzF,KAAK,EAAE,CAAC;aACT,CAAC,CAAC;YACH,aAAa,GAAG,MAAM,CAAC,GAAG,CAAC;YAC3B,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC;QAC5B,CAAC;QAED,MAAM,EAAE,GAAG,IAAI,4BAAc,CAAC,IAAI,EAAE,SAAS,EAAE;YAC7C,YAAY,EAAE,SAAS,CAAC,WAAW;YACnC,YAAY,EAAE,sBAAsB;YACpC,UAAU,EAAgC;gBACxC,QAAQ,EAAE,IAAI,CAAC,UAAU,CAAC,cAAc;gBACxC,WAAW,EAAE,OAAO,CAAC,WAAW;gBAChC,UAAU,EAAE,aAAa;aAC1B;SACF,CAAC,CAAC;QAEH,sEAAsE;QACtE,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAC/B,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACjC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC9B,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,IAAK,CAAC,CAAC;QACvC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;QAEjC,OAAO,cAAc,CAAC,CAAC,+EAA+E;IACxG,CAAC;IAEO,sBAAsB,CAAC,OAA0B,EAAE,eAA0B;QACnF,eAAe,GAAG,eAAe,IAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACtD,IAAI,eAAe,CAAC,cAAc,EAAE,IAAI,CAAC,EAAE,CAAC;YAC1C,MAAM,YAAY,GAAG,IAAI,wBAAM,CAAC,IAAI,CAAC,IAAI,EAAE,gBAAgB,EAAE;gBAC3D,WAAW,EAAE,4BAA4B,IAAI,CAAC,UAAU,CAAC,cAAc,EAAE;gBACzE,QAAQ,EAAE,wBAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC;aAChD,CAAC,CAAC;YACH,YAAY,CAAC,SAAS,CAAC,IAAI,gCAAc,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,qBAAG,CAAC,WAAW,CAAC;YACzB,cAAc,EAAE,IAAI,CAAC,cAAc;SACpC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;CACF;AArWD,kEAqWC;AAED;;GAEG;AACH,MAAa,wCAAwC;IACnD,YAAoB,KAAiB;QAAjB,UAAK,GAAL,KAAK,CAAY;IACrC,CAAC;IAEM,KAAK,CAAC,IAAgB;QAC3B,IAAI,IAAI,YAAY,2BAA2B,EAAE,CAAC;YAChD,MAAM,OAAO,GAAG,IAAmC,CAAC;YACpD,MAAM,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;YAC3D,IAAI,WAAW,EAAE,CAAC;gBAChB,MAAM,OAAO,GAAG,WAAgC,CAAC;gBACjD,OAAO,CAAC,mBAAmB,CAAC,aAAa,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YACzD,CAAC;iBAAM,CAAC;gBACN,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,UAAU,CAAC,0DAA0D,CAAC,CAAC;YACrG,CAAC;QACH,CAAC;IACH,CAAC;CACF;AAhBD,4FAgBC","sourcesContent":["import * as crypto from 'node:crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\n  aws_cloudformation as cloudformation,\n  aws_codebuild as codebuild,\n  aws_ec2 as ec2,\n  aws_ecr as ecr,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  aws_sns as sns,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct, IConstruct } from 'constructs';\nimport { defaultBaseDockerImage } from './aws-image-builder';\nimport { BuildImageFunction } from './build-image-function';\nimport { BuildImageFunctionProperties } from './build-image.lambda';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps } from './common';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers';\nimport { singletonLambda, singletonLogGroup, SingletonLogType } from '../utils';\n\n\nexport interface CodeBuildRunnerImageBuilderProps {\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * Build image to use in CodeBuild. This is the image that's going to run the code that builds the runner image.\n   *\n   * The only action taken in CodeBuild is running `docker build`. You would therefore not need to change this setting often.\n   *\n   * @default Amazon Linux 2023\n   */\n  readonly buildImage?: codebuild.IBuildImage;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * @internal\n */\nexport class CodeBuildRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: string;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc | undefined;\n  private readonly securityGroups: ec2.ISecurityGroup[] | undefined;\n  private readonly buildImage: codebuild.IBuildImage;\n  private readonly repository: ecr.Repository;\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly timeout: cdk.Duration;\n  private readonly computeType: codebuild.ComputeType;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly role: iam.Role;\n  private readonly waitOnDeploy: boolean;\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.awsImageBuilderOptions) {\n      Annotations.of(this).addWarning('awsImageBuilderOptions are ignored when using CodeBuild runner image builder.');\n    }\n\n    this.os = props?.os ?? Os.LINUX_UBUNTU;\n    this.architecture = props?.architecture ?? Architecture.X86_64;\n    this.rebuildInterval = props?.rebuildInterval ?? Duration.days(7);\n    this.logRetention = props?.logRetention ?? RetentionDays.ONE_MONTH;\n    this.logRemovalPolicy = props?.logRemovalPolicy ?? RemovalPolicy.DESTROY;\n    this.vpc = props?.vpc;\n    this.securityGroups = props?.securityGroups;\n    this.subnetSelection = props?.subnetSelection;\n    this.timeout = props?.codeBuildOptions?.timeout ?? Duration.hours(1);\n    this.computeType = props?.codeBuildOptions?.computeType ?? ComputeType.SMALL;\n    this.baseImage = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.buildImage = props?.codeBuildOptions?.buildImage ?? this.getDefaultBuildImage();\n    this.waitOnDeploy = props?.waitOnDeploy ?? true;\n\n    // warn against isolated networks\n    if (props?.subnetSelection?.subnetType == ec2.SubnetType.PRIVATE_ISOLATED) {\n      Annotations.of(this).addWarning('Private isolated subnets cannot pull from public ECR and VPC endpoint is not supported yet. ' +\n        'See https://github.com/aws/containers-roadmap/issues/1160');\n    }\n\n    // check timeout\n    if (this.timeout.toSeconds() > Duration.hours(8).toSeconds()) {\n      Annotations.of(this).addError('CodeBuild runner image builder timeout must 8 hours or less.');\n    }\n\n    // create service role for CodeBuild\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com'),\n    });\n\n    // create repository that only keeps one tag\n    this.repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      emptyOnDelete: true,\n      lifecycleRules: [\n        {\n          description: 'Remove soci indexes for replaced images',\n          tagStatus: TagStatus.TAGGED,\n          tagPrefixList: ['sha256-'],\n          maxImageCount: 1,\n        },\n        {\n          description: 'Remove untagged images that have been replaced by CodeBuild',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n        },\n      ],\n    });\n  }\n\n  bindAmi(): RunnerAmi {\n    throw new Error('CodeBuild image builder cannot be used to build AMI');\n  }\n\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    // log group for the image builds\n    const logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: this.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: this.logRemovalPolicy ?? RemovalPolicy.DESTROY,\n      },\n    );\n\n    // generate buildSpec\n    const [buildSpec, buildSpecHash] = this.getBuildSpec(this.repository);\n\n    // create CodeBuild project that builds Dockerfile and pushes to repository\n    const project = new codebuild.Project(this, 'CodeBuild', {\n      description: `Build docker image for self-hosted GitHub runner ${this.node.path} (${this.os.name}/${this.architecture.name})`,\n      buildSpec,\n      vpc: this.vpc,\n      securityGroups: this.securityGroups,\n      subnetSelection: this.subnetSelection,\n      role: this.role,\n      timeout: this.timeout,\n      environment: {\n        buildImage: this.buildImage,\n        computeType: this.computeType,\n        privileged: true,\n      },\n      logging: {\n        cloudWatch: {\n          logGroup,\n        },\n      },\n    });\n\n    // permissions\n    this.repository.grantPullPush(project);\n\n    // call CodeBuild during deployment\n    const completedImage = this.customResource(project, buildSpecHash);\n\n    // rebuild image on a schedule\n    this.rebuildImageOnSchedule(project, this.rebuildInterval);\n\n    // return the image\n    this.boundDockerImage = {\n      imageRepository: this.repository,\n      imageTag: 'latest',\n      architecture: this.architecture,\n      os: this.os,\n      logGroup,\n      runnerVersion: RunnerVersion.specific('unknown'),\n      _dependable: completedImage,\n    };\n    return this.boundDockerImage;\n  }\n\n  private getDefaultBuildImage(): codebuild.IBuildImage {\n    if (this.os.isIn(Os._ALL_LINUX_VERSIONS)) {\n      // CodeBuild just runs `docker build` so its OS doesn't really matter\n      if (this.architecture.is(Architecture.X86_64)) {\n        return codebuild.LinuxBuildImage.AMAZON_LINUX_2_5;\n      } else if (this.architecture.is(Architecture.ARM64)) {\n        return codebuild.LinuxArmBuildImage.AMAZON_LINUX_2_STANDARD_3_0;\n      }\n    }\n    if (this.os.is(Os.WINDOWS)) {\n      throw new Error('CodeBuild cannot be used to build Windows Docker images https://github.com/docker-library/docker/issues/49');\n    }\n\n    throw new Error(`Unable to find CodeBuild image for ${this.os.name}/${this.architecture.name}`);\n  }\n\n  private getDockerfileGenerationCommands(): [string[], string[]] {\n    let hashedComponents: string[] = [];\n    let commands = [];\n    let dockerfile = `FROM ${this.baseImage}\\nVOLUME /var/lib/docker\\n`;\n\n    for (let i = 0; i < this.components.length; i++) {\n      const componentName = this.components[i].name;\n      const assetDescriptors = this.components[i].getAssets(this.os, this.architecture);\n\n      for (let j = 0; j < assetDescriptors.length; j++) {\n        if (this.os.is(Os.WINDOWS)) {\n          throw new Error(\"Can't add asset as we can't build Windows Docker images on CodeBuild\");\n        }\n\n        const asset = new s3_assets.Asset(this, `Component ${i} ${componentName} Asset ${j}`, {\n          path: assetDescriptors[j].source,\n        });\n\n        if (asset.isFile) {\n          commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${componentName}-${j}`);\n        } else if (asset.isZipArchive) {\n          commands.push(`aws s3 cp ${asset.s3ObjectUrl} asset${i}-${componentName}-${j}.zip`);\n          commands.push(`unzip asset${i}-${componentName}-${j}.zip -d \"asset${i}-${componentName}-${j}\"`);\n        } else {\n          throw new Error(`Unknown asset type: ${asset}`);\n        }\n\n        dockerfile += `COPY asset${i}-${componentName}-${j} ${assetDescriptors[j].target}\\n`;\n        hashedComponents.push(`__ ASSET FILE ${asset.assetHash} ${i}-${componentName}-${j} ${assetDescriptors[j].target}`);\n\n        asset.grantRead(this);\n      }\n\n      const componentCommands = this.components[i].getCommands(this.os, this.architecture);\n      const script = '#!/bin/bash\\nset -exuo pipefail\\n' + componentCommands.join('\\n');\n      commands.push(`cat > component${i}-${componentName}.sh <<'EOFGITHUBRUNNERSDOCKERFILE'\\n${script}\\nEOFGITHUBRUNNERSDOCKERFILE`);\n      commands.push(`chmod +x component${i}-${componentName}.sh`);\n      hashedComponents.push(`__ COMMAND ${i} ${componentName} ${script}`);\n      dockerfile += `COPY component${i}-${componentName}.sh /tmp\\n`;\n      dockerfile += `RUN /tmp/component${i}-${componentName}.sh\\n`;\n\n      const dockerCommands = this.components[i].getDockerCommands(this.os, this.architecture);\n      dockerfile += dockerCommands.join('\\n') + '\\n';\n      hashedComponents.push(`__ DOCKER COMMAND ${i} ${dockerCommands.join('\\n')}`);\n    }\n\n    commands.push(`cat > Dockerfile <<'EOFGITHUBRUNNERSDOCKERFILE'\\n${dockerfile}\\nEOFGITHUBRUNNERSDOCKERFILE`);\n\n    return [commands, hashedComponents];\n  }\n\n  private getBuildSpec(repository: ecr.Repository): [codebuild.BuildSpec, string] {\n    const thisStack = cdk.Stack.of(this);\n\n    let archUrl;\n    if (this.architecture.is(Architecture.X86_64)) {\n      archUrl = 'x86_64';\n    } else if (this.architecture.is(Architecture.ARM64)) {\n      archUrl = 'arm64';\n    } else {\n      throw new Error(`Unsupported architecture for required CodeBuild: ${this.architecture.name}`);\n    }\n\n    const [commands, commandsHashedComponents] = this.getDockerfileGenerationCommands();\n\n    const buildSpecVersion = 'v1'; // change this every time the build spec changes\n    const hashedComponents = commandsHashedComponents.concat(buildSpecVersion, this.architecture.name, this.baseImage, this.os.name);\n    const hash = crypto.createHash('md5').update(hashedComponents.join('\\n')).digest('hex').slice(0, 10);\n\n    const buildSpec = codebuild.BuildSpec.fromObject({\n      version: '0.2',\n      env: {\n        variables: {\n          REPO_ARN: repository.repositoryArn,\n          REPO_URI: repository.repositoryUri,\n          WAIT_HANDLE: 'unspecified',\n          BASH_ENV: 'codebuild-log.sh',\n        },\n        shell: 'bash',\n      },\n      phases: {\n        pre_build: {\n          commands: [\n            'echo \"exec > >(tee -a /tmp/codebuild.log) 2>&1\" > codebuild-log.sh',\n            `aws ecr get-login-password --region \"$AWS_DEFAULT_REGION\" | docker login --username AWS --password-stdin ${thisStack.account}.dkr.ecr.${thisStack.region}.amazonaws.com`,\n          ],\n        },\n        build: {\n          commands: commands.concat(\n            'docker build --progress plain . -t \"$REPO_URI\"',\n            'docker push \"$REPO_URI\"',\n          ),\n        },\n        post_build: {\n          commands: [\n            'rm -f codebuild-log.sh && STATUS=\"SUCCESS\"',\n            'if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ]; then STATUS=\"FAILURE\"; fi',\n            'cat <<EOF > /tmp/payload.json\\n' +\n              '{\\n' +\n              '  \"Status\": \"$STATUS\",\\n' +\n              '  \"UniqueId\": \"build\",\\n' +\n              // we remove non-printable characters from the log because CloudFormation doesn't like them\n              // https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1601\n              '  \"Reason\": `sed \\'s/[^[:print:]]//g\\' /tmp/codebuild.log | tail -c 400 | jq -Rsa .`,\\n' +\n              // for lambda always get a new value because there is always a new image hash\n              '  \"Data\": \"$RANDOM\"\\n' +\n              '}\\n' +\n              'EOF',\n            'if [ \"$WAIT_HANDLE\" != \"unspecified\" ]; then jq . /tmp/payload.json; curl -fsSL -X PUT -H \"Content-Type:\" -d \"@/tmp/payload.json\" \"$WAIT_HANDLE\"; fi',\n            // generate and push soci index\n            // we do this after finishing the build, so we don't have to wait. it's also not required, so it's ok if it fails\n            'docker rmi \"$REPO_URI\"', // it downloads the image again to /tmp, so save on space\n            'LATEST_SOCI_VERSION=`curl -w \"%{redirect_url}\" -fsS https://github.com/CloudSnorkel/standalone-soci-indexer/releases/latest | grep -oE \"[^/]+$\"`',\n            `curl -fsSL https://github.com/CloudSnorkel/standalone-soci-indexer/releases/download/$\\{LATEST_SOCI_VERSION}/standalone-soci-indexer_Linux_${archUrl}.tar.gz | tar xz`,\n            './standalone-soci-indexer \"$REPO_URI\"',\n          ],\n        },\n      },\n    });\n\n    return [buildSpec, hash];\n  }\n\n  private customResource(project: codebuild.Project, buildSpecHash: string) {\n    const crHandler = singletonLambda(BuildImageFunction, this, 'build-image', {\n      description: 'Custom resource handler that triggers CodeBuild to build runner images, and cleans-up images on deletion',\n      timeout: cdk.Duration.minutes(3),\n      logGroup: singletonLogGroup(this, SingletonLogType.RUNNER_IMAGE_BUILD),\n      logFormat: lambda.LogFormat.JSON,\n    });\n\n    const policy = new iam.Policy(this, 'CR Policy', {\n      statements: [\n        new iam.PolicyStatement({\n          actions: ['codebuild:StartBuild'],\n          resources: [project.projectArn],\n        }),\n      ],\n    });\n    crHandler.role!.attachInlinePolicy(policy);\n\n    let waitHandleRef= 'unspecified';\n    let waitDependable = '';\n\n    if (this.waitOnDeploy) {\n      // Wait handle lets us wait for longer than an hour for the image build to complete.\n      // We generate a new wait handle for build spec changes to guarantee a new image is built.\n      // This also helps make sure the changes are good. If they have a bug, the deployment will fail instead of just the scheduled build.\n      // Finally, it's recommended by CloudFormation docs to not reuse wait handles or old responses may interfere in some cases.\n      const handle = new cloudformation.CfnWaitConditionHandle(this, `Build Wait Handle ${buildSpecHash}`);\n      const wait = new cloudformation.CfnWaitCondition(this, `Build Wait ${buildSpecHash}`, {\n        handle: handle.ref,\n        timeout: this.timeout.toSeconds().toString(), // don't wait longer than the build timeout\n        count: 1,\n      });\n      waitHandleRef = handle.ref;\n      waitDependable = wait.ref;\n    }\n\n    const cr = new CustomResource(this, 'Builder', {\n      serviceToken: crHandler.functionArn,\n      resourceType: 'Custom::ImageBuilder',\n      properties: <BuildImageFunctionProperties>{\n        RepoName: this.repository.repositoryName,\n        ProjectName: project.projectName,\n        WaitHandle: waitHandleRef,\n      },\n    });\n\n    // add dependencies to make sure resources are there when we need them\n    cr.node.addDependency(project);\n    cr.node.addDependency(this.role);\n    cr.node.addDependency(policy);\n    cr.node.addDependency(crHandler.role!);\n    cr.node.addDependency(crHandler);\n\n    return waitDependable; // user needs to wait on wait handle which is triggered when the image is built\n  }\n\n  private rebuildImageOnSchedule(project: codebuild.Project, rebuildInterval?: Duration) {\n    rebuildInterval = rebuildInterval ?? Duration.days(7);\n    if (rebuildInterval.toMilliseconds() != 0) {\n      const scheduleRule = new events.Rule(this, 'Build Schedule', {\n        description: `Rebuild runner image for ${this.repository.repositoryName}`,\n        schedule: events.Schedule.rate(rebuildInterval),\n      });\n      scheduleRule.addTarget(new events_targets.CodeBuildProject(project));\n    }\n  }\n\n  get connections(): ec2.Connections {\n    return new ec2.Connections({\n      securityGroups: this.securityGroups,\n    });\n  }\n\n  get grantPrincipal(): iam.IPrincipal {\n    return this.role;\n  }\n}\n\n/**\n * @internal\n */\nexport class CodeBuildImageBuilderFailedBuildNotifier implements cdk.IAspect {\n  constructor(private topic: sns.ITopic) {\n  }\n\n  public visit(node: IConstruct): void {\n    if (node instanceof CodeBuildRunnerImageBuilder) {\n      const builder = node as CodeBuildRunnerImageBuilder;\n      const projectNode = builder.node.tryFindChild('CodeBuild');\n      if (projectNode) {\n        const project = projectNode as codebuild.Project;\n        project.notifyOnBuildFailed('BuildFailed', this.topic);\n      } else {\n        cdk.Annotations.of(builder).addWarning('Unused builder cannot get notifications of failed builds');\n      }\n    }\n  }\n}\n"]}
@@ -195,6 +195,16 @@ export interface RunnerImageBuilderProps {
195
195
  * Options specific to AWS Image Builder. Only used when builderType is RunnerImageBuilderType.AWS_IMAGE_BUILDER.
196
196
  */
197
197
  readonly awsImageBuilderOptions?: AwsImageBuilderRunnerImageBuilderProps;
198
+ /**
199
+ * Wait for image to finish building during deployment. It's usually best to leave this enabled to ensure everything is ready once deployment is done. However, it can be disabled to speed up deployment in case where you have a lot of image components that can take a long time to build.
200
+ *
201
+ * Disabling this option means a finished deployment is not ready to be used. You will have to wait for the image to finish building before the system can be used.
202
+ *
203
+ * Disabling this option may also mean any changes to settings or components can take up to a week (default rebuild interval) to take effect.
204
+ *
205
+ * @default true
206
+ */
207
+ readonly waitOnDeploy?: boolean;
198
208
  }
199
209
  export declare enum RunnerImageBuilderType {
200
210
  /**
@@ -48,4 +48,4 @@ class RunnerImageBuilderBase extends constructs_1.Construct {
48
48
  }
49
49
  }
50
50
  exports.RunnerImageBuilderBase = RunnerImageBuilderBase;
51
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/image-builders/common.ts"],"names":[],"mappings":";;;AAAA,mCAAmC;AAEnC,2CAAuC;AAMvC;;GAEG;AACH,SAAgB,sBAAsB,CAAC,KAAgB;IACrD,OAAO,GAAG,CAAC,KAAK,CAAC,kBAAkB,CAAC,KAAK,EAAE;QACzC,SAAS,EAAE,GAAG;QACd,SAAS,EAAE,GAAG;QACd,wBAAwB,EAAE,IAAI;KAC/B,CAAC,CAAC;AACL,CAAC;AAND,wDAMC;AA4ND,IAAY,sBAcX;AAdD,WAAY,sBAAsB;IAChC;;;;OAIG;IACH,kDAAwB,CAAA;IAExB;;;;OAIG;IACH,+DAAqC,CAAA;AACvC,CAAC,EAdW,sBAAsB,sCAAtB,sBAAsB,QAcjC;AAkDD;;GAEG;AACH,MAAsB,sBAAuB,SAAQ,sBAAS;IAG5D,YAAsB,KAAgB,EAAE,EAAU,EAAE,KAA+B;QACjF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAHT,eAAU,GAA2B,EAAE,CAAC;QAKhD,IAAI,KAAK,EAAE,UAAU,EAAE,CAAC;YACtB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,UAAU,CAAC,CAAC;QAC5C,CAAC;IACH,CAAC;IASM,YAAY,CAAC,SAA+B;QACjD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAEM,eAAe,CAAC,SAA+B;QACpD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI,CAAC,CAAC;IAC3E,CAAC;CACF;AAzBD,wDAyBC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { aws_ec2 as ec2, aws_iam as iam, aws_logs as logs, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { AwsImageBuilderRunnerImageBuilderProps } from './aws-image-builder';\nimport { CodeBuildRunnerImageBuilderProps } from './codebuild';\nimport { RunnerImageComponent } from './components';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers';\n\n/**\n * @internal\n */\nexport function uniqueImageBuilderName(scope: Construct): string {\n  return cdk.Names.uniqueResourceName(scope, {\n    maxLength: 126,\n    separator: '-',\n    allowedSpecialCharacters: '_-',\n  });\n}\n\n/**\n * @internal\n */\nexport interface ImageBuilderBaseProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * List of supported architectures to be checked against {@link architecture}.\n   */\n  readonly supportedArchitectures: Architecture[];\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX_UBUNTU\n   */\n  readonly os?: Os;\n\n  /**\n   * List of supported OS to be checked against {@link os}.\n   */\n  readonly supportedOs: Os[];\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Schedule the AMI to be rebuilt every given interval. Useful for keeping the AMI up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC where builder instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security groups to assign to launched builder instances.\n   *\n   * @default new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * Pipeline and infrastructure description.\n   */\n  readonly imageTypeName: string;\n}\n\n/**\n * Asset to copy into a built image.\n */\nexport interface RunnerImageAsset {\n  /**\n   * Path on local system to copy into the image. Can be a file or a directory.\n   */\n  readonly source: string;\n\n  /**\n   * Target path in the built image.\n   */\n  readonly target: string;\n}\n\nexport interface RunnerImageBuilderProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX_UBUNTU\n   */\n  readonly os?: Os;\n\n  /**\n   * Base image from which Docker runner images will be built.\n   *\n   * @default public.ecr.aws/lts/ubuntu:22.04 for Os.LINUX_UBUNTU, public.ecr.aws/amazonlinux/amazonlinux:2 for Os.LINUX_AMAZON_2, mcr.microsoft.com/windows/servercore:ltsc2019-amd64 for Os.WINDOWS\n   */\n  readonly baseDockerImage?: string;\n\n  /**\n   * Base AMI from which runner AMIs will be built.\n   *\n   * This can be an actual AMI or an AWS Image Builder ARN that points to the latest AMI. For example `arn:aws:imagebuilder:us-east-1:aws:image/ubuntu-server-22-lts-x86/x.x.x` would always use the latest version of Ubuntu 22.04 in each build. If you want a specific version, you can replace `x.x.x` with that version.\n   *\n   * @default latest Ubuntu 22.04 AMI for Os.LINUX_UBUNTU, latest Amazon Linux 2 AMI for Os.LINUX_AMAZON_2, latest Windows Server 2022 AMI for Os.WINDOWS\n   */\n  readonly baseAmi?: string;\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Components to install on the image.\n   *\n   * @default none\n   */\n  readonly components?: RunnerImageComponent[];\n\n  /**\n   * Schedule the image to be rebuilt every given interval. Useful for keeping the image up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC to build the image in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Groups to assign to this instance.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the CodeBuild logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * @default CodeBuild for Linux Docker image, AWS Image Builder for Windows Docker image and any AMI\n   */\n  readonly builderType?: RunnerImageBuilderType;\n\n  /**\n   * Options specific to CodeBuild image builder. Only used when builderType is RunnerImageBuilderType.CODE_BUILD.\n   */\n  readonly codeBuildOptions?: CodeBuildRunnerImageBuilderProps;\n\n  /**\n   * Options specific to AWS Image Builder. Only used when builderType is RunnerImageBuilderType.AWS_IMAGE_BUILDER.\n   */\n  readonly awsImageBuilderOptions?: AwsImageBuilderRunnerImageBuilderProps;\n}\n\nexport enum RunnerImageBuilderType {\n  /**\n   * Build runner images using AWS CodeBuild.\n   *\n   * Faster than AWS Image Builder, but can only be used to build Linux Docker images.\n   */\n  CODE_BUILD = 'CodeBuild',\n\n  /**\n   * Build runner images using AWS Image Builder.\n   *\n   * Slower than CodeBuild, but can be used to build any type of image including AMIs and Windows images.\n   */\n  AWS_IMAGE_BUILDER = 'AwsImageBuilder',\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.\n *\n * An image can be a Docker image or AMI.\n */\nexport interface IRunnerImageBuilder {\n  /**\n   * Build and return a Docker image with GitHub Runner installed in it.\n   *\n   * Anything that ends up with an ECR repository containing a Docker image that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing image and nothing else.\n   *\n   * It's important that the specified image tag be available at the time the repository is available. Providers usually assume the image is ready and will fail if it's not.\n   *\n   * The image can be further updated over time manually or using a schedule as long as it is always written to the same tag.\n   */\n  bindDockerImage(): RunnerImage;\n\n  /**\n   * Build and return an AMI with GitHub Runner installed in it.\n   *\n   * Anything that ends up with a launch template pointing to an AMI that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing AMI and nothing else.\n   *\n   * The AMI can be further updated over time manually or using a schedule as long as it is always written to the same launch template.\n   */\n  bindAmi(): RunnerAmi;\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}. The image can be configured by adding or removing components. The image builder can be configured by adding grants or allowing connections.\n *\n * An image can be a Docker image or AMI.\n */\nexport interface IConfigurableRunnerImageBuilder extends IRunnerImageBuilder, ec2.IConnectable, iam.IGrantable {\n  /**\n   * Add a component to the image builder. The component will be added to the end of the list of components.\n   *\n   * @param component component to add\n   */\n  addComponent(component: RunnerImageComponent): void;\n\n  /**\n   * Remove a component from the image builder. Removal is done by component name. Multiple components with the same name will all be removed.\n   *\n   * @param component component to remove\n   */\n  removeComponent(component: RunnerImageComponent): void;\n}\n\n/**\n * @internal\n */\nexport abstract class RunnerImageBuilderBase extends Construct implements IConfigurableRunnerImageBuilder {\n  protected components: RunnerImageComponent[] = [];\n\n  protected constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id);\n\n    if (props?.components) {\n      this.components.push(...props.components);\n    }\n  }\n\n  abstract bindDockerImage(): RunnerImage;\n\n  abstract bindAmi(): RunnerAmi;\n\n  abstract get connections(): ec2.Connections;\n  abstract get grantPrincipal(): iam.IPrincipal;\n\n  public addComponent(component: RunnerImageComponent) {\n    this.components.push(component);\n  }\n\n  public removeComponent(component: RunnerImageComponent) {\n    this.components = this.components.filter(c => c.name !== component.name);\n  }\n}\n\n"]}
51
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["../../src/image-builders/common.ts"],"names":[],"mappings":";;;AAAA,mCAAmC;AAEnC,2CAAuC;AAMvC;;GAEG;AACH,SAAgB,sBAAsB,CAAC,KAAgB;IACrD,OAAO,GAAG,CAAC,KAAK,CAAC,kBAAkB,CAAC,KAAK,EAAE;QACzC,SAAS,EAAE,GAAG;QACd,SAAS,EAAE,GAAG;QACd,wBAAwB,EAAE,IAAI;KAC/B,CAAC,CAAC;AACL,CAAC;AAND,wDAMC;AAuOD,IAAY,sBAcX;AAdD,WAAY,sBAAsB;IAChC;;;;OAIG;IACH,kDAAwB,CAAA;IAExB;;;;OAIG;IACH,+DAAqC,CAAA;AACvC,CAAC,EAdW,sBAAsB,sCAAtB,sBAAsB,QAcjC;AAkDD;;GAEG;AACH,MAAsB,sBAAuB,SAAQ,sBAAS;IAG5D,YAAsB,KAAgB,EAAE,EAAU,EAAE,KAA+B;QACjF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAHT,eAAU,GAA2B,EAAE,CAAC;QAKhD,IAAI,KAAK,EAAE,UAAU,EAAE,CAAC;YACtB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,UAAU,CAAC,CAAC;QAC5C,CAAC;IACH,CAAC;IASM,YAAY,CAAC,SAA+B;QACjD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAEM,eAAe,CAAC,SAA+B;QACpD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI,CAAC,CAAC;IAC3E,CAAC;CACF;AAzBD,wDAyBC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { aws_ec2 as ec2, aws_iam as iam, aws_logs as logs, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { AwsImageBuilderRunnerImageBuilderProps } from './aws-image-builder';\nimport { CodeBuildRunnerImageBuilderProps } from './codebuild';\nimport { RunnerImageComponent } from './components';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers';\n\n/**\n * @internal\n */\nexport function uniqueImageBuilderName(scope: Construct): string {\n  return cdk.Names.uniqueResourceName(scope, {\n    maxLength: 126,\n    separator: '-',\n    allowedSpecialCharacters: '_-',\n  });\n}\n\n/**\n * @internal\n */\nexport interface ImageBuilderBaseProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * List of supported architectures to be checked against {@link architecture}.\n   */\n  readonly supportedArchitectures: Architecture[];\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX_UBUNTU\n   */\n  readonly os?: Os;\n\n  /**\n   * List of supported OS to be checked against {@link os}.\n   */\n  readonly supportedOs: Os[];\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Schedule the AMI to be rebuilt every given interval. Useful for keeping the AMI up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC where builder instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security groups to assign to launched builder instances.\n   *\n   * @default new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * Pipeline and infrastructure description.\n   */\n  readonly imageTypeName: string;\n}\n\n/**\n * Asset to copy into a built image.\n */\nexport interface RunnerImageAsset {\n  /**\n   * Path on local system to copy into the image. Can be a file or a directory.\n   */\n  readonly source: string;\n\n  /**\n   * Target path in the built image.\n   */\n  readonly target: string;\n}\n\nexport interface RunnerImageBuilderProps {\n  /**\n   * Image architecture.\n   *\n   * @default Architecture.X86_64\n   */\n  readonly architecture?: Architecture;\n\n  /**\n   * Image OS.\n   *\n   * @default OS.LINUX_UBUNTU\n   */\n  readonly os?: Os;\n\n  /**\n   * Base image from which Docker runner images will be built.\n   *\n   * @default public.ecr.aws/lts/ubuntu:22.04 for Os.LINUX_UBUNTU, public.ecr.aws/amazonlinux/amazonlinux:2 for Os.LINUX_AMAZON_2, mcr.microsoft.com/windows/servercore:ltsc2019-amd64 for Os.WINDOWS\n   */\n  readonly baseDockerImage?: string;\n\n  /**\n   * Base AMI from which runner AMIs will be built.\n   *\n   * This can be an actual AMI or an AWS Image Builder ARN that points to the latest AMI. For example `arn:aws:imagebuilder:us-east-1:aws:image/ubuntu-server-22-lts-x86/x.x.x` would always use the latest version of Ubuntu 22.04 in each build. If you want a specific version, you can replace `x.x.x` with that version.\n   *\n   * @default latest Ubuntu 22.04 AMI for Os.LINUX_UBUNTU, latest Amazon Linux 2 AMI for Os.LINUX_AMAZON_2, latest Windows Server 2022 AMI for Os.WINDOWS\n   */\n  readonly baseAmi?: string;\n\n  /**\n   * Version of GitHub Runners to install.\n   *\n   * @default latest version available\n   */\n  readonly runnerVersion?: RunnerVersion;\n\n  /**\n   * Components to install on the image.\n   *\n   * @default none\n   */\n  readonly components?: RunnerImageComponent[];\n\n  /**\n   * Schedule the image to be rebuilt every given interval. Useful for keeping the image up-do-date with the latest GitHub runner version and latest OS updates.\n   *\n   * Set to zero to disable.\n   *\n   * @default Duration.days(7)\n   */\n  readonly rebuildInterval?: Duration;\n\n  /**\n   * VPC to build the image in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Groups to assign to this instance.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs. When updating\n   * this property, unsetting it doesn't remove the log retention policy. To\n   * remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * Removal policy for logs of image builds. If deployment fails on the custom resource, try setting this to `RemovalPolicy.RETAIN`. This way the CodeBuild logs can still be viewed, and you can see why the build failed.\n   *\n   * We try to not leave anything behind when removed. But sometimes a log staying behind is useful.\n   *\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly logRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * @default CodeBuild for Linux Docker image, AWS Image Builder for Windows Docker image and any AMI\n   */\n  readonly builderType?: RunnerImageBuilderType;\n\n  /**\n   * Options specific to CodeBuild image builder. Only used when builderType is RunnerImageBuilderType.CODE_BUILD.\n   */\n  readonly codeBuildOptions?: CodeBuildRunnerImageBuilderProps;\n\n  /**\n   * Options specific to AWS Image Builder. Only used when builderType is RunnerImageBuilderType.AWS_IMAGE_BUILDER.\n   */\n  readonly awsImageBuilderOptions?: AwsImageBuilderRunnerImageBuilderProps;\n\n  /**\n   * Wait for image to finish building during deployment. It's usually best to leave this enabled to ensure everything is ready once deployment is done. However, it can be disabled to speed up deployment in case where you have a lot of image components that can take a long time to build.\n   *\n   * Disabling this option means a finished deployment is not ready to be used. You will have to wait for the image to finish building before the system can be used.\n   *\n   * Disabling this option may also mean any changes to settings or components can take up to a week (default rebuild interval) to take effect.\n   *\n   * @default true\n   */\n  readonly waitOnDeploy?: boolean;\n}\n\nexport enum RunnerImageBuilderType {\n  /**\n   * Build runner images using AWS CodeBuild.\n   *\n   * Faster than AWS Image Builder, but can only be used to build Linux Docker images.\n   */\n  CODE_BUILD = 'CodeBuild',\n\n  /**\n   * Build runner images using AWS Image Builder.\n   *\n   * Slower than CodeBuild, but can be used to build any type of image including AMIs and Windows images.\n   */\n  AWS_IMAGE_BUILDER = 'AwsImageBuilder',\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.\n *\n * An image can be a Docker image or AMI.\n */\nexport interface IRunnerImageBuilder {\n  /**\n   * Build and return a Docker image with GitHub Runner installed in it.\n   *\n   * Anything that ends up with an ECR repository containing a Docker image that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing image and nothing else.\n   *\n   * It's important that the specified image tag be available at the time the repository is available. Providers usually assume the image is ready and will fail if it's not.\n   *\n   * The image can be further updated over time manually or using a schedule as long as it is always written to the same tag.\n   */\n  bindDockerImage(): RunnerImage;\n\n  /**\n   * Build and return an AMI with GitHub Runner installed in it.\n   *\n   * Anything that ends up with a launch template pointing to an AMI that runs GitHub self-hosted runners can be used. A simple implementation could even point to an existing AMI and nothing else.\n   *\n   * The AMI can be further updated over time manually or using a schedule as long as it is always written to the same launch template.\n   */\n  bindAmi(): RunnerAmi;\n}\n\n/**\n * Interface for constructs that build an image that can be used in {@link IRunnerProvider}. The image can be configured by adding or removing components. The image builder can be configured by adding grants or allowing connections.\n *\n * An image can be a Docker image or AMI.\n */\nexport interface IConfigurableRunnerImageBuilder extends IRunnerImageBuilder, ec2.IConnectable, iam.IGrantable {\n  /**\n   * Add a component to the image builder. The component will be added to the end of the list of components.\n   *\n   * @param component component to add\n   */\n  addComponent(component: RunnerImageComponent): void;\n\n  /**\n   * Remove a component from the image builder. Removal is done by component name. Multiple components with the same name will all be removed.\n   *\n   * @param component component to remove\n   */\n  removeComponent(component: RunnerImageComponent): void;\n}\n\n/**\n * @internal\n */\nexport abstract class RunnerImageBuilderBase extends Construct implements IConfigurableRunnerImageBuilder {\n  protected components: RunnerImageComponent[] = [];\n\n  protected constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id);\n\n    if (props?.components) {\n      this.components.push(...props.components);\n    }\n  }\n\n  abstract bindDockerImage(): RunnerImage;\n\n  abstract bindAmi(): RunnerAmi;\n\n  abstract get connections(): ec2.Connections;\n  abstract get grantPrincipal(): iam.IPrincipal;\n\n  public addComponent(component: RunnerImageComponent) {\n    this.components.push(component);\n  }\n\n  public removeComponent(component: RunnerImageComponent) {\n    this.components = this.components.filter(c => c.name !== component.name);\n  }\n}\n\n"]}