@pwrdrvr/microapps-cdk 0.4.0-alpha.3 → 0.4.0-alpha.4
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 +113 -103
- package/API.md +39 -40
- package/README.md +59 -94
- package/changelog.md +1 -1
- package/lib/MicroApps.d.ts +0 -1
- package/lib/MicroApps.js +5 -5
- package/lib/MicroAppsAPIGwy.d.ts +0 -1
- package/lib/MicroAppsAPIGwy.js +1 -1
- package/lib/MicroAppsCF.d.ts +15 -33
- package/lib/MicroAppsCF.js +41 -51
- package/lib/MicroAppsChildDeployer.d.ts +0 -1
- package/lib/MicroAppsChildDeployer.js +1 -1
- package/lib/MicroAppsEdgeToOrigin.d.ts +0 -1
- package/lib/MicroAppsEdgeToOrigin.js +4 -6
- package/lib/MicroAppsS3.d.ts +15 -4
- package/lib/MicroAppsS3.js +18 -5
- package/lib/MicroAppsSvcs.d.ts +0 -1
- package/lib/MicroAppsSvcs.js +3 -4
- package/lib/MicroAppsTable.d.ts +0 -1
- package/lib/MicroAppsTable.js +3 -3
- package/lib/index.d.ts +0 -1
- package/lib/microapps-deployer/index.js +64 -65
- package/lib/microapps-deployer/index.js.map +4 -4
- package/lib/microapps-edge-to-origin/index.js +50 -57
- package/lib/microapps-edge-to-origin/index.js.map +4 -4
- package/lib/microapps-router/index.js +54 -55
- package/lib/microapps-router/index.js.map +4 -4
- package/lib/utils/ReverseDomain.d.ts +0 -1
- package/package.json +1 -1
- package/releasetag.txt +1 -1
- package/version.txt +1 -1
- package/lib/MicroApps.d.ts.map +0 -1
- package/lib/MicroAppsAPIGwy.d.ts.map +0 -1
- package/lib/MicroAppsCF.d.ts.map +0 -1
- package/lib/MicroAppsChildDeployer.d.ts.map +0 -1
- package/lib/MicroAppsEdgeToOrigin.d.ts.map +0 -1
- package/lib/MicroAppsS3.d.ts.map +0 -1
- package/lib/MicroAppsSvcs.d.ts.map +0 -1
- package/lib/MicroAppsTable.d.ts.map +0 -1
- package/lib/index.d.ts.map +0 -1
- package/lib/utils/ReverseDomain.d.ts.map +0 -1
package/API.md
CHANGED
|
@@ -406,11 +406,19 @@ CloudFront Origin Access Identity for the deployed applications bucket.
|
|
|
406
406
|
|
|
407
407
|
---
|
|
408
408
|
|
|
409
|
-
##### `
|
|
409
|
+
##### `bucketAppsOriginApp`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.MicroAppsS3.bucketAppsOriginApp"></a>
|
|
410
410
|
|
|
411
411
|
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
412
412
|
|
|
413
|
-
CloudFront Origin for the deployed applications bucket.
|
|
413
|
+
CloudFront Origin for the deployed applications bucket Marked with `x-microapps-origin: app` so the OriginRequest function knows to send the request to the application origin first, if configured for a particular application.
|
|
414
|
+
|
|
415
|
+
---
|
|
416
|
+
|
|
417
|
+
##### `bucketAppsOriginS3`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.MicroAppsS3.bucketAppsOriginS3"></a>
|
|
418
|
+
|
|
419
|
+
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
420
|
+
|
|
421
|
+
CloudFront Origin for the deployed applications bucket Marked with `x-microapps-origin: s3` so the OriginRequest function knows to NOT send the request to the application origin and instead let it fall through to the S3 bucket.
|
|
414
422
|
|
|
415
423
|
---
|
|
416
424
|
|
|
@@ -551,12 +559,15 @@ import { AddRoutesOptions } from '@pwrdrvr/microapps-cdk'
|
|
|
551
559
|
const addRoutesOptions: AddRoutesOptions = { ... }
|
|
552
560
|
```
|
|
553
561
|
|
|
554
|
-
##### `
|
|
562
|
+
##### `appOnlyOrigin`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.AddRoutesOptions.appOnlyOrigin"></a>
|
|
555
563
|
|
|
556
564
|
- *Type:* [`aws-cdk-lib.aws_cloudfront.IOrigin`](#aws-cdk-lib.aws_cloudfront.IOrigin)
|
|
557
|
-
- *Default:* invalid URL (never used)
|
|
558
565
|
|
|
559
|
-
|
|
566
|
+
Application origin.
|
|
567
|
+
|
|
568
|
+
Typically an S3 bucket with a `x-microapps-origin: app` custom header
|
|
569
|
+
|
|
570
|
+
The request never actually falls through to the S3 bucket.
|
|
560
571
|
|
|
561
572
|
---
|
|
562
573
|
|
|
@@ -568,11 +579,11 @@ Origin Request policy for API Gateway Origin.
|
|
|
568
579
|
|
|
569
580
|
---
|
|
570
581
|
|
|
571
|
-
##### `
|
|
582
|
+
##### `bucketOriginFallbackToApp`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.AddRoutesOptions.bucketOriginFallbackToApp"></a>
|
|
572
583
|
|
|
573
|
-
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.
|
|
584
|
+
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.OriginGroup`](#aws-cdk-lib.aws_cloudfront_origins.OriginGroup)
|
|
574
585
|
|
|
575
|
-
S3
|
|
586
|
+
Origin Group with Primary of S3 bucket with `x-microapps-origin: s3` custom header and Fallback of `appOnlyOrigin`.
|
|
576
587
|
|
|
577
588
|
---
|
|
578
589
|
|
|
@@ -584,34 +595,6 @@ CloudFront Distribution to add the Behaviors (Routes) to.
|
|
|
584
595
|
|
|
585
596
|
---
|
|
586
597
|
|
|
587
|
-
##### `createAPIPathRoute`<sup>Optional</sup> <a name="@pwrdrvr/microapps-cdk.AddRoutesOptions.createAPIPathRoute"></a>
|
|
588
|
-
|
|
589
|
-
- *Type:* `boolean`
|
|
590
|
-
- *Default:* false
|
|
591
|
-
|
|
592
|
-
Create an extra Behavior (Route) for /api/ that allows API routes to have a period in them.
|
|
593
|
-
|
|
594
|
-
When false API routes with a period in the path will get routed to S3.
|
|
595
|
-
|
|
596
|
-
When true API routes that contain /api/ in the path will get routed to API Gateway
|
|
597
|
-
even if they have a period in the path.
|
|
598
|
-
|
|
599
|
-
---
|
|
600
|
-
|
|
601
|
-
##### `createNextDataPathRoute`<sup>Optional</sup> <a name="@pwrdrvr/microapps-cdk.AddRoutesOptions.createNextDataPathRoute"></a>
|
|
602
|
-
|
|
603
|
-
- *Type:* `boolean`
|
|
604
|
-
- *Default:* false
|
|
605
|
-
|
|
606
|
-
Create an extra Behavior (Route) for /_next/data/ This route is used by Next.js to load data from the API Gateway on `getServerSideProps` calls. The requests can end in `.json`, which would cause them to be routed to S3 if this route is not created.
|
|
607
|
-
|
|
608
|
-
When false API routes with a period in the path will get routed to S3.
|
|
609
|
-
|
|
610
|
-
When true API routes that contain /_next/data/ in the path will get routed to API Gateway
|
|
611
|
-
even if they have a period in the path.
|
|
612
|
-
|
|
613
|
-
---
|
|
614
|
-
|
|
615
598
|
##### `edgeLambdas`<sup>Optional</sup> <a name="@pwrdrvr/microapps-cdk.AddRoutesOptions.edgeLambdas"></a>
|
|
616
599
|
|
|
617
600
|
- *Type:* [`aws-cdk-lib.aws_cloudfront.EdgeLambda`](#aws-cdk-lib.aws_cloudfront.EdgeLambda)[]
|
|
@@ -818,11 +801,19 @@ import { MicroAppsCFProps } from '@pwrdrvr/microapps-cdk'
|
|
|
818
801
|
const microAppsCFProps: MicroAppsCFProps = { ... }
|
|
819
802
|
```
|
|
820
803
|
|
|
821
|
-
##### `
|
|
804
|
+
##### `bucketAppsOriginApp`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.MicroAppsCFProps.bucketAppsOriginApp"></a>
|
|
822
805
|
|
|
823
806
|
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
824
807
|
|
|
825
|
-
S3 bucket origin for deployed applications
|
|
808
|
+
S3 bucket origin for deployed applications Marked with `x-microapps-origin: app`.
|
|
809
|
+
|
|
810
|
+
---
|
|
811
|
+
|
|
812
|
+
##### `bucketAppsOriginS3`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.MicroAppsCFProps.bucketAppsOriginS3"></a>
|
|
813
|
+
|
|
814
|
+
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
815
|
+
|
|
816
|
+
S3 bucket origin for deployed applications Marked with `x-microapps-origin: s3`.
|
|
826
817
|
|
|
827
818
|
---
|
|
828
819
|
|
|
@@ -1971,11 +1962,19 @@ CloudFront Origin Access Identity for the deployed applications bucket.
|
|
|
1971
1962
|
|
|
1972
1963
|
---
|
|
1973
1964
|
|
|
1974
|
-
##### `
|
|
1965
|
+
##### `bucketAppsOriginApp`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.IMicroAppsS3.bucketAppsOriginApp"></a>
|
|
1966
|
+
|
|
1967
|
+
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
1968
|
+
|
|
1969
|
+
CloudFront Origin for the deployed applications bucket Marked with `x-microapps-origin: app` so the OriginRequest function knows to send the request to the application origin first, if configured for a particular application.
|
|
1970
|
+
|
|
1971
|
+
---
|
|
1972
|
+
|
|
1973
|
+
##### `bucketAppsOriginS3`<sup>Required</sup> <a name="@pwrdrvr/microapps-cdk.IMicroAppsS3.bucketAppsOriginS3"></a>
|
|
1975
1974
|
|
|
1976
1975
|
- *Type:* [`aws-cdk-lib.aws_cloudfront_origins.S3Origin`](#aws-cdk-lib.aws_cloudfront_origins.S3Origin)
|
|
1977
1976
|
|
|
1978
|
-
CloudFront Origin for the deployed applications bucket.
|
|
1977
|
+
CloudFront Origin for the deployed applications bucket Marked with `x-microapps-origin: s3` so the OriginRequest function knows to NOT send the request to the application origin and instead let it fall through to the S3 bucket.
|
|
1979
1978
|
|
|
1980
1979
|
---
|
|
1981
1980
|
|
package/README.md
CHANGED
|
@@ -6,22 +6,22 @@ The MicroApps project enables rapidly deploying many web apps to AWS on a single
|
|
|
6
6
|
|
|
7
7
|
MicroApps allows many versions of an application to be deployed either as ephemeral deploys (e.g. for pull request builds) or as semi-permanent deploys. The `microapps-router` Lambda function handled routing requests to apps to the current version targeted for a particular application start request using rules as complex as one is interested in implementing (e.g. A/B testing integration, canary releases, per-user rules for logged in users, per-group, per-deparment, and default rules).
|
|
8
8
|
|
|
9
|
-
|
|
9
|
+
Users start applications via a URL such as `[/{prefix}]/{appname}/`, which hits the `microapps-router` Lambda@Edge OriginRequest handler that looks up the version of the application to be run, and either forwards the request to the target Lambda Function URL (`--startupType direct` invoke mode) or returns a transparent `iframe` (`--startupType iframe`) with a link to that version. `direct` mode works with frameworks, like Next.js, that can return pages that have build-time computed relative URLs to static resources and API calls. `iframe` mode works with frameworks that do not write computed relative URLs at build time and/or that do not use URLs that are completely relative to wherever the applications is rooted at runtime; this mode is primarily for quick prototyping as it has other complications (such as indirect access to query strings). The URL seen by the user in the browser (and available for bookmarking) has no version in it, so subsequent launches (e.g. the next day or just in another tab) will lookup the version again. All relative URL API requests (e.g. `some/api/path`) will go to the corresponding API version that matches the version of the loaded static files, eliminating issues of incompatibility between static files and API deployments.
|
|
10
10
|
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
For development / testing purposes only, each version of an applicaton can be accessed directly via a URL of the pattern `[/{prefix}]/{appname}/{semver}/`. These "versioned" URLs are not intended to be advertised to end users as they would cause a user to be stuck on a particular version of the app if the URL was bookmarked. Note that the system does not limit access to particular versions of an application, as of 2022-01-26, but that can be added as a feature.
|
|
11
|
+
For development / testing purposes only, each version of an applicaton can be accessed directly via a URL of the patterns `[/{prefix}]/{appname}?appver={semver}` for `direct` mode or `[/{prefix}]/{appname}/{semver}/` for `iframe` mode. These "versioned" URLs are not intended to be advertised to end users as they would cause a user to be stuck on a particular version of the app if the URL was bookmarked. Note that the system does not limit access to particular versions of an application, as of 2023-03-04, but that can be added as a feature.
|
|
14
12
|
|
|
15
13
|
# Table of Contents <!-- omit in toc -->
|
|
16
14
|
|
|
17
15
|
- [Overview](#overview)
|
|
16
|
+
- [Why MicroApps](#why-microapps)
|
|
17
|
+
- [Request Routing for Static Assets / App - Diagram](#request-routing-for-static-assets--app---diagram)
|
|
18
18
|
- [Request Dispatch Model for Multi-Account Deployments](#request-dispatch-model-for-multi-account-deployments)
|
|
19
19
|
- [Video Preview of the Deploying CDK Construct](#video-preview-of-the-deploying-cdk-construct)
|
|
20
20
|
- [Installation / CDK Constructs](#installation--cdk-constructs)
|
|
21
21
|
- [Tutorial - Bootstrapping a Deploy](#tutorial---bootstrapping-a-deploy)
|
|
22
|
-
- [Why MicroApps](#why-microapps)
|
|
23
22
|
- [Limitations / Future Development](#limitations--future-development)
|
|
24
23
|
- [Related Projects / Components](#related-projects--components)
|
|
24
|
+
- [Why Lambda @ Origin and Not Lambda @ Edge for Apps](#why-lambda--origin-and-not-lambda--edge-for-apps)
|
|
25
25
|
- [Architecure Diagram](#architecure-diagram)
|
|
26
26
|
- [Project Layout](#project-layout)
|
|
27
27
|
- [Creating a MicroApp Using Zip Lambda Functions](#creating-a-microapp-using-zip-lambda-functions)
|
|
@@ -31,17 +31,29 @@ For development / testing purposes only, each version of an applicaton can be ac
|
|
|
31
31
|
- [Install Dependencies](#install-dependencies)
|
|
32
32
|
- [Dockerfile](#dockerfile)
|
|
33
33
|
- [next.config.js](#nextconfigjs)
|
|
34
|
-
- [deploy.json](#deployjson)
|
|
35
|
-
- [serverless.yaml](#serverlessyaml)
|
|
36
34
|
- [Troubleshooting](#troubleshooting)
|
|
37
35
|
- [CloudFront Requests to API Gateway are Rejected with 403 Forbidden](#cloudfront-requests-to-api-gateway-are-rejected-with-403-forbidden)
|
|
38
36
|
- [SignatureV4 Headers](#signaturev4-headers)
|
|
39
37
|
|
|
38
|
+
# Why MicroApps
|
|
39
|
+
|
|
40
|
+
MicroApps are like micro services, but for Web UIs. A MicroApp allows a single functional site to be developed by many independent teams within an organization. Teams must coordinate deployments and agree upon one implementation technology and framework when building a monolithic, or even a monorepo, web application.
|
|
41
|
+
|
|
42
|
+
Teams using MicroApps can deploy independently of each other with coordination being required only at points of intentional integration (e.g. adding a feature to pass context from one MicroApp to another or coordination of a major feature release to users) and sharing UI styles, if desired (it is possible to build styles that look the same across many different UI frameworks).
|
|
43
|
+
|
|
44
|
+
MicroApps also allow each team to use a UI framework and backend language that is most appropriate for their solving their business problem. Not every app has to use React or Next.js or even Node on the backend, but instead they can use whatever framework they want and Java, Go, C#, Python, etc. for UI API calls.
|
|
45
|
+
|
|
46
|
+
For internal sites, or logged-in-customer sites, different tools or products can be hosted in entirely independent MicroApps. A menuing system / toolbar application can be created as a MicroApp and that menu app can open the apps in the system within a transparent iframe. For externally facing sites, such as for an e-commerce site, it is possible to have a MicroApp serving `/product/...`, another serving `/search/...`, another serving `/`, etc.
|
|
47
|
+
|
|
48
|
+
# Request Routing for Static Assets / App - Diagram
|
|
49
|
+
|
|
50
|
+

|
|
51
|
+
|
|
40
52
|
# Request Dispatch Model for Multi-Account Deployments
|
|
41
53
|
|
|
42
54
|
Note: requests can also be dispatched into the same account, but this model is more likely to be used by organizations with many AWS accounts.
|
|
43
55
|
|
|
44
|
-

|
|
45
57
|
|
|
46
58
|
# Video Preview of the Deploying CDK Construct
|
|
47
59
|
|
|
@@ -68,46 +80,17 @@ Note: requests can also be dispatched into the same account, but this model is m
|
|
|
68
80
|
- Of course, there are other methods of setting env vars
|
|
69
81
|
- `aws sso login`
|
|
70
82
|
- Establish an AWS SSO session
|
|
71
|
-
- `cdk-sso-sync`
|
|
72
|
-
- Using `npm i -g cdk-sso-sync`
|
|
73
|
-
- Sets AWS SSO credentials in a way that CDK can use them
|
|
74
|
-
- Not necessary if not using AWS SSO
|
|
75
83
|
- `export AWS_REGION=us-east-2`
|
|
76
84
|
- Region needs to be set for the Lambda invoke - This can be done other ways in `~/.aws/config` as well
|
|
77
85
|
- `./deploy.sh`
|
|
78
86
|
- Deploys the CDK Stack
|
|
79
87
|
- Essentially runs two commands along with extraction of outputs:
|
|
80
88
|
- `npx cdk deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`
|
|
81
|
-
- `npx microapps-publish publish -
|
|
89
|
+
- `npx microapps-publish publish --app-name release --new-version ${RELEASE_APP_PACKAGE_VERSION} --deployer-lambda-name ${DEPLOYER_LAMBDA_NAME} --app-lambda-name ${RELEASE_APP_LAMBDA_NAME} --static-assets-path node_modules/@pwrdrvr/microapps-app-release-cdk/lib/static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --no-cache`
|
|
82
90
|
- URL will be printed as last output
|
|
83
91
|
|
|
84
|
-
# Why MicroApps
|
|
85
|
-
|
|
86
|
-
MicroApps are like micro services, but for Web UIs. A MicroApp allows a single functional site to be developed by many independent teams within an organization. Teams must coordinate deployments and agree upon one implementation technology and framework when building a monolithic, or even a monorepo, web application.
|
|
87
|
-
|
|
88
|
-
Teams using MicroApps can deploy independently of each other with coordination being required only at points of intentional integration (e.g. adding a feature to pass context from one MicroApp to another or coordination of a major feature release to users) and sharing UI styles, if desired (it is possible to build styles that look the same across many different UI frameworks).
|
|
89
|
-
|
|
90
|
-
MicroApps also allow each team to use a UI framework and backend language that is most appropriate for their solving their business problem. Not every app has to use React or Next.js or even Node on the backend, but instead they can use whatever framework they want and Java, Go, C#, Python, etc. for UI API calls.
|
|
91
|
-
|
|
92
|
-
For internal sites, or logged-in-customer sites, different tools or products can be hosted in entirely independent MicroApps. A menuing system / toolbar application can be created as a MicroApp and that menu app can open the apps in the system within a transparent iframe. For externally facing sites, such as for an e-commerce site, it is possible to have a MicroApp serving `/product/...`, another serving `/search/...`, another serving `/`, etc.
|
|
93
|
-
|
|
94
92
|
# Limitations / Future Development
|
|
95
93
|
|
|
96
|
-
- `iframes`
|
|
97
|
-
- Yeah, yeah: `iframes` are not framesets and most of the hate about iframes is probably better directed at framesets
|
|
98
|
-
- The iframe serves a purpose but it stinks that it is there, primarily because it will cause issues with search bot indexing (SEO)
|
|
99
|
-
- There are other options available to implement that have their own drabacks:
|
|
100
|
-
- Using the `microapps-router` to proxy the "app start" request to a particular version of an app that then renders all of it's API resource requests to versioned URLs
|
|
101
|
-
- Works only with frameworks that support hashing filenams for each deploy to unique names
|
|
102
|
-
- This page would need to be marked as non-cachable
|
|
103
|
-
- This may work well with Next.js which wants to know the explicit path that it will be running at (it writes that path into all resource and API requests)
|
|
104
|
-
- Possible issue: the app would need to work ok being displayed at `[/{prefix}]/{appname}` when it may think that it's being displayed at `[/{prefix}]/{appname}/{semver}`
|
|
105
|
-
- Disadvantage: requires some level of UI framework features (e.g. writing the absolute resource paths) to work correctly - may not work as easily for all UI frameworks
|
|
106
|
-
- HTML5 added features to allow setting the relative path of all subsequent requests to be different than that displayed in the address bar
|
|
107
|
-
- Gotta see if this works in modern browsers
|
|
108
|
-
- Option to ditch the multiple-versions feature
|
|
109
|
-
- Works only with frameworks that support hashing filenams for each deploy to unique names
|
|
110
|
-
- Allows usage of the deploy and routing tooling without advantages and disadvantages of multiple-versions support
|
|
111
94
|
- AWS Only
|
|
112
95
|
- For the time being this has only been implemented for AWS technologies and APIs
|
|
113
96
|
- It is possible that Azure and GCP have sufficient support to enable porting the framework
|
|
@@ -134,30 +117,39 @@ For internal sites, or logged-in-customer sites, different tools or products can
|
|
|
134
117
|
- Next.js Demo App
|
|
135
118
|
- The Next.js Tutorial application deployed as a MicroApp
|
|
136
119
|
- [pwrdrvr/serverless-nextjs-demo](https://github.com/pwrdrvr/serverless-nextjs-demo)
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
120
|
+
|
|
121
|
+
|
|
122
|
+
# Why Lambda @ Origin and Not Lambda @ Edge for Apps
|
|
123
|
+
|
|
124
|
+
Calling resources (DBs and other services) and waiting for a long synchronous response is an anti-pattern in Lambda as the Lambda function will be billed for the time spent waiting for the response. This is especially true for Lambda@Edge as the cost is 3x higher than Lambda at the origin.
|
|
125
|
+
|
|
126
|
+
With Lambda@Edge (even with Origin Requests) the cost is 3x higher per GB-second and the time spent waiting for a 1 ms service response from an origin that is 250 ms away is 750x higher (250 ms / 1ms * 3x higher cost) than making that same request within the region where the resource resides.
|
|
127
|
+
|
|
128
|
+
- Lambda@Edge is _at least_ 3x more expensive than Lambda at the origin:
|
|
129
|
+
- In US East 1, the price per GB-Second is $0.00005001 for Lambda@Edge
|
|
130
|
+
- Source: https://aws.amazon.com/lambda/pricing/ (bottom of page)
|
|
131
|
+
- Updated: 2023-03-04
|
|
132
|
+
- In US East 1, the price per GB-Second is $0.0000166667 for Lambda at the origin on x86
|
|
133
|
+
- Source: https://aws.amazon.com/lambda/pricing/
|
|
134
|
+
- Updated: 2023-03-04
|
|
135
|
+
- Ratio
|
|
136
|
+
- Lambda@Edge / Lambda@Origin = $0.00005001 / $0.0000166667 = 3.0006x
|
|
137
|
+
- Any DB or services calls from Lambda@Edge back to the origin will pay that 3x higher per GB-Second cost for any time spent waiting to send the request and get a response. Example:
|
|
138
|
+
- Lambda@Edge
|
|
139
|
+
- 0.250s Round Trip Time (RTT) for EU-zone edge request to hit US-East 1 Origin
|
|
140
|
+
- 0.200s DB lookup time
|
|
141
|
+
- 0.050s CPU usage to process the DB response
|
|
142
|
+
- 0.500s total billed time @ $0.00005001 @ 128 MB
|
|
143
|
+
- $0.000003125625 total charge
|
|
144
|
+
- Lambda at Origin
|
|
145
|
+
- RTT does not apply (it's effectively 1-2 ms to hit a DB in the same region)
|
|
146
|
+
- 0.200s DB lookup time
|
|
147
|
+
- 0.050s CPU usage to process the DB response
|
|
148
|
+
- 0.250s total billed time @ $0.0000166667 @ 128 MB
|
|
149
|
+
- Half the billed time of running on Lambda@Edge
|
|
150
|
+
- 1/6th the cost of running on Lambda@Edge:
|
|
151
|
+
- $0.000000520834375 total charge (assuming no CPU time to process the response)
|
|
152
|
+
- $0.000003125625 / $0.000000520834375 = 6x more expensive in Lambda@Edge
|
|
161
153
|
|
|
162
154
|
# Architecure Diagram
|
|
163
155
|
|
|
@@ -221,11 +213,13 @@ Replace the version with `0.0.0` so it can be modified by the `microapps-publish
|
|
|
221
213
|
### Install Dependencies
|
|
222
214
|
|
|
223
215
|
```
|
|
224
|
-
npm i --save-dev @
|
|
216
|
+
npm i --save-dev @pwrdrvr/microapps-publish
|
|
225
217
|
```
|
|
226
218
|
|
|
227
219
|
### Dockerfile
|
|
228
220
|
|
|
221
|
+
FIXME: Out of date 2023-03-04
|
|
222
|
+
|
|
229
223
|
Add this file to the root of the app.
|
|
230
224
|
|
|
231
225
|
```Dockerfile
|
|
@@ -258,6 +252,8 @@ CMD [ "./index.handler" ]
|
|
|
258
252
|
|
|
259
253
|
### next.config.js
|
|
260
254
|
|
|
255
|
+
FIXME: Out of date 2023-03-04
|
|
256
|
+
|
|
261
257
|
Add this file to the root of the app.
|
|
262
258
|
|
|
263
259
|
Replace `appname` with your URL path-compatible application name.
|
|
@@ -279,37 +275,6 @@ module.exports = {
|
|
|
279
275
|
};
|
|
280
276
|
```
|
|
281
277
|
|
|
282
|
-
### deploy.json
|
|
283
|
-
|
|
284
|
-
Add this file to the root of the app.
|
|
285
|
-
|
|
286
|
-
Replace `appname` with your URL path-compatible application name.
|
|
287
|
-
|
|
288
|
-
```json
|
|
289
|
-
{
|
|
290
|
-
"AppName": "appname",
|
|
291
|
-
"SemVer": "0.0.0",
|
|
292
|
-
"DefaultFile": "",
|
|
293
|
-
"StaticAssetsPath": "./.serverless_nextjs/assets/appname/0.0.0/",
|
|
294
|
-
"LambdaARN": "arn:aws:lambda:us-east-1:123456789012:function:appname:v0_0_0",
|
|
295
|
-
"AWSAccountID": "123456789012",
|
|
296
|
-
"AWSRegion": "us-east-2",
|
|
297
|
-
"ServerlessNextRouterPath": "./node_modules/@pwrdrvr/serverless-nextjs-router/dist/index.js"
|
|
298
|
-
}
|
|
299
|
-
```
|
|
300
|
-
|
|
301
|
-
### serverless.yaml
|
|
302
|
-
|
|
303
|
-
Add this file to the root of the app.
|
|
304
|
-
|
|
305
|
-
```yaml
|
|
306
|
-
nextApp:
|
|
307
|
-
component: './node_modules/@sls-next/serverless-component'
|
|
308
|
-
inputs:
|
|
309
|
-
deploy: false
|
|
310
|
-
uploadStaticAssetsFromBuild: false
|
|
311
|
-
```
|
|
312
|
-
|
|
313
278
|
# Troubleshooting
|
|
314
279
|
|
|
315
280
|
## CloudFront Requests to API Gateway are Rejected with 403 Forbidden
|
package/changelog.md
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
|
|
2
|
-
## [0.4.0-alpha.
|
|
2
|
+
## [0.4.0-alpha.4](https://github.com/pwrdrvr/microapps-core/compare/v0.4.0-alpha.3...v0.4.0-alpha.4) (2023-03-06)
|
package/lib/MicroApps.d.ts
CHANGED
package/lib/MicroApps.js
CHANGED
|
@@ -32,7 +32,6 @@ const ReverseDomain_1 = require("./utils/ReverseDomain");
|
|
|
32
32
|
*/
|
|
33
33
|
class MicroApps extends constructs_1.Construct {
|
|
34
34
|
constructor(scope, id, props) {
|
|
35
|
-
var _b;
|
|
36
35
|
super(scope, id);
|
|
37
36
|
if (props === undefined) {
|
|
38
37
|
throw new Error('props must be set');
|
|
@@ -96,7 +95,7 @@ class MicroApps extends constructs_1.Construct {
|
|
|
96
95
|
edgeLambdas.push(...this._edgeToOrigin.edgeToOriginLambdas);
|
|
97
96
|
}
|
|
98
97
|
// Add any extra lambdas
|
|
99
|
-
if (
|
|
98
|
+
if (props.edgeLambdas?.length) {
|
|
100
99
|
edgeLambdas.push(...props.edgeLambdas);
|
|
101
100
|
}
|
|
102
101
|
this._cf = new MicroAppsCF_1.MicroAppsCF(this, 'cft', {
|
|
@@ -108,7 +107,8 @@ class MicroApps extends constructs_1.Construct {
|
|
|
108
107
|
...(this._apigwy ? { httpApi: this._apigwy.httpApi } : {}),
|
|
109
108
|
r53Zone,
|
|
110
109
|
certEdge,
|
|
111
|
-
|
|
110
|
+
bucketAppsOriginS3: this._s3.bucketAppsOriginS3,
|
|
111
|
+
bucketAppsOriginApp: this._s3.bucketAppsOriginApp,
|
|
112
112
|
bucketLogs: this._s3.bucketLogs,
|
|
113
113
|
rootPathPrefix,
|
|
114
114
|
createAPIPathRoute,
|
|
@@ -135,5 +135,5 @@ class MicroApps extends constructs_1.Construct {
|
|
|
135
135
|
}
|
|
136
136
|
exports.MicroApps = MicroApps;
|
|
137
137
|
_a = JSII_RTTI_SYMBOL_1;
|
|
138
|
-
MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.4.0-alpha.
|
|
139
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroApps.js","sourceRoot":"","sources":["../src/MicroApps.ts"],"names":[],"mappings":"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","sourcesContent":["import { RemovalPolicy } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport { Construct } from 'constructs';\nimport { IMicroAppsAPIGwy, MicroAppsAPIGwy } from './MicroAppsAPIGwy';\nimport { IMicroAppsCF, MicroAppsCF } from './MicroAppsCF';\nimport { IMicroAppsEdgeToOrigin, MicroAppsEdgeToOrigin } from './MicroAppsEdgeToOrigin';\nimport { IMicroAppsS3, MicroAppsS3 } from './MicroAppsS3';\nimport { IMicroAppsSvcs, MicroAppsSvcs } from './MicroAppsSvcs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * A CDK Construct for creating a MicroApps runtime environment used\n * to host Next.js, React, or any other sort of web application with\n * multiple versions available for comparison, quick rollbacks, quick\n * releases, and a complete lack of user disturbance on deploys.\n *\n * @remarks\n *\n * {@link MicroApps} provides a turn-key construct that creates all\n * dependencies with limited exposure of underlying AWS Resource options.\n * This construct is the easiest to use when exploring MicroApps for the\n * first time.\n *\n * {@link MicroAppsAPIGwy}, {@link MicroAppsCF}, {@link MicroAppsS3},\n * and {@link MicroAppsSvcs}, and their helper static methods, can be used\n * to create AWS Resources more directly, to provide your own AWS Resources\n * (e.g. an existing CloudFront Distribution), and to have more flexibility\n * than the {@link MicroApps} construct offers.\n *\n * @packageDocumentation\n */\n\n/**\n * Properties to initialize an instance of `MicroApps`.\n */\nexport interface MicroAppsProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Passed to NODE_ENV of Router and Deployer Lambda functions.\n   *\n   * @default dev\n   */\n  readonly appEnv: string;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Certificate in US-East-1 for the CloudFront distribution.\n   */\n  readonly certEdge?: acm.ICertificate;\n\n  /**\n   * Certificate in deployed region for the API Gateway.\n   */\n  readonly certOrigin?: acm.ICertificate;\n\n  /**\n   * Use a strict S3 Bucket Policy that prevents applications\n   * from reading/writing/modifying/deleting files in the S3 Bucket\n   * outside of the path that is specific to their app/version.\n   *\n   * This setting should be used when applications are less than\n   * fully trusted.\n   *\n   * @default false\n   */\n  readonly s3StrictBucketPolicy?: boolean;\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * IAM Role or IAM User names to exclude from the DENY rules on the S3 Bucket Policy.\n   *\n   * Roles that are Assumed must instead have their AROA added to `s3PolicyBypassAROAs`.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * @example ['arn:aws:iam::1234567890123:role/AdminAccess', 'arn:aws:iam::1234567890123:user/MyAdminUser']\n   *\n   * @see s3PolicyBypassAROAs\n   */\n  readonly s3PolicyBypassPrincipalARNs?: string[];\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * AROAs of the IAM Role to exclude from the DENY rules on the S3 Bucket Policy.\n   * This allows sessions that assume the IAM Role to be excluded from the\n   * DENY rules on the S3 Bucket Policy.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * Roles / users that are used directly, not assumed, can be added to `s3PolicyBypassRoleNames` instead.\n   *\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   *\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 wildcard matches implicitly either.\n   *\n   * The AROA must be used because there are only 3 Principal variables available:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n   *  aws:username, aws:userid, aws:PrincipalTag\n   *\n   * For an assumed role, aws:username is blank, aws:userid is:\n   *  [unique id AKA AROA for Role]:[session name]\n   *\n   * Table of unique ID prefixes such as AROA:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n   *\n   * The name of the role is simply not available for an assumed role and, if it was,\n   * a complicated comparison would be requierd to prevent exclusion\n   * of applying the Deny Rule to roles from 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   *\n   * @example [ 'AROA1234567890123' ]\n   *\n   * @see s3StrictBucketPolicy\n   */\n  readonly s3PolicyBypassAROAs?: string[];\n\n  /**\n   * Optional custom domain name for the CloudFront distribution.\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * Optional custom domain name for the API Gateway HTTPv2 API.\n   *\n   * @example apps-origin.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Create API Gateway for non-edge invocation\n   *\n   * @default false\n   */\n  readonly createAPIGateway?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /api/ that allows\n   * API routes to have a period in them.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /api/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createAPIPathRoute?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /_next/data/\n   * This route is used by Next.js to load data from the API Gateway\n   * on `getServerSideProps` calls.  The requests can end in `.json`,\n   * which would cause them to be routed to S3 if this route is not created.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /_next/data/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Adds an X-Forwarded-Host-Header when calling API Gateway\n   *\n   * Can only be trusted if `signingMode` is enabled, which restricts\n   * access to API Gateway to only IAM signed requests.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly addXForwardedHostHeader?: boolean;\n\n  /**\n   * Replaces Host header (which will be the Edge domain name) with the Origin domain name\n   * when enabled.  This is necessary when API Gateway has not been configured\n   * with a custom domain name that matches the exact domain name used by the CloudFront\n   * Distribution AND when the OriginRequestPolicy.HeadersBehavior is set\n   * to pass all headers to the origin.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly replaceHostHeader?: boolean;\n\n  /**\n   * Requires IAM auth on the API Gateway origin if not set to 'none'.\n   *\n   * 'sign' - Uses request headers for auth.\n   * 'presign' - Uses query string for auth.\n   *\n   * If enabled,\n   *\n   * Note: if 'sign' or 'presign', creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default 'sign'\n   */\n  readonly signingMode?: 'sign' | 'presign' | 'none';\n\n  /**\n   * Origin region that API Gateway or Lambda function will be deployed to, used\n   * for the config.yml on the Edge function to sign requests for\n   * the correct region\n   *\n   * @default undefined\n   */\n  readonly originRegion?: string;\n\n  /**\n   * Optional Origin Shield Region\n   *\n   * This should be the region where the DynamoDB is located so the\n   * EdgeToOrigin calls have the lowest latency (~1 ms).\n   *\n   * @default originRegion if specified, otherwise undefined\n   */\n  readonly originShieldRegion?: string;\n\n  /**\n   * Existing table for apps/versions/rules\n   *\n   * @warning - It is *strongly* suggested that production stacks create\n   * their own DynamoDB Table and pass it into this construct, for protection\n   * against data loss due to logical ID changes, the ability to configure\n   * Provisioned capacity with Auto Scaling, the ability to add additional indices, etc.\n   *\n   * Requirements:\n   * - Hash Key: `PK`\n   * - Sort Key: `SK`\n   *\n   * @default created by construct\n   */\n  readonly table?: dynamodb.ITable;\n\n  /**\n   * Pre-set table name for apps/versions/rules\n   *\n   * This is required when using v2 routing\n   */\n  readonly tableNameForEdgeToOrigin?: string;\n\n  /**\n   * Additional edge lambda functions\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\n\n  /**\n   * Account IDs allowed for cross-account Function URL invocations\n   *\n   * @default []\n   */\n  readonly allowedFunctionUrlAccounts?: string[];\n}\n\n/**\n * Represents a MicroApps\n */\nexport interface IMicroApps {\n  /** {@inheritdoc IMicroAppsCF} */\n  readonly cf: IMicroAppsCF;\n\n  /** {@inheritdoc IMicroAppsEdgeToOrigin} */\n  readonly edgeToOrigin?: IMicroAppsEdgeToOrigin;\n\n  /** {@inheritdoc IMicroAppsS3} */\n  readonly s3: IMicroAppsS3;\n\n  /** {@inheritdoc IMicroAppsSvcs} */\n  readonly svcs: IMicroAppsSvcs;\n\n  /** {@inheritdoc IMicroAppsAPIGwy} */\n  readonly apigwy?: IMicroAppsAPIGwy;\n}\n\n/**\n * Create a new MicroApps \"turnkey\" construct for simple\n * deployments and for initial evaulation of the MicroApps framework.\n *\n * Use this construct to create a PoC working entire stack.\n *\n * Do not use this construct when adding MicroApps to an existing\n * CloudFront, API Gateway, S3 Bucket, etc. or where access\n * to all features of the AWS Resources are needed (e.g. to\n * add additional Behaviors to the CloudFront distribution, set authorizors\n * on API Gateway, etc.).\n *\n * @warning This construct is not intended for production use.\n * In a production stack the DynamoDB Table, API Gateway, S3 Buckets,\n * etc. should be created in a \"durable\" stack where the IDs will not\n * change and where changes to the MicroApps construct will not\n * cause failures to deploy or data to be deleted.\n *\n *  @see {@link https://github.com/pwrdrvr/microapps-core/blob/main/packages/cdk/lib/MicroApps.ts | example usage in a CDK Stack }\n */\nexport class MicroApps extends Construct implements IMicroApps {\n  private _cf: MicroAppsCF;\n  public get cf(): IMicroAppsCF {\n    return this._cf;\n  }\n\n  private _edgeToOrigin?: MicroAppsEdgeToOrigin;\n  public get edgeToOrigin(): IMicroAppsEdgeToOrigin | undefined {\n    return this._edgeToOrigin;\n  }\n\n  private _s3: MicroAppsS3;\n  public get s3(): IMicroAppsS3 {\n    return this._s3;\n  }\n\n  private _apigwy?: MicroAppsAPIGwy;\n  public get apigwy(): IMicroAppsAPIGwy | undefined {\n    return this._apigwy;\n  }\n\n  private _svcs: MicroAppsSvcs;\n  public get svcs(): IMicroAppsSvcs {\n    return this._svcs;\n  }\n\n  constructor(scope: 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      domainNameEdge,\n      domainNameOrigin,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      certEdge,\n      appEnv = 'dev',\n      certOrigin,\n      removalPolicy,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      createAPIGateway = false,\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      signingMode = 'sign',\n      originRegion,\n      table,\n      tableNameForEdgeToOrigin,\n      originShieldRegion = originRegion,\n      allowedFunctionUrlAccounts = [],\n    } = props;\n\n    this._s3 = new MicroAppsS3(this, 's3', {\n      removalPolicy,\n      bucketLogsName: domainNameEdge ? `${reverseDomain(domainNameEdge)}-logs` : undefined,\n      bucketAppsName: domainNameEdge ? `${reverseDomain(domainNameEdge)}` : undefined,\n      bucketAppsStagingName: domainNameEdge\n        ? `${reverseDomain(domainNameEdge)}-staging`\n        : undefined,\n      assetNameRoot,\n      assetNameSuffix,\n      originShieldRegion,\n    });\n    if (createAPIGateway) {\n      this._apigwy = new MicroAppsAPIGwy(this, 'api', {\n        removalPolicy,\n        assetNameRoot,\n        assetNameSuffix,\n        domainNameEdge,\n        domainNameOrigin,\n        r53Zone,\n        certOrigin,\n        rootPathPrefix,\n        requireIAMAuthorization: signingMode !== 'none',\n      });\n    }\n    this._svcs = new MicroAppsSvcs(this, 'svcs', {\n      ...(this._apigwy ? { httpApi: this._apigwy.httpApi } : {}),\n      removalPolicy,\n      bucketApps: this._s3.bucketApps,\n      bucketAppsOAI: this._s3.bucketAppsOAI,\n      bucketAppsStaging: this._s3.bucketAppsStaging,\n      assetNameRoot,\n      assetNameSuffix,\n      appEnv,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      requireIAMAuthorization: signingMode !== 'none',\n      table,\n    });\n    const edgeLambdas: cf.EdgeLambda[] = [];\n\n    if (signingMode !== 'none' || replaceHostHeader || addXForwardedHostHeader) {\n      this._edgeToOrigin = new MicroAppsEdgeToOrigin(this, 'edgeToOrigin', {\n        assetNameRoot,\n        assetNameSuffix,\n        removalPolicy,\n        addXForwardedHostHeader,\n        setupApiGatewayPermissions: createAPIGateway,\n        replaceHostHeader,\n        originRegion,\n        signingMode,\n        rootPathPrefix,\n        tableRulesArn: tableNameForEdgeToOrigin || this._svcs.table.tableName,\n        allowedFunctionUrlAccounts,\n      });\n\n      edgeLambdas.push(...this._edgeToOrigin.edgeToOriginLambdas);\n    }\n    // Add any extra lambdas\n    if (props.edgeLambdas?.length) {\n      edgeLambdas.push(...props.edgeLambdas);\n    }\n    this._cf = new MicroAppsCF(this, 'cft', {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n      domainNameOrigin,\n      ...(this._apigwy ? { httpApi: this._apigwy.httpApi } : {}),\n      r53Zone,\n      certEdge,\n      bucketAppsOrigin: this._s3.bucketAppsOrigin,\n      bucketLogs: this._s3.bucketLogs,\n      rootPathPrefix,\n      createAPIPathRoute,\n      createNextDataPathRoute,\n      originShieldRegion,\n      ...(edgeLambdas.length ? { edgeLambdas } : {}),\n    });\n  }\n}\n"]}
|
|
138
|
+
MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.4.0-alpha.4" };
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroApps.js","sourceRoot":"","sources":["../src/MicroApps.ts"],"names":[],"mappings":"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","sourcesContent":["import { RemovalPolicy } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport { Construct } from 'constructs';\nimport { IMicroAppsAPIGwy, MicroAppsAPIGwy } from './MicroAppsAPIGwy';\nimport { IMicroAppsCF, MicroAppsCF } from './MicroAppsCF';\nimport { IMicroAppsEdgeToOrigin, MicroAppsEdgeToOrigin } from './MicroAppsEdgeToOrigin';\nimport { IMicroAppsS3, MicroAppsS3 } from './MicroAppsS3';\nimport { IMicroAppsSvcs, MicroAppsSvcs } from './MicroAppsSvcs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * A CDK Construct for creating a MicroApps runtime environment used\n * to host Next.js, React, or any other sort of web application with\n * multiple versions available for comparison, quick rollbacks, quick\n * releases, and a complete lack of user disturbance on deploys.\n *\n * @remarks\n *\n * {@link MicroApps} provides a turn-key construct that creates all\n * dependencies with limited exposure of underlying AWS Resource options.\n * This construct is the easiest to use when exploring MicroApps for the\n * first time.\n *\n * {@link MicroAppsAPIGwy}, {@link MicroAppsCF}, {@link MicroAppsS3},\n * and {@link MicroAppsSvcs}, and their helper static methods, can be used\n * to create AWS Resources more directly, to provide your own AWS Resources\n * (e.g. an existing CloudFront Distribution), and to have more flexibility\n * than the {@link MicroApps} construct offers.\n *\n * @packageDocumentation\n */\n\n/**\n * Properties to initialize an instance of `MicroApps`.\n */\nexport interface MicroAppsProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Passed to NODE_ENV of Router and Deployer Lambda functions.\n   *\n   * @default dev\n   */\n  readonly appEnv: string;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Certificate in US-East-1 for the CloudFront distribution.\n   */\n  readonly certEdge?: acm.ICertificate;\n\n  /**\n   * Certificate in deployed region for the API Gateway.\n   */\n  readonly certOrigin?: acm.ICertificate;\n\n  /**\n   * Use a strict S3 Bucket Policy that prevents applications\n   * from reading/writing/modifying/deleting files in the S3 Bucket\n   * outside of the path that is specific to their app/version.\n   *\n   * This setting should be used when applications are less than\n   * fully trusted.\n   *\n   * @default false\n   */\n  readonly s3StrictBucketPolicy?: boolean;\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * IAM Role or IAM User names to exclude from the DENY rules on the S3 Bucket Policy.\n   *\n   * Roles that are Assumed must instead have their AROA added to `s3PolicyBypassAROAs`.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * @example ['arn:aws:iam::1234567890123:role/AdminAccess', 'arn:aws:iam::1234567890123:user/MyAdminUser']\n   *\n   * @see s3PolicyBypassAROAs\n   */\n  readonly s3PolicyBypassPrincipalARNs?: string[];\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * AROAs of the IAM Role to exclude from the DENY rules on the S3 Bucket Policy.\n   * This allows sessions that assume the IAM Role to be excluded from the\n   * DENY rules on the S3 Bucket Policy.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * Roles / users that are used directly, not assumed, can be added to `s3PolicyBypassRoleNames` instead.\n   *\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   *\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 wildcard matches implicitly either.\n   *\n   * The AROA must be used because there are only 3 Principal variables available:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n   *  aws:username, aws:userid, aws:PrincipalTag\n   *\n   * For an assumed role, aws:username is blank, aws:userid is:\n   *  [unique id AKA AROA for Role]:[session name]\n   *\n   * Table of unique ID prefixes such as AROA:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n   *\n   * The name of the role is simply not available for an assumed role and, if it was,\n   * a complicated comparison would be requierd to prevent exclusion\n   * of applying the Deny Rule to roles from 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   *\n   * @example [ 'AROA1234567890123' ]\n   *\n   * @see s3StrictBucketPolicy\n   */\n  readonly s3PolicyBypassAROAs?: string[];\n\n  /**\n   * Optional custom domain name for the CloudFront distribution.\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * Optional custom domain name for the API Gateway HTTPv2 API.\n   *\n   * @example apps-origin.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Create API Gateway for non-edge invocation\n   *\n   * @default false\n   */\n  readonly createAPIGateway?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /api/ that allows\n   * API routes to have a period in them.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /api/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createAPIPathRoute?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /_next/data/\n   * This route is used by Next.js to load data from the API Gateway\n   * on `getServerSideProps` calls.  The requests can end in `.json`,\n   * which would cause them to be routed to S3 if this route is not created.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /_next/data/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Adds an X-Forwarded-Host-Header when calling API Gateway\n   *\n   * Can only be trusted if `signingMode` is enabled, which restricts\n   * access to API Gateway to only IAM signed requests.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly addXForwardedHostHeader?: boolean;\n\n  /**\n   * Replaces Host header (which will be the Edge domain name) with the Origin domain name\n   * when enabled.  This is necessary when API Gateway has not been configured\n   * with a custom domain name that matches the exact domain name used by the CloudFront\n   * Distribution AND when the OriginRequestPolicy.HeadersBehavior is set\n   * to pass all headers to the origin.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly replaceHostHeader?: boolean;\n\n  /**\n   * Requires IAM auth on the API Gateway origin if not set to 'none'.\n   *\n   * 'sign' - Uses request headers for auth.\n   * 'presign' - Uses query string for auth.\n   *\n   * If enabled,\n   *\n   * Note: if 'sign' or 'presign', creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default 'sign'\n   */\n  readonly signingMode?: 'sign' | 'presign' | 'none';\n\n  /**\n   * Origin region that API Gateway or Lambda function will be deployed to, used\n   * for the config.yml on the Edge function to sign requests for\n   * the correct region\n   *\n   * @default undefined\n   */\n  readonly originRegion?: string;\n\n  /**\n   * Optional Origin Shield Region\n   *\n   * This should be the region where the DynamoDB is located so the\n   * EdgeToOrigin calls have the lowest latency (~1 ms).\n   *\n   * @default originRegion if specified, otherwise undefined\n   */\n  readonly originShieldRegion?: string;\n\n  /**\n   * Existing table for apps/versions/rules\n   *\n   * @warning - It is *strongly* suggested that production stacks create\n   * their own DynamoDB Table and pass it into this construct, for protection\n   * against data loss due to logical ID changes, the ability to configure\n   * Provisioned capacity with Auto Scaling, the ability to add additional indices, etc.\n   *\n   * Requirements:\n   * - Hash Key: `PK`\n   * - Sort Key: `SK`\n   *\n   * @default created by construct\n   */\n  readonly table?: dynamodb.ITable;\n\n  /**\n   * Pre-set table name for apps/versions/rules\n   *\n   * This is required when using v2 routing\n   */\n  readonly tableNameForEdgeToOrigin?: string;\n\n  /**\n   * Additional edge lambda functions\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\n\n  /**\n   * Account IDs allowed for cross-account Function URL invocations\n   *\n   * @default []\n   */\n  readonly allowedFunctionUrlAccounts?: string[];\n}\n\n/**\n * Represents a MicroApps\n */\nexport interface IMicroApps {\n  /** {@inheritdoc IMicroAppsCF} */\n  readonly cf: IMicroAppsCF;\n\n  /** {@inheritdoc IMicroAppsEdgeToOrigin} */\n  readonly edgeToOrigin?: IMicroAppsEdgeToOrigin;\n\n  /** {@inheritdoc IMicroAppsS3} */\n  readonly s3: IMicroAppsS3;\n\n  /** {@inheritdoc IMicroAppsSvcs} */\n  readonly svcs: IMicroAppsSvcs;\n\n  /** {@inheritdoc IMicroAppsAPIGwy} */\n  readonly apigwy?: IMicroAppsAPIGwy;\n}\n\n/**\n * Create a new MicroApps \"turnkey\" construct for simple\n * deployments and for initial evaulation of the MicroApps framework.\n *\n * Use this construct to create a PoC working entire stack.\n *\n * Do not use this construct when adding MicroApps to an existing\n * CloudFront, API Gateway, S3 Bucket, etc. or where access\n * to all features of the AWS Resources are needed (e.g. to\n * add additional Behaviors to the CloudFront distribution, set authorizors\n * on API Gateway, etc.).\n *\n * @warning This construct is not intended for production use.\n * In a production stack the DynamoDB Table, API Gateway, S3 Buckets,\n * etc. should be created in a \"durable\" stack where the IDs will not\n * change and where changes to the MicroApps construct will not\n * cause failures to deploy or data to be deleted.\n *\n *  @see {@link https://github.com/pwrdrvr/microapps-core/blob/main/packages/cdk/lib/MicroApps.ts | example usage in a CDK Stack }\n */\nexport class MicroApps extends Construct implements IMicroApps {\n  private _cf: MicroAppsCF;\n  public get cf(): IMicroAppsCF {\n    return this._cf;\n  }\n\n  private _edgeToOrigin?: MicroAppsEdgeToOrigin;\n  public get edgeToOrigin(): IMicroAppsEdgeToOrigin | undefined {\n    return this._edgeToOrigin;\n  }\n\n  private _s3: MicroAppsS3;\n  public get s3(): IMicroAppsS3 {\n    return this._s3;\n  }\n\n  private _apigwy?: MicroAppsAPIGwy;\n  public get apigwy(): IMicroAppsAPIGwy | undefined {\n    return this._apigwy;\n  }\n\n  private _svcs: MicroAppsSvcs;\n  public get svcs(): IMicroAppsSvcs {\n    return this._svcs;\n  }\n\n  constructor(scope: 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      domainNameEdge,\n      domainNameOrigin,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      certEdge,\n      appEnv = 'dev',\n      certOrigin,\n      removalPolicy,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      createAPIGateway = false,\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      signingMode = 'sign',\n      originRegion,\n      table,\n      tableNameForEdgeToOrigin,\n      originShieldRegion = originRegion,\n      allowedFunctionUrlAccounts = [],\n    } = props;\n\n    this._s3 = new MicroAppsS3(this, 's3', {\n      removalPolicy,\n      bucketLogsName: domainNameEdge ? `${reverseDomain(domainNameEdge)}-logs` : undefined,\n      bucketAppsName: domainNameEdge ? `${reverseDomain(domainNameEdge)}` : undefined,\n      bucketAppsStagingName: domainNameEdge\n        ? `${reverseDomain(domainNameEdge)}-staging`\n        : undefined,\n      assetNameRoot,\n      assetNameSuffix,\n      originShieldRegion,\n    });\n    if (createAPIGateway) {\n      this._apigwy = new MicroAppsAPIGwy(this, 'api', {\n        removalPolicy,\n        assetNameRoot,\n        assetNameSuffix,\n        domainNameEdge,\n        domainNameOrigin,\n        r53Zone,\n        certOrigin,\n        rootPathPrefix,\n        requireIAMAuthorization: signingMode !== 'none',\n      });\n    }\n    this._svcs = new MicroAppsSvcs(this, 'svcs', {\n      ...(this._apigwy ? { httpApi: this._apigwy.httpApi } : {}),\n      removalPolicy,\n      bucketApps: this._s3.bucketApps,\n      bucketAppsOAI: this._s3.bucketAppsOAI,\n      bucketAppsStaging: this._s3.bucketAppsStaging,\n      assetNameRoot,\n      assetNameSuffix,\n      appEnv,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      requireIAMAuthorization: signingMode !== 'none',\n      table,\n    });\n    const edgeLambdas: cf.EdgeLambda[] = [];\n\n    if (signingMode !== 'none' || replaceHostHeader || addXForwardedHostHeader) {\n      this._edgeToOrigin = new MicroAppsEdgeToOrigin(this, 'edgeToOrigin', {\n        assetNameRoot,\n        assetNameSuffix,\n        removalPolicy,\n        addXForwardedHostHeader,\n        setupApiGatewayPermissions: createAPIGateway,\n        replaceHostHeader,\n        originRegion,\n        signingMode,\n        rootPathPrefix,\n        tableRulesArn: tableNameForEdgeToOrigin || this._svcs.table.tableName,\n        allowedFunctionUrlAccounts,\n      });\n\n      edgeLambdas.push(...this._edgeToOrigin.edgeToOriginLambdas);\n    }\n    // Add any extra lambdas\n    if (props.edgeLambdas?.length) {\n      edgeLambdas.push(...props.edgeLambdas);\n    }\n    this._cf = new MicroAppsCF(this, 'cft', {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n      domainNameOrigin,\n      ...(this._apigwy ? { httpApi: this._apigwy.httpApi } : {}),\n      r53Zone,\n      certEdge,\n      bucketAppsOriginS3: this._s3.bucketAppsOriginS3,\n      bucketAppsOriginApp: this._s3.bucketAppsOriginApp,\n      bucketLogs: this._s3.bucketLogs,\n      rootPathPrefix,\n      createAPIPathRoute,\n      createNextDataPathRoute,\n      originShieldRegion,\n      ...(edgeLambdas.length ? { edgeLambdas } : {}),\n    });\n  }\n}\n"]}
|
package/lib/MicroAppsAPIGwy.d.ts
CHANGED
package/lib/MicroAppsAPIGwy.js
CHANGED
|
@@ -167,5 +167,5 @@ class MicroAppsAPIGwy extends constructs_1.Construct {
|
|
|
167
167
|
}
|
|
168
168
|
exports.MicroAppsAPIGwy = MicroAppsAPIGwy;
|
|
169
169
|
_a = JSII_RTTI_SYMBOL_1;
|
|
170
|
-
MicroAppsAPIGwy[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsAPIGwy", version: "0.4.0-alpha.
|
|
170
|
+
MicroAppsAPIGwy[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsAPIGwy", version: "0.4.0-alpha.4" };
|
|
171
171
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsAPIGwy.js","sourceRoot":"","sources":["../src/MicroAppsAPIGwy.ts"],"names":[],"mappings":"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","sourcesContent":["import * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport * as apigwyAuth from '@aws-cdk/aws-apigatewayv2-authorizers-alpha';\nimport { RemovalPolicy, Stack, Tags } from 'aws-cdk-lib';\nimport * as apigwycfn from 'aws-cdk-lib/aws-apigatewayv2';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport { Construct } from 'constructs';\n\n/**\n * Properties to initialize an instance of `MicroAppsAPIGwy`.\n */\nexport interface MicroAppsAPIGwyProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * CloudFront edge domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * API Gateway origin domain name\n   *\n   * @example apps-origin.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Optional local region ACM certificate to use for API Gateway\n   * Note: required when using a custom domain\n   *\n   * @default none\n   */\n  readonly certOrigin?: acm.ICertificate;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Path prefix on the root of the API Gateway Stage\n   *\n   * @example dev/\n   * @default none\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Require IAM auth on API Gateway\n   *\n   * @default true\n   */\n  readonly requireIAMAuthorization?: boolean;\n}\n\n/**\n * Represents a MicroApps API Gateway\n */\nexport interface IMicroAppsAPIGwy {\n  /**\n   * Domain Name applied to API Gateway origin\n   */\n  readonly dnAppsOrigin?: apigwy.IDomainName;\n\n  /**\n   * API Gateway\n   */\n  readonly httpApi: apigwy.HttpApi;\n}\n\n/**\n * Create a new MicroApps API Gateway HTTP API endpoint, optionally\n * requiring IAM authorization\n */\nexport class MicroAppsAPIGwy extends Construct implements IMicroAppsAPIGwy {\n  private _dnAppsOrigin: apigwy.DomainName | undefined;\n  public get dnAppsOrigin(): apigwy.IDomainName | undefined {\n    return this._dnAppsOrigin;\n  }\n\n  private _httpApi: apigwy.HttpApi;\n  public get httpApi(): apigwy.HttpApi {\n    return this._httpApi;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsAPIGwyProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    if (\n      (props.r53Zone === undefined && props.domainNameEdge !== undefined) ||\n      (props.r53Zone !== undefined && props.domainNameEdge === undefined)\n    ) {\n      throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');\n    }\n\n    if (props.domainNameOrigin !== undefined && props.certOrigin === undefined) {\n      throw new Error('If domainNameOrigin is set then certOrigin must be set');\n    }\n\n    if (\n      (props.domainNameEdge === undefined && props.certOrigin !== undefined) ||\n      (props.domainNameEdge !== undefined && props.certOrigin === undefined)\n    ) {\n      throw new Error(\n        'If either of domainNameEdge or certOrigin are set then the other must be set',\n      );\n    }\n\n    const {\n      r53Zone,\n      domainNameEdge,\n      domainNameOrigin,\n      certOrigin,\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      rootPathPrefix,\n      requireIAMAuthorization = true,\n    } = props;\n\n    // API Gateway uses the `id` string as the gateway name without\n    // any randomization... we have to make sure the name is unique-ish\n    const apigatewayName = assetNameRoot\n      ? `${assetNameRoot}${assetNameSuffix}`\n      : `${Stack.of(this).stackName}-microapps`;\n\n    //\n    // APIGateway domain names for CloudFront and origin\n    //\n    this._httpApi = new apigwy.HttpApi(this, 'gwy', {\n      apiName: apigatewayName,\n      createDefaultStage: false,\n      defaultAuthorizer: requireIAMAuthorization ? new apigwyAuth.HttpIamAuthorizer() : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._httpApi.applyRemovalPolicy(removalPolicy);\n    }\n    // This allows the Lambda @ Edge function to execute this api\n    Tags.of(this._httpApi).add('microapp-managed', 'true');\n\n    // Create the stage\n    const stage = new apigwy.HttpStage(this, 'stage', {\n      httpApi: this._httpApi,\n      autoDeploy: true,\n      // If rootPathPrefix is not defined this will be the $default stage\n      stageName: rootPathPrefix,\n    });\n    // This allows the Lambda @ Edge function to execute this api\n    Tags.of(stage).add('microapp-managed', 'true');\n\n    if (domainNameEdge !== undefined && certOrigin !== undefined) {\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 (removalPolicy !== undefined) {\n        dnAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n\n      // Create the edge domain name mapping for the API\n      const apiMapping = new apigwy.ApiMapping(this, 'mapping', {\n        api: this._httpApi,\n        domainName: dnAppsEdge,\n        stage,\n      });\n      // 2022-01-16 - CDK is still generating CloudFormation with no dependency\n      // between the R53 RecordSet and the Mapping\n      apiMapping.node.addDependency(dnAppsEdge);\n    }\n\n    if (domainNameOrigin !== undefined && certOrigin !== undefined) {\n      this._dnAppsOrigin = new apigwy.DomainName(this, 'origin-dn', {\n        domainName: domainNameOrigin,\n        certificate: certOrigin,\n      });\n      if (removalPolicy !== undefined) {\n        this._dnAppsOrigin.applyRemovalPolicy(removalPolicy);\n      }\n    }\n\n    // Enable access logs on API Gateway\n    const apiAccessLogs = new logs.LogGroup(this, 'logs', {\n      logGroupName: apigatewayName\n        ? `/aws/apigwy/${apigatewayName}`\n        : `/aws/apigwy/${this.httpApi.httpApiName}`,\n      retention: logs.RetentionDays.TWO_WEEKS,\n    });\n    if (removalPolicy !== undefined) {\n      apiAccessLogs.applyRemovalPolicy(removalPolicy);\n    }\n    const cfnStage = stage.node.defaultChild as apigwycfn.CfnStage;\n    cfnStage.accessLogSettings = {\n      destinationArn: apiAccessLogs.logGroupArn,\n      format: JSON.stringify({\n        requestId: '$context.requestId',\n        userAgent: '$context.identity.userAgent',\n        sourceIp: '$context.identity.sourceIp',\n        requestTime: '$context.requestTime',\n        requestTimeEpoch: '$context.requestTimeEpoch',\n        httpMethod: '$context.httpMethod',\n        path: '$context.path',\n        status: '$context.status',\n        protocol: '$context.protocol',\n        responseLength: '$context.responseLength',\n        domainName: '$context.domainName',\n        authorizerError: '$context.authorizer.error',\n      }),\n    };\n\n    // Create a logging role\n    // Tips: https://github.com/aws/aws-cdk/issues/11100\n    const apiGwyLogRole = new iam.Role(this, 'logs-role', {\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          'service-role/AmazonAPIGatewayPushToCloudWatchLogs',\n        ),\n      ],\n      assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'),\n    });\n    apiAccessLogs.grantWrite(apiGwyLogRole);\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    if (this._dnAppsOrigin !== undefined) {\n      const mappingAppsApis = new apigwy.ApiMapping(this, 'api-map-origin', {\n        api: this._httpApi,\n        domainName: this._dnAppsOrigin,\n        stage,\n      });\n      // 2022-01-16 - CDK is still generating CloudFormation with no dependency\n      // between the R53 RecordSet and the Mapping\n      mappingAppsApis.node.addDependency(this._dnAppsOrigin);\n      if (removalPolicy !== undefined) {\n        mappingAppsApis.applyRemovalPolicy(removalPolicy);\n      }\n    }\n\n    //\n    // Create the origin name for API Gateway\n    //\n    if (r53Zone !== undefined && this._dnAppsOrigin) {\n      const rrAppsOrigin = new r53.ARecord(this, 'origin-arecord', {\n        zone: r53Zone,\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 (removalPolicy !== undefined) {\n        rrAppsOrigin.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
|