@pwrdrvr/microapps-cdk 0.0.18 → 0.0.22

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 CHANGED
@@ -1239,7 +1239,7 @@
1239
1239
  },
1240
1240
  "name": "@pwrdrvr/microapps-cdk",
1241
1241
  "readme": {
1242
- "markdown": "# Overview\n\nThe MicroApps project...\n\n# Project Layout\n\n- [src/cdk]() - CDK Stacks\n - MicroAppsS3\n - Creates S3 buckets\n - MicroAppsRepos\n - Creates the ECR repos for components to be published into;\n - Deployer\n - Router\n - MicroAppsSvcs\n - Create DynamoDB table\n - Create Deployer Lambda function\n - Create Router Lambda function\n - Create APIGateway HTTP API\n - MicroAppsCF\n - Creates Cloudfront distribution\n - MicroAppsR53\n - Creates domain names to point to the edge (Cloudfront) and origin (API Gateway)\n- [src/microapps-deployer]()\n - Lambda service invoked by `microapps-publish` to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n- [src/microapps-publish]()\n - Node executable that updates versions in config files, deploys static assets to the S3 staging bucket, optionally compiles and deploys a new Lambda function version, and invokes `microapps-deployer`\n - Permissions required:\n - Lambda invoke\n - S3 publish to the staging bucket\n - ECR write\n - Lambda version publish\n- [src/microapps-router]()\n - Lambda function that determines which version of an app to point a user to on a particular invocation\n\n# Useful Commands\n\n- `npm run build` compiles TypeSript to JavaScript\n- `npm run lint` checks TypeScript for compliance with Lint rules\n- `cdk list` list the stack names\n- `cdk deploy` deploy this stack to your default AWS account/region\n- `cdk diff` compare deployed stack with current state\n- `cdk synth` emits the synthesized CloudFormation template\n\n# Running CDK\n\nAlways run CDK from the root of the git repo, which is the directory containing `cdk.json`.\n\n## Set AWS Profile\n\n`export AWS_PROFILE=pwrdrvr`\n\n## Set NVM Version\n\n`nvm use`\n\n# Deployer Service\n\nCopies static assets from staging to deployed directory, creates record of application / version in DynamoDB Table.\n\n## Build and Deploy Update\n\n```\nmake aws-ecr-login\nmake aws-ecr-publish-deployer\nmake aws-lambda-update-deployer\n```\n\n# Notes on Selection of Docker Image Lambdas\n\nThe Router and Deployer services are very small (0.5 MB) after tree shaking, minification, and uglification performed by `rollup`. The router has the tightest performance requirement and performed just as well as a docker image vs a zip file. However, docker image start up is up to 2x longer vs the zip file for the router; this should not be a problem for any live system with continuous usage and for demos the router can be initialized or pre-provisioned beforehand. The development benefits of docker images for Lambda outweigh the small init time impact on cold starts.\n\n# Notes on Performance\n\n## Router\n\nFor best demo performance (and real user performance), the memory for the Router Lambda should be set to 1024 MB as this gives the fastest cold start at the lowest cost. The cost per warm request is actually lower at 1024 MB than at 128 MB, so 1024 MB is just the ideal size.\n\nFor supremely optimum demo performance the Router Lambda should be deployed as a .zip file as that saves about 50% of the cold start time, or about 200 ms, but once it's the cold start has happened they are equally as fast as each other.\n\n- Lambda Memory (which linearly scales CPU) Speeds\n - Docker Image Lambda\n - Note: All times captured with Rollup ~400 KB Docker Layer\n - 128 MB\n - Duration Warm: 118 ms\n - Duration Cold: 763 ms\n - Init Duration: 518 ms\n - Billed Duration Warm: 119 ms\n - Billed Duration Init: 1,282 ms\n - Warm Cost: 0.025 millicents\n - Init Cost: 0.26 millicents\n - 256 MB\n - Duration Warm: 30 ms\n - Duration Cold: 363 ms\n - Init Duration: 488 ms\n - Billed Duration Warm: 30 ms\n - Billed Duration Init: 853 ms\n - Warm Cost: 0.013 millicents\n - Init Cost: 0.36 millicents\n - 512 MB\n - Duration Warm: 10 ms\n - Duration Cold: 176 ms\n - Init Duration: 572 ms\n - Billed Duration Warm: 10 ms\n - Billed Duration Init: 749 ms\n - Warm Cost: 0.0083 millicents\n - Init Cost: 0.62 millicents\n - 1024 MB\n - Duration Warm: 9 ms\n - Duration Cold: 84.5 ms\n - Init Duration: 497 ms\n - Billed Duration Warm: 9 ms\n - Billed Duration Init: 585 ms\n - Warm Cost: 0.015 millicents\n - Init Cost: 0.97 millicents\n - _Init performance scales linearly up to and including 1024 MB_\n - 1769 MB\n - This is the point at which a Lambda has 100% of 1 CPU\n - https://docs.aws.amazon.com/lambda/latest/dg/configuration-memory.html\n - Duration Warm: 8.31 ms\n - Duration Cold: 73 ms\n - Init Duration: 514 ms\n - Billed Duration Warm: 10 ms\n - Billed Duration Cold: 587 ms\n - Warm Cost: 0.029 millicents\n - Init Cost: 1.7 millicents\n - 2048 MB\n - Duration Warm: 10 ms\n - Duration Cold: 67 ms\n - Init Duration: 497 ms\n - Billed Duration Warm: 11 ms\n - Billed Duration Init: 566 ms\n - Warm Cost: 0.037 millicents\n - Init Cost: 1.89 millicents\n - Zip File Lambda\n - 128 MB\n - Duration Warm: 110 ms\n - Duration Cold: 761 ms\n - Init Duration: 210 ms\n - Billed Duration Warm: 120 ms\n - Billed Duration Init: 762 ms\n - Warm Cost: 0.025 millicents\n - Init Cost: 0.16 millicents\n - 512 MB\n - Duration Warm: 10 ms\n - Duration Cold: 179 ms\n - Init Duration: 201 ms\n - Billed Duration Warm: 12 ms\n - Billed Duration Init: 185 ms\n - Warm Cost: 0.01 millicents\n - Init Cost: 0.15 millicents\n - 1024 MB\n - Duration Warm: 10 ms\n - Duration Cold: 85 ms\n - Init Duration: 185 ms\n - Billed Duration Warm: 12 ms\n - Billed Duration Init: 85 ms\n - Warm Cost: 0.02 millicents\n - Init Cost: 0.14 millicents\n"
1242
+ "markdown": "# Overview\n\nThe MicroApps project....\n\n# Project Layout\n\n- [packages/cdk]() - CDK Stacks\n - MicroAppsS3\n - Creates S3 buckets\n - MicroAppsRepos\n - Creates the ECR repos for components to be published into;\n - Deployer\n - Router\n - MicroAppsSvcs\n - Create DynamoDB table\n - Create Deployer Lambda function\n - Create Router Lambda function\n - Create APIGateway HTTP API\n - MicroAppsCF\n - Creates Cloudfront distribution\n - MicroAppsR53\n - Creates domain names to point to the edge (Cloudfront) and origin (API Gateway)\n- [packages/microapps-deployer]()\n - Lambda service invoked by `microapps-publish` to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n- [packages/microapps-publish]()\n - Node executable that updates versions in config files, deploys static assets to the S3 staging bucket, optionally compiles and deploys a new Lambda function version, and invokes `microapps-deployer`\n - Permissions required:\n - Lambda invoke\n - S3 publish to the staging bucket\n - ECR write\n - Lambda version publish\n- [packages/microapps-router]()\n - Lambda function that determines which version of an app to point a user to on a particular invocation\n\n# Useful Commands\n\n- `npm run build` compiles TypeSript to JavaScript\n- `npm run lint` checks TypeScript for compliance with Lint rules\n- `cdk list` list the stack names\n- `cdk deploy` deploy this stack to your default AWS account/region\n- `cdk diff` compare deployed stack with current state\n- `cdk synth` emits the synthesized CloudFormation template\n\n# Running CDK\n\nAlways run CDK from the root of the git repo, which is the directory containing `cdk.json`.\n\n## Set AWS Profile\n\n`export AWS_PROFILE=pwrdrvr`\n\n## Set NVM Version\n\n`nvm use`\n\n# Deployer Service\n\nCopies static assets from staging to deployed directory, creates record of application / version in DynamoDB Table.\n\n## Build and Deploy Update\n\n```\nmake aws-ecr-login\nmake aws-ecr-publish-deployer\nmake aws-lambda-update-deployer\n```\n\n# Notes on Selection of Docker Image Lambdas\n\nThe Router and Deployer services are very small (0.5 MB) after tree shaking, minification, and uglification performed by `rollup`. The router has the tightest performance requirement and performed just as well as a docker image vs a zip file. However, docker image start up is up to 2x longer vs the zip file for the router; this should not be a problem for any live system with continuous usage and for demos the router can be initialized or pre-provisioned beforehand. The development benefits of docker images for Lambda outweigh the small init time impact on cold starts.\n\n# Notes on Performance\n\n## Router\n\nFor best demo performance (and real user performance), the memory for the Router Lambda should be set to 1024 MB as this gives the fastest cold start at the lowest cost. The cost per warm request is actually lower at 1024 MB than at 128 MB, so 1024 MB is just the ideal size.\n\nFor supremely optimum demo performance the Router Lambda should be deployed as a .zip file as that saves about 50% of the cold start time, or about 200 ms, but once it's the cold start has happened they are equally as fast as each other.\n\n- Lambda Memory (which linearly scales CPU) Speeds\n - Docker Image Lambda\n - Note: All times captured with Rollup ~400 KB Docker Layer\n - 128 MB\n - Duration Warm: 118 ms\n - Duration Cold: 763 ms\n - Init Duration: 518 ms\n - Billed Duration Warm: 119 ms\n - Billed Duration Init: 1,282 ms\n - Warm Cost: 0.025 millicents\n - Init Cost: 0.26 millicents\n - 256 MB\n - Duration Warm: 30 ms\n - Duration Cold: 363 ms\n - Init Duration: 488 ms\n - Billed Duration Warm: 30 ms\n - Billed Duration Init: 853 ms\n - Warm Cost: 0.013 millicents\n - Init Cost: 0.36 millicents\n - 512 MB\n - Duration Warm: 10 ms\n - Duration Cold: 176 ms\n - Init Duration: 572 ms\n - Billed Duration Warm: 10 ms\n - Billed Duration Init: 749 ms\n - Warm Cost: 0.0083 millicents\n - Init Cost: 0.62 millicents\n - 1024 MB\n - Duration Warm: 9 ms\n - Duration Cold: 84.5 ms\n - Init Duration: 497 ms\n - Billed Duration Warm: 9 ms\n - Billed Duration Init: 585 ms\n - Warm Cost: 0.015 millicents\n - Init Cost: 0.97 millicents\n - _Init performance scales linearly up to and including 1024 MB_\n - 1769 MB\n - This is the point at which a Lambda has 100% of 1 CPU\n - https://docs.aws.amazon.com/lambda/latest/dg/configuration-memory.html\n - Duration Warm: 8.31 ms\n - Duration Cold: 73 ms\n - Init Duration: 514 ms\n - Billed Duration Warm: 10 ms\n - Billed Duration Cold: 587 ms\n - Warm Cost: 0.029 millicents\n - Init Cost: 1.7 millicents\n - 2048 MB\n - Duration Warm: 10 ms\n - Duration Cold: 67 ms\n - Init Duration: 497 ms\n - Billed Duration Warm: 11 ms\n - Billed Duration Init: 566 ms\n - Warm Cost: 0.037 millicents\n - Init Cost: 1.89 millicents\n - Zip File Lambda\n - 128 MB\n - Duration Warm: 110 ms\n - Duration Cold: 761 ms\n - Init Duration: 210 ms\n - Billed Duration Warm: 120 ms\n - Billed Duration Init: 762 ms\n - Warm Cost: 0.025 millicents\n - Init Cost: 0.16 millicents\n - 512 MB\n - Duration Warm: 10 ms\n - Duration Cold: 179 ms\n - Init Duration: 201 ms\n - Billed Duration Warm: 12 ms\n - Billed Duration Init: 185 ms\n - Warm Cost: 0.01 millicents\n - Init Cost: 0.15 millicents\n - 1024 MB\n - Duration Warm: 10 ms\n - Duration Cold: 85 ms\n - Init Duration: 185 ms\n - Billed Duration Warm: 12 ms\n - Billed Duration Init: 85 ms\n - Warm Cost: 0.02 millicents\n - Init Cost: 0.14 millicents\n"
1243
1243
  },
1244
1244
  "repository": {
1245
1245
  "type": "git",
@@ -1585,6 +1585,6 @@
1585
1585
  ]
1586
1586
  }
1587
1587
  },
1588
- "version": "0.0.18",
1589
- "fingerprint": "dz8DUjQqsyCPvBa1izkmiMLSACZviYvS4jpR/9BVBEQ="
1588
+ "version": "0.0.22",
1589
+ "fingerprint": "uUx2VQh0n/qu4k/BX+ryV/oCN37TP9JZ4AI1g4mnwa0="
1590
1590
  }
package/README.md CHANGED
@@ -1,10 +1,10 @@
1
1
  # Overview
2
2
 
3
- The MicroApps project...
3
+ The MicroApps project....
4
4
 
5
5
  # Project Layout
6
6
 
7
- - [src/cdk]() - CDK Stacks
7
+ - [packages/cdk]() - CDK Stacks
8
8
  - MicroAppsS3
9
9
  - Creates S3 buckets
10
10
  - MicroAppsRepos
@@ -20,16 +20,16 @@ The MicroApps project...
20
20
  - Creates Cloudfront distribution
21
21
  - MicroAppsR53
22
22
  - Creates domain names to point to the edge (Cloudfront) and origin (API Gateway)
23
- - [src/microapps-deployer]()
23
+ - [packages/microapps-deployer]()
24
24
  - Lambda service invoked by `microapps-publish` to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.
25
- - [src/microapps-publish]()
25
+ - [packages/microapps-publish]()
26
26
  - Node executable that updates versions in config files, deploys static assets to the S3 staging bucket, optionally compiles and deploys a new Lambda function version, and invokes `microapps-deployer`
27
27
  - Permissions required:
28
28
  - Lambda invoke
29
29
  - S3 publish to the staging bucket
30
30
  - ECR write
31
31
  - Lambda version publish
32
- - [src/microapps-router]()
32
+ - [packages/microapps-router]()
33
33
  - Lambda function that determines which version of an app to point a user to on a particular invocation
34
34
 
35
35
  # Useful Commands
package/lib/MicroApps.js CHANGED
@@ -73,5 +73,5 @@ class MicroApps extends cdk.Construct {
73
73
  }
74
74
  exports.MicroApps = MicroApps;
75
75
  _a = JSII_RTTI_SYMBOL_1;
76
- MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.0.18" };
76
+ MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.0.22" };
77
77
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroApps.js","sourceRoot":"","sources":["../src/MicroApps.ts"],"names":[],"mappings":";;;;;AACA,qCAAqC;AACrC,+CAA4C;AAC5C,+CAA4C;AAC5C,mDAAgD;;;;;;AAmDhD,MAAa,SAAU,SAAQ,GAAG,CAAC,SAAS;;;;IAU1C,YAAY,KAAoB,EAAE,EAAU,EAAE,KAAsB;QAClE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,KAAK,KAAK,SAAS,EAAE;YACvB,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;SACtC;QAED,MAAM,EACJ,UAAU,EACV,cAAc,EACd,gBAAgB,EAChB,aAAa,GAAG,WAAW,EAC3B,eAAe,GAAG,EAAE,EACpB,oBAAoB,GAAG,KAAK,EAC5B,SAAS,EACT,WAAW,EACX,QAAQ,EACR,OAAO,EACP,MAAM,EACN,MAAM,GAAG,KAAK,EACd,UAAU,EACV,kBAAkB,EAClB,sBAAsB,GAAG,aAAa,GACvC,GAAG,KAAK,CAAC;QACV,MAAM,iBAAiB,GAAG,SAAS,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QAE9D,MAAM,EAAE,GAAG,IAAI,yBAAW,CAAC,IAAI,EAAE,cAAc,EAAE;YAC/C,oBAAoB;YACpB,iBAAiB;YACjB,aAAa;YACb,eAAe;SAChB,CAAC,CAAC;QACH,MAAM,EAAE,GAAG,IAAI,yBAAW,CAAC,IAAI,EAAE,sBAAsB,EAAE;YACvD,SAAS,EAAE,EAAE;YACb,aAAa;YACb,eAAe;YACf,UAAU;YACV,iBAAiB;YACjB,cAAc;YACd,gBAAgB;YAChB,oBAAoB;YACpB,SAAS;YACT,WAAW;YACX,QAAQ;SACT,CAAC,CAAC;QACH,IAAI,6BAAa,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACxC,cAAc,EAAE,EAAE;YAClB,SAAS,EAAE,EAAE;YACb,aAAa;YACb,eAAe;YACf,UAAU;YACV,iBAAiB;YACjB,cAAc;YACd,gBAAgB;YAChB,oBAAoB;YACpB,SAAS;YACT,WAAW;YACX,OAAO;YACP,MAAM;YACN,MAAM;YACN,UAAU;YACV,kBAAkB;YAClB,sBAAsB;SACvB,CAAC,CAAC;IACL,CAAC;IAzED,yDAAyD;IACjD,MAAM,CAAC,aAAa,CAAC,MAAc;QACzC,IAAI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;QACxC,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SACxB;QACD,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;;AARH,8BA2EC","sourcesContent":["import * as acm from '@aws-cdk/aws-certificatemanager';\nimport * as cdk from '@aws-cdk/core';\nimport { MicroAppsCF } from './MicroAppsCF';\nimport { MicroAppsS3 } from './MicroAppsS3';\nimport { MicroAppsSvcs } from './MicroAppsSvcs';\n\n                                 \nexport interface MicroAppsProps {\n                                                                                                \n  readonly autoDeleteEverything?: boolean;\n\n                                                                                                   \n  readonly appEnv: string;\n\n                                                                   \n  readonly assetNameRoot: string;\n\n                                                                                                  \n  readonly assetNameSuffix?: string;\n\n                                                                                                                 \n  readonly domainName: string;\n\n                                                                                                \n  readonly r53ZoneName: string;\n\n                                                              \n  readonly r53ZoneID: string;\n\n                                                                               \n  readonly certEdge: acm.ICertificate;\n\n                                                                         \n  readonly certOrigin: acm.ICertificate;\n\n                                                                                                                     \n  readonly s3PolicyBypassRoleName: string;\n\n                                                                                                                                                                                                                                                      \n  readonly s3PolicyBypassAROA: string;\n\n                                                                                                                                                         \n  readonly account: string;\n\n                                                                                                                                               \n  readonly region: string;\n\n                                                                                           \n  readonly domainNameEdge: string;\n\n                                                                                                 \n  readonly domainNameOrigin: string;\n}\n\n                                                          \nexport class MicroApps extends cdk.Construct {\n  // input like 'example.com.' will return as 'com.example'\n  private static reverseDomain(domain: string): string {\n    let parts = domain.split('.').reverse();\n    if (parts[0] === '') {\n      parts = parts.slice(1);\n    }\n    return parts.join('.');\n  }\n\n  constructor(scope: cdk.Construct, id: string, props?: MicroAppsProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const {\n      domainName,\n      domainNameEdge,\n      domainNameOrigin,\n      assetNameRoot = 'microapps',\n      assetNameSuffix = '',\n      autoDeleteEverything = false,\n      r53ZoneID,\n      r53ZoneName,\n      certEdge,\n      account,\n      region,\n      appEnv = 'dev',\n      certOrigin,\n      s3PolicyBypassAROA,\n      s3PolicyBypassRoleName = 'AdminAccess',\n    } = props;\n    const reverseDomainName = MicroApps.reverseDomain(domainName);\n\n    const s3 = new MicroAppsS3(this, 'microapps-s3', {\n      autoDeleteEverything,\n      reverseDomainName,\n      assetNameRoot,\n      assetNameSuffix,\n    });\n    const cf = new MicroAppsCF(this, 'microapps-cloudfront', {\n      s3Exports: s3,\n      assetNameRoot,\n      assetNameSuffix,\n      domainName,\n      reverseDomainName,\n      domainNameEdge,\n      domainNameOrigin,\n      autoDeleteEverything,\n      r53ZoneID,\n      r53ZoneName,\n      certEdge,\n    });\n    new MicroAppsSvcs(this, 'microapps-svcs', {\n      cfStackExports: cf,\n      s3Exports: s3,\n      assetNameRoot,\n      assetNameSuffix,\n      domainName,\n      reverseDomainName,\n      domainNameEdge,\n      domainNameOrigin,\n      autoDeleteEverything,\n      r53ZoneID,\n      r53ZoneName,\n      account,\n      region,\n      appEnv,\n      certOrigin,\n      s3PolicyBypassAROA,\n      s3PolicyBypassRoleName,\n    });\n  }\n}\n"]}
@@ -23,7 +23,7 @@ class MicroAppsSvcs extends cdk.Construct {
23
23
  }
24
24
  const { bucketApps, bucketAppsName, bucketAppsOAI, bucketAppsStaging, bucketAppsStagingName } = props.s3Exports;
25
25
  const { r53ZoneID, r53ZoneName, s3PolicyBypassAROA, s3PolicyBypassRoleName, autoDeleteEverything, appEnv, domainNameEdge, domainNameOrigin, certOrigin, account, region, assetNameRoot, assetNameSuffix, } = props;
26
- const apigatewayName = assetNameRoot;
26
+ const apigatewayName = `${assetNameRoot}${assetNameSuffix}`;
27
27
  //
28
28
  // DynamoDB Table
29
29
  //
@@ -43,6 +43,130 @@ class MicroAppsSvcs extends cdk.Construct {
43
43
  table.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
44
44
  }
45
45
  //
46
+ // Router Lambda Function
47
+ //
48
+ // Create Router Lambda Function
49
+ let routerFunc;
50
+ const routerFuncProps = {
51
+ functionName: `${assetNameRoot}-router${assetNameSuffix}`,
52
+ memorySize: 1024,
53
+ logRetention: logs.RetentionDays.ONE_MONTH,
54
+ runtime: lambda.Runtime.NODEJS_14_X,
55
+ timeout: cdk.Duration.seconds(15),
56
+ environment: {
57
+ NODE_ENV: appEnv,
58
+ DATABASE_TABLE_NAME: table.tableName,
59
+ AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',
60
+ },
61
+ };
62
+ if (fs_1.existsSync(`${path.resolve(__dirname)}/../dist/microapps-router/index.js`)) {
63
+ // This is for local dev
64
+ routerFunc = new lambda.Function(this, 'microapps-router-func', {
65
+ code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-router/`),
66
+ handler: 'index.handler',
67
+ ...routerFuncProps,
68
+ });
69
+ }
70
+ else if (fs_1.existsSync(`${path.resolve(__dirname)}/microapps-router/index.js`)) {
71
+ // This is for built apps packaged with the CDK construct
72
+ routerFunc = new lambda.Function(this, 'microapps-router-func', {
73
+ code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-router/`),
74
+ handler: 'index.handler',
75
+ ...routerFuncProps,
76
+ });
77
+ }
78
+ else {
79
+ // Create Router Lambda Layer
80
+ const routerDataFiles = new lambda.LayerVersion(this, 'microapps-router-layer', {
81
+ code: lambda.Code.fromAsset('./packages/microapps-router/templates/'),
82
+ });
83
+ if (autoDeleteEverything) {
84
+ routerDataFiles.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
85
+ }
86
+ routerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-router-func', {
87
+ entry: './packages/microapps-router/src/index.ts',
88
+ handler: 'handler',
89
+ bundling: {
90
+ minify: true,
91
+ sourceMap: true,
92
+ },
93
+ layers: [routerDataFiles],
94
+ ...routerFuncProps,
95
+ });
96
+ }
97
+ if (autoDeleteEverything) {
98
+ routerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
99
+ }
100
+ const policyReadTarget = new iam.PolicyStatement({
101
+ effect: iam.Effect.ALLOW,
102
+ actions: ['s3:GetObject'],
103
+ resources: [`${bucketApps.bucketArn}/*`],
104
+ });
105
+ for (const router of [routerFunc]) {
106
+ router.addToRolePolicy(policyReadTarget);
107
+ // Give the Router access to DynamoDB table
108
+ table.grantReadData(router);
109
+ table.grant(router, 'dynamodb:DescribeTable');
110
+ }
111
+ // TODO: Add Last Route for /*/{proxy+}
112
+ // Note: That might not work, may need a Behavior in CloudFront
113
+ // or a Lambda @ Edge function that detects these and routes
114
+ // to origin Lambda Router function.
115
+ //
116
+ // APIGateway domain names for CloudFront and origin
117
+ //
118
+ // Create Custom Domains for API Gateway
119
+ const dnAppsEdge = new apigwy.DomainName(this, 'microapps-apps-edge-dn', {
120
+ domainName: domainNameEdge,
121
+ certificate: certOrigin,
122
+ });
123
+ if (autoDeleteEverything) {
124
+ dnAppsEdge.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
125
+ }
126
+ this._dnAppsOrigin = new apigwy.DomainName(this, 'microapps-apps-origin-dn', {
127
+ domainName: domainNameOrigin,
128
+ certificate: certOrigin,
129
+ });
130
+ if (autoDeleteEverything) {
131
+ this._dnAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
132
+ }
133
+ // Create an integration for the Router
134
+ // Do this here since it's the default route
135
+ const intRouter = new apigwyint.LambdaProxyIntegration({
136
+ handler: routerFunc,
137
+ });
138
+ // Create APIGateway for the Edge name
139
+ const httpApiDomainMapping = {
140
+ domainName: dnAppsEdge,
141
+ };
142
+ const httpApi = new apigwy.HttpApi(this, 'microapps-api', {
143
+ defaultDomainMapping: httpApiDomainMapping,
144
+ defaultIntegration: intRouter,
145
+ apiName: apigatewayName,
146
+ });
147
+ if (autoDeleteEverything) {
148
+ httpApi.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
149
+ }
150
+ // Add default route on API Gateway to point to the router
151
+ // httpApi.addRoutes({
152
+ // path: '$default',
153
+ // integration: intRouter,
154
+ // });
155
+ //
156
+ // Let API Gateway accept requests using domainNameOrigin
157
+ // That is the origin URI that CloudFront uses for this gateway.
158
+ // The gateway will refuse the traffic if it doesn't have the
159
+ // domain name registered.
160
+ //
161
+ const mappingAppsApis = new apigwy.ApiMapping(this, 'microapps-api-mapping-origin', {
162
+ api: httpApi,
163
+ domainName: this.dnAppsOrigin,
164
+ });
165
+ mappingAppsApis.node.addDependency(this.dnAppsOrigin);
166
+ if (autoDeleteEverything) {
167
+ mappingAppsApis.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
168
+ }
169
+ //
46
170
  // Deployer Lambda Function
47
171
  //
48
172
  // Create Deployer Lambda Function
@@ -57,7 +181,7 @@ class MicroAppsSvcs extends cdk.Construct {
57
181
  timeout: cdk.Duration.seconds(15),
58
182
  environment: {
59
183
  NODE_ENV: appEnv,
60
- APIGWY_NAME: apigatewayName,
184
+ APIGWY_ID: httpApi.httpApiId,
61
185
  DATABASE_TABLE_NAME: table.tableName,
62
186
  FILESTORE_STAGING_BUCKET: bucketAppsStagingName,
63
187
  FILESTORE_DEST_BUCKET: bucketAppsName,
@@ -83,7 +207,7 @@ class MicroAppsSvcs extends cdk.Construct {
83
207
  }
84
208
  else {
85
209
  deployerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-deployer-func', {
86
- entry: './src/microapps-deployer/src/index.ts',
210
+ entry: './packages/microapps-deployer/src/index.ts',
87
211
  handler: 'handler',
88
212
  bundling: {
89
213
  minify: true,
@@ -243,125 +367,6 @@ class MicroAppsSvcs extends cdk.Construct {
243
367
  });
244
368
  deployerFunc.addToRolePolicy(policyAssumeUpload);
245
369
  //
246
- // Router Lambda Function
247
- //
248
- // Create Router Lambda Function
249
- let routerFunc;
250
- const routerFuncProps = {
251
- functionName: `${assetNameRoot}-router${assetNameSuffix}`,
252
- memorySize: 1024,
253
- logRetention: logs.RetentionDays.ONE_MONTH,
254
- runtime: lambda.Runtime.NODEJS_14_X,
255
- timeout: cdk.Duration.seconds(15),
256
- environment: {
257
- NODE_ENV: appEnv,
258
- DATABASE_TABLE_NAME: table.tableName,
259
- AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',
260
- },
261
- };
262
- if (fs_1.existsSync(`${path.resolve(__dirname)}/../dist/microapps-router/index.js`)) {
263
- // This is for local dev
264
- routerFunc = new lambda.Function(this, 'microapps-router-func', {
265
- code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-router/`),
266
- handler: 'index.handler',
267
- ...routerFuncProps,
268
- });
269
- }
270
- else if (fs_1.existsSync(`${path.resolve(__dirname)}/microapps-router/index.js`)) {
271
- // This is for built apps packaged with the CDK construct
272
- routerFunc = new lambda.Function(this, 'microapps-router-func', {
273
- code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-router/`),
274
- handler: 'index.handler',
275
- ...routerFuncProps,
276
- });
277
- }
278
- else {
279
- // Create Router Lambda Layer
280
- const routerDataFiles = new lambda.LayerVersion(this, 'microapps-router-layer', {
281
- code: lambda.Code.fromAsset('./src/microapps-router/templates/'),
282
- });
283
- if (autoDeleteEverything) {
284
- routerDataFiles.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
285
- }
286
- routerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-router-func', {
287
- entry: './src/microapps-router/src/index.ts',
288
- handler: 'handler',
289
- bundling: {
290
- minify: true,
291
- sourceMap: true,
292
- },
293
- layers: [routerDataFiles],
294
- ...routerFuncProps,
295
- });
296
- }
297
- if (autoDeleteEverything) {
298
- routerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
299
- }
300
- const policyReadTarget = new iam.PolicyStatement({
301
- effect: iam.Effect.ALLOW,
302
- actions: ['s3:GetObject'],
303
- resources: [`${bucketApps.bucketArn}/*`],
304
- });
305
- for (const router of [routerFunc]) {
306
- router.addToRolePolicy(policyReadTarget);
307
- // Give the Router access to DynamoDB table
308
- table.grantReadData(router);
309
- table.grant(router, 'dynamodb:DescribeTable');
310
- }
311
- // TODO: Add Last Route for /*/{proxy+}
312
- // Note: That might not work, may need a Behavior in CloudFront
313
- // or a Lambda @ Edge function that detects these and routes
314
- // to origin Lambda Router function.
315
- //
316
- // APIGateway domain names for CloudFront and origin
317
- //
318
- // Create Custom Domains for API Gateway
319
- const dnAppsEdge = new apigwy.DomainName(this, 'microapps-apps-edge-dn', {
320
- domainName: domainNameEdge,
321
- certificate: certOrigin,
322
- });
323
- if (autoDeleteEverything) {
324
- dnAppsEdge.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
325
- }
326
- this._dnAppsOrigin = new apigwy.DomainName(this, 'microapps-apps-origin-dn', {
327
- domainName: domainNameOrigin,
328
- certificate: certOrigin,
329
- });
330
- if (autoDeleteEverything) {
331
- this._dnAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
332
- }
333
- // Create an integration for the Router
334
- // Do this here since it's the default route
335
- const intRouter = new apigwyint.LambdaProxyIntegration({
336
- handler: routerFunc,
337
- });
338
- // Create APIGateway for the Edge name
339
- const httpApiDomainMapping = {
340
- domainName: dnAppsEdge,
341
- };
342
- const httpApi = new apigwy.HttpApi(this, 'microapps-api', {
343
- defaultDomainMapping: httpApiDomainMapping,
344
- defaultIntegration: intRouter,
345
- apiName: apigatewayName,
346
- });
347
- if (autoDeleteEverything) {
348
- httpApi.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
349
- }
350
- //
351
- // Let API Gateway accept requests using domainNameOrigin
352
- // That is the origin URI that CloudFront uses for this gateway.
353
- // The gateway will refuse the traffic if it doesn't have the
354
- // domain name registered.
355
- //
356
- const mappingAppsApis = new apigwy.ApiMapping(this, 'microapps-api-mapping-origin', {
357
- api: httpApi,
358
- domainName: this.dnAppsOrigin,
359
- });
360
- mappingAppsApis.node.addDependency(this.dnAppsOrigin);
361
- if (autoDeleteEverything) {
362
- mappingAppsApis.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);
363
- }
364
- //
365
370
  // Give Deployer permissions to create routes and integrations
366
371
  // on the API Gateway API.
367
372
  //
@@ -417,4 +422,4 @@ class MicroAppsSvcs extends cdk.Construct {
417
422
  }
418
423
  }
419
424
  exports.MicroAppsSvcs = MicroAppsSvcs;
420
- //# 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 acm from '@aws-cdk/aws-certificatemanager';\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 r53 from '@aws-cdk/aws-route53';\nimport * as r53targets from '@aws-cdk/aws-route53-targets';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cdk from '@aws-cdk/core';\nimport { IMicroAppsCFExports } from './MicroAppsCF';\nimport { IMicroAppsS3Exports } from './MicroAppsS3';\n\ninterface MicroAppsSvcsStackProps extends cdk.ResourceProps {\n  readonly cfStackExports: IMicroAppsCFExports;\n  readonly s3Exports: IMicroAppsS3Exports;\n\n  readonly appEnv: string;\n  readonly autoDeleteEverything: boolean;\n  readonly reverseDomainName: string;\n  readonly domainName: string;\n  readonly domainNameEdge: string;\n  readonly domainNameOrigin: string;\n\n  readonly assetNameRoot: string;\n  readonly assetNameSuffix: string;\n\n  readonly certOrigin: acm.ICertificate;\n\n  readonly r53ZoneName: string;\n  readonly r53ZoneID: string;\n\n  readonly s3PolicyBypassAROA: string;\n  readonly s3PolicyBypassRoleName: string;\n\n  readonly account: string;\n  readonly region: string;\n}\n\nexport interface IMicroAppsSvcsExports {\n  readonly dnAppsOrigin: apigwy.DomainName;\n}\n\nexport class MicroAppsSvcs extends cdk.Construct implements IMicroAppsSvcsExports {\n  private _dnAppsOrigin: apigwy.DomainName;\n  public get dnAppsOrigin(): apigwy.DomainName {\n    return this._dnAppsOrigin;\n  }\n\n  constructor(scope: cdk.Construct, id: string, props?: MicroAppsSvcsStackProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const { bucketApps, bucketAppsName, bucketAppsOAI, bucketAppsStaging, bucketAppsStagingName } =\n      props.s3Exports;\n    const {\n      r53ZoneID,\n      r53ZoneName,\n      s3PolicyBypassAROA,\n      s3PolicyBypassRoleName,\n      autoDeleteEverything,\n      appEnv,\n      domainNameEdge,\n      domainNameOrigin,\n      certOrigin,\n      account,\n      region,\n      assetNameRoot,\n      assetNameSuffix,\n    } = props;\n\n    const apigatewayName = assetNameRoot;\n\n    //\n    // DynamoDB Table\n    //\n    const table = new dynamodb.Table(this, 'microapps-router-table', {\n      tableName: `${assetNameRoot}${assetNameSuffix}`,\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    });\n    if (autoDeleteEverything) {\n      table.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    //\n    // Deployer Lambda Function\n    //\n\n    // Create Deployer Lambda Function\n    const iamRoleUploadName = `${assetNameRoot}-deployer-upload${assetNameSuffix}`;\n    const deployerFuncName = `${assetNameRoot}-deployer${assetNameSuffix}`;\n    let deployerFunc: lambda.Function;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      memorySize: 1024,\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_NAME: apigatewayName,\n        DATABASE_TABLE_NAME: table.tableName,\n        FILESTORE_STAGING_BUCKET: bucketAppsStagingName,\n        FILESTORE_DEST_BUCKET: bucketAppsName,\n        UPLOAD_ROLE_NAME: iamRoleUploadName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n      },\n    };\n    if (existsSync(`${path.resolve(__dirname)}/../dist/microapps-deployer/index.js`)) {\n      // This is for local dev\n      deployerFunc = new lambda.Function(this, 'microapps-deployer-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-deployer/`),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(`${path.resolve(__dirname)}/microapps-deployer/index.js`)) {\n      // This is for built apps packaged with the CDK construct\n      deployerFunc = new lambda.Function(this, 'microapps-deployer-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-deployer/`),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      deployerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-deployer-func', {\n        entry: './src/microapps-deployer/src/index.ts',\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (autoDeleteEverything) {\n      deployerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n    // Give the Deployer access to DynamoDB table\n    table.grantReadWriteData(deployerFunc);\n    table.grant(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, 'microapps-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: deployerFunc.grantPrincipal,\n    });\n\n    //\n    // Update S3 permissions\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        new iam.ArnPrincipal(`arn:aws:iam::${account}:role/${s3PolicyBypassRoleName}`),\n        deployerFunc.grantPrincipal,\n        // Allow the builder user to update the buckets\n        new iam.ArnPrincipal(\n          `arn:aws:iam::${account}:user/${assetNameRoot}-builder-${props.appEnv}`,\n        ),\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 (autoDeleteEverything) {\n      policyDenyPrefixOutsideTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${assetNameRoot}-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        new iam.ArnPrincipal(`arn:aws:iam::${account}:role/${s3PolicyBypassRoleName}`),\n        deployerFunc.grantPrincipal,\n        new iam.ArnPrincipal(\n          `arn:aws:sts::${account}:assumed-role/${deployerFunc?.role?.roleName}/${deployerFunc.functionName}`,\n        ),\n        // Allow the builder user to update the buckets\n        new iam.ArnPrincipal(\n          `arn:aws:iam::${account}:user/${assetNameRoot}-builder-${props.appEnv}`,\n        ),\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 uased 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        StringNotLike: { 'aws:userid': [`${s3PolicyBypassAROA}:*`, account] },\n      },\n    });\n    if (autoDeleteEverything) {\n      policyDenyMissingTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${assetNameRoot}-core*` },\n      );\n    }\n    const policyCloudFrontAccess = new iam.PolicyStatement({\n      sid: 'cloudfront-oai-access',\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject'],\n      principals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n      ],\n      resources: [`${bucketApps.bucketArn}/*`],\n    });\n    if (bucketApps.policy === undefined) {\n      const document = new s3.BucketPolicy(this, 'CFPolicy', {\n        bucket: bucketApps,\n      }).document;\n      document.addStatements(policyCloudFrontAccess);\n      document.addStatements(policyDenyPrefixOutsideTag);\n      document.addStatements(policyDenyMissingTag);\n    } else {\n      bucketApps.policy.document.addStatements(policyCloudFrontAccess);\n      bucketApps.policy.document.addStatements(policyDenyPrefixOutsideTag);\n      bucketApps.policy.document.addStatements(policyDenyMissingTag);\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:GetObject', 's3:ListBucket'],\n      resources: [`${bucketAppsStaging.bucketArn}/*`, bucketAppsStaging.bucketArn],\n    });\n    deployerFunc.addToRolePolicy(policyReadListStaging);\n\n    // Allow the Lambda to write to the target bucket\n    const policyReadWriteListTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject', 's3:PutObject', 's3:ListBucket'],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n    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    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    deployerFunc.addToRolePolicy(policyAssumeUpload);\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}-router${assetNameSuffix}`,\n      memorySize: 1024,\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: table.tableName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n      },\n    };\n    if (existsSync(`${path.resolve(__dirname)}/../dist/microapps-router/index.js`)) {\n      // This is for local dev\n      routerFunc = new lambda.Function(this, 'microapps-router-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-router/`),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else if (existsSync(`${path.resolve(__dirname)}/microapps-router/index.js`)) {\n      // This is for built apps packaged with the CDK construct\n      routerFunc = new lambda.Function(this, 'microapps-router-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-router/`),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else {\n      // Create Router Lambda Layer\n      const routerDataFiles = new lambda.LayerVersion(this, 'microapps-router-layer', {\n        code: lambda.Code.fromAsset('./src/microapps-router/templates/'),\n      });\n      if (autoDeleteEverything) {\n        routerDataFiles.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n      }\n\n      routerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-router-func', {\n        entry: './src/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 (autoDeleteEverything) {\n      routerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\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      table.grantReadData(router);\n      table.grant(router, 'dynamodb:DescribeTable');\n    }\n\n    // TODO: Add Last Route for /*/{proxy+}\n    // Note: That might not work, may need a Behavior in CloudFront\n    //       or a Lambda @ Edge function that detects these and routes\n    //       to origin Lambda Router function.\n\n    //\n    // APIGateway domain names for CloudFront and origin\n    //\n\n    // Create Custom Domains for API Gateway\n    const dnAppsEdge = new apigwy.DomainName(this, 'microapps-apps-edge-dn', {\n      domainName: domainNameEdge,\n      certificate: certOrigin,\n    });\n    if (autoDeleteEverything) {\n      dnAppsEdge.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n    this._dnAppsOrigin = new apigwy.DomainName(this, 'microapps-apps-origin-dn', {\n      domainName: domainNameOrigin,\n      certificate: certOrigin,\n    });\n    if (autoDeleteEverything) {\n      this._dnAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    // Create an integration for the Router\n    // Do this here since it's the default route\n    const intRouter = new apigwyint.LambdaProxyIntegration({\n      handler: routerFunc,\n    });\n\n    // Create APIGateway for the Edge name\n    const httpApiDomainMapping: apigwy.DomainMappingOptions = {\n      domainName: dnAppsEdge,\n    };\n    const httpApi = new apigwy.HttpApi(this, 'microapps-api', {\n      defaultDomainMapping: httpApiDomainMapping,\n      defaultIntegration: intRouter,\n      apiName: apigatewayName,\n    });\n    if (autoDeleteEverything) {\n      httpApi.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    //\n    // Let API Gateway accept requests using domainNameOrigin\n    // That is the origin URI that CloudFront uses for this gateway.\n    // The gateway will refuse the traffic if it doesn't have the\n    // domain name registered.\n    //\n    const mappingAppsApis = new apigwy.ApiMapping(this, 'microapps-api-mapping-origin', {\n      api: httpApi,\n      domainName: this.dnAppsOrigin,\n    });\n    mappingAppsApis.node.addDependency(this.dnAppsOrigin);\n    if (autoDeleteEverything) {\n      mappingAppsApis.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\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:${region}::/apis`],\n    });\n    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:${region}:${account}:${httpApi.httpApiId}/*`,\n        `arn:aws:apigateway:${region}::/apis/${httpApi.httpApiId}/integrations`,\n        `arn:aws:apigateway:${region}::/apis/${httpApi.httpApiId}/routes`,\n      ],\n    });\n    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:${region}:${account}:function:*`,\n        `arn:aws:lambda:${region}:${account}:function:*:*`,\n      ],\n      conditions: {\n        StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n\n    //\n    // Create the origin name for API Gateway\n    //\n\n    const zone = r53.HostedZone.fromHostedZoneAttributes(this, 'microapps-zone', {\n      zoneName: r53ZoneName,\n      hostedZoneId: r53ZoneID,\n    });\n\n    const rrAppsOrigin = new r53.ARecord(this, 'microapps-origin-arecord', {\n      zone: zone,\n      recordName: domainNameOrigin,\n      target: r53.RecordTarget.fromAlias(\n        new r53targets.ApiGatewayv2DomainProperties(\n          this._dnAppsOrigin.regionalDomainName,\n          this._dnAppsOrigin.regionalHostedZoneId,\n        ),\n      ),\n    });\n    if (autoDeleteEverything) {\n      rrAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n  }\n}\n"]}
425
+ //# 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 acm from '@aws-cdk/aws-certificatemanager';\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 r53 from '@aws-cdk/aws-route53';\nimport * as r53targets from '@aws-cdk/aws-route53-targets';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cdk from '@aws-cdk/core';\nimport { IMicroAppsCFExports } from './MicroAppsCF';\nimport { IMicroAppsS3Exports } from './MicroAppsS3';\n\ninterface MicroAppsSvcsStackProps extends cdk.ResourceProps {\n  readonly cfStackExports: IMicroAppsCFExports;\n  readonly s3Exports: IMicroAppsS3Exports;\n\n  readonly appEnv: string;\n  readonly autoDeleteEverything: boolean;\n  readonly reverseDomainName: string;\n  readonly domainName: string;\n  readonly domainNameEdge: string;\n  readonly domainNameOrigin: string;\n\n  readonly assetNameRoot: string;\n  readonly assetNameSuffix: string;\n\n  readonly certOrigin: acm.ICertificate;\n\n  readonly r53ZoneName: string;\n  readonly r53ZoneID: string;\n\n  readonly s3PolicyBypassAROA: string;\n  readonly s3PolicyBypassRoleName: string;\n\n  readonly account: string;\n  readonly region: string;\n}\n\nexport interface IMicroAppsSvcsExports {\n  readonly dnAppsOrigin: apigwy.DomainName;\n}\n\nexport class MicroAppsSvcs extends cdk.Construct implements IMicroAppsSvcsExports {\n  private _dnAppsOrigin: apigwy.DomainName;\n  public get dnAppsOrigin(): apigwy.DomainName {\n    return this._dnAppsOrigin;\n  }\n\n  constructor(scope: cdk.Construct, id: string, props?: MicroAppsSvcsStackProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const { bucketApps, bucketAppsName, bucketAppsOAI, bucketAppsStaging, bucketAppsStagingName } =\n      props.s3Exports;\n    const {\n      r53ZoneID,\n      r53ZoneName,\n      s3PolicyBypassAROA,\n      s3PolicyBypassRoleName,\n      autoDeleteEverything,\n      appEnv,\n      domainNameEdge,\n      domainNameOrigin,\n      certOrigin,\n      account,\n      region,\n      assetNameRoot,\n      assetNameSuffix,\n    } = props;\n\n    const apigatewayName = `${assetNameRoot}${assetNameSuffix}`;\n\n    //\n    // DynamoDB Table\n    //\n    const table = new dynamodb.Table(this, 'microapps-router-table', {\n      tableName: `${assetNameRoot}${assetNameSuffix}`,\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    });\n    if (autoDeleteEverything) {\n      table.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\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}-router${assetNameSuffix}`,\n      memorySize: 1024,\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: table.tableName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n      },\n    };\n    if (existsSync(`${path.resolve(__dirname)}/../dist/microapps-router/index.js`)) {\n      // This is for local dev\n      routerFunc = new lambda.Function(this, 'microapps-router-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-router/`),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else if (existsSync(`${path.resolve(__dirname)}/microapps-router/index.js`)) {\n      // This is for built apps packaged with the CDK construct\n      routerFunc = new lambda.Function(this, 'microapps-router-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-router/`),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else {\n      // Create Router Lambda Layer\n      const routerDataFiles = new lambda.LayerVersion(this, 'microapps-router-layer', {\n        code: lambda.Code.fromAsset('./packages/microapps-router/templates/'),\n      });\n      if (autoDeleteEverything) {\n        routerDataFiles.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n      }\n\n      routerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-router-func', {\n        entry: './packages/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 (autoDeleteEverything) {\n      routerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\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      table.grantReadData(router);\n      table.grant(router, 'dynamodb:DescribeTable');\n    }\n\n    // TODO: Add Last Route for /*/{proxy+}\n    // Note: That might not work, may need a Behavior in CloudFront\n    //       or a Lambda @ Edge function that detects these and routes\n    //       to origin Lambda Router function.\n\n    //\n    // APIGateway domain names for CloudFront and origin\n    //\n\n    // Create Custom Domains for API Gateway\n    const dnAppsEdge = new apigwy.DomainName(this, 'microapps-apps-edge-dn', {\n      domainName: domainNameEdge,\n      certificate: certOrigin,\n    });\n    if (autoDeleteEverything) {\n      dnAppsEdge.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n    this._dnAppsOrigin = new apigwy.DomainName(this, 'microapps-apps-origin-dn', {\n      domainName: domainNameOrigin,\n      certificate: certOrigin,\n    });\n    if (autoDeleteEverything) {\n      this._dnAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    // Create an integration for the Router\n    // Do this here since it's the default route\n    const intRouter = new apigwyint.LambdaProxyIntegration({\n      handler: routerFunc,\n    });\n\n    // Create APIGateway for the Edge name\n    const httpApiDomainMapping: apigwy.DomainMappingOptions = {\n      domainName: dnAppsEdge,\n    };\n    const httpApi = new apigwy.HttpApi(this, 'microapps-api', {\n      defaultDomainMapping: httpApiDomainMapping,\n      defaultIntegration: intRouter,\n      apiName: apigatewayName,\n    });\n    if (autoDeleteEverything) {\n      httpApi.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    // Add default route on API Gateway to point to the router\n    // httpApi.addRoutes({\n    //   path: '$default',\n    //   integration: intRouter,\n    // });\n\n    //\n    // Let API Gateway accept requests using domainNameOrigin\n    // That is the origin URI that CloudFront uses for this gateway.\n    // The gateway will refuse the traffic if it doesn't have the\n    // domain name registered.\n    //\n    const mappingAppsApis = new apigwy.ApiMapping(this, 'microapps-api-mapping-origin', {\n      api: httpApi,\n      domainName: this.dnAppsOrigin,\n    });\n    mappingAppsApis.node.addDependency(this.dnAppsOrigin);\n    if (autoDeleteEverything) {\n      mappingAppsApis.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n\n    //\n    // Deployer Lambda Function\n    //\n\n    // Create Deployer Lambda Function\n    const iamRoleUploadName = `${assetNameRoot}-deployer-upload${assetNameSuffix}`;\n    const deployerFuncName = `${assetNameRoot}-deployer${assetNameSuffix}`;\n    let deployerFunc: lambda.Function;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      memorySize: 1024,\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: table.tableName,\n        FILESTORE_STAGING_BUCKET: bucketAppsStagingName,\n        FILESTORE_DEST_BUCKET: bucketAppsName,\n        UPLOAD_ROLE_NAME: iamRoleUploadName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n      },\n    };\n    if (existsSync(`${path.resolve(__dirname)}/../dist/microapps-deployer/index.js`)) {\n      // This is for local dev\n      deployerFunc = new lambda.Function(this, 'microapps-deployer-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/../dist/microapps-deployer/`),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(`${path.resolve(__dirname)}/microapps-deployer/index.js`)) {\n      // This is for built apps packaged with the CDK construct\n      deployerFunc = new lambda.Function(this, 'microapps-deployer-func', {\n        code: lambda.Code.fromAsset(`${path.resolve(__dirname)}/microapps-deployer/`),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      deployerFunc = new lambdaNodejs.NodejsFunction(this, 'microapps-deployer-func', {\n        entry: './packages/microapps-deployer/src/index.ts',\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (autoDeleteEverything) {\n      deployerFunc.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n    // Give the Deployer access to DynamoDB table\n    table.grantReadWriteData(deployerFunc);\n    table.grant(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, 'microapps-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: deployerFunc.grantPrincipal,\n    });\n\n    //\n    // Update S3 permissions\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        new iam.ArnPrincipal(`arn:aws:iam::${account}:role/${s3PolicyBypassRoleName}`),\n        deployerFunc.grantPrincipal,\n        // Allow the builder user to update the buckets\n        new iam.ArnPrincipal(\n          `arn:aws:iam::${account}:user/${assetNameRoot}-builder-${props.appEnv}`,\n        ),\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 (autoDeleteEverything) {\n      policyDenyPrefixOutsideTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${assetNameRoot}-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        new iam.ArnPrincipal(`arn:aws:iam::${account}:role/${s3PolicyBypassRoleName}`),\n        deployerFunc.grantPrincipal,\n        new iam.ArnPrincipal(\n          `arn:aws:sts::${account}:assumed-role/${deployerFunc?.role?.roleName}/${deployerFunc.functionName}`,\n        ),\n        // Allow the builder user to update the buckets\n        new iam.ArnPrincipal(\n          `arn:aws:iam::${account}:user/${assetNameRoot}-builder-${props.appEnv}`,\n        ),\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 uased 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        StringNotLike: { 'aws:userid': [`${s3PolicyBypassAROA}:*`, account] },\n      },\n    });\n    if (autoDeleteEverything) {\n      policyDenyMissingTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${assetNameRoot}-core*` },\n      );\n    }\n    const policyCloudFrontAccess = new iam.PolicyStatement({\n      sid: 'cloudfront-oai-access',\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject'],\n      principals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n      ],\n      resources: [`${bucketApps.bucketArn}/*`],\n    });\n    if (bucketApps.policy === undefined) {\n      const document = new s3.BucketPolicy(this, 'CFPolicy', {\n        bucket: bucketApps,\n      }).document;\n      document.addStatements(policyCloudFrontAccess);\n      document.addStatements(policyDenyPrefixOutsideTag);\n      document.addStatements(policyDenyMissingTag);\n    } else {\n      bucketApps.policy.document.addStatements(policyCloudFrontAccess);\n      bucketApps.policy.document.addStatements(policyDenyPrefixOutsideTag);\n      bucketApps.policy.document.addStatements(policyDenyMissingTag);\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:GetObject', 's3:ListBucket'],\n      resources: [`${bucketAppsStaging.bucketArn}/*`, bucketAppsStaging.bucketArn],\n    });\n    deployerFunc.addToRolePolicy(policyReadListStaging);\n\n    // Allow the Lambda to write to the target bucket\n    const policyReadWriteListTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject', 's3:PutObject', 's3:ListBucket'],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n    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    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    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:${region}::/apis`],\n    });\n    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:${region}:${account}:${httpApi.httpApiId}/*`,\n        `arn:aws:apigateway:${region}::/apis/${httpApi.httpApiId}/integrations`,\n        `arn:aws:apigateway:${region}::/apis/${httpApi.httpApiId}/routes`,\n      ],\n    });\n    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:${region}:${account}:function:*`,\n        `arn:aws:lambda:${region}:${account}:function:*:*`,\n      ],\n      conditions: {\n        StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n\n    //\n    // Create the origin name for API Gateway\n    //\n\n    const zone = r53.HostedZone.fromHostedZoneAttributes(this, 'microapps-zone', {\n      zoneName: r53ZoneName,\n      hostedZoneId: r53ZoneID,\n    });\n\n    const rrAppsOrigin = new r53.ARecord(this, 'microapps-origin-arecord', {\n      zone: zone,\n      recordName: domainNameOrigin,\n      target: r53.RecordTarget.fromAlias(\n        new r53targets.ApiGatewayv2DomainProperties(\n          this._dnAppsOrigin.regionalDomainName,\n          this._dnAppsOrigin.regionalHostedZoneId,\n        ),\n      ),\n    });\n    if (autoDeleteEverything) {\n      rrAppsOrigin.applyRemovalPolicy(cdk.RemovalPolicy.DESTROY);\n    }\n  }\n}\n"]}