@cloudsnorkel/cdk-github-runners 0.9.6 → 0.9.7

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 (31) hide show
  1. package/.jsii +93 -76
  2. package/API.md +16 -3
  3. package/assets/delete-runner.lambda/index.js +69 -63
  4. package/assets/image-builders/aws-image-builder/versioner.lambda/index.js +69 -63
  5. package/assets/status.lambda/index.js +69 -63
  6. package/assets/token-retriever.lambda/index.js +69 -63
  7. package/lib/access.js +1 -1
  8. package/lib/image-builders/api.js +1 -1
  9. package/lib/image-builders/aws-image-builder/builder.js +4 -7
  10. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  11. package/lib/image-builders/aws-image-builder/deprecated/container.d.ts +1 -1
  12. package/lib/image-builders/aws-image-builder/deprecated/container.js +9 -12
  13. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  14. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  15. package/lib/image-builders/codebuild-deprecated.d.ts +1 -1
  16. package/lib/image-builders/codebuild-deprecated.js +15 -19
  17. package/lib/image-builders/codebuild.js +5 -8
  18. package/lib/image-builders/components.js +1 -1
  19. package/lib/image-builders/static.d.ts +1 -1
  20. package/lib/image-builders/static.js +7 -6
  21. package/lib/providers/codebuild.js +2 -2
  22. package/lib/providers/common.d.ts +6 -0
  23. package/lib/providers/common.js +4 -4
  24. package/lib/providers/ec2.js +2 -2
  25. package/lib/providers/ecs.d.ts +6 -2
  26. package/lib/providers/ecs.js +64 -32
  27. package/lib/providers/fargate.js +2 -2
  28. package/lib/providers/lambda.js +12 -3
  29. package/lib/runner.js +1 -1
  30. package/lib/secrets.js +1 -1
  31. package/package.json +5 -5
@@ -10322,7 +10322,7 @@ var require_semver = __commonJS({
10322
10322
  version = version.version;
10323
10323
  }
10324
10324
  } else if (typeof version !== "string") {
10325
- throw new TypeError(`Invalid Version: ${version}`);
10325
+ throw new TypeError(`Invalid Version: ${require("util").inspect(version)}`);
10326
10326
  }
10327
10327
  if (version.length > MAX_LENGTH) {
10328
10328
  throw new TypeError(
@@ -10495,9 +10495,13 @@ var require_semver = __commonJS({
10495
10495
  }
10496
10496
  this.prerelease = [];
10497
10497
  break;
10498
- case "pre":
10498
+ case "pre": {
10499
+ const base = Number(identifierBase) ? 1 : 0;
10500
+ if (!identifier && identifierBase === false) {
10501
+ throw new Error("invalid increment argument: identifier is empty");
10502
+ }
10499
10503
  if (this.prerelease.length === 0) {
10500
- this.prerelease = [0];
10504
+ this.prerelease = [base];
10501
10505
  } else {
10502
10506
  let i = this.prerelease.length;
10503
10507
  while (--i >= 0) {
@@ -10507,20 +10511,27 @@ var require_semver = __commonJS({
10507
10511
  }
10508
10512
  }
10509
10513
  if (i === -1) {
10510
- this.prerelease.push(0);
10514
+ if (identifier === this.prerelease.join(".") && identifierBase === false) {
10515
+ throw new Error("invalid increment argument: identifier already exists");
10516
+ }
10517
+ this.prerelease.push(base);
10511
10518
  }
10512
10519
  }
10513
10520
  if (identifier) {
10514
- const base = Number(identifierBase) ? 1 : 0;
10521
+ let prerelease = [identifier, base];
10522
+ if (identifierBase === false) {
10523
+ prerelease = [identifier];
10524
+ }
10515
10525
  if (compareIdentifiers(this.prerelease[0], identifier) === 0) {
10516
10526
  if (isNaN(this.prerelease[1])) {
10517
- this.prerelease = [identifier, base];
10527
+ this.prerelease = prerelease;
10518
10528
  }
10519
10529
  } else {
10520
- this.prerelease = [identifier, base];
10530
+ this.prerelease = prerelease;
10521
10531
  }
10522
10532
  }
10523
10533
  break;
10534
+ }
10524
10535
  default:
10525
10536
  throw new Error(`invalid increment argument: ${release}`);
10526
10537
  }
@@ -10536,22 +10547,18 @@ var require_semver = __commonJS({
10536
10547
  // node_modules/semver/functions/parse.js
10537
10548
  var require_parse = __commonJS({
10538
10549
  "node_modules/semver/functions/parse.js"(exports2, module2) {
10539
- var { MAX_LENGTH } = require_constants();
10540
10550
  var SemVer = require_semver();
10541
- var parse = (version, options) => {
10551
+ var parse = (version, options, throwErrors = false) => {
10542
10552
  if (version instanceof SemVer) {
10543
10553
  return version;
10544
10554
  }
10545
- if (typeof version !== "string") {
10546
- return null;
10547
- }
10548
- if (version.length > MAX_LENGTH) {
10549
- return null;
10550
- }
10551
10555
  try {
10552
10556
  return new SemVer(version, options);
10553
10557
  } catch (er) {
10554
- return null;
10558
+ if (!throwErrors) {
10559
+ return null;
10560
+ }
10561
+ throw er;
10555
10562
  }
10556
10563
  };
10557
10564
  module2.exports = parse;
@@ -10605,60 +10612,41 @@ var require_inc = __commonJS({
10605
10612
  }
10606
10613
  });
10607
10614
 
10608
- // node_modules/semver/functions/compare.js
10609
- var require_compare = __commonJS({
10610
- "node_modules/semver/functions/compare.js"(exports2, module2) {
10611
- var SemVer = require_semver();
10612
- var compare = (a, b, loose) => new SemVer(a, loose).compare(new SemVer(b, loose));
10613
- module2.exports = compare;
10614
- }
10615
- });
10616
-
10617
- // node_modules/semver/functions/eq.js
10618
- var require_eq = __commonJS({
10619
- "node_modules/semver/functions/eq.js"(exports2, module2) {
10620
- var compare = require_compare();
10621
- var eq = (a, b, loose) => compare(a, b, loose) === 0;
10622
- module2.exports = eq;
10623
- }
10624
- });
10625
-
10626
10615
  // node_modules/semver/functions/diff.js
10627
10616
  var require_diff = __commonJS({
10628
10617
  "node_modules/semver/functions/diff.js"(exports2, module2) {
10629
10618
  var parse = require_parse();
10630
- var eq = require_eq();
10631
10619
  var diff = (version1, version2) => {
10632
- const v1 = parse(version1);
10633
- const v2 = parse(version2);
10634
- if (eq(v1, v2)) {
10620
+ const v1 = parse(version1, null, true);
10621
+ const v2 = parse(version2, null, true);
10622
+ const comparison = v1.compare(v2);
10623
+ if (comparison === 0) {
10635
10624
  return null;
10636
- } else {
10637
- const hasPre = v1.prerelease.length || v2.prerelease.length;
10638
- const prefix = hasPre ? "pre" : "";
10639
- const defaultResult = hasPre ? "prerelease" : "";
10640
- if (v1.major !== v2.major) {
10641
- return prefix + "major";
10642
- }
10643
- if (v1.minor !== v2.minor) {
10644
- return prefix + "minor";
10645
- }
10646
- if (v1.patch !== v2.patch) {
10647
- return prefix + "patch";
10648
- }
10649
- if (!v1.prerelease.length || !v2.prerelease.length) {
10650
- if (v1.patch) {
10651
- return "patch";
10652
- }
10653
- if (v1.minor) {
10654
- return "minor";
10655
- }
10656
- if (v1.major) {
10657
- return "major";
10658
- }
10659
- }
10660
- return defaultResult;
10661
10625
  }
10626
+ const v1Higher = comparison > 0;
10627
+ const highVersion = v1Higher ? v1 : v2;
10628
+ const lowVersion = v1Higher ? v2 : v1;
10629
+ const highHasPre = !!highVersion.prerelease.length;
10630
+ const prefix = highHasPre ? "pre" : "";
10631
+ if (v1.major !== v2.major) {
10632
+ return prefix + "major";
10633
+ }
10634
+ if (v1.minor !== v2.minor) {
10635
+ return prefix + "minor";
10636
+ }
10637
+ if (v1.patch !== v2.patch) {
10638
+ return prefix + "patch";
10639
+ }
10640
+ if (highHasPre) {
10641
+ return "prerelease";
10642
+ }
10643
+ if (lowVersion.patch) {
10644
+ return "patch";
10645
+ }
10646
+ if (lowVersion.minor) {
10647
+ return "minor";
10648
+ }
10649
+ return "major";
10662
10650
  };
10663
10651
  module2.exports = diff;
10664
10652
  }
@@ -10703,6 +10691,15 @@ var require_prerelease = __commonJS({
10703
10691
  }
10704
10692
  });
10705
10693
 
10694
+ // node_modules/semver/functions/compare.js
10695
+ var require_compare = __commonJS({
10696
+ "node_modules/semver/functions/compare.js"(exports2, module2) {
10697
+ var SemVer = require_semver();
10698
+ var compare = (a, b, loose) => new SemVer(a, loose).compare(new SemVer(b, loose));
10699
+ module2.exports = compare;
10700
+ }
10701
+ });
10702
+
10706
10703
  // node_modules/semver/functions/rcompare.js
10707
10704
  var require_rcompare = __commonJS({
10708
10705
  "node_modules/semver/functions/rcompare.js"(exports2, module2) {
@@ -10770,6 +10767,15 @@ var require_lt = __commonJS({
10770
10767
  }
10771
10768
  });
10772
10769
 
10770
+ // node_modules/semver/functions/eq.js
10771
+ var require_eq = __commonJS({
10772
+ "node_modules/semver/functions/eq.js"(exports2, module2) {
10773
+ var compare = require_compare();
10774
+ var eq = (a, b, loose) => compare(a, b, loose) === 0;
10775
+ module2.exports = eq;
10776
+ }
10777
+ });
10778
+
10773
10779
  // node_modules/semver/functions/neq.js
10774
10780
  var require_neq = __commonJS({
10775
10781
  "node_modules/semver/functions/neq.js"(exports2, module2) {
package/lib/access.js CHANGED
@@ -56,7 +56,7 @@ class LambdaAccess {
56
56
  }
57
57
  }
58
58
  _a = JSII_RTTI_SYMBOL_1;
59
- LambdaAccess[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.LambdaAccess", version: "0.9.6" };
59
+ LambdaAccess[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.LambdaAccess", version: "0.9.7" };
60
60
  exports.LambdaAccess = LambdaAccess;
61
61
  /**
62
62
  * @internal
@@ -42,6 +42,6 @@ class RunnerImageBuilder extends common_1.RunnerImageBuilderBase {
42
42
  }
43
43
  }
44
44
  _a = JSII_RTTI_SYMBOL_1;
45
- RunnerImageBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerImageBuilder", version: "0.9.6" };
45
+ RunnerImageBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.RunnerImageBuilder", version: "0.9.7" };
46
46
  exports.RunnerImageBuilder = RunnerImageBuilder;
47
47
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2ltYWdlLWJ1aWxkZXJzL2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUEwQztBQUUxQywyREFBd0U7QUFDeEUsMkNBQTBEO0FBQzFELHFDQUFtRztBQUNuRyxnREFBeUM7QUFFekM7Ozs7OztHQU1HO0FBQ0gsTUFBc0Isa0JBQW1CLFNBQVEsK0JBQXNCO0lBQ3JFOztPQUVHO0lBQ0gsTUFBTSxDQUFDLEdBQUcsQ0FBQyxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUErQjtRQUN0RSxJQUFJLEtBQUssRUFBRSxVQUFVLElBQUksS0FBSyxDQUFDLGFBQWEsRUFBRTtZQUM1Qyx5QkFBVyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQyxVQUFVLENBQUMsa0hBQWtILENBQUMsQ0FBQztTQUN0SjtRQUVELElBQUksS0FBSyxFQUFFLFdBQVcsS0FBSywrQkFBc0IsQ0FBQyxVQUFVLEVBQUU7WUFDNUQsT0FBTyxJQUFJLHVDQUEyQixDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7U0FDMUQ7YUFBTSxJQUFJLEtBQUssRUFBRSxXQUFXLEtBQUssK0JBQXNCLENBQUMsaUJBQWlCLEVBQUU7WUFDMUUsT0FBTyxJQUFJLHFEQUFpQyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7U0FDaEU7UUFFRCxNQUFNLEVBQUUsR0FBRyxLQUFLLEVBQUUsRUFBRSxJQUFJLFdBQUUsQ0FBQyxZQUFZLENBQUM7UUFDeEMsSUFBSSxFQUFFLENBQUMsRUFBRSxDQUFDLFdBQUUsQ0FBQyxZQUFZLENBQUMsSUFBSSxFQUFFLENBQUMsRUFBRSxDQUFDLFdBQUUsQ0FBQyxjQUFjLENBQUMsRUFBRTtZQUN0RCxPQUFPLElBQUksdUNBQTJCLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztTQUMxRDthQUFNLElBQUksRUFBRSxDQUFDLEVBQUUsQ0FBQyxXQUFFLENBQUMsT0FBTyxDQUFDLEVBQUU7WUFDNUIsT0FBTyxJQUFJLHFEQUFpQyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7U0FDaEU7YUFBTTtZQUNMLE1BQU0sSUFBSSxLQUFLLENBQUMsMERBQTBELEVBQUUsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO1NBQ3RGO0lBQ0gsQ0FBQzs7OztBQXZCbUIsZ0RBQWtCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQW5ub3RhdGlvbnMgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEF3c0ltYWdlQnVpbGRlclJ1bm5lckltYWdlQnVpbGRlciB9IGZyb20gJy4vYXdzLWltYWdlLWJ1aWxkZXInO1xuaW1wb3J0IHsgQ29kZUJ1aWxkUnVubmVySW1hZ2VCdWlsZGVyIH0gZnJvbSAnLi9jb2RlYnVpbGQnO1xuaW1wb3J0IHsgUnVubmVySW1hZ2VCdWlsZGVyQmFzZSwgUnVubmVySW1hZ2VCdWlsZGVyUHJvcHMsIFJ1bm5lckltYWdlQnVpbGRlclR5cGUgfSBmcm9tICcuL2NvbW1vbic7XG5pbXBvcnQgeyBPcyB9IGZyb20gJy4uL3Byb3ZpZGVycy9jb21tb24nO1xuXG4vKipcbiAqIEdpdEh1YiBSdW5uZXIgaW1hZ2UgYnVpbGRlci4gQnVpbGRzIGEgRG9ja2VyIGltYWdlIG9yIEFNSSB3aXRoIEdpdEh1YiBSdW5uZXIgYW5kIG90aGVyIHJlcXVpcmVtZW50cyBpbnN0YWxsZWQuXG4gKlxuICogSW1hZ2VzIGNhbiBiZSBjdXN0b21pemVkIGJlZm9yZSBwYXNzZWQgaW50byB0aGUgcHJvdmlkZXIgYnkgYWRkaW5nIG9yIHJlbW92aW5nIGNvbXBvbmVudHMgdG8gYmUgaW5zdGFsbGVkLlxuICpcbiAqIEltYWdlcyBhcmUgcmVidWlsdCBldmVyeSB3ZWVrIGJ5IGRlZmF1bHQgdG8gZW5zdXJlIHRoYXQgdGhlIGxhdGVzdCBzZWN1cml0eSBwYXRjaGVzIGFyZSBhcHBsaWVkLlxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgUnVubmVySW1hZ2VCdWlsZGVyIGV4dGVuZHMgUnVubmVySW1hZ2VCdWlsZGVyQmFzZSB7XG4gIC8qKlxuICAgKiBDcmVhdGUgYSBuZXcgaW1hZ2UgYnVpbGRlciBiYXNlZCBvbiB0aGUgcHJvdmlkZWQgcHJvcGVydGllcy4gVGhlIGltcGxlbWVudGF0aW9uIHdpbGwgZGlmZmVyIGJhc2VkIG9uIHRoZSBPUywgYXJjaGl0ZWN0dXJlLCBhbmQgcmVxdWVzdGVkIGJ1aWxkZXIgdHlwZS5cbiAgICovXG4gIHN0YXRpYyBuZXcoc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBSdW5uZXJJbWFnZUJ1aWxkZXJQcm9wcyk6IFJ1bm5lckltYWdlQnVpbGRlciB7XG4gICAgaWYgKHByb3BzPy5jb21wb25lbnRzICYmIHByb3BzLnJ1bm5lclZlcnNpb24pIHtcbiAgICAgIEFubm90YXRpb25zLm9mKHNjb3BlKS5hZGRXYXJuaW5nKCdydW5uZXJWZXJzaW9uIGlzIGlnbm9yZWQgd2hlbiBjb21wb25lbnRzIGFyZSBzcGVjaWZpZWQuIFRoZSBydW5uZXIgdmVyc2lvbiB3aWxsIGJlIGRldGVybWluZWQgYnkgdGhlIGNvbXBvbmVudHMuJyk7XG4gICAgfVxuXG4gICAgaWYgKHByb3BzPy5idWlsZGVyVHlwZSA9PT0gUnVubmVySW1hZ2VCdWlsZGVyVHlwZS5DT0RFX0JVSUxEKSB7XG4gICAgICByZXR1cm4gbmV3IENvZGVCdWlsZFJ1bm5lckltYWdlQnVpbGRlcihzY29wZSwgaWQsIHByb3BzKTtcbiAgICB9IGVsc2UgaWYgKHByb3BzPy5idWlsZGVyVHlwZSA9PT0gUnVubmVySW1hZ2VCdWlsZGVyVHlwZS5BV1NfSU1BR0VfQlVJTERFUikge1xuICAgICAgcmV0dXJuIG5ldyBBd3NJbWFnZUJ1aWxkZXJSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfVxuXG4gICAgY29uc3Qgb3MgPSBwcm9wcz8ub3MgPz8gT3MuTElOVVhfVUJVTlRVO1xuICAgIGlmIChvcy5pcyhPcy5MSU5VWF9VQlVOVFUpIHx8IG9zLmlzKE9zLkxJTlVYX0FNQVpPTl8yKSkge1xuICAgICAgcmV0dXJuIG5ldyBDb2RlQnVpbGRSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfSBlbHNlIGlmIChvcy5pcyhPcy5XSU5ET1dTKSkge1xuICAgICAgcmV0dXJuIG5ldyBBd3NJbWFnZUJ1aWxkZXJSdW5uZXJJbWFnZUJ1aWxkZXIoc2NvcGUsIGlkLCBwcm9wcyk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVW5hYmxlIHRvIGZpbmQgcnVubmVyIGltYWdlIGJ1aWxkZXIgaW1wbGVtZW50YXRpb24gZm9yICR7b3MubmFtZX1gKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -152,7 +152,7 @@ class ImageBuilderComponent extends common_1.ImageBuilderObjectBase {
152
152
  }
153
153
  }
154
154
  _a = JSII_RTTI_SYMBOL_1;
155
- ImageBuilderComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.ImageBuilderComponent", version: "0.9.6" };
155
+ ImageBuilderComponent[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.ImageBuilderComponent", version: "0.9.7" };
156
156
  exports.ImageBuilderComponent = ImageBuilderComponent;
157
157
  /**
158
158
  * @internal
@@ -268,16 +268,13 @@ class AwsImageBuilderRunnerImageBuilder extends common_2.RunnerImageBuilderBase
268
268
  this.imageCleaner(image, recipe.name, repository);
269
269
  this.reaper(recipe.name, 'Docker');
270
270
  this.boundDockerImage = {
271
- // There are simpler ways to get the ARN, but we want an image object that depends on the newly built image.
272
- // We want whoever is using this image to automatically wait for Image Builder to finish building before using the image.
273
- imageRepository: aws_cdk_lib_1.aws_ecr.Repository.fromRepositoryName(this, 'Dependable Image',
274
- // we can't use image.attrName because it comes up with upper case
275
- cdk.Fn.split(':', cdk.Fn.split('/', image.attrImageUri, 2)[1], 2)[0]),
271
+ imageRepository: repository,
276
272
  imageTag: 'latest',
277
273
  os: this.os,
278
274
  architecture: this.architecture,
279
275
  logGroup: log,
280
276
  runnerVersion: providers_1.RunnerVersion.specific('unknown'),
277
+ // no dependable as CloudFormation will fail to get image ARN once the image is deleted (we delete old images daily)
281
278
  };
282
279
  return this.boundDockerImage;
283
280
  }
@@ -565,4 +562,4 @@ class AwsImageBuilderFailedBuildNotifier {
565
562
  }
566
563
  }
567
564
  exports.AwsImageBuilderFailedBuildNotifier = AwsImageBuilderFailedBuildNotifier;
568
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../src/image-builders/aws-image-builder/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\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_imagebuilder as imagebuilder,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  aws_sns as sns,\n  aws_sns_subscriptions as subs,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct, IConstruct } from 'constructs';\nimport { AmiRecipe, defaultBaseAmi } from './ami';\nimport { ImageBuilderObjectBase } from './common';\nimport { ContainerRecipe, defaultBaseDockerImage } from './container';\nimport { DeleteAmiFunction } from './delete-ami-function';\nimport { FilterFailedBuildsFunction } from './filter-failed-builds-function';\nimport { ReaperFunction } from './reaper-function';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../providers';\nimport { BuildImageFunction } from '../../providers/build-image-function';\nimport { singletonLambda } from '../../utils';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps, uniqueImageBuilderName } from '../common';\n\nexport interface AwsImageBuilderRunnerImageBuilderProps {\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n}\n\n/**\n * An asset including file or directory to place inside the built image.\n */\nexport interface ImageBuilderAsset {\n  /**\n   * Path to place asset in the image.\n   */\n  readonly path: string;\n\n  /**\n   * Asset to place in the image.\n   */\n  readonly asset: s3_assets.Asset;\n}\n\n/**\n * Properties for ImageBuilderComponent construct.\n */\nexport interface ImageBuilderComponentProperties {\n  /**\n   * Component platform. Must match the builder platform.\n   */\n  readonly platform: 'Linux' | 'Windows';\n\n  /**\n   * Component display name.\n   */\n  readonly displayName: string;\n\n  /**\n   * Component description.\n   */\n  readonly description: string;\n\n  /**\n   * Shell commands to run when adding this component to the image.\n   *\n   * On Linux, these are bash commands. On Windows, there are PowerShell commands.\n   */\n  readonly commands: string[];\n\n  /**\n   * Optional assets to add to the built image.\n   */\n  readonly assets?: ImageBuilderAsset[];\n\n  /**\n   * Require a reboot after installing this component.\n   *\n   * @default false\n   */\n  readonly reboot?: boolean;\n}\n\n/**\n * Components are a set of commands to run and optional files to add to an image. Components are the building blocks of images built by Image Builder.\n *\n * Example:\n *\n * ```\n * new ImageBuilderComponent(this, 'AWS CLI', {\n *   platform: 'Windows',\n *   displayName: 'AWS CLI',\n *   description: 'Install latest version of AWS CLI',\n *   commands: [\n *     '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\\'',\n *     'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n *   ],\n * }\n * ```\n *\n * @deprecated Use `RunnerImageComponent` instead as this be internal soon.\n */\nexport class ImageBuilderComponent extends ImageBuilderObjectBase {\n  /**\n   * Component ARN.\n   */\n  public readonly arn: string;\n\n  /**\n   * Supported platform for the component.\n   */\n  public readonly platform: 'Windows' | 'Linux';\n\n  private readonly assets: s3_assets.Asset[] = [];\n\n  constructor(scope: Construct, id: string, props: ImageBuilderComponentProperties) {\n    super(scope, id);\n\n    this.platform = props.platform;\n\n    let steps: any[] = [];\n\n    if (props.assets) {\n      let inputs: any[] = [];\n      let extractCommands: string[] = [];\n      for (const asset of props.assets) {\n        this.assets.push(asset.asset);\n\n        if (asset.asset.isFile) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: asset.path,\n          });\n        } else if (asset.asset.isZipArchive) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: `${asset.path}.zip`,\n          });\n          if (props.platform === 'Windows') {\n            extractCommands.push(`Expand-Archive \"${asset.path}.zip\" -DestinationPath \"${asset.path}\"`);\n            extractCommands.push(`del \"${asset.path}.zip\"`);\n          } else {\n            extractCommands.push(`unzip \"${asset.path}.zip\" -d \"${asset.path}\"`);\n            extractCommands.push(`rm \"${asset.path}.zip\"`);\n          }\n        } else {\n          throw new Error(`Unknown asset type: ${asset.asset}`);\n        }\n      }\n\n      steps.push({\n        name: 'Download',\n        action: 'S3Download',\n        inputs,\n      });\n\n      if (extractCommands.length > 0) {\n        steps.push({\n          name: 'Extract',\n          action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n          inputs: {\n            commands: this.prefixCommandsWithErrorHandling(props.platform, extractCommands),\n          },\n        });\n      }\n    }\n\n    if (props.commands.length > 0) {\n      steps.push({\n        name: 'Run',\n        action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n        inputs: {\n          commands: this.prefixCommandsWithErrorHandling(props.platform, props.commands),\n        },\n      });\n    }\n\n    if (props.reboot ?? false) {\n      steps.push({\n        name: 'Reboot',\n        action: 'Reboot',\n        inputs: {},\n      });\n    }\n\n    const data = {\n      name: props.displayName,\n      schemaVersion: '1.0',\n      phases: [\n        {\n          name: 'build',\n          steps,\n        },\n      ],\n    };\n\n    const name = uniqueImageBuilderName(this);\n    const component = new imagebuilder.CfnComponent(this, 'Component', {\n      name: name,\n      description: props.description,\n      platform: props.platform,\n      version: this.version('Component', name, {\n        platform: props.platform,\n        data,\n        description: props.description,\n      }),\n      data: JSON.stringify(data),\n    });\n\n    this.arn = component.attrArn;\n  }\n\n  /**\n   * Grants read permissions to the principal on the assets buckets.\n   *\n   * @param grantee\n   */\n  grantAssetsRead(grantee: iam.IGrantable) {\n    for (const asset of this.assets) {\n      asset.grantRead(grantee);\n    }\n  }\n\n  prefixCommandsWithErrorHandling(platform: 'Windows' | 'Linux', commands: string[]) {\n    if (platform == 'Windows') {\n      return [\n        '$ErrorActionPreference = \\'Stop\\'',\n        '$ProgressPreference = \\'SilentlyContinue\\'',\n        'Set-PSDebug -Trace 1',\n      ].concat(commands);\n    } else {\n      return [\n        'set -ex',\n      ].concat(commands);\n    }\n  }\n}\n\n/**\n * @internal\n */\nexport class AwsImageBuilderRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private boundAmi?: RunnerAmi;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: string;\n  private readonly baseAmi: string;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc;\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly boundComponents: ImageBuilderComponent[] = [];\n  private readonly instanceType: ec2.InstanceType;\n  private infrastructure: imagebuilder.CfnInfrastructureConfiguration | undefined;\n  private readonly role: iam.Role;\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.codeBuildOptions) {\n      Annotations.of(this).addWarning('codeBuildOptions are ignored when using AWS Image Builder to build runner images.');\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 ?? ec2.Vpc.fromLookup(this, 'VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];\n    this.subnetSelection = props?.subnetSelection;\n    this.baseImage = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.baseAmi = props?.baseAmi ?? defaultBaseAmi(this, this.os, this.architecture);\n    this.instanceType = props?.awsImageBuilderOptions?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n\n    // confirm instance type\n    if (!this.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`Builder architecture (${this.architecture.name}) doesn't match selected instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\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    // role to be used by AWS Image Builder\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n  }\n\n  private platform() {\n    if (this.os.is(Os.WINDOWS)) {\n      return 'Windows';\n    }\n    if (this.os.is(Os.LINUX_AMAZON_2) || this.os.is(Os.LINUX_UBUNTU)) {\n      return 'Linux';\n    }\n    throw new Error(`OS ${this.os.name} is not supported by AWS Image Builder`);\n  }\n\n  /**\n   * Called by IRunnerProvider to finalize settings and create the image builder.\n   */\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    // create repository that only keeps one tag\n    const repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      lifecycleRules: [\n        {\n          description: 'Remove untagged images that have been replaced by AWS Image Builder',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n          rulePriority: 1,\n        },\n        {\n          description: 'Remove non-latest images',\n          tagStatus: TagStatus.TAGGED,\n          tagPrefixList: ['1'], // all versions start with '1.0.'\n          maxImageCount: 2, // keep two in case of rollback\n          rulePriority: 2,\n        },\n      ],\n    });\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'Docker Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          containerDistributionConfiguration: {\n            ContainerTags: ['latest'],\n            TargetRepository: {\n              Service: 'ECR',\n              RepositoryName: repository.repositoryName,\n            },\n          },\n        },\n      ],\n    });\n\n    let dockerfileTemplate = `FROM {{{ imagebuilder:parentImage }}}\n{{{ imagebuilder:environments }}}\n{{{ imagebuilder:components }}}`;\n\n    for (const c of this.components) {\n      const commands = c.getDockerCommands(this.os, this.architecture);\n      if (commands.length > 0) {\n        dockerfileTemplate += '\\n' + commands.join('\\n') + '\\n';\n      }\n    }\n\n    const recipe = new ContainerRecipe(this, 'Container Recipe', {\n      platform: this.platform(),\n      components: this.bindComponents(),\n      targetRepository: repository,\n      dockerfileTemplate: dockerfileTemplate,\n      parentImage: this.baseImage,\n    });\n\n    const log = this.createLog('Docker Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilderECRContainerBuilds'),\n    ]);\n    const image = this.createImage(infra, dist, log, undefined, recipe.arn);\n    this.createPipeline(infra, dist, log, undefined, recipe.arn);\n\n    this.imageCleaner(image, recipe.name, repository);\n    this.reaper(recipe.name, 'Docker');\n\n    this.boundDockerImage = {\n      // There are simpler ways to get the ARN, but we want an image object that depends on the newly built image.\n      // We want whoever is using this image to automatically wait for Image Builder to finish building before using the image.\n      imageRepository: ecr.Repository.fromRepositoryName(\n        this, 'Dependable Image',\n        // we can't use image.attrName because it comes up with upper case\n        cdk.Fn.split(':', cdk.Fn.split('/', image.attrImageUri, 2)[1], 2)[0],\n      ),\n      imageTag: 'latest',\n      os: this.os,\n      architecture: this.architecture,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n    };\n\n    return this.boundDockerImage;\n  }\n\n  private imageCleaner(image: imagebuilder.CfnImage, recipeName: string, repository: ecr.IRepository) {\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: ['ecr:BatchDeleteImage', 'ecr:ListImages'],\n          resources: [repository.repositoryArn],\n        }),\n        new iam.PolicyStatement({\n          actions: ['imagebuilder:ListImages', 'imagebuilder:ListImageBuildVersions', 'imagebuilder:DeleteImage'],\n          resources: ['*'], // Image Builder doesn't support scoping this :(\n        }),\n      ],\n    });\n    crHandler.role?.attachInlinePolicy(policy);\n\n    const cr = new CustomResource(this, 'Deleter', {\n      serviceToken: crHandler.functionArn,\n      resourceType: 'Custom::ImageDeleter',\n      properties: {\n        RepoName: repository.repositoryName,\n        ImageBuilderName: recipeName, // we don't use image.name because CloudFormation complains if it was deleted already\n        DeleteOnly: true,\n      },\n    });\n\n    // add dependencies to make sure resources are there when we need them\n    cr.node.addDependency(image);\n    cr.node.addDependency(policy);\n    cr.node.addDependency(crHandler);\n\n    return cr;\n  }\n\n  protected createLog(id: string, recipeName: string): logs.LogGroup {\n    return new logs.LogGroup(this, id, {\n      logGroupName: `/aws/imagebuilder/${recipeName}`,\n      retention: this.logRetention,\n      removalPolicy: this.logRemovalPolicy,\n    });\n  }\n\n  protected createInfrastructure(managedPolicies: iam.IManagedPolicy[]): imagebuilder.CfnInfrastructureConfiguration {\n    if (this.infrastructure) {\n      return this.infrastructure;\n    }\n\n    for (const managedPolicy of managedPolicies) {\n      this.role.addManagedPolicy(managedPolicy);\n    }\n\n    for (const component of this.boundComponents) {\n      component.grantAssetsRead(this.role);\n    }\n\n    this.infrastructure = new imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      subnetId: this.vpc?.selectSubnets(this.subnetSelection).subnetIds[0],\n      securityGroupIds: this.securityGroups?.map(sg => sg.securityGroupId),\n      instanceTypes: [this.instanceType.toString()],\n      instanceMetadataOptions: {\n        httpTokens: 'required',\n        // Container builds require a minimum of two hops.\n        httpPutResponseHopLimit: 2,\n      },\n      instanceProfileName: new iam.CfnInstanceProfile(this, 'Instance Profile', {\n        roles: [\n          this.role.roleName,\n        ],\n      }).ref,\n    });\n\n    return this.infrastructure;\n  }\n\n  protected createImage(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImage {\n    const image = new imagebuilder.CfnImage(this, this.amiOrContainerId('Image', imageRecipeArn, containerRecipeArn), {\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    image.node.addDependency(infra);\n    image.node.addDependency(log);\n\n    return image;\n  }\n\n  private amiOrContainerId(baseId: string, imageRecipeArn?: string, containerRecipeArn?: string) {\n    if (imageRecipeArn) {\n      return `AMI ${baseId}`;\n    }\n    if (containerRecipeArn) {\n      return `Docker ${baseId}`;\n    }\n    throw new Error('Either imageRecipeArn or containerRecipeArn must be defined');\n  }\n\n  protected createPipeline(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImagePipeline {\n    let scheduleOptions: imagebuilder.CfnImagePipeline.ScheduleProperty | undefined;\n    if (this.rebuildInterval.toDays() > 0) {\n      scheduleOptions = {\n        scheduleExpression: events.Schedule.rate(this.rebuildInterval).expressionString,\n        pipelineExecutionStartCondition: 'EXPRESSION_MATCH_ONLY',\n      };\n    }\n    const pipeline = new imagebuilder.CfnImagePipeline(this, this.amiOrContainerId('Pipeline', imageRecipeArn, containerRecipeArn), {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      schedule: scheduleOptions,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    pipeline.node.addDependency(infra);\n    pipeline.node.addDependency(log);\n\n    return pipeline;\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n\n  public get grantPrincipal(): iam.IPrincipal {\n    return this.role;\n  }\n\n  bindAmi(): RunnerAmi {\n    if (this.boundAmi) {\n      return this.boundAmi;\n    }\n\n    const launchTemplate = new ec2.LaunchTemplate(this, 'Launch template', {\n      requireImdsv2: true,\n    });\n\n    const stackName = cdk.Stack.of(this).stackName;\n    const builderName = this.node.path;\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'AMI Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          amiDistributionConfiguration: {\n            Name: `${cdk.Names.uniqueResourceName(this, {\n              maxLength: 100,\n              separator: '-',\n              allowedSpecialCharacters: '_-',\n            })}-{{ imagebuilder:buildDate }}`,\n            AmiTags: {\n              'Name': this.node.id,\n              'GitHubRunners:Stack': stackName,\n              'GitHubRunners:Builder': builderName,\n            },\n          },\n          launchTemplateConfigurations: [\n            {\n              launchTemplateId: launchTemplate.launchTemplateId,\n            },\n          ],\n        },\n      ],\n    });\n\n    const recipe = new AmiRecipe(this, 'Ami Recipe', {\n      platform: this.platform(),\n      components: this.bindComponents(),\n      architecture: this.architecture,\n      baseAmi: this.baseAmi,\n    });\n\n    const log = this.createLog('Ami Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilder'),\n    ]);\n    this.createImage(infra, dist, log, recipe.arn, undefined);\n    this.createPipeline(infra, dist, log, recipe.arn, undefined);\n\n    this.boundAmi = {\n      launchTemplate: launchTemplate,\n      architecture: this.architecture,\n      os: this.os,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n    };\n\n    this.amiCleaner(launchTemplate, stackName, builderName);\n    this.reaper(recipe.name, 'AMI');\n\n    return this.boundAmi;\n  }\n\n  private amiCleaner(launchTemplate: ec2.LaunchTemplate, stackName: string, builderName: string) {\n    const deleter = singletonLambda(DeleteAmiFunction, this, 'delete-ami', {\n      description: 'Delete old GitHub Runner AMIs',\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: ['ec2:DescribeLaunchTemplateVersions', 'ec2:DescribeImages', 'ec2:DeregisterImage', 'ec2:DeleteSnapshot'],\n          resources: ['*'],\n        }),\n      ],\n      timeout: cdk.Duration.minutes(5),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    // delete old AMIs on schedule\n    const eventRule = new events.Rule(this, 'Delete AMI Schedule', {\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n      description: `Delete old AMIs for ${builderName}`,\n    });\n    eventRule.addTarget(new events_targets.LambdaFunction(deleter, {\n      event: events.RuleTargetInput.fromObject({\n        RequestType: 'Scheduled',\n        LaunchTemplateId: launchTemplate.launchTemplateId,\n        StackName: stackName,\n        BuilderName: builderName,\n      }),\n    }));\n\n    // delete all AMIs when this construct is removed\n    new CustomResource(this, 'AMI Deleter', {\n      serviceToken: deleter.functionArn,\n      resourceType: 'Custom::AmiDeleter',\n      properties: {\n        StackName: stackName,\n        BuilderName: builderName,\n      },\n    });\n  }\n\n  private bindComponents(): ImageBuilderComponent[] {\n    if (this.boundComponents.length == 0) {\n      this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i} ${c.name}`, this.os, this.architecture)));\n    }\n\n    return this.boundComponents;\n  }\n\n  private reaper(recipeName: string, imageType: 'Docker' | 'AMI') {\n    const reaper = singletonLambda(ReaperFunction, this, 'Reaper', {\n      description: 'AWS Image Builder version reaper deletes old image build versions pointing to deleted AMIs/Docker images',\n      timeout: cdk.Duration.minutes(3),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: [\n            'imagebuilder:ListImages',\n            'imagebuilder:ListImageBuildVersions',\n            'imagebuilder:DeleteImage',\n            'ec2:DescribeImages',\n            'ecr:DescribeImages',\n          ],\n          resources: ['*'],\n        }),\n      ],\n    });\n\n    const scheduleRule = new events.Rule(this, `Reaper Schedule ${imageType}`, {\n      description: `Delete old image build versions for ${recipeName}`,\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n    });\n\n    scheduleRule.addTarget(new events_targets.LambdaFunction(reaper, {\n      event: events.RuleTargetInput.fromObject({\n        RecipeName: recipeName,\n      }),\n    }));\n  }\n}\n\n/**\n * @internal\n */\nexport class AwsImageBuilderFailedBuildNotifier implements cdk.IAspect {\n  public static createFilteringTopic(scope: Construct, targetTopic: sns.Topic) {\n    const topic = new sns.Topic(scope, 'Image Builder Builds');\n    const filter = new FilterFailedBuildsFunction(scope, 'Image Builder Builds Filter', {\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      environment: {\n        TARGET_TOPIC_ARN: targetTopic.topicArn,\n      },\n    });\n\n    topic.addSubscription(new subs.LambdaSubscription(filter));\n    targetTopic.grantPublish(filter);\n\n    return topic;\n  }\n\n  constructor(private topic: sns.ITopic) {\n  }\n\n  public visit(node: IConstruct): void {\n    if (node instanceof AwsImageBuilderRunnerImageBuilder) {\n      const builder = node as AwsImageBuilderRunnerImageBuilder;\n      const infra = builder.node.findChild('Infrastructure') as imagebuilder.CfnInfrastructureConfiguration;\n      infra.snsTopicArn = this.topic.topicArn;\n    }\n  }\n}\n"]}
565
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../src/image-builders/aws-image-builder/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  Annotations,\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_imagebuilder as imagebuilder,\n  aws_logs as logs,\n  aws_s3_assets as s3_assets,\n  aws_sns as sns,\n  aws_sns_subscriptions as subs,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { TagMutability, TagStatus } from 'aws-cdk-lib/aws-ecr';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Construct, IConstruct } from 'constructs';\nimport { AmiRecipe, defaultBaseAmi } from './ami';\nimport { ImageBuilderObjectBase } from './common';\nimport { ContainerRecipe, defaultBaseDockerImage } from './container';\nimport { DeleteAmiFunction } from './delete-ami-function';\nimport { FilterFailedBuildsFunction } from './filter-failed-builds-function';\nimport { ReaperFunction } from './reaper-function';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../providers';\nimport { BuildImageFunction } from '../../providers/build-image-function';\nimport { singletonLambda } from '../../utils';\nimport { RunnerImageBuilderBase, RunnerImageBuilderProps, uniqueImageBuilderName } from '../common';\n\nexport interface AwsImageBuilderRunnerImageBuilderProps {\n  /**\n   * The instance type used to build the image.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n}\n\n/**\n * An asset including file or directory to place inside the built image.\n */\nexport interface ImageBuilderAsset {\n  /**\n   * Path to place asset in the image.\n   */\n  readonly path: string;\n\n  /**\n   * Asset to place in the image.\n   */\n  readonly asset: s3_assets.Asset;\n}\n\n/**\n * Properties for ImageBuilderComponent construct.\n */\nexport interface ImageBuilderComponentProperties {\n  /**\n   * Component platform. Must match the builder platform.\n   */\n  readonly platform: 'Linux' | 'Windows';\n\n  /**\n   * Component display name.\n   */\n  readonly displayName: string;\n\n  /**\n   * Component description.\n   */\n  readonly description: string;\n\n  /**\n   * Shell commands to run when adding this component to the image.\n   *\n   * On Linux, these are bash commands. On Windows, there are PowerShell commands.\n   */\n  readonly commands: string[];\n\n  /**\n   * Optional assets to add to the built image.\n   */\n  readonly assets?: ImageBuilderAsset[];\n\n  /**\n   * Require a reboot after installing this component.\n   *\n   * @default false\n   */\n  readonly reboot?: boolean;\n}\n\n/**\n * Components are a set of commands to run and optional files to add to an image. Components are the building blocks of images built by Image Builder.\n *\n * Example:\n *\n * ```\n * new ImageBuilderComponent(this, 'AWS CLI', {\n *   platform: 'Windows',\n *   displayName: 'AWS CLI',\n *   description: 'Install latest version of AWS CLI',\n *   commands: [\n *     '$p = Start-Process msiexec.exe -PassThru -Wait -ArgumentList \\'/i https://awscli.amazonaws.com/AWSCLIV2.msi /qn\\'',\n *     'if ($p.ExitCode -ne 0) { throw \"Exit code is $p.ExitCode\" }',\n *   ],\n * }\n * ```\n *\n * @deprecated Use `RunnerImageComponent` instead as this be internal soon.\n */\nexport class ImageBuilderComponent extends ImageBuilderObjectBase {\n  /**\n   * Component ARN.\n   */\n  public readonly arn: string;\n\n  /**\n   * Supported platform for the component.\n   */\n  public readonly platform: 'Windows' | 'Linux';\n\n  private readonly assets: s3_assets.Asset[] = [];\n\n  constructor(scope: Construct, id: string, props: ImageBuilderComponentProperties) {\n    super(scope, id);\n\n    this.platform = props.platform;\n\n    let steps: any[] = [];\n\n    if (props.assets) {\n      let inputs: any[] = [];\n      let extractCommands: string[] = [];\n      for (const asset of props.assets) {\n        this.assets.push(asset.asset);\n\n        if (asset.asset.isFile) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: asset.path,\n          });\n        } else if (asset.asset.isZipArchive) {\n          inputs.push({\n            source: asset.asset.s3ObjectUrl,\n            destination: `${asset.path}.zip`,\n          });\n          if (props.platform === 'Windows') {\n            extractCommands.push(`Expand-Archive \"${asset.path}.zip\" -DestinationPath \"${asset.path}\"`);\n            extractCommands.push(`del \"${asset.path}.zip\"`);\n          } else {\n            extractCommands.push(`unzip \"${asset.path}.zip\" -d \"${asset.path}\"`);\n            extractCommands.push(`rm \"${asset.path}.zip\"`);\n          }\n        } else {\n          throw new Error(`Unknown asset type: ${asset.asset}`);\n        }\n      }\n\n      steps.push({\n        name: 'Download',\n        action: 'S3Download',\n        inputs,\n      });\n\n      if (extractCommands.length > 0) {\n        steps.push({\n          name: 'Extract',\n          action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n          inputs: {\n            commands: this.prefixCommandsWithErrorHandling(props.platform, extractCommands),\n          },\n        });\n      }\n    }\n\n    if (props.commands.length > 0) {\n      steps.push({\n        name: 'Run',\n        action: props.platform === 'Linux' ? 'ExecuteBash' : 'ExecutePowerShell',\n        inputs: {\n          commands: this.prefixCommandsWithErrorHandling(props.platform, props.commands),\n        },\n      });\n    }\n\n    if (props.reboot ?? false) {\n      steps.push({\n        name: 'Reboot',\n        action: 'Reboot',\n        inputs: {},\n      });\n    }\n\n    const data = {\n      name: props.displayName,\n      schemaVersion: '1.0',\n      phases: [\n        {\n          name: 'build',\n          steps,\n        },\n      ],\n    };\n\n    const name = uniqueImageBuilderName(this);\n    const component = new imagebuilder.CfnComponent(this, 'Component', {\n      name: name,\n      description: props.description,\n      platform: props.platform,\n      version: this.version('Component', name, {\n        platform: props.platform,\n        data,\n        description: props.description,\n      }),\n      data: JSON.stringify(data),\n    });\n\n    this.arn = component.attrArn;\n  }\n\n  /**\n   * Grants read permissions to the principal on the assets buckets.\n   *\n   * @param grantee\n   */\n  grantAssetsRead(grantee: iam.IGrantable) {\n    for (const asset of this.assets) {\n      asset.grantRead(grantee);\n    }\n  }\n\n  prefixCommandsWithErrorHandling(platform: 'Windows' | 'Linux', commands: string[]) {\n    if (platform == 'Windows') {\n      return [\n        '$ErrorActionPreference = \\'Stop\\'',\n        '$ProgressPreference = \\'SilentlyContinue\\'',\n        'Set-PSDebug -Trace 1',\n      ].concat(commands);\n    } else {\n      return [\n        'set -ex',\n      ].concat(commands);\n    }\n  }\n}\n\n/**\n * @internal\n */\nexport class AwsImageBuilderRunnerImageBuilder extends RunnerImageBuilderBase {\n  private boundDockerImage?: RunnerImage;\n  private boundAmi?: RunnerAmi;\n  private readonly os: Os;\n  private readonly architecture: Architecture;\n  private readonly baseImage: string;\n  private readonly baseAmi: string;\n  private readonly logRetention: RetentionDays;\n  private readonly logRemovalPolicy: RemovalPolicy;\n  private readonly vpc: ec2.IVpc;\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly subnetSelection: ec2.SubnetSelection | undefined;\n  private readonly rebuildInterval: cdk.Duration;\n  private readonly boundComponents: ImageBuilderComponent[] = [];\n  private readonly instanceType: ec2.InstanceType;\n  private infrastructure: imagebuilder.CfnInfrastructureConfiguration | undefined;\n  private readonly role: iam.Role;\n\n  constructor(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    super(scope, id, props);\n\n    if (props?.codeBuildOptions) {\n      Annotations.of(this).addWarning('codeBuildOptions are ignored when using AWS Image Builder to build runner images.');\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 ?? ec2.Vpc.fromLookup(this, 'VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })];\n    this.subnetSelection = props?.subnetSelection;\n    this.baseImage = props?.baseDockerImage ?? defaultBaseDockerImage(this.os);\n    this.baseAmi = props?.baseAmi ?? defaultBaseAmi(this, this.os, this.architecture);\n    this.instanceType = props?.awsImageBuilderOptions?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n\n    // confirm instance type\n    if (!this.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`Builder architecture (${this.architecture.name}) doesn't match selected instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\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    // role to be used by AWS Image Builder\n    this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n  }\n\n  private platform() {\n    if (this.os.is(Os.WINDOWS)) {\n      return 'Windows';\n    }\n    if (this.os.is(Os.LINUX_AMAZON_2) || this.os.is(Os.LINUX_UBUNTU)) {\n      return 'Linux';\n    }\n    throw new Error(`OS ${this.os.name} is not supported by AWS Image Builder`);\n  }\n\n  /**\n   * Called by IRunnerProvider to finalize settings and create the image builder.\n   */\n  bindDockerImage(): RunnerImage {\n    if (this.boundDockerImage) {\n      return this.boundDockerImage;\n    }\n\n    // create repository that only keeps one tag\n    const repository = new ecr.Repository(this, 'Repository', {\n      imageScanOnPush: true,\n      imageTagMutability: TagMutability.MUTABLE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      lifecycleRules: [\n        {\n          description: 'Remove untagged images that have been replaced by AWS Image Builder',\n          tagStatus: TagStatus.UNTAGGED,\n          maxImageAge: Duration.days(1),\n          rulePriority: 1,\n        },\n        {\n          description: 'Remove non-latest images',\n          tagStatus: TagStatus.TAGGED,\n          tagPrefixList: ['1'], // all versions start with '1.0.'\n          maxImageCount: 2, // keep two in case of rollback\n          rulePriority: 2,\n        },\n      ],\n    });\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'Docker Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          containerDistributionConfiguration: {\n            ContainerTags: ['latest'],\n            TargetRepository: {\n              Service: 'ECR',\n              RepositoryName: repository.repositoryName,\n            },\n          },\n        },\n      ],\n    });\n\n    let dockerfileTemplate = `FROM {{{ imagebuilder:parentImage }}}\n{{{ imagebuilder:environments }}}\n{{{ imagebuilder:components }}}`;\n\n    for (const c of this.components) {\n      const commands = c.getDockerCommands(this.os, this.architecture);\n      if (commands.length > 0) {\n        dockerfileTemplate += '\\n' + commands.join('\\n') + '\\n';\n      }\n    }\n\n    const recipe = new ContainerRecipe(this, 'Container Recipe', {\n      platform: this.platform(),\n      components: this.bindComponents(),\n      targetRepository: repository,\n      dockerfileTemplate: dockerfileTemplate,\n      parentImage: this.baseImage,\n    });\n\n    const log = this.createLog('Docker Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilderECRContainerBuilds'),\n    ]);\n    const image = this.createImage(infra, dist, log, undefined, recipe.arn);\n    this.createPipeline(infra, dist, log, undefined, recipe.arn);\n\n    this.imageCleaner(image, recipe.name, repository);\n    this.reaper(recipe.name, 'Docker');\n\n    this.boundDockerImage = {\n      imageRepository: repository,\n      imageTag: 'latest',\n      os: this.os,\n      architecture: this.architecture,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n      // no dependable as CloudFormation will fail to get image ARN once the image is deleted (we delete old images daily)\n    };\n\n    return this.boundDockerImage;\n  }\n\n  private imageCleaner(image: imagebuilder.CfnImage, recipeName: string, repository: ecr.IRepository) {\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: ['ecr:BatchDeleteImage', 'ecr:ListImages'],\n          resources: [repository.repositoryArn],\n        }),\n        new iam.PolicyStatement({\n          actions: ['imagebuilder:ListImages', 'imagebuilder:ListImageBuildVersions', 'imagebuilder:DeleteImage'],\n          resources: ['*'], // Image Builder doesn't support scoping this :(\n        }),\n      ],\n    });\n    crHandler.role?.attachInlinePolicy(policy);\n\n    const cr = new CustomResource(this, 'Deleter', {\n      serviceToken: crHandler.functionArn,\n      resourceType: 'Custom::ImageDeleter',\n      properties: {\n        RepoName: repository.repositoryName,\n        ImageBuilderName: recipeName, // we don't use image.name because CloudFormation complains if it was deleted already\n        DeleteOnly: true,\n      },\n    });\n\n    // add dependencies to make sure resources are there when we need them\n    cr.node.addDependency(image);\n    cr.node.addDependency(policy);\n    cr.node.addDependency(crHandler);\n\n    return cr;\n  }\n\n  protected createLog(id: string, recipeName: string): logs.LogGroup {\n    return new logs.LogGroup(this, id, {\n      logGroupName: `/aws/imagebuilder/${recipeName}`,\n      retention: this.logRetention,\n      removalPolicy: this.logRemovalPolicy,\n    });\n  }\n\n  protected createInfrastructure(managedPolicies: iam.IManagedPolicy[]): imagebuilder.CfnInfrastructureConfiguration {\n    if (this.infrastructure) {\n      return this.infrastructure;\n    }\n\n    for (const managedPolicy of managedPolicies) {\n      this.role.addManagedPolicy(managedPolicy);\n    }\n\n    for (const component of this.boundComponents) {\n      component.grantAssetsRead(this.role);\n    }\n\n    this.infrastructure = new imagebuilder.CfnInfrastructureConfiguration(this, 'Infrastructure', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      subnetId: this.vpc?.selectSubnets(this.subnetSelection).subnetIds[0],\n      securityGroupIds: this.securityGroups?.map(sg => sg.securityGroupId),\n      instanceTypes: [this.instanceType.toString()],\n      instanceMetadataOptions: {\n        httpTokens: 'required',\n        // Container builds require a minimum of two hops.\n        httpPutResponseHopLimit: 2,\n      },\n      instanceProfileName: new iam.CfnInstanceProfile(this, 'Instance Profile', {\n        roles: [\n          this.role.roleName,\n        ],\n      }).ref,\n    });\n\n    return this.infrastructure;\n  }\n\n  protected createImage(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImage {\n    const image = new imagebuilder.CfnImage(this, this.amiOrContainerId('Image', imageRecipeArn, containerRecipeArn), {\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    image.node.addDependency(infra);\n    image.node.addDependency(log);\n\n    return image;\n  }\n\n  private amiOrContainerId(baseId: string, imageRecipeArn?: string, containerRecipeArn?: string) {\n    if (imageRecipeArn) {\n      return `AMI ${baseId}`;\n    }\n    if (containerRecipeArn) {\n      return `Docker ${baseId}`;\n    }\n    throw new Error('Either imageRecipeArn or containerRecipeArn must be defined');\n  }\n\n  protected createPipeline(infra: imagebuilder.CfnInfrastructureConfiguration, dist: imagebuilder.CfnDistributionConfiguration, log: logs.LogGroup,\n    imageRecipeArn?: string, containerRecipeArn?: string): imagebuilder.CfnImagePipeline {\n    let scheduleOptions: imagebuilder.CfnImagePipeline.ScheduleProperty | undefined;\n    if (this.rebuildInterval.toDays() > 0) {\n      scheduleOptions = {\n        scheduleExpression: events.Schedule.rate(this.rebuildInterval).expressionString,\n        pipelineExecutionStartCondition: 'EXPRESSION_MATCH_ONLY',\n      };\n    }\n    const pipeline = new imagebuilder.CfnImagePipeline(this, this.amiOrContainerId('Pipeline', imageRecipeArn, containerRecipeArn), {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      infrastructureConfigurationArn: infra.attrArn,\n      distributionConfigurationArn: dist.attrArn,\n      imageRecipeArn,\n      containerRecipeArn,\n      schedule: scheduleOptions,\n      imageTestsConfiguration: {\n        imageTestsEnabled: false,\n      },\n    });\n    pipeline.node.addDependency(infra);\n    pipeline.node.addDependency(log);\n\n    return pipeline;\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n\n  public get grantPrincipal(): iam.IPrincipal {\n    return this.role;\n  }\n\n  bindAmi(): RunnerAmi {\n    if (this.boundAmi) {\n      return this.boundAmi;\n    }\n\n    const launchTemplate = new ec2.LaunchTemplate(this, 'Launch template', {\n      requireImdsv2: true,\n    });\n\n    const stackName = cdk.Stack.of(this).stackName;\n    const builderName = this.node.path;\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'AMI Distribution', {\n      name: uniqueImageBuilderName(this),\n      // description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          amiDistributionConfiguration: {\n            Name: `${cdk.Names.uniqueResourceName(this, {\n              maxLength: 100,\n              separator: '-',\n              allowedSpecialCharacters: '_-',\n            })}-{{ imagebuilder:buildDate }}`,\n            AmiTags: {\n              'Name': this.node.id,\n              'GitHubRunners:Stack': stackName,\n              'GitHubRunners:Builder': builderName,\n            },\n          },\n          launchTemplateConfigurations: [\n            {\n              launchTemplateId: launchTemplate.launchTemplateId,\n            },\n          ],\n        },\n      ],\n    });\n\n    const recipe = new AmiRecipe(this, 'Ami Recipe', {\n      platform: this.platform(),\n      components: this.bindComponents(),\n      architecture: this.architecture,\n      baseAmi: this.baseAmi,\n    });\n\n    const log = this.createLog('Ami Log', recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilder'),\n    ]);\n    this.createImage(infra, dist, log, recipe.arn, undefined);\n    this.createPipeline(infra, dist, log, recipe.arn, undefined);\n\n    this.boundAmi = {\n      launchTemplate: launchTemplate,\n      architecture: this.architecture,\n      os: this.os,\n      logGroup: log,\n      runnerVersion: RunnerVersion.specific('unknown'),\n    };\n\n    this.amiCleaner(launchTemplate, stackName, builderName);\n    this.reaper(recipe.name, 'AMI');\n\n    return this.boundAmi;\n  }\n\n  private amiCleaner(launchTemplate: ec2.LaunchTemplate, stackName: string, builderName: string) {\n    const deleter = singletonLambda(DeleteAmiFunction, this, 'delete-ami', {\n      description: 'Delete old GitHub Runner AMIs',\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: ['ec2:DescribeLaunchTemplateVersions', 'ec2:DescribeImages', 'ec2:DeregisterImage', 'ec2:DeleteSnapshot'],\n          resources: ['*'],\n        }),\n      ],\n      timeout: cdk.Duration.minutes(5),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    // delete old AMIs on schedule\n    const eventRule = new events.Rule(this, 'Delete AMI Schedule', {\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n      description: `Delete old AMIs for ${builderName}`,\n    });\n    eventRule.addTarget(new events_targets.LambdaFunction(deleter, {\n      event: events.RuleTargetInput.fromObject({\n        RequestType: 'Scheduled',\n        LaunchTemplateId: launchTemplate.launchTemplateId,\n        StackName: stackName,\n        BuilderName: builderName,\n      }),\n    }));\n\n    // delete all AMIs when this construct is removed\n    new CustomResource(this, 'AMI Deleter', {\n      serviceToken: deleter.functionArn,\n      resourceType: 'Custom::AmiDeleter',\n      properties: {\n        StackName: stackName,\n        BuilderName: builderName,\n      },\n    });\n  }\n\n  private bindComponents(): ImageBuilderComponent[] {\n    if (this.boundComponents.length == 0) {\n      this.boundComponents.push(...this.components.map((c, i) => c._asAwsImageBuilderComponent(this, `Component ${i} ${c.name}`, this.os, this.architecture)));\n    }\n\n    return this.boundComponents;\n  }\n\n  private reaper(recipeName: string, imageType: 'Docker' | 'AMI') {\n    const reaper = singletonLambda(ReaperFunction, this, 'Reaper', {\n      description: 'AWS Image Builder version reaper deletes old image build versions pointing to deleted AMIs/Docker images',\n      timeout: cdk.Duration.minutes(3),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: [\n            'imagebuilder:ListImages',\n            'imagebuilder:ListImageBuildVersions',\n            'imagebuilder:DeleteImage',\n            'ec2:DescribeImages',\n            'ecr:DescribeImages',\n          ],\n          resources: ['*'],\n        }),\n      ],\n    });\n\n    const scheduleRule = new events.Rule(this, `Reaper Schedule ${imageType}`, {\n      description: `Delete old image build versions for ${recipeName}`,\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n    });\n\n    scheduleRule.addTarget(new events_targets.LambdaFunction(reaper, {\n      event: events.RuleTargetInput.fromObject({\n        RecipeName: recipeName,\n      }),\n    }));\n  }\n}\n\n/**\n * @internal\n */\nexport class AwsImageBuilderFailedBuildNotifier implements cdk.IAspect {\n  public static createFilteringTopic(scope: Construct, targetTopic: sns.Topic) {\n    const topic = new sns.Topic(scope, 'Image Builder Builds');\n    const filter = new FilterFailedBuildsFunction(scope, 'Image Builder Builds Filter', {\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      environment: {\n        TARGET_TOPIC_ARN: targetTopic.topicArn,\n      },\n    });\n\n    topic.addSubscription(new subs.LambdaSubscription(filter));\n    targetTopic.grantPublish(filter);\n\n    return topic;\n  }\n\n  constructor(private topic: sns.ITopic) {\n  }\n\n  public visit(node: IConstruct): void {\n    if (node instanceof AwsImageBuilderRunnerImageBuilder) {\n      const builder = node as AwsImageBuilderRunnerImageBuilder;\n      const infra = builder.node.findChild('Infrastructure') as imagebuilder.CfnInfrastructureConfiguration;\n      infra.snsTopicArn = this.topic.topicArn;\n    }\n  }\n}\n"]}
@@ -235,6 +235,6 @@ class AmiBuilder extends common_1.ImageBuilderBase {
235
235
  }
236
236
  }
237
237
  _a = JSII_RTTI_SYMBOL_1;
238
- AmiBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.AmiBuilder", version: "0.9.6" };
238
+ AmiBuilder[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.AmiBuilder", version: "0.9.7" };
239
239
  exports.AmiBuilder = AmiBuilder;
240
240
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ami.js","sourceRoot":"","sources":["../../../../src/image-builders/aws-image-builder/deprecated/ami.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_imagebuilder as imagebuilder,\n  aws_logs as logs,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { ImageBuilderBase } from './common';\nimport { LinuxUbuntuComponents } from './linux-components';\nimport { WindowsComponents } from './windows-components';\nimport { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../../providers/common';\nimport { singletonLambda } from '../../../utils';\nimport { uniqueImageBuilderName } from '../../common';\nimport { AmiRecipe, defaultBaseAmi } from '../ami';\nimport { ImageBuilderComponent } from '../builder';\nimport { DeleteAmiFunction } from '../delete-ami-function';\n\n/**\n * Properties for {@link AmiBuilder} construct.\n */\nexport interface AmiBuilderProps {\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\n   */\n  readonly os?: 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 group to assign to launched builder instances.\n   *\n   * @default new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\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. Only the first matched subnet will be used.\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   * Install Docker inside the image, so it can be used by the runner.\n   *\n   * @default true\n   */\n  readonly installDocker?: boolean;\n}\n\n/**\n * An AMI builder that uses AWS Image Builder to build AMIs pre-baked with all the GitHub Actions runner requirements. Builders can be used with {@link Ec2Runner}.\n *\n * Each builder re-runs automatically at a set interval to make sure the AMIs contain the latest versions of everything.\n *\n * You can create an instance of this construct to customize the AMI used to spin-up runners. Some runner providers may require custom components. Check the runner provider documentation.\n *\n * For example, to set a specific runner version, rebuild the image every 2 weeks, and add a few packages for the EC2 provider, use:\n *\n * ```\n * const builder = new AmiBuilder(this, 'Builder', {\n *     runnerVersion: RunnerVersion.specific('2.293.0'),\n *     rebuildInterval: Duration.days(14),\n * });\n * builder.addComponent(new ImageBuilderComponent(scope, id, {\n *   platform: 'Linux',\n *   displayName: 'p7zip',\n *   description: 'Install some more packages',\n *   commands: [\n *     'apt-get install p7zip',\n *   ],\n * }));\n * new Ec2RunnerProvider(this, 'EC2 provider', {\n *     labels: ['custom-ec2'],\n *     amiBuilder: builder,\n * });\n * ```\n *\n * @deprecated use RunnerImageBuilder\n */\nexport class AmiBuilder extends ImageBuilderBase {\n  private boundAmi?: RunnerAmi;\n\n  constructor(scope: Construct, id: string, props?: AmiBuilderProps) {\n    super(scope, id, {\n      os: props?.os,\n      supportedOs: [Os.LINUX, Os.LINUX_UBUNTU, Os.LINUX_AMAZON_2, Os.WINDOWS],\n      architecture: props?.architecture,\n      supportedArchitectures: [Architecture.X86_64, Architecture.ARM64],\n      instanceType: props?.instanceType,\n      vpc: props?.vpc,\n      securityGroups: props?.securityGroup ? [props.securityGroup] : props?.securityGroups,\n      subnetSelection: props?.subnetSelection,\n      logRemovalPolicy: props?.logRemovalPolicy,\n      logRetention: props?.logRetention,\n      runnerVersion: props?.runnerVersion,\n      rebuildInterval: props?.rebuildInterval,\n      imageTypeName: 'AMI',\n    });\n\n    // add all basic components\n    if (this.os.is(Os.WINDOWS)) {\n      this.addBaseWindowsComponents(props?.installDocker ?? true);\n    } else if (this.os.is(Os.LINUX) || this.os.is(Os.LINUX_UBUNTU)) {\n      this.addBaseLinuxComponents(props?.installDocker ?? true);\n    } else {\n      throw new Error(`Unsupported OS for AMI builder: ${this.os.name}`);\n    }\n  }\n\n  private addBaseWindowsComponents(installDocker: boolean) {\n    this.addComponent(WindowsComponents.cloudwatchAgent(this, 'CloudWatch agent'));\n    this.addComponent(WindowsComponents.awsCli(this, 'AWS CLI'));\n    this.addComponent(WindowsComponents.githubCli(this, 'GitHub CLI'));\n    this.addComponent(WindowsComponents.git(this, 'git'));\n    this.addComponent(WindowsComponents.githubRunner(this, 'GitHub Actions Runner', this.runnerVersion));\n    if (installDocker) {\n      this.addComponent(WindowsComponents.docker(this, 'Docker'));\n    }\n  }\n\n  private addBaseLinuxComponents(installDocker: boolean) {\n    this.addComponent(LinuxUbuntuComponents.requiredPackages(this, 'Upgrade packages and install basics', this.architecture));\n    this.addComponent(LinuxUbuntuComponents.runnerUser(this, 'User', this.architecture));\n    this.addComponent(LinuxUbuntuComponents.awsCli(this, 'AWS CLI', this.architecture));\n    this.addComponent(LinuxUbuntuComponents.githubCli(this, 'GitHub CLI', this.architecture));\n    this.addComponent(LinuxUbuntuComponents.git(this, 'git', this.architecture));\n    this.addComponent(LinuxUbuntuComponents.githubRunner(this, 'GitHub Actions Runner', this.runnerVersion, this.architecture));\n    if (installDocker) {\n      this.addComponent(LinuxUbuntuComponents.docker(this, 'Docker', this.architecture));\n    }\n  }\n\n  /**\n   * Add a component to be installed before any other components. Useful for required system settings like certificates or proxy settings.\n   * @param component\n   */\n  prependComponent(component: ImageBuilderComponent) {\n    if (this.boundAmi) {\n      throw new Error('AMI is already bound. Use this method before passing the builder to a runner provider.');\n    }\n    if (component.platform != this.platform) {\n      throw new Error('Component platform doesn\\'t match builder platform');\n    }\n    this.components = [component].concat(this.components);\n  }\n\n  /**\n   * Add a component to be installed.\n   * @param component\n   */\n  addComponent(component: ImageBuilderComponent) {\n    if (this.boundAmi) {\n      throw new Error('AMI is already bound. Use this method before passing the builder to a runner provider.');\n    }\n    if (component.platform != this.platform) {\n      throw new Error('Component platform doesn\\'t match builder platform');\n    }\n    this.components.push(component);\n  }\n\n  /**\n   * Add extra trusted certificates. This helps deal with self-signed certificates for GitHub Enterprise Server.\n   *\n   * @param path path to directory containing a file called certs.pem containing all the required certificates\n   */\n  public addExtraCertificates(path: string) {\n    if (this.platform == 'Linux') {\n      this.prependComponent(LinuxUbuntuComponents.extraCertificates(this, 'Extra Certs', path));\n    } else if (this.platform == 'Windows') {\n      this.prependComponent(WindowsComponents.extraCertificates(this, 'Extra Certs', path));\n    } else {\n      throw new Error(`Unknown platform: ${this.platform}`);\n    }\n  }\n\n  /**\n   * Called by IRunnerProvider to finalize settings and create the AMI builder.\n   */\n  bindAmi(): RunnerAmi {\n    if (this.boundAmi) {\n      return this.boundAmi;\n    }\n\n    const launchTemplate = new ec2.LaunchTemplate(this, 'Launch template', {\n      requireImdsv2: true,\n    });\n\n    const stackName = cdk.Stack.of(this).stackName;\n    const builderName = this.node.path;\n\n    const dist = new imagebuilder.CfnDistributionConfiguration(this, 'Distribution', {\n      name: uniqueImageBuilderName(this),\n      description: this.description,\n      distributions: [\n        {\n          region: Stack.of(this).region,\n          amiDistributionConfiguration: {\n            Name: `${cdk.Names.uniqueResourceName(this, {\n              maxLength: 100,\n              separator: '-',\n              allowedSpecialCharacters: '_-',\n            })}-{{ imagebuilder:buildDate }}`,\n            AmiTags: {\n              'Name': this.node.id,\n              'GitHubRunners:Stack': stackName,\n              'GitHubRunners:Builder': builderName,\n            },\n          },\n          launchTemplateConfigurations: [\n            {\n              launchTemplateId: launchTemplate.launchTemplateId,\n            },\n          ],\n        },\n      ],\n    });\n\n    const recipe = new AmiRecipe(this, 'Ami Recipe', {\n      platform: this.platform,\n      components: this.components,\n      architecture: this.architecture,\n      baseAmi: defaultBaseAmi(this, this.os, this.architecture),\n    });\n\n    const log = this.createLog(recipe.name);\n    const infra = this.createInfrastructure([\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      iam.ManagedPolicy.fromAwsManagedPolicyName('EC2InstanceProfileForImageBuilder'),\n    ]);\n    this.createImage(infra, dist, log, recipe.arn, undefined);\n    this.createPipeline(infra, dist, log, recipe.arn, undefined);\n\n    this.boundAmi = {\n      launchTemplate: launchTemplate,\n      architecture: this.architecture,\n      os: this.os,\n      logGroup: log,\n      runnerVersion: this.runnerVersion,\n    };\n\n    this.imageCleaner(launchTemplate, stackName, builderName);\n\n    return this.boundAmi;\n  }\n\n  private imageCleaner(launchTemplate: ec2.LaunchTemplate, stackName: string, builderName: string) {\n    const deleter = singletonLambda(DeleteAmiFunction, this, 'delete-ami', {\n      description: 'Delete old GitHub Runner AMIs',\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: ['ec2:DescribeLaunchTemplateVersions', 'ec2:DescribeImages', 'ec2:DeregisterImage', 'ec2:DeleteSnapshot'],\n          resources: ['*'],\n        }),\n      ],\n      timeout: cdk.Duration.minutes(5),\n      logRetention: logs.RetentionDays.ONE_MONTH,\n    });\n\n    // delete old AMIs on schedule\n    const eventRule = new events.Rule(this, 'Delete AMI Schedule', {\n      schedule: events.Schedule.rate(cdk.Duration.days(1)),\n      description: `Delete old AMIs for ${builderName}`,\n    });\n    eventRule.addTarget(new events_targets.LambdaFunction(deleter, {\n      event: events.RuleTargetInput.fromObject({\n        RequestType: 'Scheduled',\n        LaunchTemplateId: launchTemplate.launchTemplateId,\n        StackName: stackName,\n        BuilderName: builderName,\n      }),\n    }));\n\n    // delete all AMIs when this construct is removed\n    new CustomResource(this, 'AMI Deleter', {\n      serviceToken: deleter.functionArn,\n      resourceType: 'Custom::AmiDeleter',\n      properties: {\n        StackName: stackName,\n        BuilderName: builderName,\n      },\n    });\n  }\n\n  bindDockerImage(): RunnerImage {\n    throw new Error('AmiBuilder cannot be used to build Docker images');\n  }\n}\n"]}
@@ -1,7 +1,7 @@
1
1
  import { aws_ec2 as ec2, aws_ecr as ecr, aws_logs as logs, Duration, RemovalPolicy } from 'aws-cdk-lib';
2
2
  import { Construct } from 'constructs';
3
3
  import { ImageBuilderBase } from './common';
4
- import { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../../providers/common';
4
+ import { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../../../providers';
5
5
  import { ImageBuilderComponent } from '../builder';
6
6
  /**
7
7
  * Properties for ContainerImageBuilder construct.