@pwrdrvr/microapps-cdk 1.2.0-beta.3 → 1.2.0-beta.5

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
@@ -3891,7 +3891,7 @@
3891
3891
  },
3892
3892
  "name": "@pwrdrvr/microapps-cdk",
3893
3893
  "readme": {
3894
- "markdown": "[![CI](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml) [![Merge to Main Build](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml) [![Release Packages](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml)\n\n# Overview\n\nThe MicroApps project enables rapidly deploying many web apps to AWS on a single shared host name, fronted by a CloudFront Distribution, serving static assets from an S3 Bucket, and routing application requests via API Gateway. MicroApps is delivered as a CDK Construct for deployment, although alternative deployment methods can be used if desired and implemented.\n\nMicroApps 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).\n\nUsers 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.\n\nFor 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.\n\n# Table of Contents <!-- omit in toc -->\n\n- [Overview](#overview)\n- [Why MicroApps](#why-microapps)\n- [Request Routing for Static Assets / App - Diagram](#request-routing-for-static-assets--app---diagram)\n- [Request Dispatch Model for Multi-Account Deployments](#request-dispatch-model-for-multi-account-deployments)\n- [Video Preview of the Deploying CDK Construct](#video-preview-of-the-deploying-cdk-construct)\n- [Installation / CDK Constructs](#installation--cdk-constructs)\n- [Tutorial - Bootstrapping a Deploy](#tutorial---bootstrapping-a-deploy)\n- [Limitations / Future Development](#limitations--future-development)\n- [Related Projects / Components](#related-projects--components)\n- [Why Lambda @ Origin and Not Lambda @ Edge for Apps](#why-lambda--origin-and-not-lambda--edge-for-apps)\n- [Architecure Diagram](#architecure-diagram)\n- [Project Layout](#project-layout)\n- [Creating a MicroApp Using Zip Lambda Functions](#creating-a-microapp-using-zip-lambda-functions)\n- [Creating a MicroApp Using Docker Lambda Functions](#creating-a-microapp-using-docker-lambda-functions)\n - [Next.js Apps](#nextjs-apps)\n - [Modify package.json](#modify-packagejson)\n - [Install Dependencies](#install-dependencies)\n - [Dockerfile](#dockerfile)\n - [next.config.js](#nextconfigjs)\n- [Troubleshooting](#troubleshooting)\n - [CloudFront Requests to API Gateway are Rejected with 403 Forbidden](#cloudfront-requests-to-api-gateway-are-rejected-with-403-forbidden)\n - [SignatureV4 Headers](#signaturev4-headers)\n\n# Why MicroApps\n\nMicroApps 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.\n\nTeams 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).\n\nMicroApps 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.\n\nFor 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.\n\n# Request Routing for Static Assets / App - Diagram\n\n![Request Routing for Static Assets and App](https://user-images.githubusercontent.com/5617868/222913451-0e6ed906-b6ee-461f-99a7-61db13135ce1.png)\n\n# Request Dispatch Model for Multi-Account Deployments\n\nNote: requests can also be dispatched into the same account, but this model is more likely to be used by organizations with many AWS accounts.\n\n![Request Dispatch Model for Mulit-Account Deployments](https://user-images.githubusercontent.com/5617868/218237120-65b3ae44-31ba-4b6d-8722-4d3fb7da5577.png)\n\n# Video Preview of the Deploying CDK Construct\n\n![Video Preview of Deploying](https://raw.githubusercontent.com/pwrdrvr/microapps-core/main/assets/videos/microapps-core-demo-deploy.gif)\n\n# Installation / CDK Constructs\n\n- `npm i --save-dev @pwrdrvr/microapps-cdk`\n- Add `MicroApps` construct to your stack\n- The `MicroApps` construct does a \"turn-key\" deployment complete with the Release app\n- [Construct Hub](https://constructs.dev/packages/@pwrdrvr/microapps-cdk/)\n - CDK API docs\n - Python, DotNet, Java, JS/TS installation instructions\n\n# Tutorial - Bootstrapping a Deploy\n\n- `git clone https://github.com/pwrdrvr/microapps-core.git`\n - Note: the repo is only being for the example CDK Stack, it is not necessary to clone the repo when used in a custom CDK Stack\n- `cd microapps-core`\n- `npm i -g aws-cdk`\n - Install AWS CDK v2 CLI\n- `asp [my-sso-profile-name]`\n - Using the `aws` plugin from `oh-my-zsh` for AWS SSO\n - Of course, there are other methods of setting env vars\n- `aws sso login`\n - Establish an AWS SSO session\n- `export AWS_REGION=us-east-2`\n - Region needs to be set for the Lambda invoke - This can be done other ways in `~/.aws/config` as well\n- `./deploy.sh`\n - Deploys the CDK Stack\n - Essentially runs two commands along with extraction of outputs:\n - `npx cdk deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`\n - `npx pwrdrvr 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`\n - URL will be printed as last output\n\n# Limitations / Future Development\n\n- AWS Only\n - For the time being this has only been implemented for AWS technologies and APIs\n - It is possible that Azure and GCP have sufficient support to enable porting the framework\n - CDK would have to be replaced as well (unless it's made available for Azure and GCP in the near future)\n- Release Rules\n - Currently only a Default rule is supported\n - Need to evaluate if a generic implementation can be made, possibly allowing plugins or webhooks to support arbitrary rules\n - If not possible to make it perfectly generic, consider providing a more complete reference implementation of examples\n\n# Related Projects / Components\n\n- Release App\n - The Release app is an initial, rudimentary, release control console for setting the default version of an application\n - Built with Next.js\n - [pwrdrvr/microapps-app-release](https://github.com/pwrdrvr/microapps-app-release)\n- Next.js Demo App\n - The Next.js Tutorial application deployed as a MicroApp\n - [pwrdrvr/serverless-nextjs-demo](https://github.com/pwrdrvr/serverless-nextjs-demo)\n\n\n# Why Lambda @ Origin and Not Lambda @ Edge for Apps\n\nCalling 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.\n\nWith 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.\n\n- Lambda@Edge is _at least_ 3x more expensive than Lambda at the origin:\n - In US East 1, the price per GB-Second is $0.00005001 for Lambda@Edge\n - Source: https://aws.amazon.com/lambda/pricing/ (bottom of page)\n - Updated: 2023-03-04\n - In US East 1, the price per GB-Second is $0.0000166667 for Lambda at the origin on x86\n - Source: https://aws.amazon.com/lambda/pricing/\n - Updated: 2023-03-04\n - Ratio\n - Lambda@Edge / Lambda@Origin = $0.00005001 / $0.0000166667 = 3.0006x\n- 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:\n - Lambda@Edge\n - 0.250s Round Trip Time (RTT) for EU-zone edge request to hit US-East 1 Origin\n - 0.200s DB lookup time\n - 0.050s CPU usage to process the DB response\n - 0.500s total billed time @ $0.00005001 @ 128 MB\n - $0.000003125625 total charge\n - Lambda at Origin\n - RTT does not apply (it's effectively 1-2 ms to hit a DB in the same region)\n - 0.200s DB lookup time\n - 0.050s CPU usage to process the DB response\n - 0.250s total billed time @ $0.0000166667 @ 128 MB\n - Half the billed time of running on Lambda@Edge\n - 1/6th the cost of running on Lambda@Edge:\n - $0.000000520834375 total charge (assuming no CPU time to process the response)\n - $0.000003125625 / $0.000000520834375 = 6x more expensive in Lambda@Edge\n\n# Architecure Diagram\n\n![Architecure Diagram](https://raw.githubusercontent.com/pwrdrvr/microapps-core/main/assets/images/architecture-diagram.png)\n\n# Project Layout\n\n- [packages/cdk](https://github.com/pwrdrvr/microapps-core/tree/main/packages/cdk)\n - Example CDK Stack\n - Deploys MicroApps CDK stack for the GitHub Workflows\n - Can be used as an example of how to use the MicroApps CDK Construct\n- [packages/demo-app](https://github.com/pwrdrvr/microapps-core/tree/main/packages/demo-app)\n - Example app with static resources and a Lambda function\n - Does not use any Web UI framework at all\n- [packages/microapps-cdk](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-cdk)\n - MicroApps\n - \"Turn key\" CDK Construct that creates all assets needed for a working MicroApps deployment\n - MicroAppsAPIGwy\n - Create APIGateway HTTP API\n - Creates domain names to point to the edge (Cloudfront) and origin (API Gateway)\n - MicroAppsCF\n - Creates Cloudfront distribution\n - MicroAppsS3\n - Creates S3 buckets\n - MicroAppsSvcs\n - Create DynamoDB table\n - Create Deployer Lambda function\n - Create Router Lambda function\n- [packages/microapps-datalib](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-datalib)\n - Installed from `npm`:\n - `npm i -g @pwrdrvr/microapps-datalib`\n - APIs for access to the DynamoDB Table used by the `pwrdrvr` CLI, `microapps-deployer`, and `@pwrdrvr/microapps-app-release-cdk`\n- [packages/microapps-deployer](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-deployer)\n - Lambda service invoked by the `pwrdrvr` CLI to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n - Returns a temporary S3 token with restricted access to the staging S3 bucket for upload of the static files for one app/semver\n- [packages/pwrdrvr](https://github.com/pwrdrvr/microapps-core/tree/main/packages/pwrdrvr)\n - Installed from `npm`:\n - `npm i -g pwrdrvr`\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 - AWS IAM permissions required:\n - `lambda:InvokeFunction`\n- [packages/microapps-router](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-router)\n - Lambda function that determines which version of an app to point a user to on a particular invocation\n\n# Creating a MicroApp Using Zip Lambda Functions\n\n[TBC]\n\n# Creating a MicroApp Using Docker Lambda Functions\n\nDocker Lambdas are great for large applications. These used to be slower to cold start but as of early 2023 that appears to no longer be an issue.\n\n## Next.js Apps\n\nCreate a Next.js app then follow the steps in this section to set it up for publishing to AWS Lambda @ Origin as a MicroApp. To publish new versions of the app use `npx pwrdrvr --new-version x.y.z` when logged in to the target AWS account.\n\n### Modify package.json\n\nReplace the version with `0.0.0` so it can be modified by the `pwrdrvr` CLI.\n\n### Install Dependencies\n\n```\nnpm i --save-dev pwrdrvr\n```\n\n### Dockerfile\n\nFIXME: Out of date 2023-03-04\n\nAdd this file to the root of the app.\n\n```Dockerfile\nFROM node:15-slim as base\n\nWORKDIR /app\n\n# Download the sharp libs once to save time\n# Do this before copying anything else in\nRUN mkdir -p image-lambda-npms && \\\n cd image-lambda-npms && npm i sharp && \\\n rm -rf node_modules/sharp/vendor/*/include/\n\n# Copy in the build output from `npx serverless`\nCOPY .serverless_nextjs .\nCOPY config.json .\n\n# Move the sharp libs into place\nRUN rm -rf image-lambda/node_modules/ && \\\n mv image-lambda-npms/node_modules image-labmda/ && \\\n rm -rf image-lambda-npms\n\nFROM public.ecr.aws/lambda/nodejs:14 AS final\n\n# Copy in the munged code\nCOPY --from=base /app .\n\nCMD [ \"./index.handler\" ]\n```\n\n### next.config.js\n\nFIXME: Out of date 2023-03-04\n\nAdd this file to the root of the app.\n\nReplace `appname` with your URL path-compatible application name.\n\n```js\nconst appRoot = '/appname/0.0.0';\n\n// eslint-disable-next-line no-undef\nmodule.exports = {\n target: 'serverless',\n webpack: (config, _options) => {\n return config;\n },\n basePath: appRoot,\n publicRuntimeConfig: {\n // Will be available on both server and client\n staticFolder: appRoot,\n },\n};\n```\n\n# Troubleshooting\n\n## CloudFront Requests to API Gateway are Rejected with 403 Forbidden\n\nRequests to the API Gateway origin can be rejected with a 403 Forbidden error if the signed request headers are not sent to the origin by CloudFront.\n\nThe error in the API Gateway CloudWatch logs will show up as:\n\n```log\n\"authorizerError\": \"The request for the IAM Authorizer doesn't match the format that API Gateway expects.\"\n```\n\nThis can be simulated by simply running `curl [api-gateway-url]`, with no headers.\n\nTo confirm that API Gateway is allowing signed requests when the IAM Authorizer is configured, establish credentials as a user that is allowed to execute the API Gateay, install `awscurl` with `pip3 install awscurl`, then then use `awscurl --service execute-api --region [api-gateway-region] [api-gateway-url]`.\n\nSignature headers will not be sent from CloudFront to API Gateway unless the `OriginRequestPolicy` is set to specifically include those headers on requests to the origin, or the `headersBehavior` is set to `cfront.OriginRequestHeaderBehavior.all()`.\n\nSimilarly, if `presign` is used, the `OriginRequestPolicy` must be set to `cfront.OriginRequestQueryStringBehavior.all()` or to specifically forward the query string parameters used by the presigned URL.\n\n### SignatureV4 Headers\n- `authorization`\n- `x-amz-date`\n- `x-amz-security-token`\n- `x-amz-content-sha256`\n"
3894
+ "markdown": "[![CI](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml) [![Merge to Main Build](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml) [![Release Packages](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml/badge.svg)](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml)\n\n# Overview\n\nThe MicroApps project enables rapidly deploying many web apps to AWS on a single shared host name, fronted by a CloudFront Distribution, serving static assets from an S3 Bucket, and routing application requests via API Gateway. MicroApps is delivered as a CDK Construct for deployment, although alternative deployment methods can be used if desired and implemented.\n\nMicroApps 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).\n\nUsers 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.\n\nFor 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.\n\n# Table of Contents <!-- omit in toc -->\n\n- [Overview](#overview)\n- [Why MicroApps](#why-microapps)\n- [Request Routing for Static Assets / App - Diagram](#request-routing-for-static-assets--app---diagram)\n- [Request Dispatch Model for Multi-Account Deployments](#request-dispatch-model-for-multi-account-deployments)\n- [Video Preview of the Deploying CDK Construct](#video-preview-of-the-deploying-cdk-construct)\n- [Installation / CDK Constructs](#installation--cdk-constructs)\n- [Monorepo Boundary Enforcement](#monorepo-boundary-enforcement)\n- [Tutorial - Bootstrapping a Deploy](#tutorial---bootstrapping-a-deploy)\n- [Limitations / Future Development](#limitations--future-development)\n- [Related Projects / Components](#related-projects--components)\n- [Why Lambda @ Origin and Not Lambda @ Edge for Apps](#why-lambda--origin-and-not-lambda--edge-for-apps)\n- [Architecure Diagram](#architecure-diagram)\n- [Project Layout](#project-layout)\n- [Creating a MicroApp Using Zip Lambda Functions](#creating-a-microapp-using-zip-lambda-functions)\n- [Creating a MicroApp Using Docker Lambda Functions](#creating-a-microapp-using-docker-lambda-functions)\n - [Next.js Apps](#nextjs-apps)\n - [Modify package.json](#modify-packagejson)\n - [Install Dependencies](#install-dependencies)\n - [Dockerfile](#dockerfile)\n - [next.config.js](#nextconfigjs)\n- [Troubleshooting](#troubleshooting)\n - [CloudFront Requests to API Gateway are Rejected with 403 Forbidden](#cloudfront-requests-to-api-gateway-are-rejected-with-403-forbidden)\n - [SignatureV4 Headers](#signaturev4-headers)\n\n# Why MicroApps\n\nMicroApps 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.\n\nTeams 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).\n\nMicroApps 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.\n\nFor 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.\n\n# Request Routing for Static Assets / App - Diagram\n\n![Request Routing for Static Assets and App](https://user-images.githubusercontent.com/5617868/222913451-0e6ed906-b6ee-461f-99a7-61db13135ce1.png)\n\n# Request Dispatch Model for Multi-Account Deployments\n\nNote: requests can also be dispatched into the same account, but this model is more likely to be used by organizations with many AWS accounts.\n\n![Request Dispatch Model for Mulit-Account Deployments](https://user-images.githubusercontent.com/5617868/218237120-65b3ae44-31ba-4b6d-8722-4d3fb7da5577.png)\n\n# Video Preview of the Deploying CDK Construct\n\n![Video Preview of Deploying](https://raw.githubusercontent.com/pwrdrvr/microapps-core/main/assets/videos/microapps-core-demo-deploy.gif)\n\n# Installation / CDK Constructs\n\n- `npm i --save-dev @pwrdrvr/microapps-cdk`\n- Add `MicroApps` construct to your stack\n- The `MicroApps` construct does a \"turn-key\" deployment complete with the Release app\n- [Construct Hub](https://constructs.dev/packages/@pwrdrvr/microapps-cdk/)\n - CDK API docs\n - Python, DotNet, Java, JS/TS installation instructions\n\n# Monorepo Boundary Enforcement\n\nThis repo now relies on two complementary guardrails for workspace boundaries:\n\n- `pnpm` uses its default isolated workspace layout at the repo root, so undeclared dependencies are much less likely to appear accidentally through a flattened install.\n- ESLint enforces `import/no-extraneous-dependencies`, including type-only imports, so invalid package-to-package imports fail with a file-level diagnostic during normal lint runs and in CI.\n\nIn practice, if a package imports another workspace package, declare that dependency in the importing package's `package.json`. Test-only helpers such as `jest-dynalite` should also be declared in the specific package that uses them instead of relying on a root-only dev dependency.\n\n# Tutorial - Bootstrapping a Deploy\n\n- `git clone https://github.com/pwrdrvr/microapps-core.git`\n - Note: the repo is only being for the example CDK Stack, it is not necessary to clone the repo when used in a custom CDK Stack\n- `cd microapps-core`\n- `npm i -g aws-cdk`\n - Install AWS CDK v2 CLI\n- `asp [my-sso-profile-name]`\n - Using the `aws` plugin from `oh-my-zsh` for AWS SSO\n - Of course, there are other methods of setting env vars\n- `aws sso login`\n - Establish an AWS SSO session\n- `export AWS_REGION=us-east-2`\n - Region needs to be set for the Lambda invoke - This can be done other ways in `~/.aws/config` as well\n- `./deploy.sh`\n - Deploys the CDK Stack\n - Essentially runs two commands along with extraction of outputs:\n - `npx cdk deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`\n - `npx pwrdrvr 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 packages/cdk/node_modules/@pwrdrvr/microapps-app-release-cdk/lib/static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --no-cache`\n - URL will be printed as last output\n\n# Limitations / Future Development\n\n- AWS Only\n - For the time being this has only been implemented for AWS technologies and APIs\n - It is possible that Azure and GCP have sufficient support to enable porting the framework\n - CDK would have to be replaced as well (unless it's made available for Azure and GCP in the near future)\n- Release Rules\n - Currently only a Default rule is supported\n - Need to evaluate if a generic implementation can be made, possibly allowing plugins or webhooks to support arbitrary rules\n - If not possible to make it perfectly generic, consider providing a more complete reference implementation of examples\n\n# Related Projects / Components\n\n- Release App\n - The Release app is an initial, rudimentary, release control console for setting the default version of an application\n - Built with Next.js\n - [pwrdrvr/microapps-app-release](https://github.com/pwrdrvr/microapps-app-release)\n- Next.js Demo App\n - The Next.js Tutorial application deployed as a MicroApp\n - [pwrdrvr/serverless-nextjs-demo](https://github.com/pwrdrvr/serverless-nextjs-demo)\n\n\n# Why Lambda @ Origin and Not Lambda @ Edge for Apps\n\nCalling 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.\n\nWith 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.\n\n- Lambda@Edge is _at least_ 3x more expensive than Lambda at the origin:\n - In US East 1, the price per GB-Second is $0.00005001 for Lambda@Edge\n - Source: https://aws.amazon.com/lambda/pricing/ (bottom of page)\n - Updated: 2023-03-04\n - In US East 1, the price per GB-Second is $0.0000166667 for Lambda at the origin on x86\n - Source: https://aws.amazon.com/lambda/pricing/\n - Updated: 2023-03-04\n - Ratio\n - Lambda@Edge / Lambda@Origin = $0.00005001 / $0.0000166667 = 3.0006x\n- 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:\n - Lambda@Edge\n - 0.250s Round Trip Time (RTT) for EU-zone edge request to hit US-East 1 Origin\n - 0.200s DB lookup time\n - 0.050s CPU usage to process the DB response\n - 0.500s total billed time @ $0.00005001 @ 128 MB\n - $0.000003125625 total charge\n - Lambda at Origin\n - RTT does not apply (it's effectively 1-2 ms to hit a DB in the same region)\n - 0.200s DB lookup time\n - 0.050s CPU usage to process the DB response\n - 0.250s total billed time @ $0.0000166667 @ 128 MB\n - Half the billed time of running on Lambda@Edge\n - 1/6th the cost of running on Lambda@Edge:\n - $0.000000520834375 total charge (assuming no CPU time to process the response)\n - $0.000003125625 / $0.000000520834375 = 6x more expensive in Lambda@Edge\n\n# Architecure Diagram\n\n![Architecure Diagram](https://raw.githubusercontent.com/pwrdrvr/microapps-core/main/assets/images/architecture-diagram.png)\n\n# Project Layout\n\n- [packages/cdk](https://github.com/pwrdrvr/microapps-core/tree/main/packages/cdk)\n - Example CDK Stack\n - Deploys MicroApps CDK stack for the GitHub Workflows\n - Can be used as an example of how to use the MicroApps CDK Construct\n- [packages/demo-app](https://github.com/pwrdrvr/microapps-core/tree/main/packages/demo-app)\n - Example app with static resources and a Lambda function\n - Does not use any Web UI framework at all\n- [packages/microapps-cdk](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-cdk)\n - MicroApps\n - \"Turn key\" CDK Construct that creates all assets needed for a working MicroApps deployment\n - MicroAppsAPIGwy\n - Create APIGateway HTTP API\n - Creates domain names to point to the edge (Cloudfront) and origin (API Gateway)\n - MicroAppsCF\n - Creates Cloudfront distribution\n - MicroAppsS3\n - Creates S3 buckets\n - MicroAppsSvcs\n - Create DynamoDB table\n - Create Deployer Lambda function\n - Create Router Lambda function\n- [packages/microapps-datalib](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-datalib)\n - Installed from `npm`:\n - `npm i -g @pwrdrvr/microapps-datalib`\n - APIs for access to the DynamoDB Table used by the `pwrdrvr` CLI, `microapps-deployer`, and `@pwrdrvr/microapps-app-release-cdk`\n- [packages/microapps-deployer](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-deployer)\n - Lambda service invoked by the `pwrdrvr` CLI to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n - Returns a temporary S3 token with restricted access to the staging S3 bucket for upload of the static files for one app/semver\n- [packages/pwrdrvr](https://github.com/pwrdrvr/microapps-core/tree/main/packages/pwrdrvr)\n - Installed from `npm`:\n - `npm i -g pwrdrvr`\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 - AWS IAM permissions required:\n - `lambda:InvokeFunction`\n- [packages/microapps-router](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-router)\n - Lambda function that determines which version of an app to point a user to on a particular invocation\n\n# Creating a MicroApp Using Zip Lambda Functions\n\n[TBC]\n\n# Creating a MicroApp Using Docker Lambda Functions\n\nDocker Lambdas are great for large applications. These used to be slower to cold start but as of early 2023 that appears to no longer be an issue.\n\n## Next.js Apps\n\nCreate a Next.js app then follow the steps in this section to set it up for publishing to AWS Lambda @ Origin as a MicroApp. To publish new versions of the app use `npx pwrdrvr --new-version x.y.z` when logged in to the target AWS account.\n\n### Modify package.json\n\nReplace the version with `0.0.0` so it can be modified by the `pwrdrvr` CLI.\n\n### Install Dependencies\n\n```\nnpm i --save-dev pwrdrvr\n```\n\n### Dockerfile\n\nFIXME: Out of date 2023-03-04\n\nAdd this file to the root of the app.\n\n```Dockerfile\nFROM node:15-slim as base\n\nWORKDIR /app\n\n# Download the sharp libs once to save time\n# Do this before copying anything else in\nRUN mkdir -p image-lambda-npms && \\\n cd image-lambda-npms && npm i sharp && \\\n rm -rf node_modules/sharp/vendor/*/include/\n\n# Copy in the build output from `npx serverless`\nCOPY .serverless_nextjs .\nCOPY config.json .\n\n# Move the sharp libs into place\nRUN rm -rf image-lambda/node_modules/ && \\\n mv image-lambda-npms/node_modules image-labmda/ && \\\n rm -rf image-lambda-npms\n\nFROM public.ecr.aws/lambda/nodejs:14 AS final\n\n# Copy in the munged code\nCOPY --from=base /app .\n\nCMD [ \"./index.handler\" ]\n```\n\n### next.config.js\n\nFIXME: Out of date 2023-03-04\n\nAdd this file to the root of the app.\n\nReplace `appname` with your URL path-compatible application name.\n\n```js\nconst appRoot = '/appname/0.0.0';\n\n// eslint-disable-next-line no-undef\nmodule.exports = {\n target: 'serverless',\n webpack: (config, _options) => {\n return config;\n },\n basePath: appRoot,\n publicRuntimeConfig: {\n // Will be available on both server and client\n staticFolder: appRoot,\n },\n};\n```\n\n# Troubleshooting\n\n## CloudFront Requests to API Gateway are Rejected with 403 Forbidden\n\nRequests to the API Gateway origin can be rejected with a 403 Forbidden error if the signed request headers are not sent to the origin by CloudFront.\n\nThe error in the API Gateway CloudWatch logs will show up as:\n\n```log\n\"authorizerError\": \"The request for the IAM Authorizer doesn't match the format that API Gateway expects.\"\n```\n\nThis can be simulated by simply running `curl [api-gateway-url]`, with no headers.\n\nTo confirm that API Gateway is allowing signed requests when the IAM Authorizer is configured, establish credentials as a user that is allowed to execute the API Gateay, install `awscurl` with `pip3 install awscurl`, then then use `awscurl --service execute-api --region [api-gateway-region] [api-gateway-url]`.\n\nSignature headers will not be sent from CloudFront to API Gateway unless the `OriginRequestPolicy` is set to specifically include those headers on requests to the origin, or the `headersBehavior` is set to `cfront.OriginRequestHeaderBehavior.all()`.\n\nSimilarly, if `presign` is used, the `OriginRequestPolicy` must be set to `cfront.OriginRequestQueryStringBehavior.all()` or to specifically forward the query string parameters used by the presigned URL.\n\n### SignatureV4 Headers\n- `authorization`\n- `x-amz-date`\n- `x-amz-security-token`\n- `x-amz-content-sha256`\n"
3895
3895
  },
3896
3896
  "repository": {
3897
3897
  "type": "git",
@@ -3999,6 +3999,40 @@
3999
3999
  "fqn": "aws-cdk-lib.aws_cloudfront.Distribution"
4000
4000
  }
4001
4001
  },
4002
+ {
4003
+ "abstract": true,
4004
+ "docs": {
4005
+ "stability": "experimental",
4006
+ "summary": "When true, routes that contain `/api/` get sent to the app origin even when other path segments contain periods."
4007
+ },
4008
+ "immutable": true,
4009
+ "locationInModule": {
4010
+ "filename": "src/MicroAppsCF.ts",
4011
+ "line": 215
4012
+ },
4013
+ "name": "createAPIPathRoute",
4014
+ "optional": true,
4015
+ "type": {
4016
+ "primitive": "boolean"
4017
+ }
4018
+ },
4019
+ {
4020
+ "abstract": true,
4021
+ "docs": {
4022
+ "stability": "experimental",
4023
+ "summary": "When true, routes that contain `/_next/data/` get sent to the app origin even when the request path ends in `.json`."
4024
+ },
4025
+ "immutable": true,
4026
+ "locationInModule": {
4027
+ "filename": "src/MicroAppsCF.ts",
4028
+ "line": 221
4029
+ },
4030
+ "name": "createNextDataPathRoute",
4031
+ "optional": true,
4032
+ "type": {
4033
+ "primitive": "boolean"
4034
+ }
4035
+ },
4002
4036
  {
4003
4037
  "abstract": true,
4004
4038
  "docs": {
@@ -4008,7 +4042,7 @@
4008
4042
  "immutable": true,
4009
4043
  "locationInModule": {
4010
4044
  "filename": "src/MicroAppsCF.ts",
4011
- "line": 214
4045
+ "line": 226
4012
4046
  },
4013
4047
  "name": "edgeLambdas",
4014
4048
  "optional": true,
@@ -4820,7 +4854,7 @@
4820
4854
  },
4821
4855
  "locationInModule": {
4822
4856
  "filename": "src/MicroAppsCF.ts",
4823
- "line": 350
4857
+ "line": 394
4824
4858
  },
4825
4859
  "parameters": [
4826
4860
  {
@@ -4849,7 +4883,7 @@
4849
4883
  "kind": "class",
4850
4884
  "locationInModule": {
4851
4885
  "filename": "src/MicroAppsCF.ts",
4852
- "line": 220
4886
+ "line": 232
4853
4887
  },
4854
4888
  "methods": [
4855
4889
  {
@@ -4859,7 +4893,7 @@
4859
4893
  },
4860
4894
  "locationInModule": {
4861
4895
  "filename": "src/MicroAppsCF.ts",
4862
- "line": 279
4896
+ "line": 291
4863
4897
  },
4864
4898
  "name": "addRoutes",
4865
4899
  "parameters": [
@@ -4886,7 +4920,7 @@
4886
4920
  },
4887
4921
  "locationInModule": {
4888
4922
  "filename": "src/MicroAppsCF.ts",
4889
- "line": 234
4923
+ "line": 246
4890
4924
  },
4891
4925
  "name": "createAPIOriginPolicy",
4892
4926
  "parameters": [
@@ -4921,7 +4955,7 @@
4921
4955
  "immutable": true,
4922
4956
  "locationInModule": {
4923
4957
  "filename": "src/MicroAppsCF.ts",
4924
- "line": 346
4958
+ "line": 390
4925
4959
  },
4926
4960
  "name": "cloudFrontDistro",
4927
4961
  "overrides": "@pwrdrvr/microapps-cdk.IMicroAppsCF",
@@ -7236,6 +7270,6 @@
7236
7270
  "symbolId": "src/MicroAppsTable:MicroAppsTableProps"
7237
7271
  }
7238
7272
  },
7239
- "version": "1.2.0-beta.3",
7240
- "fingerprint": "lcGtODKl072d9v+OJWG7nFPhk65ICoE0dvRu4WZt8No="
7273
+ "version": "1.2.0-beta.5",
7274
+ "fingerprint": "goEweZ8IxkdZsynGJihzX0HbmkbY2U7RUHcLqTJQ/bQ="
7241
7275
  }
package/API.md CHANGED
@@ -1077,6 +1077,8 @@ const addRoutesOptions: AddRoutesOptions = { ... }
1077
1077
  | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.appOriginRequestPolicy">appOriginRequestPolicy</a></code> | <code>aws-cdk-lib.aws_cloudfront.IOriginRequestPolicy</code> | Origin Request policy for API Gateway Origin. |
1078
1078
  | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.bucketOriginFallbackToApp">bucketOriginFallbackToApp</a></code> | <code>aws-cdk-lib.aws_cloudfront_origins.OriginGroup</code> | Origin Group with Primary of S3 bucket with `x-microapps-origin: s3` custom header and Fallback of `appOnlyOrigin`. |
1079
1079
  | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.distro">distro</a></code> | <code>aws-cdk-lib.aws_cloudfront.Distribution</code> | CloudFront Distribution to add the Behaviors (Routes) to. |
1080
+ | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.createAPIPathRoute">createAPIPathRoute</a></code> | <code>boolean</code> | When true, routes that contain `/api/` get sent to the app origin even when other path segments contain periods. |
1081
+ | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.createNextDataPathRoute">createNextDataPathRoute</a></code> | <code>boolean</code> | When true, routes that contain `/_next/data/` get sent to the app origin even when the request path ends in `.json`. |
1080
1082
  | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.edgeLambdas">edgeLambdas</a></code> | <code>aws-cdk-lib.aws_cloudfront.EdgeLambda[]</code> | Edge lambdas to associate with the API Gateway routes. |
1081
1083
  | <code><a href="#@pwrdrvr/microapps-cdk.AddRoutesOptions.property.rootPathPrefix">rootPathPrefix</a></code> | <code>string</code> | Path prefix on the root of the CloudFront distribution. |
1082
1084
 
@@ -1134,6 +1136,30 @@ CloudFront Distribution to add the Behaviors (Routes) to.
1134
1136
 
1135
1137
  ---
1136
1138
 
1139
+ ##### `createAPIPathRoute`<sup>Optional</sup> <a name="createAPIPathRoute" id="@pwrdrvr/microapps-cdk.AddRoutesOptions.property.createAPIPathRoute"></a>
1140
+
1141
+ ```typescript
1142
+ public readonly createAPIPathRoute: boolean;
1143
+ ```
1144
+
1145
+ - *Type:* boolean
1146
+
1147
+ When true, routes that contain `/api/` get sent to the app origin even when other path segments contain periods.
1148
+
1149
+ ---
1150
+
1151
+ ##### `createNextDataPathRoute`<sup>Optional</sup> <a name="createNextDataPathRoute" id="@pwrdrvr/microapps-cdk.AddRoutesOptions.property.createNextDataPathRoute"></a>
1152
+
1153
+ ```typescript
1154
+ public readonly createNextDataPathRoute: boolean;
1155
+ ```
1156
+
1157
+ - *Type:* boolean
1158
+
1159
+ When true, routes that contain `/_next/data/` get sent to the app origin even when the request path ends in `.json`.
1160
+
1161
+ ---
1162
+
1137
1163
  ##### `edgeLambdas`<sup>Optional</sup> <a name="edgeLambdas" id="@pwrdrvr/microapps-cdk.AddRoutesOptions.property.edgeLambdas"></a>
1138
1164
 
1139
1165
  ```typescript
package/README.md CHANGED
@@ -18,6 +18,7 @@ For development / testing purposes only, each version of an applicaton can be ac
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
+ - [Monorepo Boundary Enforcement](#monorepo-boundary-enforcement)
21
22
  - [Tutorial - Bootstrapping a Deploy](#tutorial---bootstrapping-a-deploy)
22
23
  - [Limitations / Future Development](#limitations--future-development)
23
24
  - [Related Projects / Components](#related-projects--components)
@@ -68,6 +69,15 @@ Note: requests can also be dispatched into the same account, but this model is m
68
69
  - CDK API docs
69
70
  - Python, DotNet, Java, JS/TS installation instructions
70
71
 
72
+ # Monorepo Boundary Enforcement
73
+
74
+ This repo now relies on two complementary guardrails for workspace boundaries:
75
+
76
+ - `pnpm` uses its default isolated workspace layout at the repo root, so undeclared dependencies are much less likely to appear accidentally through a flattened install.
77
+ - ESLint enforces `import/no-extraneous-dependencies`, including type-only imports, so invalid package-to-package imports fail with a file-level diagnostic during normal lint runs and in CI.
78
+
79
+ In practice, if a package imports another workspace package, declare that dependency in the importing package's `package.json`. Test-only helpers such as `jest-dynalite` should also be declared in the specific package that uses them instead of relying on a root-only dev dependency.
80
+
71
81
  # Tutorial - Bootstrapping a Deploy
72
82
 
73
83
  - `git clone https://github.com/pwrdrvr/microapps-core.git`
@@ -86,7 +96,7 @@ Note: requests can also be dispatched into the same account, but this model is m
86
96
  - Deploys the CDK Stack
87
97
  - Essentially runs two commands along with extraction of outputs:
88
98
  - `npx cdk deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`
89
- - `npx pwrdrvr 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`
99
+ - `npx pwrdrvr 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 packages/cdk/node_modules/@pwrdrvr/microapps-app-release-cdk/lib/static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --no-cache`
90
100
  - URL will be printed as last output
91
101
 
92
102
  # Limitations / Future Development
package/lib/MicroApps.js CHANGED
@@ -118,5 +118,5 @@ class MicroApps extends constructs_1.Construct {
118
118
  }
119
119
  exports.MicroApps = MicroApps;
120
120
  _a = JSII_RTTI_SYMBOL_1;
121
- MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "1.2.0-beta.3" };
121
+ MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "1.2.0-beta.5" };
122
122
  //# 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 { 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 | dynamodb.ITableV2;\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   * List of allowed locale prefixes for pages\n   *\n   * @example: ['en', 'fr', 'es']\n   * @default none\n   */\n  readonly allowedLocalePrefixes?: string[];\n\n  /**\n   * Additional IAM Role ARNs that should be allowed to invoke apps in child accounts\n   */\n  readonly edgeToOriginRoleARNs?: 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\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 _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      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      edgeToOriginRoleARNs = [],\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    this._svcs = new MicroAppsSvcs(this, 'svcs', {\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      edgeToOriginRoleARN: edgeToOriginRoleARNs,\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        allowedLocalePrefixes: props.allowedLocalePrefixes,\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      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"]}
@@ -180,6 +180,16 @@ export interface AddRoutesOptions {
180
180
  * @example dev/
181
181
  */
182
182
  readonly rootPathPrefix?: string;
183
+ /**
184
+ * When true, routes that contain `/api/` get sent to the app origin
185
+ * even when other path segments contain periods.
186
+ */
187
+ readonly createAPIPathRoute?: boolean;
188
+ /**
189
+ * When true, routes that contain `/_next/data/` get sent to the app origin
190
+ * even when the request path ends in `.json`.
191
+ */
192
+ readonly createNextDataPathRoute?: boolean;
183
193
  /**
184
194
  * Edge lambdas to associate with the API Gateway routes
185
195
  */
@@ -66,7 +66,7 @@ class MicroAppsCF extends constructs_1.Construct {
66
66
  * @param props
67
67
  */
68
68
  static addRoutes(_scope, props) {
69
- const { appOnlyOrigin, bucketOriginFallbackToApp, distro, appOriginRequestPolicy, rootPathPrefix = '', } = props;
69
+ const { appOnlyOrigin, bucketOriginFallbackToApp, distro, appOriginRequestPolicy, rootPathPrefix = '', createAPIPathRoute = true, createNextDataPathRoute = true, } = props;
70
70
  //
71
71
  // Add Behaviors
72
72
  //
@@ -97,6 +97,26 @@ class MicroAppsCF extends constructs_1.Construct {
97
97
  edgeLambdas: props.edgeLambdas,
98
98
  };
99
99
  //
100
+ // If a route specifically contains `/api/`, send it to the app origin.
101
+ // Without this behavior, versioned API routes like
102
+ // `/release/0.0.0-pr.106/api/default-version` match the dot-based
103
+ // `*/*.*` static-file behavior below because the version segment contains periods.
104
+ // CloudFront then treats the request as cacheable-only traffic and rejects
105
+ // POST requests before they ever reach the app origin.
106
+ //
107
+ if (createAPIPathRoute) {
108
+ distro.addBehavior(path_1.posix.join(rootPathPrefix, '*/api/*'), appOnlyOrigin, appOnlyBehaviorOptions);
109
+ }
110
+ //
111
+ // If a route specifically contains `/_next/data/`, send it to the app origin.
112
+ // These requests can end in `.json`, so they also need to bypass the
113
+ // dot-based static-file behavior below instead of being mistaken for
114
+ // static assets.
115
+ //
116
+ if (createNextDataPathRoute) {
117
+ distro.addBehavior(path_1.posix.join(rootPathPrefix, '*/_next/data/*'), appOnlyOrigin, appOnlyBehaviorOptions);
118
+ }
119
+ //
100
120
  // Handle designated static assets
101
121
  // Falls back to the app on 403/404
102
122
  //
@@ -126,7 +146,7 @@ class MicroAppsCF extends constructs_1.Construct {
126
146
  (props.r53Zone !== undefined && props.domainNameEdge === undefined)) {
127
147
  throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');
128
148
  }
129
- const { domainNameEdge, removalPolicy, certEdge, assetNameRoot, assetNameSuffix, r53Zone, bucketLogs, bucketAppsOriginS3, bucketAppsOriginApp, rootPathPrefix, edgeLambdas, } = props;
149
+ const { domainNameEdge, removalPolicy, certEdge, assetNameRoot, assetNameSuffix, r53Zone, bucketLogs, bucketAppsOriginS3, bucketAppsOriginApp, rootPathPrefix, createAPIPathRoute = true, createNextDataPathRoute = true, edgeLambdas, } = props;
130
150
  const appOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {
131
151
  assetNameRoot,
132
152
  assetNameSuffix,
@@ -175,6 +195,8 @@ class MicroAppsCF extends constructs_1.Construct {
175
195
  distro: this._cloudFrontDistro,
176
196
  appOriginRequestPolicy,
177
197
  rootPathPrefix,
198
+ createAPIPathRoute,
199
+ createNextDataPathRoute,
178
200
  edgeLambdas,
179
201
  });
180
202
  //
@@ -195,5 +217,5 @@ class MicroAppsCF extends constructs_1.Construct {
195
217
  }
196
218
  exports.MicroAppsCF = MicroAppsCF;
197
219
  _a = JSII_RTTI_SYMBOL_1;
198
- MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "1.2.0-beta.3" };
199
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsCF.js","sourceRoot":"","sources":["../src/MicroAppsCF.ts"],"names":[],"mappings":"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","sourcesContent":["import { posix as posixPath } from 'path';\nimport { 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 cforigins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * Represents a MicroApps CloudFront\n */\nexport interface IMicroAppsCF {\n  /**\n   * The CloudFront distribution\n   */\n  readonly cloudFrontDistro: cf.Distribution;\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsCF`.\n */\nexport interface MicroAppsCFProps {\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   * S3 bucket origin for deployed applications\n   * Marked with `x-microapps-origin: s3`\n   */\n  readonly bucketAppsOriginS3: cforigins.S3Origin;\n\n  /**\n   * S3 bucket origin for deployed applications\n   * Marked with `x-microapps-origin: app`\n   */\n  readonly bucketAppsOriginApp: cforigins.S3Origin;\n\n  /**\n   * S3 bucket for CloudFront logs\n   */\n  readonly bucketLogs?: s3.IBucket;\n\n  /**\n   * CloudFront Distribution domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * API Gateway custom origin domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default - retrieved from httpApi, if possible\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   * ACM Certificate that covers `domainNameEdge` name\n   */\n  readonly certEdge?: 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 CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\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 if httpApi is provided\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 if httpApi is provided\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   *\n   * @default - no edge to API Gateway origin functions added\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\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 - none\n   */\n  readonly originShieldRegion?: string;\n}\n\n/**\n * Options for the `CreateAPIOriginPolicy`\n */\nexport interface CreateAPIOriginPolicyOptions {\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   * Edge domain name used by CloudFront - If set a custom\n   * OriginRequestPolicy will be created that prevents\n   * the Host header from being passed to the origin.\n   */\n  readonly domainNameEdge?: string;\n}\n\n/**\n * Options for `AddRoutes`\n */\nexport interface AddRoutesOptions {\n  /**\n   * Application origin\n   *\n   * Typically an S3 bucket with a `x-microapps-origin: app` custom header\n   *\n   * The request never actually falls through to the S3 bucket.\n   */\n  readonly appOnlyOrigin: cf.IOrigin;\n\n  /**\n   * Origin Group with Primary of S3 bucket with `x-microapps-origin: s3` custom header\n   * and Fallback of `appOnlyOrigin`\n   */\n  readonly bucketOriginFallbackToApp: cforigins.OriginGroup;\n\n  /**\n   * CloudFront Distribution to add the Behaviors (Routes) to\n   */\n  readonly distro: cf.Distribution;\n\n  /**\n   * Origin Request policy for API Gateway Origin\n   */\n  readonly appOriginRequestPolicy: cf.IOriginRequestPolicy;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Edge lambdas to associate with the API Gateway routes\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\n}\n\n/**\n * Create a new MicroApps CloudFront Distribution.\n */\nexport class MicroAppsCF extends Construct implements IMicroAppsCF {\n  /**\n   * Create or get the origin request policy\n   *\n   * If a custom domain name is NOT used for the origin then a policy\n   * will be created.\n   *\n   * If a custom domain name IS used for the origin then the ALL_VIEWER\n   * policy will be returned.  This policy passes the Host header to the\n   * origin, which is fine when using a custom domain name on the origin.\n   *\n   * @param _scope\n   * @param _props\n   */\n  public static createAPIOriginPolicy(\n    _scope: Construct,\n    _props: CreateAPIOriginPolicyOptions,\n  ): cf.IOriginRequestPolicy {\n    // const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;\n\n    // let apigwyOriginRequestPolicy: cf.IOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;\n    // if (domainNameEdge === undefined) {\n    //   // When not using a custom domain name we must limit down the origin policy to\n    //   // prevent it from passing the Host header (distribution_id.cloudfront.net) to\n    //   // apigwy which will then reject it with a 403 because it does not match the\n    //   // execute-api name that apigwy is expecting.\n    //   //\n    //   // 2021-12-28 - There is a bug in the name generation that causes the same asset\n    //   // in different stacks to have the same generated name.  We have to make the id\n    //   // in all cases to ensure the generated name is unique.\n    //   apigwyOriginRequestPolicy = new cf.OriginRequestPolicy(\n    //     scope,\n    //     `apigwy-origin-policy-${Stack.of(scope).stackName}`,\n    //     {\n    //       comment: assetNameRoot ? `${assetNameRoot}-apigwy${assetNameSuffix}` : undefined,\n\n    //       originRequestPolicyName: assetNameRoot\n    //         ? `${assetNameRoot}-apigwy${assetNameSuffix}`\n    //         : undefined,\n    //       cookieBehavior: cf.OriginRequestCookieBehavior.all(),\n    //       queryStringBehavior: cf.OriginRequestQueryStringBehavior.all(),\n    //       // TODO: If signing is enabled this should forward all signature headers\n    //       // TODO: If set to \"cfront.OriginRequestHeaderBehavior.all()\" then\n    //       // `replaceHostHeader` must be set to true to prevent API Gateway from rejecting\n    //       // the request\n    //       // headerBehavior: cf.OriginRequestHeaderBehavior.allowList('user-agent', 'referer'),\n    //       headerBehavior: cf.OriginRequestHeaderBehavior.all(),\n    //     },\n    //   );\n    // }\n\n    return cf.OriginRequestPolicy.ALL_VIEWER;\n  }\n\n  /**\n   * Add API Gateway and S3 routes to an existing CloudFront Distribution\n   * @param _scope\n   * @param props\n   */\n  public static addRoutes(_scope: Construct, props: AddRoutesOptions) {\n    const {\n      appOnlyOrigin,\n      bucketOriginFallbackToApp,\n      distro,\n      appOriginRequestPolicy,\n      rootPathPrefix = '',\n    } = props;\n\n    //\n    // Add Behaviors\n    //\n    const s3BehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachePolicy: cf.CachePolicy.CACHING_OPTIMIZED,\n      compress: true,\n      originRequestPolicy: cf.OriginRequestPolicy.ALL_VIEWER,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n    const s3FallbackToAppOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: appOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n    const appOnlyBehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: appOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n\n    //\n    // Handle designated static assets\n    // Falls back to the app on 403/404\n    //\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '*/static/*.*'),\n      // TODO: 2023-03-04 - This could be the bucket origin without fallback, at least as an option\n      // which would allow skipping the OriginRequest (which has invoke cost and throughtput limits).\n      // This would be a distinct config from the `*/*.*` route below, which always has to have\n      // the OriginRequest function.\n      bucketOriginFallbackToApp,\n      s3BehaviorOptions,\n    );\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '*/*.*'),\n      bucketOriginFallbackToApp,\n      s3FallbackToAppOptions,\n    );\n\n    //\n    // Default to sending everything else to the app only\n    // This is necessary because we allow all methods for the app but that is not\n    // allowed for an OriginGroup\n    //\n    distro.addBehavior(posixPath.join(rootPathPrefix, '/*'), appOnlyOrigin, appOnlyBehaviorOptions);\n  }\n\n  private _cloudFrontDistro: cf.Distribution;\n  public get cloudFrontDistro(): cf.Distribution {\n    return this._cloudFrontDistro;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsCFProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\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    const {\n      domainNameEdge,\n      removalPolicy,\n      certEdge,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      bucketLogs,\n      bucketAppsOriginS3,\n      bucketAppsOriginApp,\n      rootPathPrefix,\n      edgeLambdas,\n    } = props;\n\n    const appOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n    });\n\n    //\n    // Create fallback to S3 origin group\n    //\n    const appOrigin = bucketAppsOriginApp ?? bucketAppsOriginS3;\n    const bucketOriginFallbackToApp = new cforigins.OriginGroup({\n      primaryOrigin: bucketAppsOriginS3,\n      fallbackOrigin: appOrigin,\n      fallbackStatusCodes: [403, 404],\n    });\n\n    //\n    // CloudFront Distro\n    //\n    this._cloudFrontDistro = new cf.Distribution(this, 'cft', {\n      comment: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : domainNameEdge,\n      domainNames: domainNameEdge !== undefined ? [domainNameEdge] : undefined,\n      certificate: certEdge,\n      httpVersion: cf.HttpVersion.HTTP2,\n      defaultBehavior: {\n        allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n        cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n        compress: true,\n        originRequestPolicy: appOriginRequestPolicy,\n        origin: appOrigin,\n        viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        edgeLambdas,\n      },\n      enableIpv6: true,\n      priceClass: cf.PriceClass.PRICE_CLASS_100,\n      logBucket: bucketLogs,\n      logFilePrefix: props.domainNameEdge\n        ? `${reverseDomain(props.domainNameEdge)}/cloudfront-raw/`\n        : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._cloudFrontDistro.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Add routes to the CloudFront Distribution\n    MicroAppsCF.addRoutes(scope, {\n      appOnlyOrigin: appOrigin,\n      bucketOriginFallbackToApp,\n      distro: this._cloudFrontDistro,\n      appOriginRequestPolicy,\n      rootPathPrefix,\n      edgeLambdas,\n    });\n\n    //\n    // Create the edge name for the CloudFront distro\n    //\n\n    if (r53Zone !== undefined) {\n      const rrAppsEdge = new r53.RecordSet(this, 'edge-arecord', {\n        recordName: domainNameEdge,\n        recordType: r53.RecordType.A,\n        target: r53.RecordTarget.fromAlias(new r53targets.CloudFrontTarget(this._cloudFrontDistro)),\n        zone: r53Zone,\n      });\n      if (removalPolicy !== undefined) {\n        rrAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
220
+ MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "1.2.0-beta.5" };
221
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsCF.js","sourceRoot":"","sources":["../src/MicroAppsCF.ts"],"names":[],"mappings":"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","sourcesContent":["import { posix as posixPath } from 'path';\nimport { 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 cforigins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * Represents a MicroApps CloudFront\n */\nexport interface IMicroAppsCF {\n  /**\n   * The CloudFront distribution\n   */\n  readonly cloudFrontDistro: cf.Distribution;\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsCF`.\n */\nexport interface MicroAppsCFProps {\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   * S3 bucket origin for deployed applications\n   * Marked with `x-microapps-origin: s3`\n   */\n  readonly bucketAppsOriginS3: cforigins.S3Origin;\n\n  /**\n   * S3 bucket origin for deployed applications\n   * Marked with `x-microapps-origin: app`\n   */\n  readonly bucketAppsOriginApp: cforigins.S3Origin;\n\n  /**\n   * S3 bucket for CloudFront logs\n   */\n  readonly bucketLogs?: s3.IBucket;\n\n  /**\n   * CloudFront Distribution domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * API Gateway custom origin domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default - retrieved from httpApi, if possible\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   * ACM Certificate that covers `domainNameEdge` name\n   */\n  readonly certEdge?: 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 CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\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 if httpApi is provided\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 if httpApi is provided\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   *\n   * @default - no edge to API Gateway origin functions added\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\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 - none\n   */\n  readonly originShieldRegion?: string;\n}\n\n/**\n * Options for the `CreateAPIOriginPolicy`\n */\nexport interface CreateAPIOriginPolicyOptions {\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   * Edge domain name used by CloudFront - If set a custom\n   * OriginRequestPolicy will be created that prevents\n   * the Host header from being passed to the origin.\n   */\n  readonly domainNameEdge?: string;\n}\n\n/**\n * Options for `AddRoutes`\n */\nexport interface AddRoutesOptions {\n  /**\n   * Application origin\n   *\n   * Typically an S3 bucket with a `x-microapps-origin: app` custom header\n   *\n   * The request never actually falls through to the S3 bucket.\n   */\n  readonly appOnlyOrigin: cf.IOrigin;\n\n  /**\n   * Origin Group with Primary of S3 bucket with `x-microapps-origin: s3` custom header\n   * and Fallback of `appOnlyOrigin`\n   */\n  readonly bucketOriginFallbackToApp: cforigins.OriginGroup;\n\n  /**\n   * CloudFront Distribution to add the Behaviors (Routes) to\n   */\n  readonly distro: cf.Distribution;\n\n  /**\n   * Origin Request policy for API Gateway Origin\n   */\n  readonly appOriginRequestPolicy: cf.IOriginRequestPolicy;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * When true, routes that contain `/api/` get sent to the app origin\n   * even when other path segments contain periods.\n   */\n  readonly createAPIPathRoute?: boolean;\n\n  /**\n   * When true, routes that contain `/_next/data/` get sent to the app origin\n   * even when the request path ends in `.json`.\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Edge lambdas to associate with the API Gateway routes\n   */\n  readonly edgeLambdas?: cf.EdgeLambda[];\n}\n\n/**\n * Create a new MicroApps CloudFront Distribution.\n */\nexport class MicroAppsCF extends Construct implements IMicroAppsCF {\n  /**\n   * Create or get the origin request policy\n   *\n   * If a custom domain name is NOT used for the origin then a policy\n   * will be created.\n   *\n   * If a custom domain name IS used for the origin then the ALL_VIEWER\n   * policy will be returned.  This policy passes the Host header to the\n   * origin, which is fine when using a custom domain name on the origin.\n   *\n   * @param _scope\n   * @param _props\n   */\n  public static createAPIOriginPolicy(\n    _scope: Construct,\n    _props: CreateAPIOriginPolicyOptions,\n  ): cf.IOriginRequestPolicy {\n    // const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;\n\n    // let apigwyOriginRequestPolicy: cf.IOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;\n    // if (domainNameEdge === undefined) {\n    //   // When not using a custom domain name we must limit down the origin policy to\n    //   // prevent it from passing the Host header (distribution_id.cloudfront.net) to\n    //   // apigwy which will then reject it with a 403 because it does not match the\n    //   // execute-api name that apigwy is expecting.\n    //   //\n    //   // 2021-12-28 - There is a bug in the name generation that causes the same asset\n    //   // in different stacks to have the same generated name.  We have to make the id\n    //   // in all cases to ensure the generated name is unique.\n    //   apigwyOriginRequestPolicy = new cf.OriginRequestPolicy(\n    //     scope,\n    //     `apigwy-origin-policy-${Stack.of(scope).stackName}`,\n    //     {\n    //       comment: assetNameRoot ? `${assetNameRoot}-apigwy${assetNameSuffix}` : undefined,\n\n    //       originRequestPolicyName: assetNameRoot\n    //         ? `${assetNameRoot}-apigwy${assetNameSuffix}`\n    //         : undefined,\n    //       cookieBehavior: cf.OriginRequestCookieBehavior.all(),\n    //       queryStringBehavior: cf.OriginRequestQueryStringBehavior.all(),\n    //       // TODO: If signing is enabled this should forward all signature headers\n    //       // TODO: If set to \"cfront.OriginRequestHeaderBehavior.all()\" then\n    //       // `replaceHostHeader` must be set to true to prevent API Gateway from rejecting\n    //       // the request\n    //       // headerBehavior: cf.OriginRequestHeaderBehavior.allowList('user-agent', 'referer'),\n    //       headerBehavior: cf.OriginRequestHeaderBehavior.all(),\n    //     },\n    //   );\n    // }\n\n    return cf.OriginRequestPolicy.ALL_VIEWER;\n  }\n\n  /**\n   * Add API Gateway and S3 routes to an existing CloudFront Distribution\n   * @param _scope\n   * @param props\n   */\n  public static addRoutes(_scope: Construct, props: AddRoutesOptions) {\n    const {\n      appOnlyOrigin,\n      bucketOriginFallbackToApp,\n      distro,\n      appOriginRequestPolicy,\n      rootPathPrefix = '',\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n    } = props;\n\n    //\n    // Add Behaviors\n    //\n    const s3BehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachePolicy: cf.CachePolicy.CACHING_OPTIMIZED,\n      compress: true,\n      originRequestPolicy: cf.OriginRequestPolicy.ALL_VIEWER,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n    const s3FallbackToAppOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: appOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n    const appOnlyBehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: appOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.edgeLambdas,\n    };\n\n    //\n    // If a route specifically contains `/api/`, send it to the app origin.\n    // Without this behavior, versioned API routes like\n    // `/release/0.0.0-pr.106/api/default-version` match the dot-based\n    // `*/*.*` static-file behavior below because the version segment contains periods.\n    // CloudFront then treats the request as cacheable-only traffic and rejects\n    // POST requests before they ever reach the app origin.\n    //\n    if (createAPIPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '*/api/*'),\n        appOnlyOrigin,\n        appOnlyBehaviorOptions,\n      );\n    }\n\n    //\n    // If a route specifically contains `/_next/data/`, send it to the app origin.\n    // These requests can end in `.json`, so they also need to bypass the\n    // dot-based static-file behavior below instead of being mistaken for\n    // static assets.\n    //\n    if (createNextDataPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '*/_next/data/*'),\n        appOnlyOrigin,\n        appOnlyBehaviorOptions,\n      );\n    }\n\n    //\n    // Handle designated static assets\n    // Falls back to the app on 403/404\n    //\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '*/static/*.*'),\n      // TODO: 2023-03-04 - This could be the bucket origin without fallback, at least as an option\n      // which would allow skipping the OriginRequest (which has invoke cost and throughtput limits).\n      // This would be a distinct config from the `*/*.*` route below, which always has to have\n      // the OriginRequest function.\n      bucketOriginFallbackToApp,\n      s3BehaviorOptions,\n    );\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '*/*.*'),\n      bucketOriginFallbackToApp,\n      s3FallbackToAppOptions,\n    );\n\n    //\n    // Default to sending everything else to the app only\n    // This is necessary because we allow all methods for the app but that is not\n    // allowed for an OriginGroup\n    //\n    distro.addBehavior(posixPath.join(rootPathPrefix, '/*'), appOnlyOrigin, appOnlyBehaviorOptions);\n  }\n\n  private _cloudFrontDistro: cf.Distribution;\n  public get cloudFrontDistro(): cf.Distribution {\n    return this._cloudFrontDistro;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsCFProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\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    const {\n      domainNameEdge,\n      removalPolicy,\n      certEdge,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      bucketLogs,\n      bucketAppsOriginS3,\n      bucketAppsOriginApp,\n      rootPathPrefix,\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n      edgeLambdas,\n    } = props;\n\n    const appOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n    });\n\n    //\n    // Create fallback to S3 origin group\n    //\n    const appOrigin = bucketAppsOriginApp ?? bucketAppsOriginS3;\n    const bucketOriginFallbackToApp = new cforigins.OriginGroup({\n      primaryOrigin: bucketAppsOriginS3,\n      fallbackOrigin: appOrigin,\n      fallbackStatusCodes: [403, 404],\n    });\n\n    //\n    // CloudFront Distro\n    //\n    this._cloudFrontDistro = new cf.Distribution(this, 'cft', {\n      comment: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : domainNameEdge,\n      domainNames: domainNameEdge !== undefined ? [domainNameEdge] : undefined,\n      certificate: certEdge,\n      httpVersion: cf.HttpVersion.HTTP2,\n      defaultBehavior: {\n        allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n        cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n        compress: true,\n        originRequestPolicy: appOriginRequestPolicy,\n        origin: appOrigin,\n        viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        edgeLambdas,\n      },\n      enableIpv6: true,\n      priceClass: cf.PriceClass.PRICE_CLASS_100,\n      logBucket: bucketLogs,\n      logFilePrefix: props.domainNameEdge\n        ? `${reverseDomain(props.domainNameEdge)}/cloudfront-raw/`\n        : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._cloudFrontDistro.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Add routes to the CloudFront Distribution\n    MicroAppsCF.addRoutes(scope, {\n      appOnlyOrigin: appOrigin,\n      bucketOriginFallbackToApp,\n      distro: this._cloudFrontDistro,\n      appOriginRequestPolicy,\n      rootPathPrefix,\n      createAPIPathRoute,\n      createNextDataPathRoute,\n      edgeLambdas,\n    });\n\n    //\n    // Create the edge name for the CloudFront distro\n    //\n\n    if (r53Zone !== undefined) {\n      const rrAppsEdge = new r53.RecordSet(this, 'edge-arecord', {\n        recordName: domainNameEdge,\n        recordType: r53.RecordType.A,\n        target: r53.RecordTarget.fromAlias(new r53targets.CloudFrontTarget(this._cloudFrontDistro)),\n        zone: r53Zone,\n      });\n      if (removalPolicy !== undefined) {\n        rrAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
@@ -123,5 +123,5 @@ class MicroAppsChildDeployer extends constructs_1.Construct {
123
123
  }
124
124
  exports.MicroAppsChildDeployer = MicroAppsChildDeployer;
125
125
  _a = JSII_RTTI_SYMBOL_1;
126
- MicroAppsChildDeployer[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsChildDeployer", version: "1.2.0-beta.3" };
126
+ MicroAppsChildDeployer[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsChildDeployer", version: "1.2.0-beta.5" };
127
127
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsChildDeployer.js","sourceRoot":"","sources":["../src/MicroAppsChildDeployer.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from 'fs';\nimport * as path from 'path';\nimport { Aws, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport { Construct } from 'constructs';\n\n/**\n * Properties to initialize an instance of `MicroAppsChildDeployer`.\n */\nexport interface MicroAppsChildDeployerProps {\n  /**\n   * ARN of the parent Deployer Lambda Function\n   */\n  readonly parentDeployerLambdaARN: string;\n\n  /**\n   * ARN of the IAM Role for the Edge to Origin Lambda Function\n   *\n   * For child accounts this can be blank as it is retrieved from the parent Deployer\n   */\n  readonly edgeToOriginRoleARN?: string;\n\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   * Application environment, passed as `NODE_ENV`\n   * to the Router and Deployer Lambda functions\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   * Deployer timeout\n   *\n   * For larger applications this needs to be set up to 2-5 minutes for the S3 copy\n   *\n   * @default 2 minutes\n   */\n  readonly deployerTimeout?: Duration;\n}\n\n/**\n * Represents a MicroApps Child Deployer\n */\nexport interface IMicroAppsChildDeployer {\n  /**\n   * Lambda function for the Deployer\n   */\n  readonly deployerFunc: lambda.IFunction;\n}\n\n/**\n * Create a new MicroApps Child Deployer construct.\n */\nexport class MicroAppsChildDeployer extends Construct implements IMicroAppsChildDeployer {\n  private _deployerFunc: lambda.Function;\n  public get deployerFunc(): lambda.IFunction {\n    return this._deployerFunc;\n  }\n\n  constructor(scope: Construct, id: string, props?: MicroAppsChildDeployerProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const {\n      appEnv,\n      deployerTimeout = Duration.minutes(2),\n      assetNameRoot,\n      assetNameSuffix,\n      removalPolicy,\n      parentDeployerLambdaARN,\n      edgeToOriginRoleARN,\n    } = props;\n\n    //\n    // Deployer Lambda Function\n    //\n\n    const iamRoleDeployerName = assetNameRoot\n      ? `${assetNameRoot}-deployer${assetNameSuffix}`\n      : undefined;\n    const iamRoleDeployer = new iam.Role(this, 'deployer-role', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      roleName: iamRoleDeployerName,\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),\n      ],\n      inlinePolicies: {\n        deployPolicy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              effect: iam.Effect.ALLOW,\n              actions: ['lambda:InvokeFunction'],\n              resources: [`${parentDeployerLambdaARN}:currentVersion`],\n            }),\n          ],\n        }),\n      },\n    });\n\n    // Create Deployer Lambda Function\n    const deployerFuncName = assetNameRoot\n      ? `${assetNameRoot}-deployer${assetNameSuffix}`\n      : undefined;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      role: iamRoleDeployer,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_22_X,\n      timeout: deployerTimeout,\n      environment: {\n        NODE_ENV: appEnv,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        PARENT_DEPLOYER_LAMBDA_ARN: parentDeployerLambdaARN,\n        ...(edgeToOriginRoleARN ? { EDGE_TO_ORIGIN_ROLE_ARN: edgeToOriginRoleARN } : {}),\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-deployer', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-deployer')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      this._deployerFunc = new lambdaNodejs.NodejsFunction(this, 'deployer-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-deployer', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._deployerFunc.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Grant full control over lambdas that indicate they are microapps\n    const policyAPIManageLambdas = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['lambda:*'],\n      resources: [\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*`,\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*:*`,\n      ],\n      conditions: {\n        StringEquals: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n    const policyReadonlyLambdas = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['lambda:GetFunction', 'lambda:GetAlias'],\n      resources: [\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*`,\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*:*`,\n      ],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadonlyLambdas);\n  }\n}\n"]}