@cloudsnorkel/cdk-github-runners 0.8.1 → 0.8.2

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 (96) hide show
  1. package/.gitattributes +9 -0
  2. package/.jsii +368 -184
  3. package/API.md +1718 -512
  4. package/README.md +2 -2
  5. package/{lib/providers → assets}/docker-images/lambda/linux-arm64/runner.sh +1 -1
  6. package/{lib/providers → assets}/docker-images/lambda/linux-x64/runner.sh +1 -1
  7. package/{lib/lambdas/aws-image-builder-versioner → assets/lambdas/aws-image-builder-versioner.lambda}/index.js +5 -5
  8. package/{lib/lambdas/build-image → assets/lambdas/build-image.lambda}/index.js +5 -5
  9. package/{lib/lambdas/delete-ami → assets/lambdas/delete-ami.lambda}/index.js +2 -2
  10. package/{lib/lambdas/delete-runner → assets/lambdas/delete-runner.lambda}/index.js +1 -1
  11. package/{lib/lambdas/setup → assets/lambdas/setup.lambda}/index.js +2 -2
  12. package/{lib/lambdas/status → assets/lambdas/status.lambda}/index.js +2 -2
  13. package/{lib/lambdas/token-retriever → assets/lambdas/token-retriever.lambda}/index.js +1 -1
  14. package/{lib/lambdas/update-lambda → assets/lambdas/update-lambda.lambda}/index.js +4 -4
  15. package/{lib/lambdas/webhook-handler → assets/lambdas/webhook-handler.lambda}/index.js +2 -2
  16. package/lib/lambdas/aws-image-builder-versioner-function.d.ts +13 -0
  17. package/lib/lambdas/aws-image-builder-versioner-function.js +23 -0
  18. package/lib/lambdas/aws-image-builder-versioner.lambda.d.ts +2 -0
  19. package/lib/lambdas/aws-image-builder-versioner.lambda.js +80 -0
  20. package/lib/lambdas/build-image-function.d.ts +13 -0
  21. package/lib/lambdas/build-image-function.js +23 -0
  22. package/lib/lambdas/build-image.lambda.d.ts +2 -0
  23. package/lib/lambdas/build-image.lambda.js +92 -0
  24. package/lib/lambdas/delete-ami-function.d.ts +13 -0
  25. package/lib/lambdas/delete-ami-function.js +23 -0
  26. package/lib/lambdas/delete-ami.lambda.d.ts +1 -0
  27. package/lib/lambdas/delete-ami.lambda.js +87 -0
  28. package/lib/lambdas/delete-runner-function.d.ts +13 -0
  29. package/lib/lambdas/delete-runner-function.js +23 -0
  30. package/lib/lambdas/delete-runner.lambda.d.ts +1 -0
  31. package/lib/lambdas/delete-runner.lambda.js +69 -0
  32. package/lib/lambdas/github.d.ts +7 -0
  33. package/lib/lambdas/github.js +50 -0
  34. package/lib/lambdas/helpers.d.ts +12 -0
  35. package/lib/lambdas/helpers.js +66 -0
  36. package/lib/lambdas/setup-function.d.ts +13 -0
  37. package/lib/lambdas/setup-function.js +23 -0
  38. package/lib/lambdas/setup.lambda.d.ts +1 -0
  39. package/lib/lambdas/setup.lambda.js +148 -0
  40. package/lib/lambdas/status-function.d.ts +13 -0
  41. package/lib/lambdas/status-function.js +23 -0
  42. package/lib/lambdas/status.lambda.d.ts +1 -0
  43. package/lib/lambdas/status.lambda.js +285 -0
  44. package/lib/lambdas/token-retriever-function.d.ts +13 -0
  45. package/lib/lambdas/token-retriever-function.js +23 -0
  46. package/lib/lambdas/token-retriever.lambda.d.ts +1 -0
  47. package/lib/lambdas/token-retriever.lambda.js +15 -0
  48. package/lib/lambdas/update-lambda-function.d.ts +13 -0
  49. package/lib/lambdas/update-lambda-function.js +23 -0
  50. package/lib/lambdas/update-lambda.lambda.d.ts +7 -0
  51. package/lib/lambdas/update-lambda.lambda.js +34 -0
  52. package/lib/lambdas/webhook-handler-function.d.ts +13 -0
  53. package/lib/lambdas/webhook-handler-function.js +23 -0
  54. package/lib/lambdas/webhook-handler.lambda.d.ts +1 -0
  55. package/lib/lambdas/webhook-handler.lambda.js +107 -0
  56. package/lib/providers/codebuild.d.ts +8 -3
  57. package/lib/providers/codebuild.js +17 -9
  58. package/lib/providers/common.js +3 -3
  59. package/lib/providers/ec2.d.ts +9 -4
  60. package/lib/providers/ec2.js +14 -6
  61. package/lib/providers/fargate.d.ts +8 -3
  62. package/lib/providers/fargate.js +17 -9
  63. package/lib/providers/image-builders/ami.js +6 -3
  64. package/lib/providers/image-builders/codebuild.js +5 -3
  65. package/lib/providers/image-builders/common.js +5 -3
  66. package/lib/providers/image-builders/container.js +5 -3
  67. package/lib/providers/image-builders/linux-components.js +1 -1
  68. package/lib/providers/image-builders/static.js +3 -3
  69. package/lib/providers/image-builders/windows-components.js +1 -1
  70. package/lib/providers/lambda.d.ts +8 -3
  71. package/lib/providers/lambda.js +20 -10
  72. package/lib/runner.js +17 -10
  73. package/lib/secrets.js +1 -1
  74. package/lib/utils.d.ts +2 -6
  75. package/lib/utils.js +11 -26
  76. package/lib/webhook.d.ts +2 -2
  77. package/lib/webhook.js +5 -3
  78. package/package.json +28 -12
  79. package/setup/index.html +0 -12
  80. package/setup/src/App.svelte +0 -291
  81. package/setup/src/app.scss +0 -15
  82. package/setup/src/main.ts +0 -8
  83. package/setup/src/vite-env.d.ts +0 -2
  84. package/setup/svelte.config.mjs +0 -7
  85. package/setup/tsconfig.json +0 -21
  86. package/setup/tsconfig.node.json +0 -8
  87. package/setup/vite.config.ts +0 -15
  88. /package/{lib/providers → assets}/docker-images/codebuild/linux-arm64/Dockerfile +0 -0
  89. /package/{lib/providers → assets}/docker-images/codebuild/linux-x64/Dockerfile +0 -0
  90. /package/{lib/providers → assets}/docker-images/fargate/linux-arm64/Dockerfile +0 -0
  91. /package/{lib/providers → assets}/docker-images/fargate/linux-x64/Dockerfile +0 -0
  92. /package/{lib/providers → assets}/docker-images/lambda/linux-arm64/Dockerfile +0 -0
  93. /package/{lib/providers → assets}/docker-images/lambda/linux-arm64/runner.js +0 -0
  94. /package/{lib/providers → assets}/docker-images/lambda/linux-x64/Dockerfile +0 -0
  95. /package/{lib/providers → assets}/docker-images/lambda/linux-x64/runner.js +0 -0
  96. /package/{lib/lambdas/setup → assets/lambdas/setup.lambda}/index.html +0 -0
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.StatusFunction = 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/lambdas/status.
9
+ */
10
+ class StatusFunction extends lambda.Function {
11
+ constructor(scope, id, props) {
12
+ super(scope, id, {
13
+ description: 'src/lambdas/status.lambda.ts',
14
+ ...props,
15
+ runtime: new lambda.Runtime('nodejs14.x', lambda.RuntimeFamily.NODEJS),
16
+ handler: 'index.handler',
17
+ code: lambda.Code.fromAsset(path.join(__dirname, '../../assets/lambdas/status.lambda')),
18
+ });
19
+ this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
20
+ }
21
+ }
22
+ exports.StatusFunction = StatusFunction;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdHVzLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xhbWJkYXMvc3RhdHVzLWZ1bmN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZFQUE2RTtBQUM3RSw2QkFBNkI7QUFDN0IsaURBQWlEO0FBU2pEOztHQUVHO0FBQ0gsTUFBYSxjQUFlLFNBQVEsTUFBTSxDQUFDLFFBQVE7SUFDakQsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUEyQjtRQUNuRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRTtZQUNmLFdBQVcsRUFBRSw4QkFBOEI7WUFDM0MsR0FBRyxLQUFLO1lBQ1IsT0FBTyxFQUFFLElBQUksTUFBTSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsTUFBTSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUM7WUFDdEUsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLG9DQUFvQyxDQUFDLENBQUM7U0FDeEYsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQ0FBcUMsRUFBRSxHQUFHLEVBQUUsRUFBRSxZQUFZLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUMxRixDQUFDO0NBQ0Y7QUFYRCx3Q0FXQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIH5+IEdlbmVyYXRlZCBieSBwcm9qZW4uIFRvIG1vZGlmeSwgZWRpdCAucHJvamVucmMuanMgYW5kIHJ1biBcIm5weCBwcm9qZW5cIi5cbmltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuLyoqXG4gKiBQcm9wcyBmb3IgU3RhdHVzRnVuY3Rpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTdGF0dXNGdW5jdGlvblByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbi8qKlxuICogQW4gQVdTIExhbWJkYSBmdW5jdGlvbiB3aGljaCBleGVjdXRlcyBzcmMvbGFtYmRhcy9zdGF0dXMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTdGF0dXNGdW5jdGlvbiBleHRlbmRzIGxhbWJkYS5GdW5jdGlvbiB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzPzogU3RhdHVzRnVuY3Rpb25Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwge1xuICAgICAgZGVzY3JpcHRpb246ICdzcmMvbGFtYmRhcy9zdGF0dXMubGFtYmRhLnRzJyxcbiAgICAgIC4uLnByb3BzLFxuICAgICAgcnVudGltZTogbmV3IGxhbWJkYS5SdW50aW1lKCdub2RlanMxNC54JywgbGFtYmRhLlJ1bnRpbWVGYW1pbHkuTk9ERUpTKSxcbiAgICAgIGhhbmRsZXI6ICdpbmRleC5oYW5kbGVyJyxcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vLi4vYXNzZXRzL2xhbWJkYXMvc3RhdHVzLmxhbWJkYScpKSxcbiAgICB9KTtcbiAgICB0aGlzLmFkZEVudmlyb25tZW50KCdBV1NfTk9ERUpTX0NPTk5FQ1RJT05fUkVVU0VfRU5BQkxFRCcsICcxJywgeyByZW1vdmVJbkVkZ2U6IHRydWUgfSk7XG4gIH1cbn0iXX0=
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,285 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ /* eslint-disable import/no-extraneous-dependencies */
4
+ const auth_app_1 = require("@octokit/auth-app");
5
+ const core_1 = require("@octokit/core");
6
+ const AWS = require("aws-sdk");
7
+ const github_1 = require("./github");
8
+ const helpers_1 = require("./helpers");
9
+ const cfn = new AWS.CloudFormation();
10
+ const ec2 = new AWS.EC2();
11
+ const ecr = new AWS.ECR();
12
+ const sf = new AWS.StepFunctions();
13
+ function secretArnToUrl(arn) {
14
+ const parts = arn.split(':'); // arn:aws:secretsmanager:us-east-1:12345678:secret:secret-name-REVISION
15
+ const region = parts[3];
16
+ const fullName = parts[6];
17
+ const name = fullName.slice(0, fullName.lastIndexOf('-'));
18
+ return `https://${region}.console.aws.amazon.com/secretsmanager/home?region=${region}#!/secret?name=${name}`;
19
+ }
20
+ function lambdaArnToUrl(arn) {
21
+ const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ
22
+ const region = parts[3];
23
+ const name = parts[6];
24
+ return `https://${region}.console.aws.amazon.com/lambda/home?region=${region}#/functions/${name}?tab=monitoring`;
25
+ }
26
+ function stepFunctionArnToUrl(arn) {
27
+ const parts = arn.split(':'); // arn:aws:states:us-east-1:12345678:stateMachine:name-XYZ
28
+ const region = parts[3];
29
+ return `https://${region}.console.aws.amazon.com/states/home?region=${region}#/statemachines/view/${arn}`;
30
+ }
31
+ async function generateProvidersStatus(stack, logicalId) {
32
+ const resource = await cfn.describeStackResource({ StackName: stack, LogicalResourceId: logicalId }).promise();
33
+ const providers = JSON.parse(resource.StackResourceDetail?.Metadata ?? '{}').providers;
34
+ if (!providers) {
35
+ return {};
36
+ }
37
+ return Promise.all(providers.map(async (p) => {
38
+ // add ECR data, if image is from ECR
39
+ if (p.image?.imageRepository?.match(/[0-9]+\.dkr\.ecr\.[a-z0-9\-]+\.amazonaws\.com\/.+/)) {
40
+ const tags = await ecr.describeImages({
41
+ repositoryName: p.image.imageRepository.split('/')[1],
42
+ filter: {
43
+ tagStatus: 'TAGGED',
44
+ },
45
+ maxResults: 1,
46
+ }).promise();
47
+ if (tags.imageDetails && tags.imageDetails?.length >= 1) {
48
+ p.image.latestImage = {
49
+ tags: tags.imageDetails[0].imageTags,
50
+ digest: tags.imageDetails[0].imageDigest,
51
+ date: tags.imageDetails[0].imagePushedAt,
52
+ };
53
+ }
54
+ }
55
+ // add AMI data, if image is AMI
56
+ if (p.ami?.launchTemplate) {
57
+ const versions = await ec2.describeLaunchTemplateVersions({
58
+ LaunchTemplateId: p.ami.launchTemplate,
59
+ Versions: ['$Default'],
60
+ }).promise();
61
+ if (versions.LaunchTemplateVersions && versions.LaunchTemplateVersions.length >= 1) {
62
+ p.ami.latestAmi = versions.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId;
63
+ }
64
+ }
65
+ return p;
66
+ }));
67
+ }
68
+ exports.handler = async function () {
69
+ // confirm required environment variables
70
+ if (!process.env.WEBHOOK_SECRET_ARN || !process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN || !process.env.LOGICAL_ID ||
71
+ !process.env.WEBHOOK_HANDLER_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SETUP_SECRET_ARN || !process.env.SETUP_FUNCTION_URL ||
72
+ !process.env.STACK_NAME) {
73
+ throw new Error('Missing environment variables');
74
+ }
75
+ // base status
76
+ const status = {
77
+ github: {
78
+ setup: {
79
+ status: 'Unknown',
80
+ url: '',
81
+ secretArn: process.env.SETUP_SECRET_ARN,
82
+ secretUrl: secretArnToUrl(process.env.SETUP_SECRET_ARN),
83
+ },
84
+ domain: 'Unknown',
85
+ webhook: {
86
+ url: process.env.WEBHOOK_URL,
87
+ status: 'Unable to check',
88
+ secretArn: process.env.WEBHOOK_SECRET_ARN,
89
+ secretUrl: secretArnToUrl(process.env.WEBHOOK_SECRET_ARN),
90
+ },
91
+ auth: {
92
+ type: 'Unknown',
93
+ status: 'Unknown',
94
+ secretArn: process.env.GITHUB_SECRET_ARN,
95
+ secretUrl: secretArnToUrl(process.env.GITHUB_SECRET_ARN),
96
+ privateKeySecretArn: process.env.GITHUB_PRIVATE_KEY_SECRET_ARN,
97
+ privateKeySecretUrl: secretArnToUrl(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN),
98
+ app: {
99
+ id: '',
100
+ url: '',
101
+ installations: [],
102
+ },
103
+ personalAuthToken: '',
104
+ },
105
+ },
106
+ providers: await generateProvidersStatus(process.env.STACK_NAME, process.env.LOGICAL_ID),
107
+ troubleshooting: {
108
+ webhookHandlerArn: process.env.WEBHOOK_HANDLER_ARN,
109
+ webhookHandlerUrl: lambdaArnToUrl(process.env.WEBHOOK_HANDLER_ARN),
110
+ stepFunctionArn: process.env.STEP_FUNCTION_ARN,
111
+ stepFunctionUrl: stepFunctionArnToUrl(process.env.STEP_FUNCTION_ARN),
112
+ stepFunctionLogGroup: process.env.STEP_FUNCTION_LOG_GROUP,
113
+ recentRuns: [],
114
+ },
115
+ };
116
+ // setup url
117
+ const setupToken = (await helpers_1.getSecretJsonValue(process.env.SETUP_SECRET_ARN)).token;
118
+ if (setupToken) {
119
+ status.github.setup.status = 'Pending';
120
+ status.github.setup.url = `${process.env.SETUP_FUNCTION_URL}?token=${setupToken}`;
121
+ }
122
+ else {
123
+ status.github.setup.status = 'Complete';
124
+ }
125
+ // list last 10 executions and their status
126
+ try {
127
+ const executions = await sf.listExecutions({
128
+ stateMachineArn: process.env.STEP_FUNCTION_ARN,
129
+ maxResults: 10,
130
+ }).promise();
131
+ for (const execution of executions.executions) {
132
+ const executionDetails = await sf.describeExecution({
133
+ executionArn: execution.executionArn,
134
+ }).promise();
135
+ const input = JSON.parse(executionDetails.input || '{}');
136
+ status.troubleshooting.recentRuns.push({
137
+ executionArn: execution.executionArn,
138
+ status: execution.status,
139
+ owner: input.owner,
140
+ repo: input.repo,
141
+ runId: input.runId,
142
+ });
143
+ }
144
+ }
145
+ catch (e) {
146
+ status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` });
147
+ }
148
+ // get secrets
149
+ let githubSecrets;
150
+ try {
151
+ githubSecrets = await helpers_1.getSecretJsonValue(process.env.GITHUB_SECRET_ARN);
152
+ }
153
+ catch (e) {
154
+ status.github.auth.status = `Unable to read secret: ${e}`;
155
+ return status;
156
+ }
157
+ let privateKey;
158
+ try {
159
+ privateKey = await helpers_1.getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
160
+ }
161
+ catch (e) {
162
+ status.github.auth.status = `Unable to read private key secret: ${e}`;
163
+ return status;
164
+ }
165
+ // calculate base url
166
+ let baseUrl = github_1.baseUrlFromDomain(githubSecrets.domain);
167
+ status.github.domain = githubSecrets.domain;
168
+ if (githubSecrets.personalAuthToken) {
169
+ // try authenticating with personal authentication token
170
+ status.github.auth.type = 'Personal Auth Token';
171
+ status.github.auth.personalAuthToken = '*redacted*';
172
+ let octokit;
173
+ try {
174
+ octokit = new core_1.Octokit({ baseUrl, auth: githubSecrets.personalAuthToken });
175
+ }
176
+ catch (e) {
177
+ status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`;
178
+ return status;
179
+ }
180
+ try {
181
+ const user = await octokit.request('GET /user');
182
+ status.github.auth.personalAuthToken = `username: ${user.data.login}`;
183
+ }
184
+ catch (e) {
185
+ status.github.auth.status = `Unable to call /user with personal auth token: ${e}`;
186
+ return status;
187
+ }
188
+ status.github.auth.status = 'OK';
189
+ status.github.webhook.status = 'Unable to verify automatically';
190
+ }
191
+ else {
192
+ // try authenticating with GitHub app
193
+ status.github.auth.type = 'GitHub App';
194
+ status.github.auth.app.id = githubSecrets.appId;
195
+ let appOctokit;
196
+ try {
197
+ appOctokit = new core_1.Octokit({
198
+ baseUrl,
199
+ authStrategy: auth_app_1.createAppAuth,
200
+ auth: {
201
+ appId: githubSecrets.appId,
202
+ privateKey: privateKey,
203
+ },
204
+ });
205
+ }
206
+ catch (e) {
207
+ status.github.auth.status = `Unable to authenticate app: ${e}`;
208
+ return status;
209
+ }
210
+ // get app url
211
+ try {
212
+ const app = (await appOctokit.request('GET /app')).data;
213
+ status.github.auth.app.url = app.html_url;
214
+ }
215
+ catch (e) {
216
+ status.github.auth.status = `Unable to get app details: ${e}`;
217
+ return status;
218
+ }
219
+ // list all app installations
220
+ try {
221
+ const installations = (await appOctokit.request('GET /app/installations')).data;
222
+ for (const installation of installations) {
223
+ let installationDetails = {
224
+ id: installation.id,
225
+ url: installation.html_url,
226
+ status: 'Unable to query',
227
+ repositories: [],
228
+ };
229
+ let token;
230
+ try {
231
+ token = (await appOctokit.auth({
232
+ type: 'installation',
233
+ installationId: installation.id,
234
+ })).token;
235
+ }
236
+ catch (e) {
237
+ installationDetails.status = `Unable to authenticate app installation: ${e}`;
238
+ continue;
239
+ }
240
+ let octokit;
241
+ try {
242
+ octokit = new core_1.Octokit({ baseUrl, auth: token });
243
+ }
244
+ catch (e) {
245
+ installationDetails.status = `Unable to authenticate using app: ${e}`;
246
+ continue;
247
+ }
248
+ try {
249
+ const repositories = (await octokit.request('GET /installation/repositories')).data.repositories;
250
+ for (const repo of repositories) {
251
+ installationDetails.repositories.push(repo.full_name);
252
+ }
253
+ }
254
+ catch (e) {
255
+ installationDetails.status = `Unable to authenticate using installation token: ${e}`;
256
+ continue;
257
+ }
258
+ installationDetails.status = 'OK';
259
+ status.github.auth.app.installations.push(installationDetails);
260
+ }
261
+ }
262
+ catch (e) {
263
+ status.github.auth.status = 'Unable to list app installations';
264
+ return status;
265
+ }
266
+ status.github.auth.status = 'OK';
267
+ // check webhook config
268
+ try {
269
+ const response = await appOctokit.request('GET /app/hook/config', {});
270
+ if (response.data.url !== process.env.WEBHOOK_URL) {
271
+ status.github.webhook.status = 'GitHub has wrong webhook URL configured';
272
+ }
273
+ else {
274
+ // TODO check secret by doing a dummy delivery? force apply secret?
275
+ status.github.webhook.status = 'OK (note that secret cannot be checked automatically)';
276
+ }
277
+ }
278
+ catch (e) {
279
+ status.github.webhook.status = `Unable to check app configuration: ${e}`;
280
+ return status;
281
+ }
282
+ }
283
+ return status;
284
+ };
285
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.lambda.js","sourceRoot":"","sources":["../../src/lambdas/status.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/no-extraneous-dependencies */\nimport { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/core';\nimport * as AWS from 'aws-sdk';\nimport { baseUrlFromDomain } from './github';\nimport { getSecretJsonValue, getSecretValue } from './helpers';\n\nconst cfn = new AWS.CloudFormation();\nconst ec2 = new AWS.EC2();\nconst ecr = new AWS.ECR();\nconst sf = new AWS.StepFunctions();\n\nfunction secretArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:secretsmanager:us-east-1:12345678:secret:secret-name-REVISION\n  const region = parts[3];\n  const fullName = parts[6];\n  const name = fullName.slice(0, fullName.lastIndexOf('-'));\n\n  return `https://${region}.console.aws.amazon.com/secretsmanager/home?region=${region}#!/secret?name=${name}`;\n}\n\nfunction lambdaArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:lambda:us-east-1:12345678:function:name-XYZ\n  const region = parts[3];\n  const name = parts[6];\n\n  return `https://${region}.console.aws.amazon.com/lambda/home?region=${region}#/functions/${name}?tab=monitoring`;\n}\n\nfunction stepFunctionArnToUrl(arn: string) {\n  const parts = arn.split(':'); // arn:aws:states:us-east-1:12345678:stateMachine:name-XYZ\n  const region = parts[3];\n\n  return `https://${region}.console.aws.amazon.com/states/home?region=${region}#/statemachines/view/${arn}`;\n}\n\nasync function generateProvidersStatus(stack: string, logicalId: string) {\n  const resource = await cfn.describeStackResource({ StackName: stack, LogicalResourceId: logicalId }).promise();\n  const providers = JSON.parse(resource.StackResourceDetail?.Metadata ?? '{}').providers as any[] | undefined;\n\n  if (!providers) {\n    return {};\n  }\n\n  return Promise.all(providers.map(async (p) => {\n    // add ECR data, if image is from ECR\n    if (p.image?.imageRepository?.match(/[0-9]+\\.dkr\\.ecr\\.[a-z0-9\\-]+\\.amazonaws\\.com\\/.+/)) {\n      const tags = await ecr.describeImages({\n        repositoryName: p.image.imageRepository.split('/')[1],\n        filter: {\n          tagStatus: 'TAGGED',\n        },\n        maxResults: 1,\n      }).promise();\n      if (tags.imageDetails && tags.imageDetails?.length >= 1) {\n        p.image.latestImage = {\n          tags: tags.imageDetails[0].imageTags,\n          digest: tags.imageDetails[0].imageDigest,\n          date: tags.imageDetails[0].imagePushedAt,\n        };\n      }\n    }\n    // add AMI data, if image is AMI\n    if (p.ami?.launchTemplate) {\n      const versions = await ec2.describeLaunchTemplateVersions({\n        LaunchTemplateId: p.ami.launchTemplate,\n        Versions: ['$Default'],\n      }).promise();\n      if (versions.LaunchTemplateVersions && versions.LaunchTemplateVersions.length >= 1) {\n        p.ami.latestAmi = versions.LaunchTemplateVersions[0].LaunchTemplateData?.ImageId;\n      }\n    }\n    return p;\n  }));\n}\n\ninterface AppInstallation {\n  readonly id: number;\n  readonly url: string;\n  readonly status: string;\n  readonly repositories: string[];\n}\n\ninterface RecentRun {\n  readonly owner?: string;\n  readonly repo?: string;\n  readonly runId?: string;\n  readonly executionArn?: string;\n  readonly status: string;\n}\n\nexports.handler = async function () {\n  // confirm required environment variables\n  if (!process.env.WEBHOOK_SECRET_ARN || !process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN || !process.env.LOGICAL_ID ||\n      !process.env.WEBHOOK_HANDLER_ARN || !process.env.STEP_FUNCTION_ARN || !process.env.SETUP_SECRET_ARN || !process.env.SETUP_FUNCTION_URL ||\n      !process.env.STACK_NAME) {\n    throw new Error('Missing environment variables');\n  }\n\n  // base status\n  const status = {\n    github: {\n      setup: {\n        status: 'Unknown',\n        url: '',\n        secretArn: process.env.SETUP_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.SETUP_SECRET_ARN),\n      },\n      domain: 'Unknown',\n      webhook: {\n        url: process.env.WEBHOOK_URL,\n        status: 'Unable to check',\n        secretArn: process.env.WEBHOOK_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.WEBHOOK_SECRET_ARN),\n      },\n      auth: {\n        type: 'Unknown',\n        status: 'Unknown',\n        secretArn: process.env.GITHUB_SECRET_ARN,\n        secretUrl: secretArnToUrl(process.env.GITHUB_SECRET_ARN),\n        privateKeySecretArn: process.env.GITHUB_PRIVATE_KEY_SECRET_ARN,\n        privateKeySecretUrl: secretArnToUrl(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN),\n        app: {\n          id: '',\n          url: '',\n          installations: [] as AppInstallation[],\n        },\n        personalAuthToken: '',\n      },\n    },\n    providers: await generateProvidersStatus(process.env.STACK_NAME, process.env.LOGICAL_ID),\n    troubleshooting: {\n      webhookHandlerArn: process.env.WEBHOOK_HANDLER_ARN,\n      webhookHandlerUrl: lambdaArnToUrl(process.env.WEBHOOK_HANDLER_ARN),\n      stepFunctionArn: process.env.STEP_FUNCTION_ARN,\n      stepFunctionUrl: stepFunctionArnToUrl(process.env.STEP_FUNCTION_ARN),\n      stepFunctionLogGroup: process.env.STEP_FUNCTION_LOG_GROUP,\n      recentRuns: [] as RecentRun[],\n    },\n  };\n\n  // setup url\n  const setupToken = (await getSecretJsonValue(process.env.SETUP_SECRET_ARN)).token;\n  if (setupToken) {\n    status.github.setup.status = 'Pending';\n    status.github.setup.url = `${process.env.SETUP_FUNCTION_URL}?token=${setupToken}`;\n  } else {\n    status.github.setup.status = 'Complete';\n  }\n\n  // list last 10 executions and their status\n  try {\n    const executions = await sf.listExecutions({\n      stateMachineArn: process.env.STEP_FUNCTION_ARN,\n      maxResults: 10,\n    }).promise();\n    for (const execution of executions.executions) {\n      const executionDetails = await sf.describeExecution({\n        executionArn: execution.executionArn,\n      }).promise();\n      const input = JSON.parse(executionDetails.input || '{}');\n\n      status.troubleshooting.recentRuns.push({\n        executionArn: execution.executionArn,\n        status: execution.status,\n        owner: input.owner,\n        repo: input.repo,\n        runId: input.runId,\n      });\n    }\n  } catch (e) {\n    status.troubleshooting.recentRuns.push({ status: `Error getting executions: ${e}` });\n  }\n\n  // get secrets\n  let githubSecrets;\n  try {\n    githubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  } catch (e) {\n    status.github.auth.status = `Unable to read secret: ${e}`;\n    return status;\n  }\n\n  let privateKey;\n  try {\n    privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n  } catch (e) {\n    status.github.auth.status = `Unable to read private key secret: ${e}`;\n    return status;\n  }\n\n  // calculate base url\n  let baseUrl = baseUrlFromDomain(githubSecrets.domain);\n  status.github.domain = githubSecrets.domain;\n\n  if (githubSecrets.personalAuthToken) {\n    // try authenticating with personal authentication token\n    status.github.auth.type = 'Personal Auth Token';\n    status.github.auth.personalAuthToken = '*redacted*';\n\n    let octokit;\n    try {\n      octokit = new Octokit({ baseUrl, auth: githubSecrets.personalAuthToken });\n    } catch (e) {\n      status.github.auth.status = `Unable to authenticate using personal auth token: ${e}`;\n      return status;\n    }\n\n    try {\n      const user = await octokit.request('GET /user');\n      status.github.auth.personalAuthToken = `username: ${user.data.login}`;\n    } catch (e) {\n      status.github.auth.status = `Unable to call /user with personal auth token: ${e}`;\n      return status;\n    }\n\n    status.github.auth.status = 'OK';\n    status.github.webhook.status = 'Unable to verify automatically';\n  } else {\n    // try authenticating with GitHub app\n    status.github.auth.type = 'GitHub App';\n    status.github.auth.app.id = githubSecrets.appId;\n\n    let appOctokit;\n    try {\n      appOctokit = new Octokit({\n        baseUrl,\n        authStrategy: createAppAuth,\n        auth: {\n          appId: githubSecrets.appId,\n          privateKey: privateKey,\n        },\n      });\n    } catch (e) {\n      status.github.auth.status = `Unable to authenticate app: ${e}`;\n      return status;\n    }\n\n    // get app url\n    try {\n      const app = (await appOctokit.request('GET /app')).data;\n      status.github.auth.app.url = app.html_url;\n    } catch (e) {\n      status.github.auth.status = `Unable to get app details: ${e}`;\n      return status;\n    }\n\n    // list all app installations\n    try {\n      const installations = (await appOctokit.request('GET /app/installations')).data;\n      for (const installation of installations) {\n        let installationDetails = {\n          id: installation.id,\n          url: installation.html_url,\n          status: 'Unable to query',\n          repositories: [] as string[],\n        };\n\n        let token;\n        try {\n          token = (await appOctokit.auth({\n            type: 'installation',\n            installationId: installation.id,\n          }) as any).token;\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate app installation: ${e}`;\n          continue;\n        }\n\n        let octokit;\n        try {\n          octokit = new Octokit({ baseUrl, auth: token });\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate using app: ${e}`;\n          continue;\n        }\n\n        try {\n          const repositories = (await octokit.request('GET /installation/repositories')).data.repositories;\n          for (const repo of repositories) {\n            installationDetails.repositories.push(repo.full_name as string);\n          }\n        } catch (e) {\n          installationDetails.status = `Unable to authenticate using installation token: ${e}`;\n          continue;\n        }\n\n        installationDetails.status = 'OK';\n        status.github.auth.app.installations.push(installationDetails);\n      }\n    } catch (e) {\n      status.github.auth.status = 'Unable to list app installations';\n      return status;\n    }\n\n    status.github.auth.status = 'OK';\n\n    // check webhook config\n    try {\n      const response = await appOctokit.request('GET /app/hook/config', {});\n\n      if (response.data.url !== process.env.WEBHOOK_URL) {\n        status.github.webhook.status = 'GitHub has wrong webhook URL configured';\n      } else {\n        // TODO check secret by doing a dummy delivery? force apply secret?\n        status.github.webhook.status = 'OK (note that secret cannot be checked automatically)';\n      }\n    } catch (e) {\n      status.github.webhook.status = `Unable to check app configuration: ${e}`;\n      return status;\n    }\n  }\n\n  return status;\n};\n"]}
@@ -0,0 +1,13 @@
1
+ import * as lambda from 'aws-cdk-lib/aws-lambda';
2
+ import { Construct } from 'constructs';
3
+ /**
4
+ * Props for TokenRetrieverFunction
5
+ */
6
+ export interface TokenRetrieverFunctionProps extends lambda.FunctionOptions {
7
+ }
8
+ /**
9
+ * An AWS Lambda function which executes src/lambdas/token-retriever.
10
+ */
11
+ export declare class TokenRetrieverFunction extends lambda.Function {
12
+ constructor(scope: Construct, id: string, props?: TokenRetrieverFunctionProps);
13
+ }
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TokenRetrieverFunction = 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/lambdas/token-retriever.
9
+ */
10
+ class TokenRetrieverFunction extends lambda.Function {
11
+ constructor(scope, id, props) {
12
+ super(scope, id, {
13
+ description: 'src/lambdas/token-retriever.lambda.ts',
14
+ ...props,
15
+ runtime: new lambda.Runtime('nodejs14.x', lambda.RuntimeFamily.NODEJS),
16
+ handler: 'index.handler',
17
+ code: lambda.Code.fromAsset(path.join(__dirname, '../../assets/lambdas/token-retriever.lambda')),
18
+ });
19
+ this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
20
+ }
21
+ }
22
+ exports.TokenRetrieverFunction = TokenRetrieverFunction;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9rZW4tcmV0cmlldmVyLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xhbWJkYXMvdG9rZW4tcmV0cmlldmVyLWZ1bmN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZFQUE2RTtBQUM3RSw2QkFBNkI7QUFDN0IsaURBQWlEO0FBU2pEOztHQUVHO0FBQ0gsTUFBYSxzQkFBdUIsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUN6RCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQW1DO1FBQzNFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLHVDQUF1QztZQUNwRCxHQUFHLEtBQUs7WUFDUixPQUFPLEVBQUUsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQztZQUN0RSxPQUFPLEVBQUUsZUFBZTtZQUN4QixJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsNkNBQTZDLENBQUMsQ0FBQztTQUNqRyxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxFQUFFLEdBQUcsRUFBRSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQzFGLENBQUM7Q0FDRjtBQVhELHdEQVdDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gfn4gR2VuZXJhdGVkIGJ5IHByb2plbi4gVG8gbW9kaWZ5LCBlZGl0IC5wcm9qZW5yYy5qcyBhbmQgcnVuIFwibnB4IHByb2plblwiLlxuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIFByb3BzIGZvciBUb2tlblJldHJpZXZlckZ1bmN0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVG9rZW5SZXRyaWV2ZXJGdW5jdGlvblByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbi8qKlxuICogQW4gQVdTIExhbWJkYSBmdW5jdGlvbiB3aGljaCBleGVjdXRlcyBzcmMvbGFtYmRhcy90b2tlbi1yZXRyaWV2ZXIuXG4gKi9cbmV4cG9ydCBjbGFzcyBUb2tlblJldHJpZXZlckZ1bmN0aW9uIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBUb2tlblJldHJpZXZlckZ1bmN0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnc3JjL2xhbWJkYXMvdG9rZW4tcmV0cmlldmVyLmxhbWJkYS50cycsXG4gICAgICAuLi5wcm9wcyxcbiAgICAgIHJ1bnRpbWU6IG5ldyBsYW1iZGEuUnVudGltZSgnbm9kZWpzMTQueCcsIGxhbWJkYS5SdW50aW1lRmFtaWx5Lk5PREVKUyksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy4uLy4uL2Fzc2V0cy9sYW1iZGFzL3Rva2VuLXJldHJpZXZlci5sYW1iZGEnKSksXG4gICAgfSk7XG4gICAgdGhpcy5hZGRFbnZpcm9ubWVudCgnQVdTX05PREVKU19DT05ORUNUSU9OX1JFVVNFX0VOQUJMRUQnLCAnMScsIHsgcmVtb3ZlSW5FZGdlOiB0cnVlIH0pO1xuICB9XG59Il19
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,15 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const github_1 = require("./github");
4
+ exports.handler = async function (event) {
5
+ const { githubSecrets, octokit } = await github_1.getOctokit(event.installationId);
6
+ const response = await octokit.rest.actions.createRegistrationTokenForRepo({
7
+ owner: event.owner,
8
+ repo: event.repo,
9
+ });
10
+ return {
11
+ domain: githubSecrets.domain,
12
+ token: response.data.token,
13
+ };
14
+ };
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9rZW4tcmV0cmlldmVyLmxhbWJkYS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sYW1iZGFzL3Rva2VuLXJldHJpZXZlci5sYW1iZGEudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSxxQ0FBc0M7QUFHdEMsT0FBTyxDQUFDLE9BQU8sR0FBRyxLQUFLLFdBQVcsS0FBOEI7SUFDOUQsTUFBTSxFQUFFLGFBQWEsRUFBRSxPQUFPLEVBQUUsR0FBRyxNQUFNLG1CQUFVLENBQUMsS0FBSyxDQUFDLGNBQWMsQ0FBQyxDQUFDO0lBRTFFLE1BQU0sUUFBUSxHQUFHLE1BQU0sT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsOEJBQThCLENBQUM7UUFDekUsS0FBSyxFQUFFLEtBQUssQ0FBQyxLQUFLO1FBQ2xCLElBQUksRUFBRSxLQUFLLENBQUMsSUFBSTtLQUNqQixDQUFDLENBQUM7SUFFSCxPQUFPO1FBQ0wsTUFBTSxFQUFFLGFBQWEsQ0FBQyxNQUFNO1FBQzVCLEtBQUssRUFBRSxRQUFRLENBQUMsSUFBSSxDQUFDLEtBQUs7S0FDM0IsQ0FBQztBQUNKLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGdldE9jdG9raXQgfSBmcm9tICcuL2dpdGh1Yic7XG5pbXBvcnQgeyBTdGVwRnVuY3Rpb25MYW1iZGFJbnB1dCB9IGZyb20gJy4vaGVscGVycyc7XG5cbmV4cG9ydHMuaGFuZGxlciA9IGFzeW5jIGZ1bmN0aW9uIChldmVudDogU3RlcEZ1bmN0aW9uTGFtYmRhSW5wdXQpIHtcbiAgY29uc3QgeyBnaXRodWJTZWNyZXRzLCBvY3Rva2l0IH0gPSBhd2FpdCBnZXRPY3Rva2l0KGV2ZW50Lmluc3RhbGxhdGlvbklkKTtcblxuICBjb25zdCByZXNwb25zZSA9IGF3YWl0IG9jdG9raXQucmVzdC5hY3Rpb25zLmNyZWF0ZVJlZ2lzdHJhdGlvblRva2VuRm9yUmVwbyh7XG4gICAgb3duZXI6IGV2ZW50Lm93bmVyLFxuICAgIHJlcG86IGV2ZW50LnJlcG8sXG4gIH0pO1xuXG4gIHJldHVybiB7XG4gICAgZG9tYWluOiBnaXRodWJTZWNyZXRzLmRvbWFpbixcbiAgICB0b2tlbjogcmVzcG9uc2UuZGF0YS50b2tlbixcbiAgfTtcbn07XG4iXX0=
@@ -0,0 +1,13 @@
1
+ import * as lambda from 'aws-cdk-lib/aws-lambda';
2
+ import { Construct } from 'constructs';
3
+ /**
4
+ * Props for UpdateLambdaFunction
5
+ */
6
+ export interface UpdateLambdaFunctionProps extends lambda.FunctionOptions {
7
+ }
8
+ /**
9
+ * An AWS Lambda function which executes src/lambdas/update-lambda.
10
+ */
11
+ export declare class UpdateLambdaFunction extends lambda.Function {
12
+ constructor(scope: Construct, id: string, props?: UpdateLambdaFunctionProps);
13
+ }
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.UpdateLambdaFunction = 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/lambdas/update-lambda.
9
+ */
10
+ class UpdateLambdaFunction extends lambda.Function {
11
+ constructor(scope, id, props) {
12
+ super(scope, id, {
13
+ description: 'src/lambdas/update-lambda.lambda.ts',
14
+ ...props,
15
+ runtime: new lambda.Runtime('nodejs14.x', lambda.RuntimeFamily.NODEJS),
16
+ handler: 'index.handler',
17
+ code: lambda.Code.fromAsset(path.join(__dirname, '../../assets/lambdas/update-lambda.lambda')),
18
+ });
19
+ this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
20
+ }
21
+ }
22
+ exports.UpdateLambdaFunction = UpdateLambdaFunction;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBkYXRlLWxhbWJkYS1mdW5jdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sYW1iZGFzL3VwZGF0ZS1sYW1iZGEtZnVuY3Rpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkVBQTZFO0FBQzdFLDZCQUE2QjtBQUM3QixpREFBaUQ7QUFTakQ7O0dBRUc7QUFDSCxNQUFhLG9CQUFxQixTQUFRLE1BQU0sQ0FBQyxRQUFRO0lBQ3ZELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBaUM7UUFDekUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDZixXQUFXLEVBQUUscUNBQXFDO1lBQ2xELEdBQUcsS0FBSztZQUNSLE9BQU8sRUFBRSxJQUFJLE1BQU0sQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLE1BQU0sQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDO1lBQ3RFLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSwyQ0FBMkMsQ0FBQyxDQUFDO1NBQy9GLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxjQUFjLENBQUMscUNBQXFDLEVBQUUsR0FBRyxFQUFFLEVBQUUsWUFBWSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7SUFDMUYsQ0FBQztDQUNGO0FBWEQsb0RBV0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyB+fiBHZW5lcmF0ZWQgYnkgcHJvamVuLiBUbyBtb2RpZnksIGVkaXQgLnByb2plbnJjLmpzIGFuZCBydW4gXCJucHggcHJvamVuXCIuXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbi8qKlxuICogUHJvcHMgZm9yIFVwZGF0ZUxhbWJkYUZ1bmN0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVXBkYXRlTGFtYmRhRnVuY3Rpb25Qcm9wcyBleHRlbmRzIGxhbWJkYS5GdW5jdGlvbk9wdGlvbnMge1xufVxuXG4vKipcbiAqIEFuIEFXUyBMYW1iZGEgZnVuY3Rpb24gd2hpY2ggZXhlY3V0ZXMgc3JjL2xhbWJkYXMvdXBkYXRlLWxhbWJkYS5cbiAqL1xuZXhwb3J0IGNsYXNzIFVwZGF0ZUxhbWJkYUZ1bmN0aW9uIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBVcGRhdGVMYW1iZGFGdW5jdGlvblByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkLCB7XG4gICAgICBkZXNjcmlwdGlvbjogJ3NyYy9sYW1iZGFzL3VwZGF0ZS1sYW1iZGEubGFtYmRhLnRzJyxcbiAgICAgIC4uLnByb3BzLFxuICAgICAgcnVudGltZTogbmV3IGxhbWJkYS5SdW50aW1lKCdub2RlanMxNC54JywgbGFtYmRhLlJ1bnRpbWVGYW1pbHkuTk9ERUpTKSxcbiAgICAgIGhhbmRsZXI6ICdpbmRleC5oYW5kbGVyJyxcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vLi4vYXNzZXRzL2xhbWJkYXMvdXBkYXRlLWxhbWJkYS5sYW1iZGEnKSksXG4gICAgfSk7XG4gICAgdGhpcy5hZGRFbnZpcm9ubWVudCgnQVdTX05PREVKU19DT05ORUNUSU9OX1JFVVNFX0VOQUJMRUQnLCAnMScsIHsgcmVtb3ZlSW5FZGdlOiB0cnVlIH0pO1xuICB9XG59Il19
@@ -0,0 +1,7 @@
1
+ interface Input {
2
+ readonly lambdaName: string;
3
+ readonly repositoryUri: string;
4
+ readonly repositoryTag: string;
5
+ }
6
+ export declare function handler(event: Input): Promise<void>;
7
+ export {};
@@ -0,0 +1,34 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handler = void 0;
4
+ /* eslint-disable-next-line import/no-extraneous-dependencies */
5
+ const AWS = require("aws-sdk");
6
+ const lambda = new AWS.Lambda();
7
+ function sleep(ms) {
8
+ return new Promise(resolve => setTimeout(resolve, ms));
9
+ }
10
+ async function handler(event) {
11
+ console.log(JSON.stringify(event));
12
+ while (true) {
13
+ try {
14
+ await lambda.updateFunctionCode({
15
+ FunctionName: event.lambdaName,
16
+ ImageUri: `${event.repositoryUri}:${event.repositoryTag}`,
17
+ Publish: true,
18
+ }).promise();
19
+ break;
20
+ }
21
+ catch (e) {
22
+ if (e.code == 'ResourceConflictException') {
23
+ // keep trying if function is already being updated by CloudFormation
24
+ // this can happen if we update some settings on the function and the image code at the same time
25
+ await sleep(10000);
26
+ }
27
+ else {
28
+ throw e;
29
+ }
30
+ }
31
+ }
32
+ }
33
+ exports.handler = handler;
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBkYXRlLWxhbWJkYS5sYW1iZGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbGFtYmRhcy91cGRhdGUtbGFtYmRhLmxhbWJkYS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnRUFBZ0U7QUFDaEUsK0JBQStCO0FBSS9CLE1BQU0sTUFBTSxHQUFHLElBQUksR0FBRyxDQUFDLE1BQU0sRUFBRSxDQUFDO0FBUWhDLFNBQVMsS0FBSyxDQUFDLEVBQVU7SUFDdkIsT0FBTyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLFVBQVUsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQztBQUN6RCxDQUFDO0FBRU0sS0FBSyxVQUFVLE9BQU8sQ0FBQyxLQUFZO0lBQ3hDLE9BQU8sQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBRW5DLE9BQU8sSUFBSSxFQUFFO1FBQ1gsSUFBSTtZQUNGLE1BQU0sTUFBTSxDQUFDLGtCQUFrQixDQUFDO2dCQUM5QixZQUFZLEVBQUUsS0FBSyxDQUFDLFVBQVU7Z0JBQzlCLFFBQVEsRUFBRSxHQUFHLEtBQUssQ0FBQyxhQUFhLElBQUksS0FBSyxDQUFDLGFBQWEsRUFBRTtnQkFDekQsT0FBTyxFQUFFLElBQUk7YUFDZCxDQUFDLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDYixNQUFNO1NBQ1A7UUFBQyxPQUFPLENBQUMsRUFBRTtZQUNWLElBQWUsQ0FBRSxDQUFDLElBQUksSUFBSSwyQkFBMkIsRUFBRTtnQkFDckQscUVBQXFFO2dCQUNyRSxpR0FBaUc7Z0JBQ2pHLE1BQU0sS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO2FBQ3BCO2lCQUFNO2dCQUNMLE1BQU0sQ0FBQyxDQUFDO2FBQ1Q7U0FDRjtLQUNGO0FBQ0gsQ0FBQztBQXJCRCwwQkFxQkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzICovXG5pbXBvcnQgKiBhcyBBV1MgZnJvbSAnYXdzLXNkayc7XG4vKiBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzICovXG5pbXBvcnQgeyBBV1NFcnJvciB9IGZyb20gJ2F3cy1zZGsvbGliL2Vycm9yJztcblxuY29uc3QgbGFtYmRhID0gbmV3IEFXUy5MYW1iZGEoKTtcblxuaW50ZXJmYWNlIElucHV0IHtcbiAgcmVhZG9ubHkgbGFtYmRhTmFtZTogc3RyaW5nO1xuICByZWFkb25seSByZXBvc2l0b3J5VXJpOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHJlcG9zaXRvcnlUYWc6IHN0cmluZztcbn1cblxuZnVuY3Rpb24gc2xlZXAobXM6IG51bWJlcikge1xuICByZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIG1zKSk7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBoYW5kbGVyKGV2ZW50OiBJbnB1dCkge1xuICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShldmVudCkpO1xuXG4gIHdoaWxlICh0cnVlKSB7XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGxhbWJkYS51cGRhdGVGdW5jdGlvbkNvZGUoe1xuICAgICAgICBGdW5jdGlvbk5hbWU6IGV2ZW50LmxhbWJkYU5hbWUsXG4gICAgICAgIEltYWdlVXJpOiBgJHtldmVudC5yZXBvc2l0b3J5VXJpfToke2V2ZW50LnJlcG9zaXRvcnlUYWd9YCxcbiAgICAgICAgUHVibGlzaDogdHJ1ZSxcbiAgICAgIH0pLnByb21pc2UoKTtcbiAgICAgIGJyZWFrO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIGlmICgoPEFXU0Vycm9yPmUpLmNvZGUgPT0gJ1Jlc291cmNlQ29uZmxpY3RFeGNlcHRpb24nKSB7XG4gICAgICAgIC8vIGtlZXAgdHJ5aW5nIGlmIGZ1bmN0aW9uIGlzIGFscmVhZHkgYmVpbmcgdXBkYXRlZCBieSBDbG91ZEZvcm1hdGlvblxuICAgICAgICAvLyB0aGlzIGNhbiBoYXBwZW4gaWYgd2UgdXBkYXRlIHNvbWUgc2V0dGluZ3Mgb24gdGhlIGZ1bmN0aW9uIGFuZCB0aGUgaW1hZ2UgY29kZSBhdCB0aGUgc2FtZSB0aW1lXG4gICAgICAgIGF3YWl0IHNsZWVwKDEwMDAwKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iXX0=
@@ -0,0 +1,13 @@
1
+ import * as lambda from 'aws-cdk-lib/aws-lambda';
2
+ import { Construct } from 'constructs';
3
+ /**
4
+ * Props for WebhookHandlerFunction
5
+ */
6
+ export interface WebhookHandlerFunctionProps extends lambda.FunctionOptions {
7
+ }
8
+ /**
9
+ * An AWS Lambda function which executes src/lambdas/webhook-handler.
10
+ */
11
+ export declare class WebhookHandlerFunction extends lambda.Function {
12
+ constructor(scope: Construct, id: string, props?: WebhookHandlerFunctionProps);
13
+ }
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.WebhookHandlerFunction = 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/lambdas/webhook-handler.
9
+ */
10
+ class WebhookHandlerFunction extends lambda.Function {
11
+ constructor(scope, id, props) {
12
+ super(scope, id, {
13
+ description: 'src/lambdas/webhook-handler.lambda.ts',
14
+ ...props,
15
+ runtime: new lambda.Runtime('nodejs14.x', lambda.RuntimeFamily.NODEJS),
16
+ handler: 'index.handler',
17
+ code: lambda.Code.fromAsset(path.join(__dirname, '../../assets/lambdas/webhook-handler.lambda')),
18
+ });
19
+ this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
20
+ }
21
+ }
22
+ exports.WebhookHandlerFunction = WebhookHandlerFunction;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2ViaG9vay1oYW5kbGVyLWZ1bmN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xhbWJkYXMvd2ViaG9vay1oYW5kbGVyLWZ1bmN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZFQUE2RTtBQUM3RSw2QkFBNkI7QUFDN0IsaURBQWlEO0FBU2pEOztHQUVHO0FBQ0gsTUFBYSxzQkFBdUIsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUN6RCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQW1DO1FBQzNFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLHVDQUF1QztZQUNwRCxHQUFHLEtBQUs7WUFDUixPQUFPLEVBQUUsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQztZQUN0RSxPQUFPLEVBQUUsZUFBZTtZQUN4QixJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsNkNBQTZDLENBQUMsQ0FBQztTQUNqRyxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxFQUFFLEdBQUcsRUFBRSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQzFGLENBQUM7Q0FDRjtBQVhELHdEQVdDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gfn4gR2VuZXJhdGVkIGJ5IHByb2plbi4gVG8gbW9kaWZ5LCBlZGl0IC5wcm9qZW5yYy5qcyBhbmQgcnVuIFwibnB4IHByb2plblwiLlxuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIFByb3BzIGZvciBXZWJob29rSGFuZGxlckZ1bmN0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgV2ViaG9va0hhbmRsZXJGdW5jdGlvblByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbi8qKlxuICogQW4gQVdTIExhbWJkYSBmdW5jdGlvbiB3aGljaCBleGVjdXRlcyBzcmMvbGFtYmRhcy93ZWJob29rLWhhbmRsZXIuXG4gKi9cbmV4cG9ydCBjbGFzcyBXZWJob29rSGFuZGxlckZ1bmN0aW9uIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBXZWJob29rSGFuZGxlckZ1bmN0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnc3JjL2xhbWJkYXMvd2ViaG9vay1oYW5kbGVyLmxhbWJkYS50cycsXG4gICAgICAuLi5wcm9wcyxcbiAgICAgIHJ1bnRpbWU6IG5ldyBsYW1iZGEuUnVudGltZSgnbm9kZWpzMTQueCcsIGxhbWJkYS5SdW50aW1lRmFtaWx5Lk5PREVKUyksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy4uLy4uL2Fzc2V0cy9sYW1iZGFzL3dlYmhvb2staGFuZGxlci5sYW1iZGEnKSksXG4gICAgfSk7XG4gICAgdGhpcy5hZGRFbnZpcm9ubWVudCgnQVdTX05PREVKU19DT05ORUNUSU9OX1JFVVNFX0VOQUJMRUQnLCAnMScsIHsgcmVtb3ZlSW5FZGdlOiB0cnVlIH0pO1xuICB9XG59Il19
@@ -0,0 +1 @@
1
+ export {};