@cloudsnorkel/cdk-github-runners 0.9.4 → 0.9.6
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.
- package/.gitattributes +5 -3
- package/.jsii +332 -284
- package/API.md +55 -19
- package/README.md +135 -65
- package/assets/{providers/image-builders → image-builders}/aws-image-builder/delete-ami.lambda/index.js +2 -2
- package/assets/{providers/image-builders → image-builders}/aws-image-builder/filter-failed-builds.lambda/index.js +1 -1
- package/assets/image-builders/aws-image-builder/reaper.lambda/index.js +163 -0
- package/assets/{providers/image-builders → image-builders}/aws-image-builder/versioner.lambda/index.js +2 -2
- package/cdk.json +10 -0
- package/lib/access.js +1 -1
- package/lib/image-builders/api.js +47 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/ami.d.ts +2 -3
- package/lib/image-builders/aws-image-builder/ami.js +93 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/builder.d.ts +10 -3
- package/lib/image-builders/aws-image-builder/builder.js +568 -0
- package/lib/image-builders/aws-image-builder/common.js +46 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/container.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/container.js +63 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/delete-ami-function.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/delete-ami-function.js +23 -0
- package/lib/image-builders/aws-image-builder/delete-ami.lambda.js +87 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/ami.d.ts +4 -4
- package/lib/image-builders/aws-image-builder/deprecated/ami.js +240 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/common.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/common.js +144 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/container.d.ts +3 -3
- package/lib/image-builders/aws-image-builder/deprecated/container.js +222 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/index.js +1 -1
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/linux-components.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +172 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/windows-components.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +126 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/filter-failed-builds-function.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/filter-failed-builds-function.js +23 -0
- package/lib/image-builders/aws-image-builder/filter-failed-builds.lambda.js +18 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/index.js +1 -1
- package/lib/image-builders/aws-image-builder/reaper-function.d.ts +13 -0
- package/lib/image-builders/aws-image-builder/reaper-function.js +23 -0
- package/lib/image-builders/aws-image-builder/reaper.lambda.d.ts +1 -0
- package/lib/image-builders/aws-image-builder/reaper.lambda.js +149 -0
- package/lib/{providers/image-builders → image-builders}/aws-image-builder/versioner-function.d.ts +1 -1
- package/lib/image-builders/aws-image-builder/versioner-function.js +23 -0
- package/lib/image-builders/aws-image-builder/versioner.lambda.js +96 -0
- package/lib/{providers/image-builders → image-builders}/codebuild-deprecated.d.ts +5 -5
- package/lib/image-builders/codebuild-deprecated.js +373 -0
- package/lib/{providers/image-builders → image-builders}/codebuild.d.ts +2 -2
- package/lib/image-builders/codebuild.js +289 -0
- package/lib/{providers/image-builders → image-builders}/common.d.ts +6 -4
- package/lib/{providers/image-builders → image-builders}/common.js +1 -1
- package/lib/{providers/image-builders → image-builders}/components.d.ts +8 -2
- package/lib/image-builders/components.js +568 -0
- package/lib/{providers/image-builders → image-builders}/index.js +1 -1
- package/lib/{providers/image-builders → image-builders}/static.d.ts +1 -1
- package/lib/image-builders/static.js +58 -0
- package/lib/providers/codebuild.d.ts +1 -1
- package/lib/providers/codebuild.js +4 -4
- package/lib/providers/common.js +3 -3
- package/lib/providers/ec2.d.ts +2 -2
- package/lib/providers/ec2.js +4 -4
- package/lib/providers/ecs.d.ts +1 -1
- package/lib/providers/ecs.js +3 -3
- package/lib/providers/fargate.d.ts +1 -1
- package/lib/providers/fargate.js +4 -4
- package/lib/providers/index.d.ts +1 -1
- package/lib/providers/index.js +2 -2
- package/lib/providers/lambda.d.ts +1 -1
- package/lib/providers/lambda.js +4 -4
- package/lib/runner.d.ts +3 -3
- package/lib/runner.js +5 -5
- package/lib/secrets.js +1 -1
- package/package.json +12 -10
- package/lib/providers/image-builders/api.js +0 -47
- package/lib/providers/image-builders/aws-image-builder/ami.js +0 -81
- package/lib/providers/image-builders/aws-image-builder/builder.js +0 -520
- package/lib/providers/image-builders/aws-image-builder/common.js +0 -46
- package/lib/providers/image-builders/aws-image-builder/container.js +0 -63
- package/lib/providers/image-builders/aws-image-builder/delete-ami-function.js +0 -23
- package/lib/providers/image-builders/aws-image-builder/delete-ami.lambda.js +0 -87
- package/lib/providers/image-builders/aws-image-builder/deprecated/ami.js +0 -240
- package/lib/providers/image-builders/aws-image-builder/deprecated/common.js +0 -144
- package/lib/providers/image-builders/aws-image-builder/deprecated/container.js +0 -222
- package/lib/providers/image-builders/aws-image-builder/deprecated/linux-components.js +0 -172
- package/lib/providers/image-builders/aws-image-builder/deprecated/windows-components.js +0 -129
- package/lib/providers/image-builders/aws-image-builder/filter-failed-builds-function.js +0 -23
- package/lib/providers/image-builders/aws-image-builder/filter-failed-builds.lambda.js +0 -18
- package/lib/providers/image-builders/aws-image-builder/versioner-function.js +0 -23
- package/lib/providers/image-builders/aws-image-builder/versioner.lambda.js +0 -96
- package/lib/providers/image-builders/codebuild-deprecated.js +0 -373
- package/lib/providers/image-builders/codebuild.js +0 -287
- package/lib/providers/image-builders/components.js +0 -535
- package/lib/providers/image-builders/static.js +0 -58
- /package/lib/{providers/image-builders → image-builders}/api.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/common.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/delete-ami.lambda.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/deprecated/index.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/filter-failed-builds.lambda.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/index.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/aws-image-builder/versioner.lambda.d.ts +0 -0
- /package/lib/{providers/image-builders → image-builders}/index.d.ts +0 -0
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
/* eslint-disable-next-line import/no-extraneous-dependencies */
|
|
4
|
+
const AWS = require("aws-sdk");
|
|
5
|
+
const ec2 = new AWS.EC2();
|
|
6
|
+
const ecr = new AWS.ECR();
|
|
7
|
+
const ib = new AWS.Imagebuilder();
|
|
8
|
+
async function iterateImageVersions(imageName) {
|
|
9
|
+
let result = [];
|
|
10
|
+
let params = {
|
|
11
|
+
owner: 'Self',
|
|
12
|
+
filters: [
|
|
13
|
+
{
|
|
14
|
+
name: 'name',
|
|
15
|
+
values: [imageName],
|
|
16
|
+
},
|
|
17
|
+
],
|
|
18
|
+
};
|
|
19
|
+
while (true) {
|
|
20
|
+
const response = await ib.listImages(params).promise();
|
|
21
|
+
if (response.imageVersionList) {
|
|
22
|
+
for (const imageVersion of response.imageVersionList) {
|
|
23
|
+
if (imageVersion.arn) {
|
|
24
|
+
result.push(imageVersion.arn);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
if (!response.nextToken) {
|
|
29
|
+
break;
|
|
30
|
+
}
|
|
31
|
+
params.nextToken = response.nextToken;
|
|
32
|
+
}
|
|
33
|
+
return result;
|
|
34
|
+
}
|
|
35
|
+
async function iterateImageBuildVersions(imageVersionArn) {
|
|
36
|
+
let result = [];
|
|
37
|
+
let params = {
|
|
38
|
+
imageVersionArn: imageVersionArn,
|
|
39
|
+
};
|
|
40
|
+
while (true) {
|
|
41
|
+
const response = await ib.listImageBuildVersions(params).promise();
|
|
42
|
+
if (response.imageSummaryList) {
|
|
43
|
+
for (const imageBuildVersion of response.imageSummaryList) {
|
|
44
|
+
if (imageBuildVersion.state?.status !== 'AVAILABLE') {
|
|
45
|
+
// if it's not available, then it's still being created
|
|
46
|
+
console.log(`${imageBuildVersion.arn} is still being created, so we can't delete it`);
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
result.push(imageBuildVersion);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
if (!response.nextToken) {
|
|
53
|
+
break;
|
|
54
|
+
}
|
|
55
|
+
params.nextToken = response.nextToken;
|
|
56
|
+
}
|
|
57
|
+
return result;
|
|
58
|
+
}
|
|
59
|
+
async function amisGone(amis) {
|
|
60
|
+
if (!amis) {
|
|
61
|
+
console.log('No AMIs found, so we can delete the image version build');
|
|
62
|
+
return true;
|
|
63
|
+
}
|
|
64
|
+
for (const ami of amis) {
|
|
65
|
+
console.log(`Checking if ${ami.image} exists`);
|
|
66
|
+
if (!ami.image) {
|
|
67
|
+
// no image? delete it
|
|
68
|
+
console.log('No AMI, so we can delete it');
|
|
69
|
+
continue;
|
|
70
|
+
}
|
|
71
|
+
try {
|
|
72
|
+
const response = await ec2.describeImages({
|
|
73
|
+
ImageIds: [ami.image],
|
|
74
|
+
}).promise();
|
|
75
|
+
if (response.Images?.length ?? 0 > 0) {
|
|
76
|
+
// image still available
|
|
77
|
+
console.log('AMI still available, so we can\'t delete it');
|
|
78
|
+
return false;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
catch (e) {
|
|
82
|
+
if (e.code != 'InvalidAMIID.NotFound') {
|
|
83
|
+
// unknown exception, keep image for now
|
|
84
|
+
console.error(`Unknown exception while checking if ${ami.image} exists:`, e);
|
|
85
|
+
return false;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
console.log('All AMIs are gone, so we can delete the image version build');
|
|
90
|
+
return true;
|
|
91
|
+
}
|
|
92
|
+
async function dockerImagesGone(dockerImages) {
|
|
93
|
+
if (!dockerImages) {
|
|
94
|
+
console.log('No docker images, so we can delete the image version build');
|
|
95
|
+
return true;
|
|
96
|
+
}
|
|
97
|
+
for (const images of dockerImages) {
|
|
98
|
+
for (const image of images.imageUris ?? []) {
|
|
99
|
+
const [repo, version] = image.split(':', 2);
|
|
100
|
+
const [_, repoName] = repo.split('/', 2);
|
|
101
|
+
if (version === 'latest') {
|
|
102
|
+
// don't check latest as image builder sometimes keeps that tag on forever
|
|
103
|
+
// we want to check if the image is still tagged as latest in ECR, not Image Builder
|
|
104
|
+
continue;
|
|
105
|
+
}
|
|
106
|
+
console.log(`Checking if ${repoName}:${version} exists`);
|
|
107
|
+
try {
|
|
108
|
+
const response = await ecr.describeImages({
|
|
109
|
+
repositoryName: repoName,
|
|
110
|
+
imageIds: [{ imageTag: version }],
|
|
111
|
+
}).promise();
|
|
112
|
+
if (response.imageDetails && response.imageDetails.length > 0) {
|
|
113
|
+
// image still available
|
|
114
|
+
if (response.imageDetails[0].imageTags?.includes('latest')) {
|
|
115
|
+
console.log(`Docker image ${repoName}:${version} still available and tagged latest, so we can't delete it`);
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
catch (e) {
|
|
121
|
+
if (e.code != 'RepositoryNotFoundException' && e.code != 'ImageNotFoundException') {
|
|
122
|
+
// unknown exception, keep image for now
|
|
123
|
+
console.error(`Unknown exception while checking if ${repoName}:${version} exists:`, e);
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
console.log('All Docker images are gone, so we can delete the image version build');
|
|
130
|
+
return true;
|
|
131
|
+
}
|
|
132
|
+
/* eslint-disable @typescript-eslint/no-require-imports, import/no-extraneous-dependencies */
|
|
133
|
+
exports.handler = async function (event, _context) {
|
|
134
|
+
for (const imageVersion of await iterateImageVersions(event.RecipeName)) {
|
|
135
|
+
for (const imageBuildVersion of await iterateImageBuildVersions(imageVersion)) {
|
|
136
|
+
if (!imageBuildVersion.arn) {
|
|
137
|
+
continue;
|
|
138
|
+
}
|
|
139
|
+
console.log(`Checking ${imageBuildVersion.name}/${imageBuildVersion.version}`);
|
|
140
|
+
if (await amisGone(imageBuildVersion.outputResources?.amis) && await dockerImagesGone(imageBuildVersion.outputResources?.containers)) {
|
|
141
|
+
console.log('Deleting image version build', imageBuildVersion.arn);
|
|
142
|
+
await ib.deleteImage({
|
|
143
|
+
imageBuildVersionArn: imageBuildVersion.arn,
|
|
144
|
+
}).promise();
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
};
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reaper.lambda.js","sourceRoot":"","sources":["../../../src/image-builders/aws-image-builder/reaper.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable-next-line import/no-extraneous-dependencies,import/no-unresolved */\nimport * as AWSLambda from 'aws-lambda';\n/* eslint-disable-next-line import/no-extraneous-dependencies */\nimport * as AWS from 'aws-sdk';\n\ninterface ReaperInput {\n  /**\n   * Recipe/image name.\n   */\n  RecipeName: string;\n}\n\nconst ec2 = new AWS.EC2();\nconst ecr = new AWS.ECR();\nconst ib = new AWS.Imagebuilder();\n\nasync function iterateImageVersions(imageName: string) {\n  let result: string[] = [];\n\n  let params: AWS.Imagebuilder.ListImagesRequest = {\n    owner: 'Self',\n    filters: [\n      {\n        name: 'name',\n        values: [imageName],\n      },\n    ],\n  };\n\n  while (true) {\n    const response = await ib.listImages(params).promise();\n\n    if (response.imageVersionList) {\n      for (const imageVersion of response.imageVersionList) {\n        if (imageVersion.arn) {\n          result.push(imageVersion.arn);\n        }\n      }\n    }\n\n    if (!response.nextToken) {\n      break;\n    }\n    params.nextToken = response.nextToken;\n  }\n\n  return result;\n}\n\nasync function iterateImageBuildVersions(imageVersionArn: string) {\n  let result: AWS.Imagebuilder.ImageSummary[] = [];\n\n  let params: AWS.Imagebuilder.ListImageBuildVersionsRequest = {\n    imageVersionArn: imageVersionArn,\n  };\n\n  while (true) {\n    const response = await ib.listImageBuildVersions(params).promise();\n\n    if (response.imageSummaryList) {\n      for (const imageBuildVersion of response.imageSummaryList) {\n        if (imageBuildVersion.state?.status !== 'AVAILABLE') {\n          // if it's not available, then it's still being created\n          console.log(`${imageBuildVersion.arn} is still being created, so we can't delete it`);\n          continue;\n        }\n\n        result.push(imageBuildVersion);\n      }\n    }\n\n    if (!response.nextToken) {\n      break;\n    }\n    params.nextToken = response.nextToken;\n  }\n\n  return result;\n}\n\nasync function amisGone(amis?: AWS.Imagebuilder.AmiList) {\n  if (!amis) {\n    console.log('No AMIs found, so we can delete the image version build');\n    return true;\n  }\n\n  for (const ami of amis) {\n    console.log(`Checking if ${ami.image} exists`);\n\n    if (!ami.image) {\n      // no image? delete it\n      console.log('No AMI, so we can delete it');\n      continue;\n    }\n\n    try {\n      const response = await ec2.describeImages({\n        ImageIds: [ami.image],\n      }).promise();\n\n      if (response.Images?.length ?? 0 > 0) {\n        // image still available\n        console.log('AMI still available, so we can\\'t delete it');\n        return false;\n      }\n    } catch (e: any) {\n      if (e.code != 'InvalidAMIID.NotFound') {\n        // unknown exception, keep image for now\n        console.error(`Unknown exception while checking if ${ami.image} exists:`, e);\n        return false;\n      }\n    }\n  }\n\n  console.log('All AMIs are gone, so we can delete the image version build');\n\n  return true;\n}\n\nasync function dockerImagesGone(dockerImages?: AWS.Imagebuilder.ContainerList) {\n  if (!dockerImages) {\n    console.log('No docker images, so we can delete the image version build');\n    return true;\n  }\n\n  for (const images of dockerImages) {\n    for (const image of images.imageUris ?? []) {\n      const [repo, version] = image.split(':', 2);\n      const [_, repoName] = repo.split('/', 2);\n\n      if (version === 'latest') {\n        // don't check latest as image builder sometimes keeps that tag on forever\n        // we want to check if the image is still tagged as latest in ECR, not Image Builder\n        continue;\n      }\n\n      console.log(`Checking if ${repoName}:${version} exists`);\n\n      try {\n        const response = await ecr.describeImages({\n          repositoryName: repoName,\n          imageIds: [{ imageTag: version }],\n        }).promise();\n\n        if (response.imageDetails && response.imageDetails.length > 0) {\n          // image still available\n          if (response.imageDetails[0].imageTags?.includes('latest')) {\n            console.log(`Docker image ${repoName}:${version} still available and tagged latest, so we can't delete it`);\n            return false;\n          }\n        }\n      } catch (e: any) {\n        if (e.code != 'RepositoryNotFoundException' && e.code != 'ImageNotFoundException') {\n          // unknown exception, keep image for now\n          console.error(`Unknown exception while checking if ${repoName}:${version} exists:`, e);\n          return false;\n        }\n      }\n    }\n  }\n\n  console.log('All Docker images are gone, so we can delete the image version build');\n\n  return true;\n}\n\n/* eslint-disable @typescript-eslint/no-require-imports, import/no-extraneous-dependencies */\nexports.handler = async function (event: ReaperInput, _context: AWSLambda.Context) {\n  for (const imageVersion of await iterateImageVersions(event.RecipeName)) {\n    for (const imageBuildVersion of await iterateImageBuildVersions(imageVersion)) {\n      if (!imageBuildVersion.arn) {\n        continue;\n      }\n\n      console.log(`Checking ${imageBuildVersion.name}/${imageBuildVersion.version}`);\n\n      if (await amisGone(imageBuildVersion.outputResources?.amis) && await dockerImagesGone(imageBuildVersion.outputResources?.containers)) {\n        console.log('Deleting image version build', imageBuildVersion.arn);\n        await ib.deleteImage({\n          imageBuildVersionArn: imageBuildVersion.arn,\n        }).promise();\n      }\n    }\n  }\n};\n"]}
|
package/lib/{providers/image-builders → image-builders}/aws-image-builder/versioner-function.d.ts
RENAMED
|
@@ -6,7 +6,7 @@ import { Construct } from 'constructs';
|
|
|
6
6
|
export interface VersionerFunctionProps extends lambda.FunctionOptions {
|
|
7
7
|
}
|
|
8
8
|
/**
|
|
9
|
-
* An AWS Lambda function which executes src/
|
|
9
|
+
* An AWS Lambda function which executes src/image-builders/aws-image-builder/versioner.
|
|
10
10
|
*/
|
|
11
11
|
export declare class VersionerFunction extends lambda.Function {
|
|
12
12
|
constructor(scope: Construct, id: string, props?: VersionerFunctionProps);
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VersionerFunction = void 0;
|
|
4
|
+
// ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
7
|
+
/**
|
|
8
|
+
* An AWS Lambda function which executes src/image-builders/aws-image-builder/versioner.
|
|
9
|
+
*/
|
|
10
|
+
class VersionerFunction extends lambda.Function {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id, {
|
|
13
|
+
description: 'src/image-builders/aws-image-builder/versioner.lambda.ts',
|
|
14
|
+
...props,
|
|
15
|
+
runtime: new lambda.Runtime('nodejs16.x', lambda.RuntimeFamily.NODEJS),
|
|
16
|
+
handler: 'index.handler',
|
|
17
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../../../assets/image-builders/aws-image-builder/versioner.lambda')),
|
|
18
|
+
});
|
|
19
|
+
this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.VersionerFunction = VersionerFunction;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbmVyLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2ltYWdlLWJ1aWxkZXJzL2F3cy1pbWFnZS1idWlsZGVyL3ZlcnNpb25lci1mdW5jdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2RUFBNkU7QUFDN0UsNkJBQTZCO0FBQzdCLGlEQUFpRDtBQVNqRDs7R0FFRztBQUNILE1BQWEsaUJBQWtCLFNBQVEsTUFBTSxDQUFDLFFBQVE7SUFDcEQsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUE4QjtRQUN0RSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRTtZQUNmLFdBQVcsRUFBRSwwREFBMEQ7WUFDdkUsR0FBRyxLQUFLO1lBQ1IsT0FBTyxFQUFFLElBQUksTUFBTSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsTUFBTSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUM7WUFDdEUsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLG1FQUFtRSxDQUFDLENBQUM7U0FDdkgsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQ0FBcUMsRUFBRSxHQUFHLEVBQUUsRUFBRSxZQUFZLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUMxRixDQUFDO0NBQ0Y7QUFYRCw4Q0FXQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIH5+IEdlbmVyYXRlZCBieSBwcm9qZW4uIFRvIG1vZGlmeSwgZWRpdCAucHJvamVucmMuanMgYW5kIHJ1biBcIm5weCBwcm9qZW5cIi5cbmltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuLyoqXG4gKiBQcm9wcyBmb3IgVmVyc2lvbmVyRnVuY3Rpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBWZXJzaW9uZXJGdW5jdGlvblByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbi8qKlxuICogQW4gQVdTIExhbWJkYSBmdW5jdGlvbiB3aGljaCBleGVjdXRlcyBzcmMvaW1hZ2UtYnVpbGRlcnMvYXdzLWltYWdlLWJ1aWxkZXIvdmVyc2lvbmVyLlxuICovXG5leHBvcnQgY2xhc3MgVmVyc2lvbmVyRnVuY3Rpb24gZXh0ZW5kcyBsYW1iZGEuRnVuY3Rpb24ge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wcz86IFZlcnNpb25lckZ1bmN0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnc3JjL2ltYWdlLWJ1aWxkZXJzL2F3cy1pbWFnZS1idWlsZGVyL3ZlcnNpb25lci5sYW1iZGEudHMnLFxuICAgICAgLi4ucHJvcHMsXG4gICAgICBydW50aW1lOiBuZXcgbGFtYmRhLlJ1bnRpbWUoJ25vZGVqczE2LngnLCBsYW1iZGEuUnVudGltZUZhbWlseS5OT0RFSlMpLFxuICAgICAgaGFuZGxlcjogJ2luZGV4LmhhbmRsZXInLFxuICAgICAgY29kZTogbGFtYmRhLkNvZGUuZnJvbUFzc2V0KHBhdGguam9pbihfX2Rpcm5hbWUsICcuLi8uLi8uLi9hc3NldHMvaW1hZ2UtYnVpbGRlcnMvYXdzLWltYWdlLWJ1aWxkZXIvdmVyc2lvbmVyLmxhbWJkYScpKSxcbiAgICB9KTtcbiAgICB0aGlzLmFkZEVudmlyb25tZW50KCdBV1NfTk9ERUpTX0NPTk5FQ1RJT05fUkVVU0VfRU5BQkxFRCcsICcxJywgeyByZW1vdmVJbkVkZ2U6IHRydWUgfSk7XG4gIH1cbn0iXX0=
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.handler = void 0;
|
|
4
|
+
/* eslint-disable import/no-extraneous-dependencies */
|
|
5
|
+
const AWS = require("aws-sdk");
|
|
6
|
+
const semver_1 = require("semver");
|
|
7
|
+
const lambda_helpers_1 = require("../../lambda-helpers");
|
|
8
|
+
const ib = new AWS.Imagebuilder();
|
|
9
|
+
/* eslint-disable @typescript-eslint/no-require-imports, import/no-extraneous-dependencies */
|
|
10
|
+
async function handler(event, context) {
|
|
11
|
+
console.log(JSON.stringify({ ...event, ResponseURL: '...' }));
|
|
12
|
+
try {
|
|
13
|
+
const objectType = event.ResourceProperties.ObjectType;
|
|
14
|
+
const objectName = event.ResourceProperties.ObjectName;
|
|
15
|
+
switch (event.RequestType) {
|
|
16
|
+
case 'Create':
|
|
17
|
+
case 'Update':
|
|
18
|
+
let version = '1.0.0';
|
|
19
|
+
let allVersions = [];
|
|
20
|
+
try {
|
|
21
|
+
switch (objectType) {
|
|
22
|
+
case 'Component': {
|
|
23
|
+
let result = {};
|
|
24
|
+
do {
|
|
25
|
+
result = await ib.listComponents({
|
|
26
|
+
filters: [{
|
|
27
|
+
name: 'name',
|
|
28
|
+
values: [objectName],
|
|
29
|
+
}],
|
|
30
|
+
nextToken: result.nextToken,
|
|
31
|
+
}).promise();
|
|
32
|
+
allVersions = allVersions.concat(result.componentVersionList.map(i => i.version || '1.0.0'));
|
|
33
|
+
} while (result.nextToken);
|
|
34
|
+
break;
|
|
35
|
+
}
|
|
36
|
+
case 'ImageRecipe': {
|
|
37
|
+
let result = {};
|
|
38
|
+
do {
|
|
39
|
+
result = await ib.listImageRecipes({
|
|
40
|
+
filters: [{
|
|
41
|
+
name: 'name',
|
|
42
|
+
values: [objectName],
|
|
43
|
+
}],
|
|
44
|
+
nextToken: result.nextToken,
|
|
45
|
+
}).promise();
|
|
46
|
+
allVersions = allVersions.concat(result.imageRecipeSummaryList.map(i => i.arn?.split('/').pop() || '1.0.0'));
|
|
47
|
+
} while (result.nextToken);
|
|
48
|
+
break;
|
|
49
|
+
}
|
|
50
|
+
case 'ContainerRecipe': {
|
|
51
|
+
let result = {};
|
|
52
|
+
do {
|
|
53
|
+
result = await ib.listContainerRecipes({
|
|
54
|
+
filters: [{
|
|
55
|
+
name: 'name',
|
|
56
|
+
values: [objectName],
|
|
57
|
+
}],
|
|
58
|
+
nextToken: result.nextToken,
|
|
59
|
+
}).promise();
|
|
60
|
+
allVersions = allVersions.concat(result.containerRecipeSummaryList.map(i => i.arn?.split('/').pop() || '1.0.0'));
|
|
61
|
+
} while (result.nextToken);
|
|
62
|
+
break;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
catch (e) {
|
|
67
|
+
if (e.code !== 'ResourceNotFoundException') {
|
|
68
|
+
throw e;
|
|
69
|
+
}
|
|
70
|
+
else {
|
|
71
|
+
console.log('Resource not found, assuming first version');
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
version = (0, semver_1.maxSatisfying)(allVersions, '>=0.0.0');
|
|
75
|
+
if (version === null) {
|
|
76
|
+
version = '1.0.0';
|
|
77
|
+
}
|
|
78
|
+
console.log(`Found versions ${allVersions} -- latest is ${version}`);
|
|
79
|
+
version = (0, semver_1.inc)(version, 'patch');
|
|
80
|
+
if (version === null) {
|
|
81
|
+
throw new Error('Unable to bump version');
|
|
82
|
+
}
|
|
83
|
+
await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', version, {});
|
|
84
|
+
break;
|
|
85
|
+
case 'Delete':
|
|
86
|
+
await (0, lambda_helpers_1.customResourceRespond)(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});
|
|
87
|
+
break;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
catch (e) {
|
|
91
|
+
console.log(e);
|
|
92
|
+
await (0, lambda_helpers_1.customResourceRespond)(event, 'FAILED', e.message || 'Internal Error', context.logStreamName, {});
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
exports.handler = handler;
|
|
96
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"versioner.lambda.js","sourceRoot":"","sources":["../../../src/image-builders/aws-image-builder/versioner.lambda.ts"],"names":[],"mappings":";;;AAEA,sDAAsD;AACtD,+BAA+B;AAC/B,mCAA4C;AAC5C,yDAA6D;AAE7D,MAAM,EAAE,GAAG,IAAI,GAAG,CAAC,YAAY,EAAE,CAAC;AAElC,6FAA6F;AACtF,KAAK,UAAU,OAAO,CAAC,KAAkD,EAAE,OAA0B;IAC1G,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;IAE9D,IAAI;QACF,MAAM,UAAU,GAAG,KAAK,CAAC,kBAAkB,CAAC,UAAU,CAAC;QACvD,MAAM,UAAU,GAAG,KAAK,CAAC,kBAAkB,CAAC,UAAU,CAAC;QAEvD,QAAQ,KAAK,CAAC,WAAW,EAAE;YACzB,KAAK,QAAQ,CAAC;YACd,KAAK,QAAQ;gBACX,IAAI,OAAO,GAAkB,OAAO,CAAC;gBACrC,IAAI,WAAW,GAAa,EAAE,CAAC;gBAC/B,IAAI;oBACF,QAAQ,UAAU,EAAE;wBAClB,KAAK,WAAW,CAAC,CAAC;4BAChB,IAAI,MAAM,GAA4C,EAAE,CAAC;4BACzD,GAAG;gCACD,MAAM,GAAG,MAAM,EAAE,CAAC,cAAc,CAAC;oCAC/B,OAAO,EAAE,CAAC;4CACR,IAAI,EAAE,MAAM;4CACZ,MAAM,EAAE,CAAC,UAAU,CAAC;yCACrB,CAAC;oCACF,SAAS,EAAE,MAAM,CAAC,SAAS;iCAC5B,CAAC,CAAC,OAAO,EAAE,CAAC;gCACb,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,oBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,IAAI,OAAO,CAAC,CAAC,CAAC;6BAC/F,QAAQ,MAAM,CAAC,SAAS,EAAE;4BAC3B,MAAM;yBACP;wBACD,KAAK,aAAa,CAAC,CAAC;4BAClB,IAAI,MAAM,GAA8C,EAAE,CAAC;4BAC3D,GAAG;gCACD,MAAM,GAAG,MAAM,EAAE,CAAC,gBAAgB,CAAC;oCACjC,OAAO,EAAE,CAAC;4CACR,IAAI,EAAE,MAAM;4CACZ,MAAM,EAAE,CAAC,UAAU,CAAC;yCACrB,CAAC;oCACF,SAAS,EAAE,MAAM,CAAC,SAAS;iCAC5B,CAAC,CAAC,OAAO,EAAE,CAAC;gCACb,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,sBAAuB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,OAAO,CAAC,CAAC,CAAC;6BAC/G,QAAQ,MAAM,CAAC,SAAS,EAAE;4BAC3B,MAAM;yBACP;wBACD,KAAK,iBAAiB,CAAC,CAAC;4BACtB,IAAI,MAAM,GAAkD,EAAE,CAAC;4BAC/D,GAAG;gCACD,MAAM,GAAG,MAAM,EAAE,CAAC,oBAAoB,CAAC;oCACrC,OAAO,EAAE,CAAC;4CACR,IAAI,EAAE,MAAM;4CACZ,MAAM,EAAE,CAAC,UAAU,CAAC;yCACrB,CAAC;oCACF,SAAS,EAAE,MAAM,CAAC,SAAS;iCAC5B,CAAC,CAAC,OAAO,EAAE,CAAC;gCACb,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,0BAA2B,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,OAAO,CAAC,CAAC,CAAC;6BACnH,QAAQ,MAAM,CAAC,SAAS,EAAE;4BAC3B,MAAM;yBACP;qBACF;iBACF;gBAAC,OAAO,CAAC,EAAE;oBACV,IAAK,CAAS,CAAC,IAAI,KAAK,2BAA2B,EAAE;wBACnD,MAAM,CAAC,CAAC;qBACT;yBAAM;wBACL,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAC;qBAC3D;iBACF;gBAED,OAAO,GAAG,IAAA,sBAAa,EAAC,WAAW,EAAE,SAAS,CAAC,CAAC;gBAChD,IAAI,OAAO,KAAK,IAAI,EAAE;oBACpB,OAAO,GAAG,OAAO,CAAC;iBACnB;gBACD,OAAO,CAAC,GAAG,CAAC,kBAAkB,WAAW,iBAAiB,OAAO,EAAE,CAAC,CAAC;gBAErE,OAAO,GAAG,IAAA,YAAG,EAAC,OAAO,EAAE,OAAO,CAAC,CAAC;gBAChC,IAAI,OAAO,KAAK,IAAI,EAAE;oBACpB,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;iBAC3C;gBACD,MAAM,IAAA,sCAAqB,EAAC,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC;gBAEjE,MAAM;YACR,KAAK,QAAQ;gBACX,MAAM,IAAA,sCAAqB,EAAC,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC;gBAClF,MAAM;SACT;KACF;IAAC,OAAO,CAAC,EAAE;QACV,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACf,MAAM,IAAA,sCAAqB,EAAC,KAAK,EAAE,QAAQ,EAAG,CAAW,CAAC,OAAO,IAAI,gBAAgB,EAAE,OAAO,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC;KACnH;AACH,CAAC;AAtFD,0BAsFC","sourcesContent":["/* eslint-disable-next-line import/no-extraneous-dependencies,import/no-unresolved */\nimport * as AWSLambda from 'aws-lambda';\n/* eslint-disable import/no-extraneous-dependencies */\nimport * as AWS from 'aws-sdk';\nimport { inc, maxSatisfying } from 'semver';\nimport { customResourceRespond } from '../../lambda-helpers';\n\nconst ib = new AWS.Imagebuilder();\n\n/* eslint-disable @typescript-eslint/no-require-imports, import/no-extraneous-dependencies */\nexport async function handler(event: AWSLambda.CloudFormationCustomResourceEvent, context: AWSLambda.Context) {\n  console.log(JSON.stringify({ ...event, ResponseURL: '...' }));\n\n  try {\n    const objectType = event.ResourceProperties.ObjectType;\n    const objectName = event.ResourceProperties.ObjectName;\n\n    switch (event.RequestType) {\n      case 'Create':\n      case 'Update':\n        let version: string | null = '1.0.0';\n        let allVersions: string[] = [];\n        try {\n          switch (objectType) {\n            case 'Component': {\n              let result: AWS.Imagebuilder.ListComponentsResponse = {};\n              do {\n                result = await ib.listComponents({\n                  filters: [{\n                    name: 'name',\n                    values: [objectName],\n                  }],\n                  nextToken: result.nextToken,\n                }).promise();\n                allVersions = allVersions.concat(result.componentVersionList!.map(i => i.version || '1.0.0'));\n              } while (result.nextToken);\n              break;\n            }\n            case 'ImageRecipe': {\n              let result: AWS.Imagebuilder.ListImageRecipesResponse = {};\n              do {\n                result = await ib.listImageRecipes({\n                  filters: [{\n                    name: 'name',\n                    values: [objectName],\n                  }],\n                  nextToken: result.nextToken,\n                }).promise();\n                allVersions = allVersions.concat(result.imageRecipeSummaryList!.map(i => i.arn?.split('/').pop() || '1.0.0'));\n              } while (result.nextToken);\n              break;\n            }\n            case 'ContainerRecipe': {\n              let result: AWS.Imagebuilder.ListContainerRecipesResponse = {};\n              do {\n                result = await ib.listContainerRecipes({\n                  filters: [{\n                    name: 'name',\n                    values: [objectName],\n                  }],\n                  nextToken: result.nextToken,\n                }).promise();\n                allVersions = allVersions.concat(result.containerRecipeSummaryList!.map(i => i.arn?.split('/').pop() || '1.0.0'));\n              } while (result.nextToken);\n              break;\n            }\n          }\n        } catch (e) {\n          if ((e as any).code !== 'ResourceNotFoundException') {\n            throw e;\n          } else {\n            console.log('Resource not found, assuming first version');\n          }\n        }\n\n        version = maxSatisfying(allVersions, '>=0.0.0');\n        if (version === null) {\n          version = '1.0.0';\n        }\n        console.log(`Found versions ${allVersions} -- latest is ${version}`);\n\n        version = inc(version, 'patch');\n        if (version === null) {\n          throw new Error('Unable to bump version');\n        }\n        await customResourceRespond(event, 'SUCCESS', 'OK', version, {});\n\n        break;\n      case 'Delete':\n        await customResourceRespond(event, 'SUCCESS', 'OK', event.PhysicalResourceId, {});\n        break;\n    }\n  } catch (e) {\n    console.log(e);\n    await customResourceRespond(event, 'FAILED', (e as Error).message || 'Internal Error', context.logStreamName, {});\n  }\n}\n"]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { aws_codebuild as codebuild, aws_ec2 as ec2, aws_iam as iam, aws_logs as logs, Duration, RemovalPolicy } from 'aws-cdk-lib';
|
|
2
2
|
import { Construct } from 'constructs';
|
|
3
3
|
import { IRunnerImageBuilder } from './common';
|
|
4
|
-
import { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../common';
|
|
4
|
+
import { Architecture, Os, RunnerAmi, RunnerImage, RunnerVersion } from '../providers/common';
|
|
5
5
|
/**
|
|
6
6
|
* Properties for CodeBuildImageBuilder construct.
|
|
7
7
|
*/
|
|
@@ -66,7 +66,7 @@ export interface CodeBuildImageBuilderProps {
|
|
|
66
66
|
*
|
|
67
67
|
* The only action taken in CodeBuild is running `docker build`. You would therefore not need to change this setting often.
|
|
68
68
|
*
|
|
69
|
-
* @default Ubuntu
|
|
69
|
+
* @default Ubuntu 22.04 for x64 and Amazon Linux 2 for ARM64
|
|
70
70
|
*/
|
|
71
71
|
readonly buildImage?: codebuild.IBuildImage;
|
|
72
72
|
/**
|
|
@@ -105,13 +105,13 @@ export interface CodeBuildImageBuilderProps {
|
|
|
105
105
|
*
|
|
106
106
|
* ```
|
|
107
107
|
* const builder = new CodeBuildImageBuilder(this, 'Builder', {
|
|
108
|
-
* dockerfilePath:
|
|
108
|
+
* dockerfilePath: FargateRunnerProvider.LINUX_X64_DOCKERFILE_PATH,
|
|
109
109
|
* runnerVersion: RunnerVersion.specific('2.293.0'),
|
|
110
110
|
* rebuildInterval: Duration.days(14),
|
|
111
111
|
* });
|
|
112
112
|
* builder.setBuildArg('EXTRA_PACKAGES', 'nginx xz-utils');
|
|
113
|
-
* new
|
|
114
|
-
*
|
|
113
|
+
* new FargateRunnerProvider(this, 'Fargate provider', {
|
|
114
|
+
* labels: ['customized-fargate'],
|
|
115
115
|
* imageBuilder: builder,
|
|
116
116
|
* });
|
|
117
117
|
* ```
|