@pwrdrvr/microapps-cdk 0.0.29 → 0.2.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/.jsii +3108 -1362
- package/API.md +240 -53
- package/README.md +236 -135
- package/lib/MicroApps.d.ts +86 -50
- package/lib/MicroApps.js +30 -16
- package/lib/MicroAppsAPIGwy.d.ts +42 -37
- package/lib/MicroAppsAPIGwy.js +19 -18
- package/lib/MicroAppsCF.d.ts +115 -57
- package/lib/MicroAppsCF.js +31 -21
- package/lib/MicroAppsS3.d.ts +51 -46
- package/lib/MicroAppsS3.js +23 -22
- package/lib/MicroAppsSvcs.d.ts +123 -44
- package/lib/MicroAppsSvcs.js +112 -42
- package/lib/microapps-deployer/index.js +93 -89
- package/lib/microapps-deployer/index.js.map +3 -3
- package/lib/microapps-router/index.js.map +2 -2
- package/package.json +15 -36
- package/patches/@aws-cdk+aws-apigatewayv2-alpha+2.8.0-alpha.0.patch +39 -0
package/lib/MicroAppsSvcs.js
CHANGED
|
@@ -5,23 +5,56 @@ exports.MicroAppsSvcs = void 0;
|
|
|
5
5
|
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
6
|
const fs_1 = require("fs");
|
|
7
7
|
const path = require("path");
|
|
8
|
-
const apigwy = require("@aws-cdk/aws-apigatewayv2");
|
|
9
|
-
const
|
|
10
|
-
const dynamodb = require("
|
|
11
|
-
const iam = require("
|
|
12
|
-
const lambda = require("
|
|
13
|
-
const lambdaNodejs = require("
|
|
14
|
-
const logs = require("
|
|
15
|
-
const s3 = require("
|
|
16
|
-
const
|
|
8
|
+
const apigwy = require("@aws-cdk/aws-apigatewayv2-alpha");
|
|
9
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
10
|
+
const dynamodb = require("aws-cdk-lib/aws-dynamodb");
|
|
11
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
12
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
13
|
+
const lambdaNodejs = require("aws-cdk-lib/aws-lambda-nodejs");
|
|
14
|
+
const logs = require("aws-cdk-lib/aws-logs");
|
|
15
|
+
const s3 = require("aws-cdk-lib/aws-s3");
|
|
16
|
+
const constructs_1 = require("constructs");
|
|
17
17
|
/**
|
|
18
|
-
*
|
|
18
|
+
* Class missing from `@aws-cdk/aws-apigatewayv2-alpha`.
|
|
19
19
|
*/
|
|
20
|
-
class
|
|
20
|
+
class HttpRouteIntegration extends apigwy.HttpRouteIntegration {
|
|
21
|
+
constructor(id, opts) {
|
|
22
|
+
super(id);
|
|
23
|
+
this.httpIntegrationProps = opts.integrationProps;
|
|
24
|
+
this.integration = opts.integration;
|
|
25
|
+
}
|
|
21
26
|
/**
|
|
22
|
-
*
|
|
27
|
+
* (experimental) Bind this integration to the route.
|
|
23
28
|
*
|
|
24
|
-
* @
|
|
29
|
+
* @experimental
|
|
30
|
+
*/
|
|
31
|
+
bind(_options) {
|
|
32
|
+
var _b;
|
|
33
|
+
if (this.httpIntegrationProps === undefined) {
|
|
34
|
+
throw new TypeError('bind called without IntegrationProps defined');
|
|
35
|
+
}
|
|
36
|
+
return {
|
|
37
|
+
type: this.httpIntegrationProps.integrationType,
|
|
38
|
+
payloadFormatVersion: (_b = this.httpIntegrationProps.payloadFormatVersion) !== null && _b !== void 0 ? _b : apigwy.PayloadFormatVersion.VERSION_2_0,
|
|
39
|
+
connectionType: this.httpIntegrationProps.connectionType,
|
|
40
|
+
connectionId: this.httpIntegrationProps.connectionId,
|
|
41
|
+
credentials: this.httpIntegrationProps.credentials,
|
|
42
|
+
method: this.httpIntegrationProps.method,
|
|
43
|
+
parameterMapping: this.httpIntegrationProps.parameterMapping,
|
|
44
|
+
secureServerName: this.httpIntegrationProps.secureServerName,
|
|
45
|
+
subtype: this.httpIntegrationProps.integrationSubtype,
|
|
46
|
+
uri: this.httpIntegrationProps.integrationUri,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* (experimental) Create a new MicroApps Services construct, including the Deployer and Router Lambda Functions, and the DynamoDB Table used by both.
|
|
52
|
+
*
|
|
53
|
+
* @experimental
|
|
54
|
+
*/
|
|
55
|
+
class MicroAppsSvcs extends constructs_1.Construct {
|
|
56
|
+
/**
|
|
57
|
+
* @experimental
|
|
25
58
|
*/
|
|
26
59
|
constructor(scope, id, props) {
|
|
27
60
|
var _b;
|
|
@@ -55,13 +88,12 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
55
88
|
// Router Lambda Function
|
|
56
89
|
//
|
|
57
90
|
// Create Router Lambda Function
|
|
58
|
-
let routerFunc;
|
|
59
91
|
const routerFuncProps = {
|
|
60
92
|
functionName: assetNameRoot ? `${assetNameRoot}-router${assetNameSuffix}` : undefined,
|
|
61
93
|
memorySize: 1769,
|
|
62
94
|
logRetention: logs.RetentionDays.ONE_MONTH,
|
|
63
95
|
runtime: lambda.Runtime.NODEJS_14_X,
|
|
64
|
-
timeout:
|
|
96
|
+
timeout: aws_cdk_lib_1.Duration.seconds(15),
|
|
65
97
|
environment: {
|
|
66
98
|
NODE_ENV: appEnv,
|
|
67
99
|
DATABASE_TABLE_NAME: this._table.tableName,
|
|
@@ -72,7 +104,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
72
104
|
if (process.env.NODE_ENV === 'test' &&
|
|
73
105
|
fs_1.existsSync(path.join(__dirname, '..', '..', 'microapps-router', 'dist', 'index.js'))) {
|
|
74
106
|
// This is for local dev
|
|
75
|
-
|
|
107
|
+
this._routerFunc = new lambda.Function(this, 'router-func', {
|
|
76
108
|
code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-router', 'dist')),
|
|
77
109
|
handler: 'index.handler',
|
|
78
110
|
...routerFuncProps,
|
|
@@ -80,7 +112,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
80
112
|
}
|
|
81
113
|
else if (fs_1.existsSync(path.join(__dirname, 'microapps-router', 'index.js'))) {
|
|
82
114
|
// This is for built apps packaged with the CDK construct
|
|
83
|
-
|
|
115
|
+
this._routerFunc = new lambda.Function(this, 'router-func', {
|
|
84
116
|
code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-router')),
|
|
85
117
|
handler: 'index.handler',
|
|
86
118
|
...routerFuncProps,
|
|
@@ -92,7 +124,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
92
124
|
code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-router', 'templates')),
|
|
93
125
|
removalPolicy,
|
|
94
126
|
});
|
|
95
|
-
|
|
127
|
+
this._routerFunc = new lambdaNodejs.NodejsFunction(this, 'router-func', {
|
|
96
128
|
entry: path.join(__dirname, '..', '..', 'microapps-router', 'src', 'index.ts'),
|
|
97
129
|
handler: 'handler',
|
|
98
130
|
bundling: {
|
|
@@ -104,14 +136,14 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
104
136
|
});
|
|
105
137
|
}
|
|
106
138
|
if (removalPolicy !== undefined) {
|
|
107
|
-
|
|
139
|
+
this._routerFunc.applyRemovalPolicy(removalPolicy);
|
|
108
140
|
}
|
|
109
141
|
const policyReadTarget = new iam.PolicyStatement({
|
|
110
142
|
effect: iam.Effect.ALLOW,
|
|
111
143
|
actions: ['s3:GetObject'],
|
|
112
144
|
resources: [`${bucketApps.bucketArn}/*`],
|
|
113
145
|
});
|
|
114
|
-
for (const router of [
|
|
146
|
+
for (const router of [this._routerFunc]) {
|
|
115
147
|
router.addToRolePolicy(policyReadTarget);
|
|
116
148
|
// Give the Router access to DynamoDB table
|
|
117
149
|
this._table.grantReadData(router);
|
|
@@ -132,7 +164,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
132
164
|
memorySize: 1769,
|
|
133
165
|
logRetention: logs.RetentionDays.ONE_MONTH,
|
|
134
166
|
runtime: lambda.Runtime.NODEJS_14_X,
|
|
135
|
-
timeout:
|
|
167
|
+
timeout: aws_cdk_lib_1.Duration.seconds(15),
|
|
136
168
|
environment: {
|
|
137
169
|
NODE_ENV: appEnv,
|
|
138
170
|
APIGWY_ID: httpApi.httpApiId,
|
|
@@ -238,7 +270,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
238
270
|
if (removalPolicy !== undefined) {
|
|
239
271
|
policyDenyPrefixOutsideTag.addCondition(
|
|
240
272
|
// Allows the DeletableBucket Lambda to delete items in the buckets
|
|
241
|
-
'StringNotLike', { 'aws:PrincipalTag/application': `${
|
|
273
|
+
'StringNotLike', { 'aws:PrincipalTag/application': `${aws_cdk_lib_1.Stack.of(this).stackName}-core*` });
|
|
242
274
|
}
|
|
243
275
|
const policyDenyMissingTag = new iam.PolicyStatement({
|
|
244
276
|
sid: 'deny-missing-microapp-name-tag',
|
|
@@ -251,7 +283,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
251
283
|
this._deployerFunc.grantPrincipal,
|
|
252
284
|
// 2021-12-04 - Not 100% sure that this is actually needed...
|
|
253
285
|
// Let's test this and remove if actually not necessary
|
|
254
|
-
new iam.ArnPrincipal(`arn:aws:sts::${
|
|
286
|
+
new iam.ArnPrincipal(`arn:aws:sts::${aws_cdk_lib_1.Aws.ACCOUNT_ID}:assumed-role/${(_b = this._deployerFunc.role) === null || _b === void 0 ? void 0 : _b.roleName}/${this._deployerFunc.functionName}`),
|
|
255
287
|
...s3PolicyBypassArnPrincipals,
|
|
256
288
|
],
|
|
257
289
|
resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],
|
|
@@ -276,13 +308,13 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
276
308
|
// To get the AROA with the AWS CLI:
|
|
277
309
|
// aws iam get-role --role-name ROLE-NAME
|
|
278
310
|
// aws iam get-user -–user-name USER-NAME
|
|
279
|
-
StringNotLike: { 'aws:userid': [
|
|
311
|
+
StringNotLike: { 'aws:userid': [aws_cdk_lib_1.Aws.ACCOUNT_ID, ...s3PolicyBypassAROAMatches] },
|
|
280
312
|
},
|
|
281
313
|
});
|
|
282
314
|
if (removalPolicy !== undefined) {
|
|
283
315
|
policyDenyMissingTag.addCondition(
|
|
284
316
|
// Allows the DeletableBucket Lambda to delete items in the buckets
|
|
285
|
-
'StringNotLike', { 'aws:PrincipalTag/application': `${
|
|
317
|
+
'StringNotLike', { 'aws:PrincipalTag/application': `${aws_cdk_lib_1.Stack.of(this).stackName}-core*` });
|
|
286
318
|
}
|
|
287
319
|
const policyCloudFrontAccess = new iam.PolicyStatement({
|
|
288
320
|
sid: 'cloudfront-oai-access',
|
|
@@ -347,7 +379,7 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
347
379
|
const policyAPIList = new iam.PolicyStatement({
|
|
348
380
|
effect: iam.Effect.ALLOW,
|
|
349
381
|
actions: ['apigateway:GET'],
|
|
350
|
-
resources: [`arn:aws:apigateway:${
|
|
382
|
+
resources: [`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}::/apis`],
|
|
351
383
|
});
|
|
352
384
|
this._deployerFunc.addToRolePolicy(policyAPIList);
|
|
353
385
|
// Grant full control over the API we created
|
|
@@ -355,11 +387,11 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
355
387
|
effect: iam.Effect.ALLOW,
|
|
356
388
|
actions: ['apigateway:*'],
|
|
357
389
|
resources: [
|
|
358
|
-
`arn:aws:apigateway:${
|
|
359
|
-
`arn:aws:apigateway:${
|
|
360
|
-
`arn:aws:apigateway:${
|
|
361
|
-
`arn:aws:apigateway:${
|
|
362
|
-
`arn:aws:apigateway:${
|
|
390
|
+
`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:${httpApi.httpApiId}/*`,
|
|
391
|
+
`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}::/apis/${httpApi.httpApiId}/integrations/*`,
|
|
392
|
+
`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}::/apis/${httpApi.httpApiId}/integrations`,
|
|
393
|
+
`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}::/apis/${httpApi.httpApiId}/routes`,
|
|
394
|
+
`arn:aws:apigateway:${aws_cdk_lib_1.Aws.REGION}::/apis/${httpApi.httpApiId}/routes/*`,
|
|
363
395
|
],
|
|
364
396
|
});
|
|
365
397
|
this._deployerFunc.addToRolePolicy(policyAPIManage);
|
|
@@ -368,8 +400,8 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
368
400
|
effect: iam.Effect.ALLOW,
|
|
369
401
|
actions: ['lambda:*'],
|
|
370
402
|
resources: [
|
|
371
|
-
`arn:aws:lambda:${
|
|
372
|
-
`arn:aws:lambda:${
|
|
403
|
+
`arn:aws:lambda:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:function:*`,
|
|
404
|
+
`arn:aws:lambda:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:function:*:*`,
|
|
373
405
|
],
|
|
374
406
|
conditions: {
|
|
375
407
|
StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },
|
|
@@ -378,31 +410,69 @@ class MicroAppsSvcs extends cdk.Construct {
|
|
|
378
410
|
this._deployerFunc.addToRolePolicy(policyAPIManageLambdas);
|
|
379
411
|
// Create an integration for the Router
|
|
380
412
|
// All traffic without another route goes to the Router
|
|
381
|
-
const intRouter = new
|
|
382
|
-
|
|
413
|
+
const intRouter = new apigwy.HttpIntegration(this, 'router-integration', {
|
|
414
|
+
integrationType: apigwy.HttpIntegrationType.AWS_PROXY,
|
|
415
|
+
httpApi,
|
|
416
|
+
integrationUri: this._routerFunc.functionArn,
|
|
417
|
+
payloadFormatVersion: apigwy.PayloadFormatVersion.VERSION_2_0,
|
|
418
|
+
});
|
|
419
|
+
// new apigwycfn.CfnIntegration(this, 'router-integration', {
|
|
420
|
+
// apiId: httpApi.httpApiId,
|
|
421
|
+
// integrationType: 'AWS_PROXY',
|
|
422
|
+
// integrationUri: routerFunc.functionArn,
|
|
423
|
+
// });
|
|
424
|
+
// new apigwycfn.CfnRoute(this, 'route-default', {
|
|
425
|
+
// apiId: httpApi.httpApiId,
|
|
426
|
+
// routeKey: apigwy.HttpRouteKey.DEFAULT.key,
|
|
427
|
+
// target: `integrations/${intRouter.integrationId}`,
|
|
428
|
+
// });
|
|
429
|
+
// This creates an integration and a router
|
|
430
|
+
const route = new apigwy.HttpRoute(this, 'route-default', {
|
|
383
431
|
httpApi,
|
|
384
432
|
routeKey: apigwy.HttpRouteKey.DEFAULT,
|
|
385
|
-
|
|
433
|
+
// @ts-expect-error null is needed to prevent this.bind call
|
|
434
|
+
authorizer: apigwy.Auth,
|
|
435
|
+
integration: new HttpRouteIntegration('router-integration', {
|
|
436
|
+
integration: intRouter,
|
|
437
|
+
}),
|
|
438
|
+
});
|
|
439
|
+
let routeArn = route.produceRouteArn(apigwy.HttpMethod.ANY);
|
|
440
|
+
// Remove the trailing `/` on the ARN, which is not correct
|
|
441
|
+
routeArn = routeArn.slice(0, routeArn.length - 1);
|
|
442
|
+
// Grant API Gateway permission to invoke the Lambda
|
|
443
|
+
new lambda.CfnPermission(this, 'router-invoke', {
|
|
444
|
+
action: 'lambda:InvokeFunction',
|
|
445
|
+
functionName: this._routerFunc.functionName,
|
|
446
|
+
principal: 'apigateway.amazonaws.com',
|
|
447
|
+
sourceArn: routeArn,
|
|
386
448
|
});
|
|
387
449
|
}
|
|
388
450
|
/**
|
|
389
|
-
* DynamoDB table used by Router, Deployer, and Release console app.
|
|
451
|
+
* (experimental) DynamoDB table used by Router, Deployer, and Release console app.
|
|
390
452
|
*
|
|
391
|
-
* @
|
|
453
|
+
* @experimental
|
|
392
454
|
*/
|
|
393
455
|
get table() {
|
|
394
456
|
return this._table;
|
|
395
457
|
}
|
|
396
458
|
/**
|
|
397
|
-
* Lambda function for the Deployer.
|
|
459
|
+
* (experimental) Lambda function for the Deployer.
|
|
398
460
|
*
|
|
399
|
-
* @
|
|
461
|
+
* @experimental
|
|
400
462
|
*/
|
|
401
463
|
get deployerFunc() {
|
|
402
464
|
return this._deployerFunc;
|
|
403
465
|
}
|
|
466
|
+
/**
|
|
467
|
+
* (experimental) Lambda function for the Router.
|
|
468
|
+
*
|
|
469
|
+
* @experimental
|
|
470
|
+
*/
|
|
471
|
+
get routerFunc() {
|
|
472
|
+
return this._routerFunc;
|
|
473
|
+
}
|
|
404
474
|
}
|
|
405
475
|
exports.MicroAppsSvcs = MicroAppsSvcs;
|
|
406
476
|
_a = JSII_RTTI_SYMBOL_1;
|
|
407
|
-
MicroAppsSvcs[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsSvcs", version: "0.
|
|
408
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsSvcs.js","sourceRoot":"","sources":["../src/MicroAppsSvcs.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from 'fs';\nimport * as path from 'path';\nimport * as apigwy from '@aws-cdk/aws-apigatewayv2';\nimport * as apigwyint from '@aws-cdk/aws-apigatewayv2-integrations';\nimport * as cf from '@aws-cdk/aws-cloudfront';\nimport * as dynamodb from '@aws-cdk/aws-dynamodb';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as lambda from '@aws-cdk/aws-lambda';\nimport * as lambdaNodejs from '@aws-cdk/aws-lambda-nodejs';\nimport * as logs from '@aws-cdk/aws-logs';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cdk from '@aws-cdk/core';\n\nexport interface MicroAppsSvcsProps {\n                                                                                                                                                                                                 \n  readonly removalPolicy?: cdk.RemovalPolicy;\n\n                                                    \n  readonly bucketApps: s3.IBucket;\n\n                                                                                       \n  readonly bucketAppsOAI: cf.OriginAccessIdentity;\n\n                                                                    \n  readonly bucketAppsStaging: s3.IBucket;\n\n                                                       \n  readonly httpApi: apigwy.HttpApi;\n\n  readonly appEnv: string;\n\n                                                                                                                   \n  readonly assetNameRoot?: string;\n\n                                                                                            \n  readonly assetNameSuffix?: string;\n\n  readonly s3StrictBucketPolicy?: boolean;\n  readonly s3PolicyBypassAROAs?: string[];\n  readonly s3PolicyBypassPrincipalARNs?: string[];\n\n                                                                                                     \n  readonly rootPathPrefix?: string;\n}\n\nexport interface IMicroAppsSvcs {\n                                                                                 \n  readonly table: dynamodb.ITable;\n\n                                                 \n  readonly deployerFunc: lambda.IFunction;\n}\n\nexport class MicroAppsSvcs extends cdk.Construct implements IMicroAppsSvcs {\n  private _table: dynamodb.Table;\n  public get table(): dynamodb.ITable {\n    return this._table;\n  }\n\n  private _deployerFunc: lambda.Function;\n  public get deployerFunc(): lambda.IFunction {\n    return this._deployerFunc;\n  }\n\n                                                                                                                                     \n  constructor(scope: cdk.Construct, id: string, props?: MicroAppsSvcsProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const {\n      bucketApps,\n      bucketAppsOAI,\n      bucketAppsStaging,\n      s3PolicyBypassAROAs = [],\n      s3PolicyBypassPrincipalARNs = [],\n      s3StrictBucketPolicy = false,\n      appEnv,\n      httpApi,\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      rootPathPrefix = '',\n    } = props;\n\n    if (s3StrictBucketPolicy === true) {\n      if (s3PolicyBypassAROAs.length === 0 && s3PolicyBypassPrincipalARNs.length === 0) {\n        throw new Error(\n          's3StrictBucketPolicy cannot be true without specifying at least one s3PolicyBypassAROAs or s3PolicyBypassPrincipalARNs',\n        );\n      }\n    }\n\n    //\n    // DynamoDB Table\n    //\n    this._table = new dynamodb.Table(this, 'table', {\n      tableName: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : undefined,\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      partitionKey: {\n        name: 'PK',\n        type: dynamodb.AttributeType.STRING,\n      },\n      sortKey: {\n        name: 'SK',\n        type: dynamodb.AttributeType.STRING,\n      },\n      removalPolicy,\n    });\n\n    //\n    // Router Lambda Function\n    //\n\n    // Create Router Lambda Function\n    let routerFunc: lambda.Function;\n    const routerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-router${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: cdk.Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-router', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-router', 'dist')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-router', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-router')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else {\n      // Create Router Lambda Layer\n      const routerDataFiles = new lambda.LayerVersion(this, 'router-templates', {\n        code: lambda.Code.fromAsset(\n          path.join(__dirname, '..', '..', 'microapps-router', 'templates'),\n        ),\n        removalPolicy,\n      });\n\n      routerFunc = new lambdaNodejs.NodejsFunction(this, 'router-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-router', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        layers: [routerDataFiles],\n        ...routerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      routerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    const policyReadTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject'],\n      resources: [`${bucketApps.bucketArn}/*`],\n    });\n    for (const router of [routerFunc]) {\n      router.addToRolePolicy(policyReadTarget);\n      // Give the Router access to DynamoDB table\n      this._table.grantReadData(router);\n      this._table.grant(router, 'dynamodb:DescribeTable');\n    }\n\n    //\n    // Deployer Lambda Function\n    //\n\n    // Create Deployer Lambda Function\n    const iamRoleUploadName = assetNameRoot\n      ? `${assetNameRoot}-deployer-upload${assetNameSuffix}`\n      : undefined;\n    const deployerFuncName = assetNameRoot\n      ? `${assetNameRoot}-deployer${assetNameSuffix}`\n      : undefined;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: cdk.Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        APIGWY_ID: httpApi.httpApiId,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        FILESTORE_STAGING_BUCKET: bucketAppsStaging.bucketName,\n        FILESTORE_DEST_BUCKET: bucketApps.bucketName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-deployer', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-deployer')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      this._deployerFunc = new lambdaNodejs.NodejsFunction(this, 'deployer-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-deployer', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._deployerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    // Give the Deployer access to DynamoDB table\n    this._table.grantReadWriteData(this._deployerFunc);\n    this._table.grant(this._deployerFunc, 'dynamodb:DescribeTable');\n\n    //\n    // Deloyer upload temp role\n    // Deployer assumes this role with a limited policy to generate\n    // an STS temp token to return to microapps-publish for the upload.\n    //\n    const iamRoleUpload = new iam.Role(this, 'deployer-upload-role', {\n      roleName: iamRoleUploadName,\n      inlinePolicies: {\n        uploadPolicy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              actions: ['s3:ListBucket'],\n              resources: [bucketAppsStaging.bucketArn],\n            }),\n            new iam.PolicyStatement({\n              actions: ['s3:PutObject', 's3:GetObject', 's3:AbortMultipartUpload'],\n              resources: [`${bucketAppsStaging.bucketArn}/*`],\n            }),\n          ],\n        }),\n      },\n      assumedBy: this._deployerFunc.grantPrincipal,\n    });\n    this._deployerFunc.addEnvironment('UPLOAD_ROLE_NAME', iamRoleUpload.roleName);\n\n    //\n    // Update S3 permissions\n    //\n    // Create PrincipalARN List\n    const s3PolicyBypassArnPrincipals: iam.ArnPrincipal[] = [];\n    for (const arnPrincipal of s3PolicyBypassPrincipalARNs) {\n      s3PolicyBypassArnPrincipals.push(new iam.ArnPrincipal(arnPrincipal));\n    }\n    // Create AROA List that matches assumed sessions\n    const s3PolicyBypassAROAMatches: string[] = [];\n    for (const aroa of s3PolicyBypassAROAs) {\n      s3PolicyBypassAROAMatches.push(`${aroa}:*`);\n    }\n    // Deny apps from reading:\n    // - If they are missing the microapp-name tag\n    // - Anything outside of the folder that matches their microapp-name tag\n    const policyDenyPrefixOutsideTag = new iam.PolicyStatement({\n      sid: 'deny-prefix-outside-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        ...s3PolicyBypassArnPrincipals,\n        this._deployerFunc.grantPrincipal,\n      ],\n      notResources: [\n        `${bucketApps.bucketArn}/\\${aws:PrincipalTag/microapp-name}/*`,\n        bucketApps.bucketArn,\n      ],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'false' },\n        // StringNotLike: {'aws:'}\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyPrefixOutsideTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${cdk.Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyDenyMissingTag = new iam.PolicyStatement({\n      sid: 'deny-missing-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        // Exclude the Deployer Function directly\n        this._deployerFunc.grantPrincipal,\n        // 2021-12-04 - Not 100% sure that this is actually needed...\n        // Let's test this and remove if actually not necessary\n        new iam.ArnPrincipal(\n          `arn:aws:sts::${cdk.Aws.ACCOUNT_ID}:assumed-role/${this._deployerFunc.role?.roleName}/${this._deployerFunc.functionName}`,\n        ),\n        ...s3PolicyBypassArnPrincipals,\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'true' },\n        // Note: This AROA must be specified to prevent this policy from locking\n        // out non-root sessions that have assumed the admin role.\n        // The notPrincipals will only match the role name exactly and will not match\n        // any session that has assumed the role since notPrincipals does not allow\n        // wildcard matches and does not do them implicitly either.\n        // The AROA must be used because there are only 3 Principal variables:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n        //  aws:username, aws:userid, aws:PrincipalTag\n        // For an assumed role, aws:username is blank, aws:userid is:\n        //  [unique id AKA AROA for Role]:[session name]\n        // Table of unique ID prefixes such as AROA:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n        // The name of the role is simply not available and if it was\n        // we'd need to write a complicated comparison to make sure\n        // that we didn't exclude the Deny tag from roles in other accounts.\n        //\n        // To get the AROA with the AWS CLI:\n        //   aws iam get-role --role-name ROLE-NAME\n        //   aws iam get-user -–user-name USER-NAME\n        StringNotLike: { 'aws:userid': [cdk.Aws.ACCOUNT_ID, ...s3PolicyBypassAROAMatches] },\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyMissingTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${cdk.Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyCloudFrontAccess = new iam.PolicyStatement({\n      sid: 'cloudfront-oai-access',\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject', 's3:ListBucket'],\n      principals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n\n    if (bucketApps.policy === undefined) {\n      const document = new s3.BucketPolicy(this, 's3-policy', {\n        bucket: bucketApps,\n      }).document;\n      document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        document.addStatements(policyDenyPrefixOutsideTag);\n        document.addStatements(policyDenyMissingTag);\n      }\n    } else {\n      bucketApps.policy.document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        bucketApps.policy.document.addStatements(policyDenyPrefixOutsideTag);\n        bucketApps.policy.document.addStatements(policyDenyMissingTag);\n      }\n    }\n\n    // Allow the Lambda to read from the staging bucket\n    const policyReadListStaging = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      // FIXME: Allow Deployer to delete from Staging bucket\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:ListBucket'],\n      resources: [`${bucketAppsStaging.bucketArn}/*`, bucketAppsStaging.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadListStaging);\n\n    // Allow the Lambda to write to the target bucket and delete\n    const policyReadWriteListTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:PutObject', 's3:ListBucket'],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadWriteListTarget);\n\n    // Allow the deployer to get a temporary STS token\n    const policyGetSTSToken = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:GetFederationToken'],\n      resources: ['*'],\n    });\n    this._deployerFunc.addToRolePolicy(policyGetSTSToken);\n\n    // Allow the deployer to assume the upload role\n    const policyAssumeUpload = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:AssumeRole'],\n      resources: [iamRoleUpload.roleArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyAssumeUpload);\n\n    //\n    // Give Deployer permissions to create routes and integrations\n    // on the API Gateway API.\n    //\n\n    // Grant the ability to List all APIs (we have to find it)\n    const policyAPIList = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:GET'],\n      resources: [`arn:aws:apigateway:${cdk.Aws.REGION}::/apis`],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIList);\n    // Grant full control over the API we created\n    const policyAPIManage = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:*'],\n      resources: [\n        `arn:aws:apigateway:${cdk.Aws.REGION}:${cdk.Aws.ACCOUNT_ID}:${httpApi.httpApiId}/*`,\n        `arn:aws:apigateway:${cdk.Aws.REGION}::/apis/${httpApi.httpApiId}/integrations/*`,\n        `arn:aws:apigateway:${cdk.Aws.REGION}::/apis/${httpApi.httpApiId}/integrations`,\n        `arn:aws:apigateway:${cdk.Aws.REGION}::/apis/${httpApi.httpApiId}/routes`,\n        `arn:aws:apigateway:${cdk.Aws.REGION}::/apis/${httpApi.httpApiId}/routes/*`,\n      ],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManage);\n    // Grant full control over lambdas that indicate they are microapps\n    const policyAPIManageLambdas = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['lambda:*'],\n      resources: [\n        `arn:aws:lambda:${cdk.Aws.REGION}:${cdk.Aws.ACCOUNT_ID}:function:*`,\n        `arn:aws:lambda:${cdk.Aws.REGION}:${cdk.Aws.ACCOUNT_ID}:function:*:*`,\n      ],\n      conditions: {\n        StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n\n    // Create an integration for the Router\n    // All traffic without another route goes to the Router\n    const intRouter = new apigwyint.HttpLambdaIntegration('router-integration', routerFunc);\n    new apigwy.HttpRoute(this, 'route-default', {\n      httpApi,\n      routeKey: apigwy.HttpRouteKey.DEFAULT,\n      integration: intRouter,\n    });\n  }\n}\n"]}
|
|
477
|
+
MicroAppsSvcs[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsSvcs", version: "0.2.6" };
|
|
478
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsSvcs.js","sourceRoot":"","sources":["../src/MicroAppsSvcs.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from 'fs';\nimport * as path from 'path';\nimport * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport { Aws, Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\n\n/**\n * Class missing from `@aws-cdk/aws-apigatewayv2-alpha`.\n */\nclass HttpRouteIntegration extends apigwy.HttpRouteIntegration {\n  private httpIntegrationProps?: apigwy.HttpIntegrationProps;\n\n  constructor(\n    id: string,\n    opts: { integration?: apigwy.HttpIntegration; integrationProps?: apigwy.HttpIntegrationProps },\n  ) {\n    super(id);\n    this.httpIntegrationProps = opts.integrationProps;\n    this.integration = opts.integration;\n  }\n\n  /**\n   * (experimental) Bind this integration to the route.\n   *\n   * @experimental\n   */\n  public bind(_options: apigwy.HttpRouteIntegrationBindOptions): apigwy.HttpRouteIntegrationConfig {\n    if (this.httpIntegrationProps === undefined) {\n      throw new TypeError('bind called without IntegrationProps defined');\n    }\n\n    return {\n      type: this.httpIntegrationProps.integrationType,\n      payloadFormatVersion:\n        this.httpIntegrationProps.payloadFormatVersion ?? apigwy.PayloadFormatVersion.VERSION_2_0,\n      connectionType: this.httpIntegrationProps.connectionType,\n      connectionId: this.httpIntegrationProps.connectionId,\n      credentials: this.httpIntegrationProps.credentials,\n      method: this.httpIntegrationProps.method,\n      parameterMapping: this.httpIntegrationProps.parameterMapping,\n      secureServerName: this.httpIntegrationProps.secureServerName,\n      subtype: this.httpIntegrationProps.integrationSubtype,\n      uri: this.httpIntegrationProps.integrationUri,\n    };\n  }\n}\n\n                                                                   \nexport interface MicroAppsSvcsProps {\n                                                                                                                                                                                                 \n  readonly removalPolicy?: RemovalPolicy;\n\n                                                    \n  readonly bucketApps: s3.IBucket;\n\n                                                                                       \n  readonly bucketAppsOAI: cf.OriginAccessIdentity;\n\n                                                                    \n  readonly bucketAppsStaging: s3.IBucket;\n\n                                                       \n  readonly httpApi: apigwy.HttpApi;\n\n                                                                                                               \n  readonly appEnv: string;\n\n                                                                                                                   \n  readonly assetNameRoot?: string;\n\n                                                                                            \n  readonly assetNameSuffix?: string;\n\n                                                                                                                                                                                                                                                                                                                                \n  readonly s3StrictBucketPolicy?: boolean;\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       \n  readonly s3PolicyBypassPrincipalARNs?: string[];\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     \n  readonly s3PolicyBypassAROAs?: string[];\n\n                                                                                                     \n  readonly rootPathPrefix?: string;\n}\n\n                                          \nexport interface IMicroAppsSvcs {\n                                                                                 \n  readonly table: dynamodb.ITable;\n\n                                                 \n  readonly deployerFunc: lambda.IFunction;\n\n                                               \n  readonly routerFunc: lambda.IFunction;\n}\n\n                                                                                                                                                 \nexport class MicroAppsSvcs extends Construct implements IMicroAppsSvcs {\n  private _table: dynamodb.Table;\n  public get table(): dynamodb.ITable {\n    return this._table;\n  }\n\n  private _deployerFunc: lambda.Function;\n  public get deployerFunc(): lambda.IFunction {\n    return this._deployerFunc;\n  }\n\n  private _routerFunc: lambda.Function;\n  public get routerFunc(): lambda.IFunction {\n    return this._routerFunc;\n  }\n\n  constructor(scope: Construct, id: string, props?: MicroAppsSvcsProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const {\n      bucketApps,\n      bucketAppsOAI,\n      bucketAppsStaging,\n      s3PolicyBypassAROAs = [],\n      s3PolicyBypassPrincipalARNs = [],\n      s3StrictBucketPolicy = false,\n      appEnv,\n      httpApi,\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      rootPathPrefix = '',\n    } = props;\n\n    if (s3StrictBucketPolicy === true) {\n      if (s3PolicyBypassAROAs.length === 0 && s3PolicyBypassPrincipalARNs.length === 0) {\n        throw new Error(\n          's3StrictBucketPolicy cannot be true without specifying at least one s3PolicyBypassAROAs or s3PolicyBypassPrincipalARNs',\n        );\n      }\n    }\n\n    //\n    // DynamoDB Table\n    //\n    this._table = new dynamodb.Table(this, 'table', {\n      tableName: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : undefined,\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      partitionKey: {\n        name: 'PK',\n        type: dynamodb.AttributeType.STRING,\n      },\n      sortKey: {\n        name: 'SK',\n        type: dynamodb.AttributeType.STRING,\n      },\n      removalPolicy,\n    });\n\n    //\n    // Router Lambda Function\n    //\n\n    // Create Router Lambda Function\n    const routerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-router${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-router', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-router', 'dist')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-router', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-router')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else {\n      // Create Router Lambda Layer\n      const routerDataFiles = new lambda.LayerVersion(this, 'router-templates', {\n        code: lambda.Code.fromAsset(\n          path.join(__dirname, '..', '..', 'microapps-router', 'templates'),\n        ),\n        removalPolicy,\n      });\n\n      this._routerFunc = new lambdaNodejs.NodejsFunction(this, 'router-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-router', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        layers: [routerDataFiles],\n        ...routerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._routerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    const policyReadTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject'],\n      resources: [`${bucketApps.bucketArn}/*`],\n    });\n    for (const router of [this._routerFunc]) {\n      router.addToRolePolicy(policyReadTarget);\n      // Give the Router access to DynamoDB table\n      this._table.grantReadData(router);\n      this._table.grant(router, 'dynamodb:DescribeTable');\n    }\n\n    //\n    // Deployer Lambda Function\n    //\n\n    // Create Deployer Lambda Function\n    const iamRoleUploadName = assetNameRoot\n      ? `${assetNameRoot}-deployer-upload${assetNameSuffix}`\n      : undefined;\n    const deployerFuncName = assetNameRoot\n      ? `${assetNameRoot}-deployer${assetNameSuffix}`\n      : undefined;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        APIGWY_ID: httpApi.httpApiId,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        FILESTORE_STAGING_BUCKET: bucketAppsStaging.bucketName,\n        FILESTORE_DEST_BUCKET: bucketApps.bucketName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-deployer', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-deployer')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      this._deployerFunc = new lambdaNodejs.NodejsFunction(this, 'deployer-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-deployer', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._deployerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    // Give the Deployer access to DynamoDB table\n    this._table.grantReadWriteData(this._deployerFunc);\n    this._table.grant(this._deployerFunc, 'dynamodb:DescribeTable');\n\n    //\n    // Deloyer upload temp role\n    // Deployer assumes this role with a limited policy to generate\n    // an STS temp token to return to microapps-publish for the upload.\n    //\n    const iamRoleUpload = new iam.Role(this, 'deployer-upload-role', {\n      roleName: iamRoleUploadName,\n      inlinePolicies: {\n        uploadPolicy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              actions: ['s3:ListBucket'],\n              resources: [bucketAppsStaging.bucketArn],\n            }),\n            new iam.PolicyStatement({\n              actions: ['s3:PutObject', 's3:GetObject', 's3:AbortMultipartUpload'],\n              resources: [`${bucketAppsStaging.bucketArn}/*`],\n            }),\n          ],\n        }),\n      },\n      assumedBy: this._deployerFunc.grantPrincipal,\n    });\n    this._deployerFunc.addEnvironment('UPLOAD_ROLE_NAME', iamRoleUpload.roleName);\n\n    //\n    // Update S3 permissions\n    //\n    // Create PrincipalARN List\n    const s3PolicyBypassArnPrincipals: iam.ArnPrincipal[] = [];\n    for (const arnPrincipal of s3PolicyBypassPrincipalARNs) {\n      s3PolicyBypassArnPrincipals.push(new iam.ArnPrincipal(arnPrincipal));\n    }\n    // Create AROA List that matches assumed sessions\n    const s3PolicyBypassAROAMatches: string[] = [];\n    for (const aroa of s3PolicyBypassAROAs) {\n      s3PolicyBypassAROAMatches.push(`${aroa}:*`);\n    }\n    // Deny apps from reading:\n    // - If they are missing the microapp-name tag\n    // - Anything outside of the folder that matches their microapp-name tag\n    const policyDenyPrefixOutsideTag = new iam.PolicyStatement({\n      sid: 'deny-prefix-outside-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        ...s3PolicyBypassArnPrincipals,\n        this._deployerFunc.grantPrincipal,\n      ],\n      notResources: [\n        `${bucketApps.bucketArn}/\\${aws:PrincipalTag/microapp-name}/*`,\n        bucketApps.bucketArn,\n      ],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'false' },\n        // StringNotLike: {'aws:'}\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyPrefixOutsideTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyDenyMissingTag = new iam.PolicyStatement({\n      sid: 'deny-missing-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        // Exclude the Deployer Function directly\n        this._deployerFunc.grantPrincipal,\n        // 2021-12-04 - Not 100% sure that this is actually needed...\n        // Let's test this and remove if actually not necessary\n        new iam.ArnPrincipal(\n          `arn:aws:sts::${Aws.ACCOUNT_ID}:assumed-role/${this._deployerFunc.role?.roleName}/${this._deployerFunc.functionName}`,\n        ),\n        ...s3PolicyBypassArnPrincipals,\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'true' },\n        // Note: This AROA must be specified to prevent this policy from locking\n        // out non-root sessions that have assumed the admin role.\n        // The notPrincipals will only match the role name exactly and will not match\n        // any session that has assumed the role since notPrincipals does not allow\n        // wildcard matches and does not do them implicitly either.\n        // The AROA must be used because there are only 3 Principal variables:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n        //  aws:username, aws:userid, aws:PrincipalTag\n        // For an assumed role, aws:username is blank, aws:userid is:\n        //  [unique id AKA AROA for Role]:[session name]\n        // Table of unique ID prefixes such as AROA:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n        // The name of the role is simply not available and if it was\n        // we'd need to write a complicated comparison to make sure\n        // that we didn't exclude the Deny tag from roles in other accounts.\n        //\n        // To get the AROA with the AWS CLI:\n        //   aws iam get-role --role-name ROLE-NAME\n        //   aws iam get-user -–user-name USER-NAME\n        StringNotLike: { 'aws:userid': [Aws.ACCOUNT_ID, ...s3PolicyBypassAROAMatches] },\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyMissingTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyCloudFrontAccess = new iam.PolicyStatement({\n      sid: 'cloudfront-oai-access',\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject', 's3:ListBucket'],\n      principals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n\n    if (bucketApps.policy === undefined) {\n      const document = new s3.BucketPolicy(this, 's3-policy', {\n        bucket: bucketApps,\n      }).document;\n      document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        document.addStatements(policyDenyPrefixOutsideTag);\n        document.addStatements(policyDenyMissingTag);\n      }\n    } else {\n      bucketApps.policy.document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        bucketApps.policy.document.addStatements(policyDenyPrefixOutsideTag);\n        bucketApps.policy.document.addStatements(policyDenyMissingTag);\n      }\n    }\n\n    // Allow the Lambda to read from the staging bucket\n    const policyReadListStaging = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      // FIXME: Allow Deployer to delete from Staging bucket\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:ListBucket'],\n      resources: [`${bucketAppsStaging.bucketArn}/*`, bucketAppsStaging.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadListStaging);\n\n    // Allow the Lambda to write to the target bucket and delete\n    const policyReadWriteListTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:PutObject', 's3:ListBucket'],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadWriteListTarget);\n\n    // Allow the deployer to get a temporary STS token\n    const policyGetSTSToken = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:GetFederationToken'],\n      resources: ['*'],\n    });\n    this._deployerFunc.addToRolePolicy(policyGetSTSToken);\n\n    // Allow the deployer to assume the upload role\n    const policyAssumeUpload = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:AssumeRole'],\n      resources: [iamRoleUpload.roleArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyAssumeUpload);\n\n    //\n    // Give Deployer permissions to create routes and integrations\n    // on the API Gateway API.\n    //\n\n    // Grant the ability to List all APIs (we have to find it)\n    const policyAPIList = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:GET'],\n      resources: [`arn:aws:apigateway:${Aws.REGION}::/apis`],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIList);\n    // Grant full control over the API we created\n    const policyAPIManage = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:*'],\n      resources: [\n        `arn:aws:apigateway:${Aws.REGION}:${Aws.ACCOUNT_ID}:${httpApi.httpApiId}/*`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/integrations/*`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/integrations`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/routes`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/routes/*`,\n      ],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManage);\n    // Grant full control over lambdas that indicate they are microapps\n    const policyAPIManageLambdas = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['lambda:*'],\n      resources: [\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*`,\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*:*`,\n      ],\n      conditions: {\n        StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n\n    // Create an integration for the Router\n    // All traffic without another route goes to the Router\n    const intRouter = new apigwy.HttpIntegration(this, 'router-integration', {\n      integrationType: apigwy.HttpIntegrationType.AWS_PROXY,\n      httpApi,\n      integrationUri: this._routerFunc.functionArn,\n      payloadFormatVersion: apigwy.PayloadFormatVersion.VERSION_2_0,\n    });\n    // new apigwycfn.CfnIntegration(this, 'router-integration', {\n    //   apiId: httpApi.httpApiId,\n    //   integrationType: 'AWS_PROXY',\n    //   integrationUri: routerFunc.functionArn,\n    // });\n    // new apigwycfn.CfnRoute(this, 'route-default', {\n    //   apiId: httpApi.httpApiId,\n    //   routeKey: apigwy.HttpRouteKey.DEFAULT.key,\n    //   target: `integrations/${intRouter.integrationId}`,\n    // });\n\n    // This creates an integration and a router\n    const route = new apigwy.HttpRoute(this, 'route-default', {\n      httpApi,\n      routeKey: apigwy.HttpRouteKey.DEFAULT,\n      // @ts-expect-error null is needed to prevent this.bind call\n      authorizer: apigwy.Auth,\n      integration: new HttpRouteIntegration('router-integration', {\n        integration: intRouter,\n      }),\n    });\n\n    let routeArn = route.produceRouteArn(apigwy.HttpMethod.ANY);\n    // Remove the trailing `/` on the ARN, which is not correct\n    routeArn = routeArn.slice(0, routeArn.length - 1);\n\n    // Grant API Gateway permission to invoke the Lambda\n    new lambda.CfnPermission(this, 'router-invoke', {\n      action: 'lambda:InvokeFunction',\n      functionName: this._routerFunc.functionName,\n      principal: 'apigateway.amazonaws.com',\n      sourceArn: routeArn,\n    });\n  }\n}\n"]}
|